]>
diplodocus.org Git - nmh/blob - sbr/vector.c
2 * vector.c -- dynamically sized vectors
4 * This code is Copyright (c) 2013, by the authors of nmh. See the
5 * COPYRIGHT file in the root directory of the nmh distribution for
6 * complete copyright information.
10 * This file defines several kinds of vectors:
12 * svector: vector of char arrays
13 * ivector: vector of ints
15 * The interfaces provide only the capabilities needed by nmh. The
16 * implementations rely on dynamic allocation, so that a vector
17 * can be as large as needed, as long as it fits in (virtual) memory.
23 #define VEC_INIT_SIZE 256
26 * These try to hide the type of the "bits" member of bvector. But if
27 * that is changed to a type that's wider than unsigned long, the 1ul
28 * constants in the code below must also be changed to a 1 that's at
29 * least as wide as the new type.
34 #define BVEC_WORD(vec, max) ((max) / (sizeof *vec->bits * Nbby))
35 #define BVEC_OFFSET(vec, max) ((max) % (sizeof *vec->bits * Nbby))
36 #define BVEC_BYTES(vec, n) \
37 ((BVEC_WORD (vec, n) + (BVEC_OFFSET (vec, n) == 0 ? 0 : 1)) * \
45 static void bvector_resize (bvector_t
, size_t);
48 bvector_create (size_t init_size
) {
53 bytes
= BVEC_BYTES (vec
, init_size
? init_size
: VEC_INIT_SIZE
);
55 vec
->bits
= mh_xmalloc (bytes
);
56 memset (vec
->bits
, 0, bytes
);
57 vec
->maxsize
= bytes
* Nbby
;
63 bvector_copy (bvector_t dest
, bvector_t src
) {
64 size_t bytes
= BVEC_BYTES (dest
, src
->maxsize
);
67 dest
->bits
= mh_xmalloc (bytes
);
68 memcpy (dest
->bits
, src
->bits
, bytes
);
69 dest
->maxsize
= src
->maxsize
;
73 bvector_free (bvector_t vec
) {
79 bvector_clear (bvector_t vec
, size_t n
) {
80 size_t word
= BVEC_WORD (vec
,n
);
81 size_t offset
= BVEC_OFFSET (vec
, n
);
83 if (n
>= vec
->maxsize
)
84 bvector_resize (vec
, n
);
86 assert (sizeof *vec
->bits
<= sizeof 1ul);
87 vec
->bits
[word
] &= ~(1ul << offset
);
92 bvector_clear_all (bvector_t vec
) {
93 memset (vec
->bits
, 0, BVEC_BYTES (vec
, vec
->maxsize
));
98 bvector_set (bvector_t vec
, size_t n
) {
99 size_t word
= BVEC_WORD (vec
, n
);
100 size_t offset
= BVEC_OFFSET (vec
, n
);
102 if (n
>= vec
->maxsize
)
103 bvector_resize (vec
, n
);
104 assert (sizeof *vec
->bits
<= sizeof 1ul);
105 vec
->bits
[word
] |= 1ul << offset
;
109 bvector_at (bvector_t vec
, size_t i
) {
110 size_t word
= BVEC_WORD (vec
, i
);
111 size_t offset
= BVEC_OFFSET (vec
, i
);
113 assert (sizeof *vec
->bits
<= sizeof 1ul);
114 return i
< vec
->maxsize
115 ? (vec
->bits
[word
] & (1ul << offset
) ? 1 : 0)
120 bvector_resize (bvector_t vec
, size_t maxsize
) {
121 size_t old_maxsize
= vec
->maxsize
;
125 while ((vec
->maxsize
*= 2) < maxsize
)
127 bytes
= BVEC_BYTES (vec
, vec
->maxsize
);
128 vec
->bits
= mh_xrealloc (vec
->bits
, bytes
);
129 for (i
= old_maxsize
; i
< vec
->maxsize
; ++i
)
130 bvector_clear (vec
, i
);
133 const unsigned long *
134 bvector_bits (bvector_t vec
) {
139 bvector_maxsize (bvector_t vec
) {
150 static void svector_resize (svector_t
, size_t);
153 svector_create (size_t init_size
) {
158 vec
->maxsize
= init_size
? init_size
: VEC_INIT_SIZE
;
159 bytes
= vec
->maxsize
* sizeof (char *);
160 vec
->strs
= mh_xmalloc (bytes
);
161 memset (vec
->strs
, 0, bytes
);
168 svector_free (svector_t vec
) {
174 svector_push_back (svector_t vec
, char *s
) {
175 if (++vec
->size
>= vec
->maxsize
)
176 svector_resize (vec
, vec
->size
);
177 return vec
->strs
[vec
->size
-1] = s
;
181 svector_at (svector_t vec
, size_t i
) {
182 if (i
>= vec
->maxsize
)
183 svector_resize (vec
, i
);
188 * Return address of first element that stringwise matches s.
189 * The caller can replace the contents of the return address.
192 svector_find (svector_t vec
, const char *s
) {
194 char **str
= vec
->strs
;
196 for (i
= 0; i
< vec
->size
; ++i
, ++str
) {
197 if (*str
&& ! strcmp(*str
, s
)) {
206 svector_strs (svector_t vec
) {
211 svector_size (svector_t vec
) {
216 svector_resize (svector_t vec
, size_t maxsize
) {
217 size_t old_maxsize
= vec
->maxsize
;
220 while ((vec
->maxsize
*= 2) < maxsize
)
222 vec
->strs
= mh_xrealloc (vec
->strs
, vec
->maxsize
* sizeof (char *));
223 for (i
= old_maxsize
; i
< vec
->maxsize
; ++i
)
234 static void ivector_resize (ivector_t
, size_t);
237 ivector_create (size_t init_size
) {
242 vec
->maxsize
= init_size
? init_size
: VEC_INIT_SIZE
;
243 bytes
= vec
->maxsize
* sizeof (int);
244 vec
->ints
= mh_xmalloc (bytes
);
245 memset (vec
->ints
, 0, bytes
);
252 ivector_free (ivector_t vec
) {
258 ivector_push_back (ivector_t vec
, int n
) {
259 if (++vec
->size
>= vec
->maxsize
)
260 ivector_resize (vec
, vec
->size
);
261 return vec
->ints
[vec
->size
-1] = n
;
265 ivector_at (ivector_t vec
, size_t i
) {
266 if (i
>= vec
->maxsize
)
267 ivector_resize (vec
, i
);
272 ivector_atp (ivector_t vec
, size_t i
) {
273 if (i
>= vec
->maxsize
)
274 ivector_resize (vec
, i
);
275 return &vec
->ints
[i
];
279 ivector_size (ivector_t vec
) {
284 ivector_resize (ivector_t vec
, size_t maxsize
) {
285 size_t old_maxsize
= vec
->maxsize
;
288 while ((vec
->maxsize
*= 2) < maxsize
)
290 vec
->ints
= mh_xrealloc (vec
->ints
, vec
->maxsize
* sizeof (int));
291 for (i
= old_maxsize
; i
< vec
->maxsize
; ++i
)