1*7688df22SAndroid Build Coastguard Worker /*
2*7688df22SAndroid Build Coastguard Worker * Copyright 2010 Jerome Glisse <[email protected]>
3*7688df22SAndroid Build Coastguard Worker *
4*7688df22SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
5*7688df22SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
6*7688df22SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
7*7688df22SAndroid Build Coastguard Worker * on the rights to use, copy, modify, merge, publish, distribute, sub
8*7688df22SAndroid Build Coastguard Worker * license, and/or sell copies of the Software, and to permit persons to whom
9*7688df22SAndroid Build Coastguard Worker * the Software is furnished to do so, subject to the following conditions:
10*7688df22SAndroid Build Coastguard Worker *
11*7688df22SAndroid Build Coastguard Worker * The above copyright notice and this permission notice (including the next
12*7688df22SAndroid Build Coastguard Worker * paragraph) shall be included in all copies or substantial portions of the
13*7688df22SAndroid Build Coastguard Worker * Software.
14*7688df22SAndroid Build Coastguard Worker *
15*7688df22SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*7688df22SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*7688df22SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
18*7688df22SAndroid Build Coastguard Worker * THE AUTHOR(S) AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM,
19*7688df22SAndroid Build Coastguard Worker * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
20*7688df22SAndroid Build Coastguard Worker * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
21*7688df22SAndroid Build Coastguard Worker * USE OR OTHER DEALINGS IN THE SOFTWARE.
22*7688df22SAndroid Build Coastguard Worker *
23*7688df22SAndroid Build Coastguard Worker * Authors:
24*7688df22SAndroid Build Coastguard Worker * Jerome Glisse
25*7688df22SAndroid Build Coastguard Worker */
26*7688df22SAndroid Build Coastguard Worker #include <errno.h>
27*7688df22SAndroid Build Coastguard Worker #include <stdlib.h>
28*7688df22SAndroid Build Coastguard Worker #include <string.h>
29*7688df22SAndroid Build Coastguard Worker #include "bof.h"
30*7688df22SAndroid Build Coastguard Worker
31*7688df22SAndroid Build Coastguard Worker /*
32*7688df22SAndroid Build Coastguard Worker * helpers
33*7688df22SAndroid Build Coastguard Worker */
bof_entry_grow(bof_t * bof)34*7688df22SAndroid Build Coastguard Worker static int bof_entry_grow(bof_t *bof)
35*7688df22SAndroid Build Coastguard Worker {
36*7688df22SAndroid Build Coastguard Worker bof_t **array;
37*7688df22SAndroid Build Coastguard Worker
38*7688df22SAndroid Build Coastguard Worker if (bof->array_size < bof->nentry)
39*7688df22SAndroid Build Coastguard Worker return 0;
40*7688df22SAndroid Build Coastguard Worker array = realloc(bof->array, (bof->nentry + 16) * sizeof(void*));
41*7688df22SAndroid Build Coastguard Worker if (array == NULL)
42*7688df22SAndroid Build Coastguard Worker return -ENOMEM;
43*7688df22SAndroid Build Coastguard Worker bof->array = array;
44*7688df22SAndroid Build Coastguard Worker bof->nentry += 16;
45*7688df22SAndroid Build Coastguard Worker return 0;
46*7688df22SAndroid Build Coastguard Worker }
47*7688df22SAndroid Build Coastguard Worker
48*7688df22SAndroid Build Coastguard Worker /*
49*7688df22SAndroid Build Coastguard Worker * object
50*7688df22SAndroid Build Coastguard Worker */
bof_object(void)51*7688df22SAndroid Build Coastguard Worker bof_t *bof_object(void)
52*7688df22SAndroid Build Coastguard Worker {
53*7688df22SAndroid Build Coastguard Worker bof_t *object;
54*7688df22SAndroid Build Coastguard Worker
55*7688df22SAndroid Build Coastguard Worker object = calloc(1, sizeof(bof_t));
56*7688df22SAndroid Build Coastguard Worker if (object == NULL)
57*7688df22SAndroid Build Coastguard Worker return NULL;
58*7688df22SAndroid Build Coastguard Worker object->refcount = 1;
59*7688df22SAndroid Build Coastguard Worker object->type = BOF_TYPE_OBJECT;
60*7688df22SAndroid Build Coastguard Worker object->size = 12;
61*7688df22SAndroid Build Coastguard Worker return object;
62*7688df22SAndroid Build Coastguard Worker }
63*7688df22SAndroid Build Coastguard Worker
bof_object_get(bof_t * object,const char * keyname)64*7688df22SAndroid Build Coastguard Worker bof_t *bof_object_get(bof_t *object, const char *keyname)
65*7688df22SAndroid Build Coastguard Worker {
66*7688df22SAndroid Build Coastguard Worker unsigned i;
67*7688df22SAndroid Build Coastguard Worker
68*7688df22SAndroid Build Coastguard Worker for (i = 0; i < object->array_size; i += 2) {
69*7688df22SAndroid Build Coastguard Worker if (!strcmp(object->array[i]->value, keyname)) {
70*7688df22SAndroid Build Coastguard Worker return object->array[i + 1];
71*7688df22SAndroid Build Coastguard Worker }
72*7688df22SAndroid Build Coastguard Worker }
73*7688df22SAndroid Build Coastguard Worker return NULL;
74*7688df22SAndroid Build Coastguard Worker }
75*7688df22SAndroid Build Coastguard Worker
bof_object_set(bof_t * object,const char * keyname,bof_t * value)76*7688df22SAndroid Build Coastguard Worker int bof_object_set(bof_t *object, const char *keyname, bof_t *value)
77*7688df22SAndroid Build Coastguard Worker {
78*7688df22SAndroid Build Coastguard Worker bof_t *key;
79*7688df22SAndroid Build Coastguard Worker int r;
80*7688df22SAndroid Build Coastguard Worker
81*7688df22SAndroid Build Coastguard Worker if (object->type != BOF_TYPE_OBJECT)
82*7688df22SAndroid Build Coastguard Worker return -EINVAL;
83*7688df22SAndroid Build Coastguard Worker r = bof_entry_grow(object);
84*7688df22SAndroid Build Coastguard Worker if (r)
85*7688df22SAndroid Build Coastguard Worker return r;
86*7688df22SAndroid Build Coastguard Worker key = bof_string(keyname);
87*7688df22SAndroid Build Coastguard Worker if (key == NULL)
88*7688df22SAndroid Build Coastguard Worker return -ENOMEM;
89*7688df22SAndroid Build Coastguard Worker object->array[object->array_size++] = key;
90*7688df22SAndroid Build Coastguard Worker object->array[object->array_size++] = value;
91*7688df22SAndroid Build Coastguard Worker object->size += value->size;
92*7688df22SAndroid Build Coastguard Worker object->size += key->size;
93*7688df22SAndroid Build Coastguard Worker bof_incref(value);
94*7688df22SAndroid Build Coastguard Worker return 0;
95*7688df22SAndroid Build Coastguard Worker }
96*7688df22SAndroid Build Coastguard Worker
97*7688df22SAndroid Build Coastguard Worker /*
98*7688df22SAndroid Build Coastguard Worker * array
99*7688df22SAndroid Build Coastguard Worker */
bof_array(void)100*7688df22SAndroid Build Coastguard Worker bof_t *bof_array(void)
101*7688df22SAndroid Build Coastguard Worker {
102*7688df22SAndroid Build Coastguard Worker bof_t *array = bof_object();
103*7688df22SAndroid Build Coastguard Worker
104*7688df22SAndroid Build Coastguard Worker if (array == NULL)
105*7688df22SAndroid Build Coastguard Worker return NULL;
106*7688df22SAndroid Build Coastguard Worker array->type = BOF_TYPE_ARRAY;
107*7688df22SAndroid Build Coastguard Worker array->size = 12;
108*7688df22SAndroid Build Coastguard Worker return array;
109*7688df22SAndroid Build Coastguard Worker }
110*7688df22SAndroid Build Coastguard Worker
bof_array_append(bof_t * array,bof_t * value)111*7688df22SAndroid Build Coastguard Worker int bof_array_append(bof_t *array, bof_t *value)
112*7688df22SAndroid Build Coastguard Worker {
113*7688df22SAndroid Build Coastguard Worker int r;
114*7688df22SAndroid Build Coastguard Worker if (array->type != BOF_TYPE_ARRAY)
115*7688df22SAndroid Build Coastguard Worker return -EINVAL;
116*7688df22SAndroid Build Coastguard Worker r = bof_entry_grow(array);
117*7688df22SAndroid Build Coastguard Worker if (r)
118*7688df22SAndroid Build Coastguard Worker return r;
119*7688df22SAndroid Build Coastguard Worker array->array[array->array_size++] = value;
120*7688df22SAndroid Build Coastguard Worker array->size += value->size;
121*7688df22SAndroid Build Coastguard Worker bof_incref(value);
122*7688df22SAndroid Build Coastguard Worker return 0;
123*7688df22SAndroid Build Coastguard Worker }
124*7688df22SAndroid Build Coastguard Worker
bof_array_get(bof_t * bof,unsigned i)125*7688df22SAndroid Build Coastguard Worker bof_t *bof_array_get(bof_t *bof, unsigned i)
126*7688df22SAndroid Build Coastguard Worker {
127*7688df22SAndroid Build Coastguard Worker if (!bof_is_array(bof) || i >= bof->array_size)
128*7688df22SAndroid Build Coastguard Worker return NULL;
129*7688df22SAndroid Build Coastguard Worker return bof->array[i];
130*7688df22SAndroid Build Coastguard Worker }
131*7688df22SAndroid Build Coastguard Worker
bof_array_size(bof_t * bof)132*7688df22SAndroid Build Coastguard Worker unsigned bof_array_size(bof_t *bof)
133*7688df22SAndroid Build Coastguard Worker {
134*7688df22SAndroid Build Coastguard Worker if (!bof_is_array(bof))
135*7688df22SAndroid Build Coastguard Worker return 0;
136*7688df22SAndroid Build Coastguard Worker return bof->array_size;
137*7688df22SAndroid Build Coastguard Worker }
138*7688df22SAndroid Build Coastguard Worker
139*7688df22SAndroid Build Coastguard Worker /*
140*7688df22SAndroid Build Coastguard Worker * blob
141*7688df22SAndroid Build Coastguard Worker */
bof_blob(unsigned size,void * value)142*7688df22SAndroid Build Coastguard Worker bof_t *bof_blob(unsigned size, void *value)
143*7688df22SAndroid Build Coastguard Worker {
144*7688df22SAndroid Build Coastguard Worker bof_t *blob = bof_object();
145*7688df22SAndroid Build Coastguard Worker
146*7688df22SAndroid Build Coastguard Worker if (blob == NULL)
147*7688df22SAndroid Build Coastguard Worker return NULL;
148*7688df22SAndroid Build Coastguard Worker blob->type = BOF_TYPE_BLOB;
149*7688df22SAndroid Build Coastguard Worker blob->value = calloc(1, size);
150*7688df22SAndroid Build Coastguard Worker if (blob->value == NULL) {
151*7688df22SAndroid Build Coastguard Worker bof_decref(blob);
152*7688df22SAndroid Build Coastguard Worker return NULL;
153*7688df22SAndroid Build Coastguard Worker }
154*7688df22SAndroid Build Coastguard Worker blob->size = size;
155*7688df22SAndroid Build Coastguard Worker memcpy(blob->value, value, size);
156*7688df22SAndroid Build Coastguard Worker blob->size += 12;
157*7688df22SAndroid Build Coastguard Worker return blob;
158*7688df22SAndroid Build Coastguard Worker }
159*7688df22SAndroid Build Coastguard Worker
bof_blob_size(bof_t * bof)160*7688df22SAndroid Build Coastguard Worker unsigned bof_blob_size(bof_t *bof)
161*7688df22SAndroid Build Coastguard Worker {
162*7688df22SAndroid Build Coastguard Worker if (!bof_is_blob(bof))
163*7688df22SAndroid Build Coastguard Worker return 0;
164*7688df22SAndroid Build Coastguard Worker return bof->size - 12;
165*7688df22SAndroid Build Coastguard Worker }
166*7688df22SAndroid Build Coastguard Worker
bof_blob_value(bof_t * bof)167*7688df22SAndroid Build Coastguard Worker void *bof_blob_value(bof_t *bof)
168*7688df22SAndroid Build Coastguard Worker {
169*7688df22SAndroid Build Coastguard Worker if (!bof_is_blob(bof))
170*7688df22SAndroid Build Coastguard Worker return NULL;
171*7688df22SAndroid Build Coastguard Worker return bof->value;
172*7688df22SAndroid Build Coastguard Worker }
173*7688df22SAndroid Build Coastguard Worker
174*7688df22SAndroid Build Coastguard Worker /*
175*7688df22SAndroid Build Coastguard Worker * string
176*7688df22SAndroid Build Coastguard Worker */
bof_string(const char * value)177*7688df22SAndroid Build Coastguard Worker bof_t *bof_string(const char *value)
178*7688df22SAndroid Build Coastguard Worker {
179*7688df22SAndroid Build Coastguard Worker bof_t *string = bof_object();
180*7688df22SAndroid Build Coastguard Worker
181*7688df22SAndroid Build Coastguard Worker if (string == NULL)
182*7688df22SAndroid Build Coastguard Worker return NULL;
183*7688df22SAndroid Build Coastguard Worker string->type = BOF_TYPE_STRING;
184*7688df22SAndroid Build Coastguard Worker string->size = strlen(value) + 1;
185*7688df22SAndroid Build Coastguard Worker string->value = calloc(1, string->size);
186*7688df22SAndroid Build Coastguard Worker if (string->value == NULL) {
187*7688df22SAndroid Build Coastguard Worker bof_decref(string);
188*7688df22SAndroid Build Coastguard Worker return NULL;
189*7688df22SAndroid Build Coastguard Worker }
190*7688df22SAndroid Build Coastguard Worker strcpy(string->value, value);
191*7688df22SAndroid Build Coastguard Worker string->size += 12;
192*7688df22SAndroid Build Coastguard Worker return string;
193*7688df22SAndroid Build Coastguard Worker }
194*7688df22SAndroid Build Coastguard Worker
195*7688df22SAndroid Build Coastguard Worker /*
196*7688df22SAndroid Build Coastguard Worker * int32
197*7688df22SAndroid Build Coastguard Worker */
bof_int32(int32_t value)198*7688df22SAndroid Build Coastguard Worker bof_t *bof_int32(int32_t value)
199*7688df22SAndroid Build Coastguard Worker {
200*7688df22SAndroid Build Coastguard Worker bof_t *int32 = bof_object();
201*7688df22SAndroid Build Coastguard Worker
202*7688df22SAndroid Build Coastguard Worker if (int32 == NULL)
203*7688df22SAndroid Build Coastguard Worker return NULL;
204*7688df22SAndroid Build Coastguard Worker int32->type = BOF_TYPE_INT32;
205*7688df22SAndroid Build Coastguard Worker int32->size = 4;
206*7688df22SAndroid Build Coastguard Worker int32->value = calloc(1, int32->size);
207*7688df22SAndroid Build Coastguard Worker if (int32->value == NULL) {
208*7688df22SAndroid Build Coastguard Worker bof_decref(int32);
209*7688df22SAndroid Build Coastguard Worker return NULL;
210*7688df22SAndroid Build Coastguard Worker }
211*7688df22SAndroid Build Coastguard Worker memcpy(int32->value, &value, 4);
212*7688df22SAndroid Build Coastguard Worker int32->size += 12;
213*7688df22SAndroid Build Coastguard Worker return int32;
214*7688df22SAndroid Build Coastguard Worker }
215*7688df22SAndroid Build Coastguard Worker
bof_int32_value(bof_t * bof)216*7688df22SAndroid Build Coastguard Worker int32_t bof_int32_value(bof_t *bof)
217*7688df22SAndroid Build Coastguard Worker {
218*7688df22SAndroid Build Coastguard Worker return *((uint32_t*)bof->value);
219*7688df22SAndroid Build Coastguard Worker }
220*7688df22SAndroid Build Coastguard Worker
221*7688df22SAndroid Build Coastguard Worker /*
222*7688df22SAndroid Build Coastguard Worker * common
223*7688df22SAndroid Build Coastguard Worker */
bof_indent(int level)224*7688df22SAndroid Build Coastguard Worker static void bof_indent(int level)
225*7688df22SAndroid Build Coastguard Worker {
226*7688df22SAndroid Build Coastguard Worker int i;
227*7688df22SAndroid Build Coastguard Worker
228*7688df22SAndroid Build Coastguard Worker for (i = 0; i < level; i++)
229*7688df22SAndroid Build Coastguard Worker fprintf(stderr, " ");
230*7688df22SAndroid Build Coastguard Worker }
231*7688df22SAndroid Build Coastguard Worker
bof_print_bof(bof_t * bof,int level,int entry)232*7688df22SAndroid Build Coastguard Worker static void bof_print_bof(bof_t *bof, int level, int entry)
233*7688df22SAndroid Build Coastguard Worker {
234*7688df22SAndroid Build Coastguard Worker bof_indent(level);
235*7688df22SAndroid Build Coastguard Worker if (bof == NULL) {
236*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "--NULL-- for entry %d\n", entry);
237*7688df22SAndroid Build Coastguard Worker return;
238*7688df22SAndroid Build Coastguard Worker }
239*7688df22SAndroid Build Coastguard Worker switch (bof->type) {
240*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_STRING:
241*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%p string [%s %d]\n", bof, (char*)bof->value, bof->size);
242*7688df22SAndroid Build Coastguard Worker break;
243*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_INT32:
244*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%p int32 [%d %d]\n", bof, *(int*)bof->value, bof->size);
245*7688df22SAndroid Build Coastguard Worker break;
246*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_BLOB:
247*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%p blob [%d]\n", bof, bof->size);
248*7688df22SAndroid Build Coastguard Worker break;
249*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_NULL:
250*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%p null [%d]\n", bof, bof->size);
251*7688df22SAndroid Build Coastguard Worker break;
252*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_OBJECT:
253*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%p object [%d %d]\n", bof, bof->array_size / 2, bof->size);
254*7688df22SAndroid Build Coastguard Worker break;
255*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_ARRAY:
256*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%p array [%d %d]\n", bof, bof->array_size, bof->size);
257*7688df22SAndroid Build Coastguard Worker break;
258*7688df22SAndroid Build Coastguard Worker default:
259*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%p unknown [%d]\n", bof, bof->type);
260*7688df22SAndroid Build Coastguard Worker return;
261*7688df22SAndroid Build Coastguard Worker }
262*7688df22SAndroid Build Coastguard Worker }
263*7688df22SAndroid Build Coastguard Worker
bof_print_rec(bof_t * bof,int level,int entry)264*7688df22SAndroid Build Coastguard Worker static void bof_print_rec(bof_t *bof, int level, int entry)
265*7688df22SAndroid Build Coastguard Worker {
266*7688df22SAndroid Build Coastguard Worker unsigned i;
267*7688df22SAndroid Build Coastguard Worker
268*7688df22SAndroid Build Coastguard Worker bof_print_bof(bof, level, entry);
269*7688df22SAndroid Build Coastguard Worker for (i = 0; i < bof->array_size; i++) {
270*7688df22SAndroid Build Coastguard Worker bof_print_rec(bof->array[i], level + 2, i);
271*7688df22SAndroid Build Coastguard Worker }
272*7688df22SAndroid Build Coastguard Worker }
273*7688df22SAndroid Build Coastguard Worker
bof_print(bof_t * bof)274*7688df22SAndroid Build Coastguard Worker void bof_print(bof_t *bof)
275*7688df22SAndroid Build Coastguard Worker {
276*7688df22SAndroid Build Coastguard Worker bof_print_rec(bof, 0, 0);
277*7688df22SAndroid Build Coastguard Worker }
278*7688df22SAndroid Build Coastguard Worker
bof_read(bof_t * root,FILE * file,long end,int level)279*7688df22SAndroid Build Coastguard Worker static int bof_read(bof_t *root, FILE *file, long end, int level)
280*7688df22SAndroid Build Coastguard Worker {
281*7688df22SAndroid Build Coastguard Worker bof_t *bof = NULL;
282*7688df22SAndroid Build Coastguard Worker int r;
283*7688df22SAndroid Build Coastguard Worker
284*7688df22SAndroid Build Coastguard Worker if (ftell(file) >= end) {
285*7688df22SAndroid Build Coastguard Worker return 0;
286*7688df22SAndroid Build Coastguard Worker }
287*7688df22SAndroid Build Coastguard Worker r = bof_entry_grow(root);
288*7688df22SAndroid Build Coastguard Worker if (r)
289*7688df22SAndroid Build Coastguard Worker return r;
290*7688df22SAndroid Build Coastguard Worker bof = bof_object();
291*7688df22SAndroid Build Coastguard Worker if (bof == NULL)
292*7688df22SAndroid Build Coastguard Worker return -ENOMEM;
293*7688df22SAndroid Build Coastguard Worker bof->offset = ftell(file);
294*7688df22SAndroid Build Coastguard Worker r = fread(&bof->type, 4, 1, file);
295*7688df22SAndroid Build Coastguard Worker if (r != 1)
296*7688df22SAndroid Build Coastguard Worker goto out_err;
297*7688df22SAndroid Build Coastguard Worker r = fread(&bof->size, 4, 1, file);
298*7688df22SAndroid Build Coastguard Worker if (r != 1)
299*7688df22SAndroid Build Coastguard Worker goto out_err;
300*7688df22SAndroid Build Coastguard Worker r = fread(&bof->array_size, 4, 1, file);
301*7688df22SAndroid Build Coastguard Worker if (r != 1)
302*7688df22SAndroid Build Coastguard Worker goto out_err;
303*7688df22SAndroid Build Coastguard Worker switch (bof->type) {
304*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_STRING:
305*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_INT32:
306*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_BLOB:
307*7688df22SAndroid Build Coastguard Worker bof->value = calloc(1, bof->size - 12);
308*7688df22SAndroid Build Coastguard Worker if (bof->value == NULL) {
309*7688df22SAndroid Build Coastguard Worker goto out_err;
310*7688df22SAndroid Build Coastguard Worker }
311*7688df22SAndroid Build Coastguard Worker r = fread(bof->value, bof->size - 12, 1, file);
312*7688df22SAndroid Build Coastguard Worker if (r != 1) {
313*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "error reading %d\n", bof->size - 12);
314*7688df22SAndroid Build Coastguard Worker goto out_err;
315*7688df22SAndroid Build Coastguard Worker }
316*7688df22SAndroid Build Coastguard Worker break;
317*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_NULL:
318*7688df22SAndroid Build Coastguard Worker return 0;
319*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_OBJECT:
320*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_ARRAY:
321*7688df22SAndroid Build Coastguard Worker r = bof_read(bof, file, bof->offset + bof->size, level + 2);
322*7688df22SAndroid Build Coastguard Worker if (r)
323*7688df22SAndroid Build Coastguard Worker goto out_err;
324*7688df22SAndroid Build Coastguard Worker break;
325*7688df22SAndroid Build Coastguard Worker default:
326*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "invalid type %d\n", bof->type);
327*7688df22SAndroid Build Coastguard Worker goto out_err;
328*7688df22SAndroid Build Coastguard Worker }
329*7688df22SAndroid Build Coastguard Worker root->array[root->centry++] = bof;
330*7688df22SAndroid Build Coastguard Worker return bof_read(root, file, end, level);
331*7688df22SAndroid Build Coastguard Worker out_err:
332*7688df22SAndroid Build Coastguard Worker bof_decref(bof);
333*7688df22SAndroid Build Coastguard Worker return -EINVAL;
334*7688df22SAndroid Build Coastguard Worker }
335*7688df22SAndroid Build Coastguard Worker
bof_load_file(const char * filename)336*7688df22SAndroid Build Coastguard Worker bof_t *bof_load_file(const char *filename)
337*7688df22SAndroid Build Coastguard Worker {
338*7688df22SAndroid Build Coastguard Worker bof_t *root = bof_object();
339*7688df22SAndroid Build Coastguard Worker int r;
340*7688df22SAndroid Build Coastguard Worker
341*7688df22SAndroid Build Coastguard Worker if (root == NULL) {
342*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%s failed to create root object\n", __func__);
343*7688df22SAndroid Build Coastguard Worker return NULL;
344*7688df22SAndroid Build Coastguard Worker }
345*7688df22SAndroid Build Coastguard Worker root->file = fopen(filename, "r");
346*7688df22SAndroid Build Coastguard Worker if (root->file == NULL)
347*7688df22SAndroid Build Coastguard Worker goto out_err;
348*7688df22SAndroid Build Coastguard Worker r = fseek(root->file, 0L, SEEK_SET);
349*7688df22SAndroid Build Coastguard Worker if (r) {
350*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%s failed to seek into file %s\n", __func__, filename);
351*7688df22SAndroid Build Coastguard Worker goto out_err;
352*7688df22SAndroid Build Coastguard Worker }
353*7688df22SAndroid Build Coastguard Worker root->offset = ftell(root->file);
354*7688df22SAndroid Build Coastguard Worker r = fread(&root->type, 4, 1, root->file);
355*7688df22SAndroid Build Coastguard Worker if (r != 1)
356*7688df22SAndroid Build Coastguard Worker goto out_err;
357*7688df22SAndroid Build Coastguard Worker r = fread(&root->size, 4, 1, root->file);
358*7688df22SAndroid Build Coastguard Worker if (r != 1)
359*7688df22SAndroid Build Coastguard Worker goto out_err;
360*7688df22SAndroid Build Coastguard Worker r = fread(&root->array_size, 4, 1, root->file);
361*7688df22SAndroid Build Coastguard Worker if (r != 1)
362*7688df22SAndroid Build Coastguard Worker goto out_err;
363*7688df22SAndroid Build Coastguard Worker r = bof_read(root, root->file, root->offset + root->size, 2);
364*7688df22SAndroid Build Coastguard Worker if (r)
365*7688df22SAndroid Build Coastguard Worker goto out_err;
366*7688df22SAndroid Build Coastguard Worker return root;
367*7688df22SAndroid Build Coastguard Worker out_err:
368*7688df22SAndroid Build Coastguard Worker bof_decref(root);
369*7688df22SAndroid Build Coastguard Worker return NULL;
370*7688df22SAndroid Build Coastguard Worker }
371*7688df22SAndroid Build Coastguard Worker
bof_incref(bof_t * bof)372*7688df22SAndroid Build Coastguard Worker void bof_incref(bof_t *bof)
373*7688df22SAndroid Build Coastguard Worker {
374*7688df22SAndroid Build Coastguard Worker bof->refcount++;
375*7688df22SAndroid Build Coastguard Worker }
376*7688df22SAndroid Build Coastguard Worker
bof_decref(bof_t * bof)377*7688df22SAndroid Build Coastguard Worker void bof_decref(bof_t *bof)
378*7688df22SAndroid Build Coastguard Worker {
379*7688df22SAndroid Build Coastguard Worker unsigned i;
380*7688df22SAndroid Build Coastguard Worker
381*7688df22SAndroid Build Coastguard Worker if (bof == NULL)
382*7688df22SAndroid Build Coastguard Worker return;
383*7688df22SAndroid Build Coastguard Worker if (--bof->refcount > 0)
384*7688df22SAndroid Build Coastguard Worker return;
385*7688df22SAndroid Build Coastguard Worker for (i = 0; i < bof->array_size; i++) {
386*7688df22SAndroid Build Coastguard Worker bof_decref(bof->array[i]);
387*7688df22SAndroid Build Coastguard Worker bof->array[i] = NULL;
388*7688df22SAndroid Build Coastguard Worker }
389*7688df22SAndroid Build Coastguard Worker bof->array_size = 0;
390*7688df22SAndroid Build Coastguard Worker if (bof->file) {
391*7688df22SAndroid Build Coastguard Worker fclose(bof->file);
392*7688df22SAndroid Build Coastguard Worker bof->file = NULL;
393*7688df22SAndroid Build Coastguard Worker }
394*7688df22SAndroid Build Coastguard Worker free(bof->array);
395*7688df22SAndroid Build Coastguard Worker free(bof->value);
396*7688df22SAndroid Build Coastguard Worker free(bof);
397*7688df22SAndroid Build Coastguard Worker }
398*7688df22SAndroid Build Coastguard Worker
bof_file_write(bof_t * bof,FILE * file)399*7688df22SAndroid Build Coastguard Worker static int bof_file_write(bof_t *bof, FILE *file)
400*7688df22SAndroid Build Coastguard Worker {
401*7688df22SAndroid Build Coastguard Worker unsigned i;
402*7688df22SAndroid Build Coastguard Worker int r;
403*7688df22SAndroid Build Coastguard Worker
404*7688df22SAndroid Build Coastguard Worker r = fwrite(&bof->type, 4, 1, file);
405*7688df22SAndroid Build Coastguard Worker if (r != 1)
406*7688df22SAndroid Build Coastguard Worker return -EINVAL;
407*7688df22SAndroid Build Coastguard Worker r = fwrite(&bof->size, 4, 1, file);
408*7688df22SAndroid Build Coastguard Worker if (r != 1)
409*7688df22SAndroid Build Coastguard Worker return -EINVAL;
410*7688df22SAndroid Build Coastguard Worker r = fwrite(&bof->array_size, 4, 1, file);
411*7688df22SAndroid Build Coastguard Worker if (r != 1)
412*7688df22SAndroid Build Coastguard Worker return -EINVAL;
413*7688df22SAndroid Build Coastguard Worker switch (bof->type) {
414*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_NULL:
415*7688df22SAndroid Build Coastguard Worker if (bof->size)
416*7688df22SAndroid Build Coastguard Worker return -EINVAL;
417*7688df22SAndroid Build Coastguard Worker break;
418*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_STRING:
419*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_INT32:
420*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_BLOB:
421*7688df22SAndroid Build Coastguard Worker r = fwrite(bof->value, bof->size - 12, 1, file);
422*7688df22SAndroid Build Coastguard Worker if (r != 1)
423*7688df22SAndroid Build Coastguard Worker return -EINVAL;
424*7688df22SAndroid Build Coastguard Worker break;
425*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_OBJECT:
426*7688df22SAndroid Build Coastguard Worker case BOF_TYPE_ARRAY:
427*7688df22SAndroid Build Coastguard Worker for (i = 0; i < bof->array_size; i++) {
428*7688df22SAndroid Build Coastguard Worker r = bof_file_write(bof->array[i], file);
429*7688df22SAndroid Build Coastguard Worker if (r)
430*7688df22SAndroid Build Coastguard Worker return r;
431*7688df22SAndroid Build Coastguard Worker }
432*7688df22SAndroid Build Coastguard Worker break;
433*7688df22SAndroid Build Coastguard Worker default:
434*7688df22SAndroid Build Coastguard Worker return -EINVAL;
435*7688df22SAndroid Build Coastguard Worker }
436*7688df22SAndroid Build Coastguard Worker return 0;
437*7688df22SAndroid Build Coastguard Worker }
438*7688df22SAndroid Build Coastguard Worker
bof_dump_file(bof_t * bof,const char * filename)439*7688df22SAndroid Build Coastguard Worker int bof_dump_file(bof_t *bof, const char *filename)
440*7688df22SAndroid Build Coastguard Worker {
441*7688df22SAndroid Build Coastguard Worker unsigned i;
442*7688df22SAndroid Build Coastguard Worker int r = 0;
443*7688df22SAndroid Build Coastguard Worker
444*7688df22SAndroid Build Coastguard Worker if (bof->file) {
445*7688df22SAndroid Build Coastguard Worker fclose(bof->file);
446*7688df22SAndroid Build Coastguard Worker bof->file = NULL;
447*7688df22SAndroid Build Coastguard Worker }
448*7688df22SAndroid Build Coastguard Worker bof->file = fopen(filename, "w");
449*7688df22SAndroid Build Coastguard Worker if (bof->file == NULL) {
450*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%s failed to open file %s\n", __func__, filename);
451*7688df22SAndroid Build Coastguard Worker r = -EINVAL;
452*7688df22SAndroid Build Coastguard Worker goto out_err;
453*7688df22SAndroid Build Coastguard Worker }
454*7688df22SAndroid Build Coastguard Worker r = fseek(bof->file, 0L, SEEK_SET);
455*7688df22SAndroid Build Coastguard Worker if (r) {
456*7688df22SAndroid Build Coastguard Worker fprintf(stderr, "%s failed to seek into file %s\n", __func__, filename);
457*7688df22SAndroid Build Coastguard Worker goto out_err;
458*7688df22SAndroid Build Coastguard Worker }
459*7688df22SAndroid Build Coastguard Worker r = fwrite(&bof->type, 4, 1, bof->file);
460*7688df22SAndroid Build Coastguard Worker if (r != 1)
461*7688df22SAndroid Build Coastguard Worker goto out_err;
462*7688df22SAndroid Build Coastguard Worker r = fwrite(&bof->size, 4, 1, bof->file);
463*7688df22SAndroid Build Coastguard Worker if (r != 1)
464*7688df22SAndroid Build Coastguard Worker goto out_err;
465*7688df22SAndroid Build Coastguard Worker r = fwrite(&bof->array_size, 4, 1, bof->file);
466*7688df22SAndroid Build Coastguard Worker if (r != 1)
467*7688df22SAndroid Build Coastguard Worker goto out_err;
468*7688df22SAndroid Build Coastguard Worker for (i = 0; i < bof->array_size; i++) {
469*7688df22SAndroid Build Coastguard Worker r = bof_file_write(bof->array[i], bof->file);
470*7688df22SAndroid Build Coastguard Worker if (r)
471*7688df22SAndroid Build Coastguard Worker return r;
472*7688df22SAndroid Build Coastguard Worker }
473*7688df22SAndroid Build Coastguard Worker out_err:
474*7688df22SAndroid Build Coastguard Worker fclose(bof->file);
475*7688df22SAndroid Build Coastguard Worker bof->file = NULL;
476*7688df22SAndroid Build Coastguard Worker return r;
477*7688df22SAndroid Build Coastguard Worker }
478