Coverage Report

Created: 2025-07-18 06:24

/src/dovecot/src/lib/array.h
Line
Count
Source (jump to first uncovered line)
1
#ifndef ARRAY_H
2
#define ARRAY_H
3
4
/* Array is a buffer accessible using fixed size elements. As long as the
5
   compiler provides a typeof() operator, the array provides type safety. If
6
   a wrong type is tried to be added to the array, or if the array's contents
7
   are tried to be used using a wrong type, the compiler will give a warning.
8
9
   Example usage:
10
11
   struct foo {
12
  ARRAY(struct bar) bars;
13
  ...
14
   };
15
16
   i_array_init(&foo->bars, 10);
17
18
   struct bar *bar = array_idx(&foo->bars, 5);
19
   struct baz *baz = array_idx(&foo->bars, 5); // compiler warning
20
21
   If you want to pass an array as a parameter to a function, you'll need to
22
   create a type for the array using ARRAY_DEFINE_TYPE() and use the type in
23
   the parameter using ARRAY_TYPE(). Any arrays that you want to be passing
24
   around, such as structure members as in the above example, must also be
25
   defined using ARRAY_TYPE() too, rather than ARRAY().
26
27
   Example:
28
29
   ARRAY_DEFINE_TYPE(foo, struct foo);
30
   void do_foo(ARRAY_TYPE(foo) *foos) {
31
  struct foo *foo = array_idx(foos, 0);
32
   }
33
   struct foo_manager {
34
  ARRAY_TYPE(foo) foos; // pedantically, ARRAY(struct foo) is a different type
35
   };
36
   // ...
37
  do_foo(&my_foo_manager->foos); // No compiler warning about mismatched types
38
39
*/
40
#include "array-decl.h"
41
#include "buffer.h"
42
43
#define p_array_init(array, pool, init_count) \
44
15.8M
  array_create(array, pool, sizeof(**(array)->v), init_count) // NOLINT(bugprone-sizeof-expression)
45
#define i_array_init(array, init_count) \
46
8
  p_array_init(array, default_pool, init_count)
47
#define t_array_init(array, init_count) \
48
0
  p_array_init(array, pool_datastack_create(), init_count)
49
50
#ifdef HAVE_TYPEOF
51
#  define ARRAY_TYPE_CAST_CONST(array) \
52
7.15M
  (typeof(*(array)->v))
53
#  define ARRAY_TYPE_CAST_MODIFIABLE(array) \
54
70.6M
  (typeof(*(array)->v_modifiable))
55
#  define ARRAY_TYPE_CHECK(array, data) \
56
  COMPILE_ERROR_IF_TYPES_NOT_COMPATIBLE( \
57
    **(array)->v_modifiable, *(data))
58
#  define ARRAY_TYPES_CHECK(array1, array2) \
59
  COMPILE_ERROR_IF_TYPES_NOT_COMPATIBLE( \
60
    **(array1)->v_modifiable, **(array2)->v_modifiable)
61
62
#else
63
#  define ARRAY_TYPE_CAST_CONST(array)
64
#  define ARRAY_TYPE_CAST_MODIFIABLE(array)
65
#  define ARRAY_TYPE_CHECK(array, data) 0
66
#  define ARRAY_TYPES_CHECK(array1, array2) 0
67
#endif
68
69
/* Usage:
70
   ARRAY(struct foo) foo_arr;
71
   struct foo *foo;
72
73
   array_foreach(&foo_arr, foo) {
74
     ..
75
   }
76
77
   Note that deleting an element while iterating will cause the iteration to
78
   skip over the next element. So deleting a single element and breaking out
79
   of the loop is fine, but continuing the loop is likely a bug. Use
80
   array_foreach_reverse() instead when deleting multiple elements.
81
*/
82
#define array_foreach(array, elem) \
83
0
  for (const void *elem ## __foreach_end = \
84
0
    (const char *)(elem = *(array)->v) + (array)->arr.buffer->used; \
85
0
       elem != elem ## __foreach_end; (elem)++)
86
#define array_foreach_modifiable(array, elem) \
87
39
  for (const void *elem ## _end = \
88
39
    (const char *)(elem = ARRAY_TYPE_CAST_MODIFIABLE(array) \
89
39
      buffer_get_modifiable_data((array)->arr.buffer, NULL)) + \
90
39
      (array)->arr.buffer->used; \
91
119
       elem != elem ## _end; (elem)++)
92
93
/* Iterate the array in reverse order. */
94
#define array_foreach_reverse(array, elem) \
95
  for (elem = CONST_PTR_OFFSET(*(array)->v, (array)->arr.buffer->used); \
96
       (const char *)(elem--) > (const char *)*(array)->v; )
97
#define array_foreach_reverse_modifiable(array, elem) \
98
  for (elem = ARRAY_TYPE_CAST_MODIFIABLE(array) \
99
    ((char *)buffer_get_modifiable_data((array)->arr.buffer, NULL) + \
100
     (array)->arr.buffer->used); \
101
       (const char *)(elem--) > (const char *)*(array)->v; )
102
103
/* Usage:
104
   ARRAY(struct foo *) foo_ptrs_arr;
105
   struct foo *foo;
106
107
   array_foreach_elem(&foo_ptrs_arr, foo) {
108
     ..
109
   } */
110
#define array_foreach_elem(array, elem) \
111
1
  for (const void *_foreach_end = \
112
1
    /* NOLINTBEGIN(bugprone-sizeof-expression) */ \
113
1
    CONST_PTR_OFFSET(*(array)->v, (array)->arr.buffer->used), \
114
1
       *_foreach_ptr = CONST_PTR_OFFSET(*(array)->v, ARRAY_TYPE_CHECK(array, &elem) + \
115
1
    COMPILE_ERROR_IF_TRUE(sizeof(elem) > sizeof(void *))) \
116
1
         ;              \
117
1
       (_foreach_ptr != _foreach_end &&   \
118
1
       (memcpy(&elem, _foreach_ptr, sizeof(elem)), TRUE)) \
119
1
    ;             \
120
1
       _foreach_ptr = CONST_PTR_OFFSET(_foreach_ptr, sizeof(elem))) \
121
       /* NOLINTEND(bugprone-sizeof-expression) */ \
122
123
124
#define array_ptr_to_idx(array, elem) \
125
0
  ((elem) - (array)->v[0])
126
/* Return index of iterated element inside array_foreach() or
127
   array_foreach_modifiable() loop. Note that this doesn't work inside
128
   array_foreach_elem() loop. */
129
#define array_foreach_idx(array, elem) \
130
0
  array_ptr_to_idx(array, elem)
131
132
static inline void
133
array_create_from_buffer_i(struct array *array, buffer_t *buffer,
134
         size_t element_size)
135
15.8M
{
136
15.8M
  array->buffer = buffer;
137
15.8M
  array->element_size = element_size;
138
15.8M
}
Unexecuted instantiation: imap-bodystructure.c:array_create_from_buffer_i
Unexecuted instantiation: imap-envelope.c:array_create_from_buffer_i
imap-parser.c:array_create_from_buffer_i
Line
Count
Source
135
15.8M
{
136
15.8M
  array->buffer = buffer;
137
15.8M
  array->element_size = element_size;
138
15.8M
}
Unexecuted instantiation: imap-quote.c:array_create_from_buffer_i
Unexecuted instantiation: imap-arg.c:array_create_from_buffer_i
Unexecuted instantiation: istream.c:array_create_from_buffer_i
ioloop.c:array_create_from_buffer_i
Line
Count
Source
135
2
{
136
2
  array->buffer = buffer;
137
2
  array->element_size = element_size;
138
2
}
Unexecuted instantiation: ioloop-epoll.c:array_create_from_buffer_i
lib.c:array_create_from_buffer_i
Line
Count
Source
135
1
{
136
1
  array->buffer = buffer;
137
1
  array->element_size = element_size;
138
1
}
lib-event.c:array_create_from_buffer_i
Line
Count
Source
135
5
{
136
5
  array->buffer = buffer;
137
5
  array->element_size = element_size;
138
5
}
Unexecuted instantiation: lib-signals.c:array_create_from_buffer_i
Unexecuted instantiation: mempool.c:array_create_from_buffer_i
Unexecuted instantiation: priorityq.c:array_create_from_buffer_i
Unexecuted instantiation: strfuncs.c:array_create_from_buffer_i
Unexecuted instantiation: unichar.c:array_create_from_buffer_i
Unexecuted instantiation: array.c:array_create_from_buffer_i
Unexecuted instantiation: env-util.c:array_create_from_buffer_i
Unexecuted instantiation: event-filter.c:array_create_from_buffer_i
Unexecuted instantiation: iostream.c:array_create_from_buffer_i
139
#define array_create_from_buffer(array, buffer, element_size) \
140
  array_create_from_buffer_i(&(array)->arr, buffer, element_size)
141
142
static inline void
143
array_create_i(struct array *array, pool_t pool,
144
         size_t element_size, unsigned int init_count)
145
15.8M
{
146
15.8M
  buffer_t *buffer;
147
148
15.8M
  buffer = buffer_create_dynamic_max(pool, init_count * element_size,
149
15.8M
    SIZE_MAX / element_size < UINT_MAX ? SIZE_MAX :
150
15.8M
    UINT_MAX * element_size);
151
15.8M
  array_create_from_buffer_i(array, buffer, element_size);
152
15.8M
}
Unexecuted instantiation: imap-bodystructure.c:array_create_i
Unexecuted instantiation: imap-envelope.c:array_create_i
imap-parser.c:array_create_i
Line
Count
Source
145
15.8M
{
146
15.8M
  buffer_t *buffer;
147
148
15.8M
  buffer = buffer_create_dynamic_max(pool, init_count * element_size,
149
15.8M
    SIZE_MAX / element_size < UINT_MAX ? SIZE_MAX :
150
15.8M
    UINT_MAX * element_size);
151
15.8M
  array_create_from_buffer_i(array, buffer, element_size);
152
15.8M
}
Unexecuted instantiation: imap-quote.c:array_create_i
Unexecuted instantiation: imap-arg.c:array_create_i
Unexecuted instantiation: istream.c:array_create_i
ioloop.c:array_create_i
Line
Count
Source
145
2
{
146
2
  buffer_t *buffer;
147
148
2
  buffer = buffer_create_dynamic_max(pool, init_count * element_size,
149
2
    SIZE_MAX / element_size < UINT_MAX ? SIZE_MAX :
150
2
    UINT_MAX * element_size);
151
2
  array_create_from_buffer_i(array, buffer, element_size);
152
2
}
Unexecuted instantiation: ioloop-epoll.c:array_create_i
lib.c:array_create_i
Line
Count
Source
145
1
{
146
1
  buffer_t *buffer;
147
148
1
  buffer = buffer_create_dynamic_max(pool, init_count * element_size,
149
1
    SIZE_MAX / element_size < UINT_MAX ? SIZE_MAX :
150
1
    UINT_MAX * element_size);
151
1
  array_create_from_buffer_i(array, buffer, element_size);
152
1
}
lib-event.c:array_create_i
Line
Count
Source
145
5
{
146
5
  buffer_t *buffer;
147
148
5
  buffer = buffer_create_dynamic_max(pool, init_count * element_size,
149
5
    SIZE_MAX / element_size < UINT_MAX ? SIZE_MAX :
150
5
    UINT_MAX * element_size);
151
5
  array_create_from_buffer_i(array, buffer, element_size);
152
5
}
Unexecuted instantiation: lib-signals.c:array_create_i
Unexecuted instantiation: mempool.c:array_create_i
Unexecuted instantiation: priorityq.c:array_create_i
Unexecuted instantiation: strfuncs.c:array_create_i
Unexecuted instantiation: unichar.c:array_create_i
Unexecuted instantiation: array.c:array_create_i
Unexecuted instantiation: env-util.c:array_create_i
Unexecuted instantiation: event-filter.c:array_create_i
Unexecuted instantiation: iostream.c:array_create_i
153
#define array_create(array, pool, element_size, init_count) \
154
15.8M
  array_create_i(&(array)->arr, pool, element_size, init_count)
155
156
static inline void
157
array_free_i(struct array *array)
158
0
{
159
0
  buffer_free(&array->buffer);
160
0
}
Unexecuted instantiation: imap-bodystructure.c:array_free_i
Unexecuted instantiation: imap-envelope.c:array_free_i
Unexecuted instantiation: imap-parser.c:array_free_i
Unexecuted instantiation: imap-quote.c:array_free_i
Unexecuted instantiation: imap-arg.c:array_free_i
Unexecuted instantiation: istream.c:array_free_i
Unexecuted instantiation: ioloop.c:array_free_i
Unexecuted instantiation: ioloop-epoll.c:array_free_i
Unexecuted instantiation: lib.c:array_free_i
Unexecuted instantiation: lib-event.c:array_free_i
Unexecuted instantiation: lib-signals.c:array_free_i
Unexecuted instantiation: mempool.c:array_free_i
Unexecuted instantiation: priorityq.c:array_free_i
Unexecuted instantiation: strfuncs.c:array_free_i
Unexecuted instantiation: unichar.c:array_free_i
Unexecuted instantiation: array.c:array_free_i
Unexecuted instantiation: env-util.c:array_free_i
Unexecuted instantiation: event-filter.c:array_free_i
Unexecuted instantiation: iostream.c:array_free_i
161
#define array_free(array) \
162
0
  array_free_i(&(array)->arr)
163
164
static inline void * ATTR_WARN_UNUSED_RESULT
165
array_free_without_data_i(struct array *array)
166
0
{
167
0
  return buffer_free_without_data(&array->buffer);
168
0
}
Unexecuted instantiation: imap-bodystructure.c:array_free_without_data_i
Unexecuted instantiation: imap-envelope.c:array_free_without_data_i
Unexecuted instantiation: imap-parser.c:array_free_without_data_i
Unexecuted instantiation: imap-quote.c:array_free_without_data_i
Unexecuted instantiation: imap-arg.c:array_free_without_data_i
Unexecuted instantiation: istream.c:array_free_without_data_i
Unexecuted instantiation: ioloop.c:array_free_without_data_i
Unexecuted instantiation: ioloop-epoll.c:array_free_without_data_i
Unexecuted instantiation: lib.c:array_free_without_data_i
Unexecuted instantiation: lib-event.c:array_free_without_data_i
Unexecuted instantiation: lib-signals.c:array_free_without_data_i
Unexecuted instantiation: mempool.c:array_free_without_data_i
Unexecuted instantiation: priorityq.c:array_free_without_data_i
Unexecuted instantiation: strfuncs.c:array_free_without_data_i
Unexecuted instantiation: unichar.c:array_free_without_data_i
Unexecuted instantiation: array.c:array_free_without_data_i
Unexecuted instantiation: env-util.c:array_free_without_data_i
Unexecuted instantiation: event-filter.c:array_free_without_data_i
Unexecuted instantiation: iostream.c:array_free_without_data_i
169
#define array_free_without_data(array) \
170
  ARRAY_TYPE_CAST_MODIFIABLE(array)array_free_without_data_i(&(array)->arr)
171
172
static inline bool
173
array_is_created_i(const struct array *array)
174
15.8k
{
175
15.8k
  return array->buffer != NULL;
176
15.8k
}
Unexecuted instantiation: imap-bodystructure.c:array_is_created_i
Unexecuted instantiation: imap-envelope.c:array_is_created_i
Unexecuted instantiation: imap-parser.c:array_is_created_i
Unexecuted instantiation: imap-quote.c:array_is_created_i
Unexecuted instantiation: imap-arg.c:array_is_created_i
Unexecuted instantiation: istream.c:array_is_created_i
ioloop.c:array_is_created_i
Line
Count
Source
174
2
{
175
2
  return array->buffer != NULL;
176
2
}
Unexecuted instantiation: ioloop-epoll.c:array_is_created_i
lib.c:array_is_created_i
Line
Count
Source
174
2
{
175
2
  return array->buffer != NULL;
176
2
}
lib-event.c:array_is_created_i
Line
Count
Source
174
45
{
175
45
  return array->buffer != NULL;
176
45
}
Unexecuted instantiation: lib-signals.c:array_is_created_i
mempool.c:array_is_created_i
Line
Count
Source
174
9.89k
{
175
9.89k
  return array->buffer != NULL;
176
9.89k
}
Unexecuted instantiation: priorityq.c:array_is_created_i
Unexecuted instantiation: strfuncs.c:array_is_created_i
Unexecuted instantiation: unichar.c:array_is_created_i
Unexecuted instantiation: array.c:array_is_created_i
Unexecuted instantiation: env-util.c:array_is_created_i
Unexecuted instantiation: event-filter.c:array_is_created_i
iostream.c:array_is_created_i
Line
Count
Source
174
5.90k
{
175
5.90k
  return array->buffer != NULL;
176
5.90k
}
177
#define array_is_created(array) \
178
15.8k
  array_is_created_i(&(array)->arr)
179
180
static inline pool_t ATTR_PURE
181
array_get_pool_i(struct array *array)
182
0
{
183
0
  return buffer_get_pool(array->buffer);
184
0
}
Unexecuted instantiation: imap-bodystructure.c:array_get_pool_i
Unexecuted instantiation: imap-envelope.c:array_get_pool_i
Unexecuted instantiation: imap-parser.c:array_get_pool_i
Unexecuted instantiation: imap-quote.c:array_get_pool_i
Unexecuted instantiation: imap-arg.c:array_get_pool_i
Unexecuted instantiation: istream.c:array_get_pool_i
Unexecuted instantiation: ioloop.c:array_get_pool_i
Unexecuted instantiation: ioloop-epoll.c:array_get_pool_i
Unexecuted instantiation: lib.c:array_get_pool_i
Unexecuted instantiation: lib-event.c:array_get_pool_i
Unexecuted instantiation: lib-signals.c:array_get_pool_i
Unexecuted instantiation: mempool.c:array_get_pool_i
Unexecuted instantiation: priorityq.c:array_get_pool_i
Unexecuted instantiation: strfuncs.c:array_get_pool_i
Unexecuted instantiation: unichar.c:array_get_pool_i
Unexecuted instantiation: array.c:array_get_pool_i
Unexecuted instantiation: env-util.c:array_get_pool_i
Unexecuted instantiation: event-filter.c:array_get_pool_i
Unexecuted instantiation: iostream.c:array_get_pool_i
185
#define array_get_pool(array) \
186
  array_get_pool_i(&(array)->arr)
187
188
static inline void
189
array_clear_i(struct array *array)
190
0
{
191
0
  buffer_set_used_size(array->buffer, 0);
192
0
}
Unexecuted instantiation: imap-bodystructure.c:array_clear_i
Unexecuted instantiation: imap-envelope.c:array_clear_i
Unexecuted instantiation: imap-parser.c:array_clear_i
Unexecuted instantiation: imap-quote.c:array_clear_i
Unexecuted instantiation: imap-arg.c:array_clear_i
Unexecuted instantiation: istream.c:array_clear_i
Unexecuted instantiation: ioloop.c:array_clear_i
Unexecuted instantiation: ioloop-epoll.c:array_clear_i
Unexecuted instantiation: lib.c:array_clear_i
Unexecuted instantiation: lib-event.c:array_clear_i
Unexecuted instantiation: lib-signals.c:array_clear_i
Unexecuted instantiation: mempool.c:array_clear_i
Unexecuted instantiation: priorityq.c:array_clear_i
Unexecuted instantiation: strfuncs.c:array_clear_i
Unexecuted instantiation: unichar.c:array_clear_i
Unexecuted instantiation: array.c:array_clear_i
Unexecuted instantiation: env-util.c:array_clear_i
Unexecuted instantiation: event-filter.c:array_clear_i
Unexecuted instantiation: iostream.c:array_clear_i
193
#define array_clear(array) \
194
0
  array_clear_i(&(array)->arr)
195
196
static inline unsigned int ATTR_PURE
197
array_count_i(const struct array *array)
198
7.15M
{
199
7.15M
  return array->buffer->used / array->element_size;
200
7.15M
}
Unexecuted instantiation: imap-bodystructure.c:array_count_i
Unexecuted instantiation: imap-envelope.c:array_count_i
imap-parser.c:array_count_i
Line
Count
Source
198
11.4k
{
199
11.4k
  return array->buffer->used / array->element_size;
200
11.4k
}
Unexecuted instantiation: imap-quote.c:array_count_i
imap-arg.c:array_count_i
Line
Count
Source
198
7.14M
{
199
7.14M
  return array->buffer->used / array->element_size;
200
7.14M
}
Unexecuted instantiation: istream.c:array_count_i
Unexecuted instantiation: ioloop.c:array_count_i
Unexecuted instantiation: ioloop-epoll.c:array_count_i
lib.c:array_count_i
Line
Count
Source
198
1
{
199
1
  return array->buffer->used / array->element_size;
200
1
}
Unexecuted instantiation: lib-event.c:array_count_i
Unexecuted instantiation: lib-signals.c:array_count_i
Unexecuted instantiation: mempool.c:array_count_i
Unexecuted instantiation: priorityq.c:array_count_i
Unexecuted instantiation: strfuncs.c:array_count_i
Unexecuted instantiation: unichar.c:array_count_i
Unexecuted instantiation: array.c:array_count_i
Unexecuted instantiation: env-util.c:array_count_i
Unexecuted instantiation: event-filter.c:array_count_i
Unexecuted instantiation: iostream.c:array_count_i
201
#define array_count(array) \
202
5.78k
  array_count_i(&(array)->arr)
203
/* No need for the real count if all we're doing is comparing against 0 */
204
#define array_is_empty(array) \
205
0
  ((!array_is_created(array)) || ((array)->arr.buffer->used == 0))
206
#define array_not_empty(array) \
207
  ((array_is_created(array)) && ((array)->arr.buffer->used > 0))
208
209
static inline void
210
array_append_i(struct array *array, const void *data, unsigned int count)
211
3
{
212
3
  buffer_append(array->buffer, data, count * array->element_size);
213
3
}
Unexecuted instantiation: imap-bodystructure.c:array_append_i
Unexecuted instantiation: imap-envelope.c:array_append_i
Unexecuted instantiation: imap-parser.c:array_append_i
Unexecuted instantiation: imap-quote.c:array_append_i
Unexecuted instantiation: imap-arg.c:array_append_i
Unexecuted instantiation: istream.c:array_append_i
ioloop.c:array_append_i
Line
Count
Source
211
2
{
212
2
  buffer_append(array->buffer, data, count * array->element_size);
213
2
}
Unexecuted instantiation: ioloop-epoll.c:array_append_i
Unexecuted instantiation: lib.c:array_append_i
lib-event.c:array_append_i
Line
Count
Source
211
1
{
212
1
  buffer_append(array->buffer, data, count * array->element_size);
213
1
}
Unexecuted instantiation: lib-signals.c:array_append_i
Unexecuted instantiation: mempool.c:array_append_i
Unexecuted instantiation: priorityq.c:array_append_i
Unexecuted instantiation: strfuncs.c:array_append_i
Unexecuted instantiation: unichar.c:array_append_i
Unexecuted instantiation: array.c:array_append_i
Unexecuted instantiation: env-util.c:array_append_i
Unexecuted instantiation: event-filter.c:array_append_i
Unexecuted instantiation: iostream.c:array_append_i
214
215
#define array_append(array, data, count) \
216
3
  TYPE_CHECKS(void, ARRAY_TYPE_CHECK(array, data), \
217
3
  array_append_i(&(array)->arr, data, count))
218
219
static inline void
220
array_append_array_i(struct array *dest_array, const struct array *src_array)
221
0
{
222
0
  i_assert(dest_array->element_size == src_array->element_size);
223
0
  buffer_append_buf(dest_array->buffer, src_array->buffer, 0, SIZE_MAX);
224
0
}
Unexecuted instantiation: imap-bodystructure.c:array_append_array_i
Unexecuted instantiation: imap-envelope.c:array_append_array_i
Unexecuted instantiation: imap-parser.c:array_append_array_i
Unexecuted instantiation: imap-quote.c:array_append_array_i
Unexecuted instantiation: imap-arg.c:array_append_array_i
Unexecuted instantiation: istream.c:array_append_array_i
Unexecuted instantiation: ioloop.c:array_append_array_i
Unexecuted instantiation: ioloop-epoll.c:array_append_array_i
Unexecuted instantiation: lib.c:array_append_array_i
Unexecuted instantiation: lib-event.c:array_append_array_i
Unexecuted instantiation: lib-signals.c:array_append_array_i
Unexecuted instantiation: mempool.c:array_append_array_i
Unexecuted instantiation: priorityq.c:array_append_array_i
Unexecuted instantiation: strfuncs.c:array_append_array_i
Unexecuted instantiation: unichar.c:array_append_array_i
Unexecuted instantiation: array.c:array_append_array_i
Unexecuted instantiation: env-util.c:array_append_array_i
Unexecuted instantiation: event-filter.c:array_append_array_i
Unexecuted instantiation: iostream.c:array_append_array_i
225
#define array_append_array(dest_array, src_array) \
226
  TYPE_CHECKS(void, ARRAY_TYPES_CHECK(dest_array, src_array), \
227
  array_append_array_i(&(dest_array)->arr, &(src_array)->arr))
228
229
static inline void
230
array_insert_i(struct array *array, unsigned int idx,
231
         const void *data, unsigned int count)
232
0
{
233
0
  buffer_insert(array->buffer, idx * array->element_size,
234
0
          data, count * array->element_size);
235
0
}
Unexecuted instantiation: imap-bodystructure.c:array_insert_i
Unexecuted instantiation: imap-envelope.c:array_insert_i
Unexecuted instantiation: imap-parser.c:array_insert_i
Unexecuted instantiation: imap-quote.c:array_insert_i
Unexecuted instantiation: imap-arg.c:array_insert_i
Unexecuted instantiation: istream.c:array_insert_i
Unexecuted instantiation: ioloop.c:array_insert_i
Unexecuted instantiation: ioloop-epoll.c:array_insert_i
Unexecuted instantiation: lib.c:array_insert_i
Unexecuted instantiation: lib-event.c:array_insert_i
Unexecuted instantiation: lib-signals.c:array_insert_i
Unexecuted instantiation: mempool.c:array_insert_i
Unexecuted instantiation: priorityq.c:array_insert_i
Unexecuted instantiation: strfuncs.c:array_insert_i
Unexecuted instantiation: unichar.c:array_insert_i
Unexecuted instantiation: array.c:array_insert_i
Unexecuted instantiation: env-util.c:array_insert_i
Unexecuted instantiation: event-filter.c:array_insert_i
Unexecuted instantiation: iostream.c:array_insert_i
236
237
#define array_insert(array, idx, data, count) \
238
  TYPE_CHECKS(void, ARRAY_TYPE_CHECK(array, data), \
239
  array_insert_i(&(array)->arr, idx, data, count))
240
241
static inline void
242
array_delete_i(struct array *array, unsigned int idx, unsigned int count)
243
0
{
244
0
  buffer_delete(array->buffer, idx * array->element_size,
245
0
          count * array->element_size);
246
0
}
Unexecuted instantiation: imap-bodystructure.c:array_delete_i
Unexecuted instantiation: imap-envelope.c:array_delete_i
Unexecuted instantiation: imap-parser.c:array_delete_i
Unexecuted instantiation: imap-quote.c:array_delete_i
Unexecuted instantiation: imap-arg.c:array_delete_i
Unexecuted instantiation: istream.c:array_delete_i
Unexecuted instantiation: ioloop.c:array_delete_i
Unexecuted instantiation: ioloop-epoll.c:array_delete_i
Unexecuted instantiation: lib.c:array_delete_i
Unexecuted instantiation: lib-event.c:array_delete_i
Unexecuted instantiation: lib-signals.c:array_delete_i
Unexecuted instantiation: mempool.c:array_delete_i
Unexecuted instantiation: priorityq.c:array_delete_i
Unexecuted instantiation: strfuncs.c:array_delete_i
Unexecuted instantiation: unichar.c:array_delete_i
Unexecuted instantiation: array.c:array_delete_i
Unexecuted instantiation: env-util.c:array_delete_i
Unexecuted instantiation: event-filter.c:array_delete_i
Unexecuted instantiation: iostream.c:array_delete_i
247
#define array_delete(array, idx, count) \
248
0
  array_delete_i(&(array)->arr, idx, count)
249
250
static inline const void *
251
array_get_i(const struct array *array, unsigned int *count_r)
252
7.15M
{
253
7.15M
  *count_r = array_count_i(array);
254
7.15M
  return array->buffer->data;
255
7.15M
}
Unexecuted instantiation: imap-bodystructure.c:array_get_i
Unexecuted instantiation: imap-envelope.c:array_get_i
imap-parser.c:array_get_i
Line
Count
Source
252
5.65k
{
253
5.65k
  *count_r = array_count_i(array);
254
5.65k
  return array->buffer->data;
255
5.65k
}
Unexecuted instantiation: imap-quote.c:array_get_i
imap-arg.c:array_get_i
Line
Count
Source
252
7.14M
{
253
7.14M
  *count_r = array_count_i(array);
254
7.14M
  return array->buffer->data;
255
7.14M
}
Unexecuted instantiation: istream.c:array_get_i
Unexecuted instantiation: ioloop.c:array_get_i
Unexecuted instantiation: ioloop-epoll.c:array_get_i
lib.c:array_get_i
Line
Count
Source
252
1
{
253
1
  *count_r = array_count_i(array);
254
1
  return array->buffer->data;
255
1
}
Unexecuted instantiation: lib-event.c:array_get_i
Unexecuted instantiation: lib-signals.c:array_get_i
Unexecuted instantiation: mempool.c:array_get_i
Unexecuted instantiation: priorityq.c:array_get_i
Unexecuted instantiation: strfuncs.c:array_get_i
Unexecuted instantiation: unichar.c:array_get_i
Unexecuted instantiation: array.c:array_get_i
Unexecuted instantiation: env-util.c:array_get_i
Unexecuted instantiation: event-filter.c:array_get_i
Unexecuted instantiation: iostream.c:array_get_i
256
#define array_get(array, count) \
257
7.15M
  ARRAY_TYPE_CAST_CONST(array)array_get_i(&(array)->arr, count)
258
259
static inline void *
260
array_get_copy_i(const struct array *array, pool_t pool, unsigned int *count_r)
261
0
{
262
0
  *count_r = array_count_i(array);
263
0
  if (array->buffer->used == 0)
264
0
    return NULL;
265
0
  return p_memdup(pool, array->buffer->data, array->buffer->used);
266
0
}
Unexecuted instantiation: imap-bodystructure.c:array_get_copy_i
Unexecuted instantiation: imap-envelope.c:array_get_copy_i
Unexecuted instantiation: imap-parser.c:array_get_copy_i
Unexecuted instantiation: imap-quote.c:array_get_copy_i
Unexecuted instantiation: imap-arg.c:array_get_copy_i
Unexecuted instantiation: istream.c:array_get_copy_i
Unexecuted instantiation: ioloop.c:array_get_copy_i
Unexecuted instantiation: ioloop-epoll.c:array_get_copy_i
Unexecuted instantiation: lib.c:array_get_copy_i
Unexecuted instantiation: lib-event.c:array_get_copy_i
Unexecuted instantiation: lib-signals.c:array_get_copy_i
Unexecuted instantiation: mempool.c:array_get_copy_i
Unexecuted instantiation: priorityq.c:array_get_copy_i
Unexecuted instantiation: strfuncs.c:array_get_copy_i
Unexecuted instantiation: unichar.c:array_get_copy_i
Unexecuted instantiation: array.c:array_get_copy_i
Unexecuted instantiation: env-util.c:array_get_copy_i
Unexecuted instantiation: event-filter.c:array_get_copy_i
Unexecuted instantiation: iostream.c:array_get_copy_i
267
#define array_get_copy(array, pool, count) \
268
  ARRAY_TYPE_CAST_MODIFIABLE(array) \
269
    array_get_copy_i(&(array)->arr, pool, count)
270
271
/* Re: i_assert() vs. pure: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=51971#c1 */
272
static inline const void * ATTR_PURE
273
array_idx_i(const struct array *array, unsigned int idx)
274
0
{
275
0
  i_assert(idx < array->buffer->used / array->element_size);
276
0
  return CONST_PTR_OFFSET(array->buffer->data, idx * array->element_size);
277
0
}
Unexecuted instantiation: imap-bodystructure.c:array_idx_i
Unexecuted instantiation: imap-envelope.c:array_idx_i
Unexecuted instantiation: imap-parser.c:array_idx_i
Unexecuted instantiation: imap-quote.c:array_idx_i
Unexecuted instantiation: imap-arg.c:array_idx_i
Unexecuted instantiation: istream.c:array_idx_i
Unexecuted instantiation: ioloop.c:array_idx_i
Unexecuted instantiation: ioloop-epoll.c:array_idx_i
Unexecuted instantiation: lib.c:array_idx_i
Unexecuted instantiation: lib-event.c:array_idx_i
Unexecuted instantiation: lib-signals.c:array_idx_i
Unexecuted instantiation: mempool.c:array_idx_i
Unexecuted instantiation: priorityq.c:array_idx_i
Unexecuted instantiation: strfuncs.c:array_idx_i
Unexecuted instantiation: unichar.c:array_idx_i
Unexecuted instantiation: array.c:array_idx_i
Unexecuted instantiation: env-util.c:array_idx_i
Unexecuted instantiation: event-filter.c:array_idx_i
Unexecuted instantiation: iostream.c:array_idx_i
278
279
0
#define array_front(array) array_idx(array, 0)
280
#define array_front_modifiable(array) array_idx_modifiable(array, 0)
281
0
#define array_back(array) array_idx(array, array_count(array)-1)
282
#define array_back_modifiable(array) array_idx_modifiable(array, array_count(array)-1)
283
0
#define array_pop_back(array) array_delete(array, array_count(array)-1, 1);
284
3
#define array_push_back(array, item) array_append(array, (item), 1)
285
#define array_pop_front(array) array_delete(array, 0, 1)
286
#define array_push_front(array, item) array_insert(array, 0, (item), 1)
287
288
#define array_idx(array, idx) \
289
0
  ARRAY_TYPE_CAST_CONST(array)array_idx_i(&(array)->arr, idx)
290
/* Using *array_idx() will fail if the compiler doesn't support typeof().
291
   The same can be done with array_idx_elem() for arrays that have pointers. */
292
#ifdef HAVE_TYPEOF
293
#  define array_idx_elem(array, idx) \
294
  (TRUE ? *array_idx(array, idx) : \
295
    COMPILE_ERROR_IF_TRUE(sizeof(**(array)->v) != sizeof(void *)))
296
#else
297
#  define array_idx_elem(array, idx) \
298
  (*(void **)array_idx_i(&(array)->arr, idx))
299
#endif
300
301
static inline void *
302
array_get_modifiable_i(struct array *array, unsigned int *count_r)
303
0
{
304
0
  *count_r = array_count_i(array);
305
0
  return buffer_get_modifiable_data(array->buffer, NULL);
306
0
}
Unexecuted instantiation: imap-bodystructure.c:array_get_modifiable_i
Unexecuted instantiation: imap-envelope.c:array_get_modifiable_i
Unexecuted instantiation: imap-parser.c:array_get_modifiable_i
Unexecuted instantiation: imap-quote.c:array_get_modifiable_i
Unexecuted instantiation: imap-arg.c:array_get_modifiable_i
Unexecuted instantiation: istream.c:array_get_modifiable_i
Unexecuted instantiation: ioloop.c:array_get_modifiable_i
Unexecuted instantiation: ioloop-epoll.c:array_get_modifiable_i
Unexecuted instantiation: lib.c:array_get_modifiable_i
Unexecuted instantiation: lib-event.c:array_get_modifiable_i
Unexecuted instantiation: lib-signals.c:array_get_modifiable_i
Unexecuted instantiation: mempool.c:array_get_modifiable_i
Unexecuted instantiation: priorityq.c:array_get_modifiable_i
Unexecuted instantiation: strfuncs.c:array_get_modifiable_i
Unexecuted instantiation: unichar.c:array_get_modifiable_i
Unexecuted instantiation: array.c:array_get_modifiable_i
Unexecuted instantiation: env-util.c:array_get_modifiable_i
Unexecuted instantiation: event-filter.c:array_get_modifiable_i
Unexecuted instantiation: iostream.c:array_get_modifiable_i
307
#define array_get_modifiable(array, count) \
308
0
  ARRAY_TYPE_CAST_MODIFIABLE(array) \
309
0
    array_get_modifiable_i(&(array)->arr, count)
310
311
void *
312
array_idx_modifiable_i(const struct array *array, unsigned int idx) ATTR_PURE;
313
#define array_idx_modifiable(array, idx) \
314
0
  ARRAY_TYPE_CAST_MODIFIABLE(array) \
315
0
    array_idx_modifiable_i(&(array)->arr, idx)
316
317
void *array_idx_get_space_i(struct array *array, unsigned int idx);
318
#define array_idx_get_space(array, idx) \
319
0
  ARRAY_TYPE_CAST_MODIFIABLE(array) \
320
0
    array_idx_get_space_i(&(array)->arr, idx)
321
322
void array_idx_set_i(struct array *array, unsigned int idx, const void *data);
323
#define array_idx_set(array, idx, data) \
324
0
  TYPE_CHECKS(void, ARRAY_TYPE_CHECK(array, data), \
325
0
  array_idx_set_i(&(array)->arr, idx, data))
326
327
void array_idx_clear_i(struct array *array, unsigned int idx);
328
#define array_idx_clear(array, idx) \
329
  array_idx_clear_i(&(array)->arr, idx)
330
331
static inline void *
332
array_append_space_i(struct array *array)
333
70.6M
{
334
70.6M
  void *data;
335
336
70.6M
  data = buffer_append_space_unsafe(array->buffer, array->element_size);
337
70.6M
  memset(data, 0, array->element_size);
338
70.6M
  return data;
339
70.6M
}
Unexecuted instantiation: imap-bodystructure.c:array_append_space_i
Unexecuted instantiation: imap-envelope.c:array_append_space_i
imap-parser.c:array_append_space_i
Line
Count
Source
333
70.6M
{
334
70.6M
  void *data;
335
336
70.6M
  data = buffer_append_space_unsafe(array->buffer, array->element_size);
337
70.6M
  memset(data, 0, array->element_size);
338
70.6M
  return data;
339
70.6M
}
Unexecuted instantiation: imap-quote.c:array_append_space_i
Unexecuted instantiation: imap-arg.c:array_append_space_i
Unexecuted instantiation: istream.c:array_append_space_i
Unexecuted instantiation: ioloop.c:array_append_space_i
Unexecuted instantiation: ioloop-epoll.c:array_append_space_i
lib.c:array_append_space_i
Line
Count
Source
333
2
{
334
2
  void *data;
335
336
2
  data = buffer_append_space_unsafe(array->buffer, array->element_size);
337
2
  memset(data, 0, array->element_size);
338
2
  return data;
339
2
}
lib-event.c:array_append_space_i
Line
Count
Source
333
5
{
334
5
  void *data;
335
336
5
  data = buffer_append_space_unsafe(array->buffer, array->element_size);
337
5
  memset(data, 0, array->element_size);
338
5
  return data;
339
5
}
Unexecuted instantiation: lib-signals.c:array_append_space_i
Unexecuted instantiation: mempool.c:array_append_space_i
Unexecuted instantiation: priorityq.c:array_append_space_i
Unexecuted instantiation: strfuncs.c:array_append_space_i
Unexecuted instantiation: unichar.c:array_append_space_i
Unexecuted instantiation: array.c:array_append_space_i
Unexecuted instantiation: env-util.c:array_append_space_i
Unexecuted instantiation: event-filter.c:array_append_space_i
Unexecuted instantiation: iostream.c:array_append_space_i
340
#define array_append_space(array) \
341
70.6M
  ARRAY_TYPE_CAST_MODIFIABLE(array)array_append_space_i(&(array)->arr)
342
#define array_append_zero(array) \
343
0
  (void)array_append_space_i(&(array)->arr)
344
345
void *array_insert_space_i(struct array *array, unsigned int idx);
346
#define array_insert_space(array, idx) \
347
  ARRAY_TYPE_CAST_MODIFIABLE(array) \
348
    array_insert_space_i(&(array)->arr, idx)
349
350
static inline void
351
array_copy(struct array *dest, unsigned int dest_idx,
352
     const struct array *src, unsigned int src_idx, unsigned int count)
353
0
{
354
0
  i_assert(dest->element_size == src->element_size);
355
0
356
0
  buffer_copy(dest->buffer, dest_idx * dest->element_size,
357
0
        src->buffer, src_idx * src->element_size,
358
0
        count * dest->element_size);
359
0
}
Unexecuted instantiation: imap-bodystructure.c:array_copy
Unexecuted instantiation: imap-envelope.c:array_copy
Unexecuted instantiation: imap-parser.c:array_copy
Unexecuted instantiation: imap-quote.c:array_copy
Unexecuted instantiation: imap-arg.c:array_copy
Unexecuted instantiation: istream.c:array_copy
Unexecuted instantiation: ioloop.c:array_copy
Unexecuted instantiation: ioloop-epoll.c:array_copy
Unexecuted instantiation: lib.c:array_copy
Unexecuted instantiation: lib-event.c:array_copy
Unexecuted instantiation: lib-signals.c:array_copy
Unexecuted instantiation: mempool.c:array_copy
Unexecuted instantiation: priorityq.c:array_copy
Unexecuted instantiation: strfuncs.c:array_copy
Unexecuted instantiation: unichar.c:array_copy
Unexecuted instantiation: array.c:array_copy
Unexecuted instantiation: env-util.c:array_copy
Unexecuted instantiation: event-filter.c:array_copy
Unexecuted instantiation: iostream.c:array_copy
360
361
bool array_cmp_i(const struct array *array1,
362
     const struct array *array2) ATTR_PURE;
363
#define array_cmp(array1, array2) \
364
  array_cmp_i(&(array1)->arr, &(array2)->arr)
365
366
/* Test equality via a comparator */
367
bool array_equal_fn_i(const struct array *array1,
368
          const struct array *array2,
369
          int (*cmp)(const void*, const void *)) ATTR_PURE;
370
#define array_equal_fn(array1, array2, cmp) \
371
  TYPE_CHECKS(bool, \
372
  ARRAY_TYPES_CHECK(array1, array2) || \
373
  CALLBACK_TYPECHECK(cmp, int (*)(typeof(*(array1)->v), \
374
          typeof(*(array2)->v))), \
375
  array_equal_fn_i(&(array1)->arr, &(array2)->arr, \
376
       (int (*)(const void *, const void *))cmp))
377
bool array_equal_fn_ctx_i(const struct array *array1,
378
        const struct array *array2,
379
        int (*cmp)(const void*, const void *, const void *),
380
        const void *context) ATTR_PURE;
381
/* Same, but with a context pointer.
382
   context can't be void* as ``const typeof(context)'' won't compile,
383
   so ``const typeof(*context)*'' is required instead, and that requires a
384
   complete type. */
385
#define array_equal_fn_ctx(array1, array2, cmp, ctx) \
386
  TYPE_CHECKS(bool, \
387
  ARRAY_TYPES_CHECK(array1, array2) || \
388
  CALLBACK_TYPECHECK(cmp, int (*)(typeof(*(array1)->v), \
389
          typeof(*(array2)->v), \
390
          const typeof(*ctx)*)), \
391
  array_equal_fn_ctx_i(&(array1)->arr, &(array2)->arr, \
392
    (int (*)(const void *, const void *, const void *))cmp, ctx))
393
394
void array_reverse_i(struct array *array);
395
#define array_reverse(array) \
396
  array_reverse_i(&(array)->arr)
397
398
void array_sort_i(struct array *array, int (*cmp)(const void *, const void *));
399
#define array_sort(array, cmp) \
400
0
  TYPE_CHECKS(void, \
401
0
  CALLBACK_TYPECHECK(cmp, int (*)(typeof(*(array)->v), \
402
0
          typeof(*(array)->v))), \
403
0
  array_sort_i(&(array)->arr, (int (*)(const void *, const void *))cmp))
404
405
#define ARRAY_SEARCH_CALL(func, array, key, cmp) \
406
0
  TYPE_CHECKS(void *, \
407
0
    CALLBACK_TYPECHECK(cmp, int (*)(typeof(const typeof(*key) *), \
408
0
            typeof(*(array)->v))), \
409
0
    array_##func##_i( \
410
0
      &(array)->arr, (const void *)key, \
411
0
      (int (*)(const void *, const void *))cmp))
412
413
const void *array_bsearch_i(const struct array *array, const void *key,
414
                int (*cmp)(const void *, const void *));
415
static inline void *array_bsearch_modifiable_i(struct array *array, const void *key,
416
                 int (*cmp)(const void *, const void *))
417
0
{
418
0
  return (void *)array_bsearch_i(array, key, cmp);
419
0
}
Unexecuted instantiation: imap-bodystructure.c:array_bsearch_modifiable_i
Unexecuted instantiation: imap-envelope.c:array_bsearch_modifiable_i
Unexecuted instantiation: imap-parser.c:array_bsearch_modifiable_i
Unexecuted instantiation: imap-quote.c:array_bsearch_modifiable_i
Unexecuted instantiation: imap-arg.c:array_bsearch_modifiable_i
Unexecuted instantiation: istream.c:array_bsearch_modifiable_i
Unexecuted instantiation: ioloop.c:array_bsearch_modifiable_i
Unexecuted instantiation: ioloop-epoll.c:array_bsearch_modifiable_i
Unexecuted instantiation: lib.c:array_bsearch_modifiable_i
Unexecuted instantiation: lib-event.c:array_bsearch_modifiable_i
Unexecuted instantiation: lib-signals.c:array_bsearch_modifiable_i
Unexecuted instantiation: mempool.c:array_bsearch_modifiable_i
Unexecuted instantiation: priorityq.c:array_bsearch_modifiable_i
Unexecuted instantiation: strfuncs.c:array_bsearch_modifiable_i
Unexecuted instantiation: unichar.c:array_bsearch_modifiable_i
Unexecuted instantiation: array.c:array_bsearch_modifiable_i
Unexecuted instantiation: env-util.c:array_bsearch_modifiable_i
Unexecuted instantiation: event-filter.c:array_bsearch_modifiable_i
Unexecuted instantiation: iostream.c:array_bsearch_modifiable_i
420
421
#define array_bsearch(array, key, cmp) \
422
  ARRAY_TYPE_CAST_CONST(array) \
423
  ARRAY_SEARCH_CALL(bsearch, array, key, cmp)
424
#define array_bsearch_modifiable(array, key, cmp) \
425
  ARRAY_TYPE_CAST_MODIFIABLE(array) \
426
  ARRAY_SEARCH_CALL(bsearch_modifiable, array, key, cmp)
427
428
/* Returns pointer to first element for which cmp(key,elem)==0, or NULL */
429
const void *array_lsearch_i(const struct array *array, const void *key,
430
          int (*cmp)(const void *, const void *));
431
static inline void *array_lsearch_modifiable_i(struct array *array, const void *key,
432
                 int (*cmp)(const void *, const void *))
433
0
{
434
0
  return (void *)array_lsearch_i(array, key, cmp);
435
0
}
Unexecuted instantiation: imap-bodystructure.c:array_lsearch_modifiable_i
Unexecuted instantiation: imap-envelope.c:array_lsearch_modifiable_i
Unexecuted instantiation: imap-parser.c:array_lsearch_modifiable_i
Unexecuted instantiation: imap-quote.c:array_lsearch_modifiable_i
Unexecuted instantiation: imap-arg.c:array_lsearch_modifiable_i
Unexecuted instantiation: istream.c:array_lsearch_modifiable_i
Unexecuted instantiation: ioloop.c:array_lsearch_modifiable_i
Unexecuted instantiation: ioloop-epoll.c:array_lsearch_modifiable_i
Unexecuted instantiation: lib.c:array_lsearch_modifiable_i
Unexecuted instantiation: lib-event.c:array_lsearch_modifiable_i
Unexecuted instantiation: lib-signals.c:array_lsearch_modifiable_i
Unexecuted instantiation: mempool.c:array_lsearch_modifiable_i
Unexecuted instantiation: priorityq.c:array_lsearch_modifiable_i
Unexecuted instantiation: strfuncs.c:array_lsearch_modifiable_i
Unexecuted instantiation: unichar.c:array_lsearch_modifiable_i
Unexecuted instantiation: array.c:array_lsearch_modifiable_i
Unexecuted instantiation: env-util.c:array_lsearch_modifiable_i
Unexecuted instantiation: event-filter.c:array_lsearch_modifiable_i
Unexecuted instantiation: iostream.c:array_lsearch_modifiable_i
436
437
#define array_lsearch(array, key, cmp) \
438
0
  ARRAY_TYPE_CAST_CONST(array) \
439
0
  ARRAY_SEARCH_CALL(lsearch, array, key, cmp)
440
#define array_lsearch_modifiable(array, key, cmp) \
441
  ARRAY_TYPE_CAST_MODIFIABLE(array) \
442
  ARRAY_SEARCH_CALL(lsearch_modifiable, array, key, cmp)
443
444
/* Search a pointer from an array */
445
const void *array_lsearch_ptr_i(const struct array *array, const void *key);
446
#define array_lsearch_ptr(array, key) \
447
0
  TYPE_CHECKS(const void *, \
448
0
  COMPILE_ERROR_IF_TYPES_NOT_COMPATIBLE(***(array)->v, *(key)), \
449
0
  array_lsearch_ptr_i(&(array)->arr, key))
450
#define array_lsearch_ptr_modifiable(array, key) \
451
  (void *)array_lsearch_ptr(array, key)
452
453
bool array_lsearch_ptr_idx_i(const struct array *array, const void *key,
454
           unsigned int *idx_r);
455
#define array_lsearch_ptr_idx(array, key, idx_r) \
456
0
  TYPE_CHECKS(bool, \
457
  COMPILE_ERROR_IF_TYPES_NOT_COMPATIBLE(***(array)->v, *(key)), \
458
  array_lsearch_ptr_idx_i(&(array)->arr, key, idx_r))
459
460
#endif