1*0f4c859eSApple OSS Distributions /*
2*0f4c859eSApple OSS Distributions * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
3*0f4c859eSApple OSS Distributions *
4*0f4c859eSApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*0f4c859eSApple OSS Distributions *
6*0f4c859eSApple OSS Distributions * This file contains Original Code and/or Modifications of Original Code
7*0f4c859eSApple OSS Distributions * as defined in and that are subject to the Apple Public Source License
8*0f4c859eSApple OSS Distributions * Version 2.0 (the 'License'). You may not use this file except in
9*0f4c859eSApple OSS Distributions * compliance with the License. The rights granted to you under the License
10*0f4c859eSApple OSS Distributions * may not be used to create, or enable the creation or redistribution of,
11*0f4c859eSApple OSS Distributions * unlawful or unlicensed copies of an Apple operating system, or to
12*0f4c859eSApple OSS Distributions * circumvent, violate, or enable the circumvention or violation of, any
13*0f4c859eSApple OSS Distributions * terms of an Apple operating system software license agreement.
14*0f4c859eSApple OSS Distributions *
15*0f4c859eSApple OSS Distributions * Please obtain a copy of the License at
16*0f4c859eSApple OSS Distributions * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*0f4c859eSApple OSS Distributions *
18*0f4c859eSApple OSS Distributions * The Original Code and all software distributed under the License are
19*0f4c859eSApple OSS Distributions * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*0f4c859eSApple OSS Distributions * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*0f4c859eSApple OSS Distributions * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*0f4c859eSApple OSS Distributions * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*0f4c859eSApple OSS Distributions * Please see the License for the specific language governing rights and
24*0f4c859eSApple OSS Distributions * limitations under the License.
25*0f4c859eSApple OSS Distributions *
26*0f4c859eSApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*0f4c859eSApple OSS Distributions */
28*0f4c859eSApple OSS Distributions
29*0f4c859eSApple OSS Distributions /* OSUnserialize.y created by rsulack on Nov 21 1998 */
30*0f4c859eSApple OSS Distributions
31*0f4c859eSApple OSS Distributions // "classic" parser for unserializing OSContainer objects
32*0f4c859eSApple OSS Distributions //
33*0f4c859eSApple OSS Distributions // XXX - this code should really be removed!
34*0f4c859eSApple OSS Distributions // - the XML format is now prefered
35*0f4c859eSApple OSS Distributions // - this code leaks on syntax errors, the XML doesn't
36*0f4c859eSApple OSS Distributions // - "classic" looks, reads, ... much better than XML :-(
37*0f4c859eSApple OSS Distributions // - well except the XML is more efficent on OSData
38*0f4c859eSApple OSS Distributions //
39*0f4c859eSApple OSS Distributions //
40*0f4c859eSApple OSS Distributions // to build :
41*0f4c859eSApple OSS Distributions // bison -p OSUnserialize OSUnserialize.y
42*0f4c859eSApple OSS Distributions // head -50 OSUnserialize.y > OSUnserialize.cpp
43*0f4c859eSApple OSS Distributions // sed -e "s/stdio.h/stddef.h/" < OSUnserialize.tab.c >> OSUnserialize.cpp
44*0f4c859eSApple OSS Distributions //
45*0f4c859eSApple OSS Distributions // when changing code check in both OSUnserialize.y and OSUnserialize.cpp
46*0f4c859eSApple OSS Distributions //
47*0f4c859eSApple OSS Distributions //
48*0f4c859eSApple OSS Distributions //
49*0f4c859eSApple OSS Distributions //
50*0f4c859eSApple OSS Distributions // DO NOT EDIT OSUnserialize.tab.cpp!
51*0f4c859eSApple OSS Distributions //
52*0f4c859eSApple OSS Distributions // this means you!
53*0f4c859eSApple OSS Distributions //
54*0f4c859eSApple OSS Distributions //
55*0f4c859eSApple OSS Distributions //
56*0f4c859eSApple OSS Distributions //
57*0f4c859eSApple OSS Distributions //
58*0f4c859eSApple OSS Distributions
59*0f4c859eSApple OSS Distributions
60*0f4c859eSApple OSS Distributions %{
61*0f4c859eSApple OSS Distributions #include <libkern/c++/OSMetaClass.h>
62*0f4c859eSApple OSS Distributions #include <libkern/c++/OSContainers.h>
63*0f4c859eSApple OSS Distributions #include <libkern/c++/OSLib.h>
64*0f4c859eSApple OSS Distributions
65*0f4c859eSApple OSS Distributions typedef struct object {
66*0f4c859eSApple OSS Distributions struct object *next;
67*0f4c859eSApple OSS Distributions struct object *prev;
68*0f4c859eSApple OSS Distributions void *object;
69*0f4c859eSApple OSS Distributions int size; // for data
70*0f4c859eSApple OSS Distributions union {
71*0f4c859eSApple OSS Distributions void *key; // for dictionary
72*0f4c859eSApple OSS Distributions long long offset; // for offset
73*0f4c859eSApple OSS Distributions } u;
74*0f4c859eSApple OSS Distributions
75*0f4c859eSApple OSS Distributions } object_t;
76*0f4c859eSApple OSS Distributions
77*0f4c859eSApple OSS Distributions static int yyerror(const char *s);
78*0f4c859eSApple OSS Distributions static int yylex();
79*0f4c859eSApple OSS Distributions
80*0f4c859eSApple OSS Distributions static object_t * newObject();
81*0f4c859eSApple OSS Distributions static void freeObject(object_t *o);
82*0f4c859eSApple OSS Distributions
83*0f4c859eSApple OSS Distributions static OSObject *buildOSDictionary(object_t *);
84*0f4c859eSApple OSS Distributions static OSObject *buildOSArray(object_t *);
85*0f4c859eSApple OSS Distributions static OSObject *buildOSSet(object_t *);
86*0f4c859eSApple OSS Distributions static OSObject *buildOSString(object_t *);
87*0f4c859eSApple OSS Distributions static OSObject *buildOSData(object_t *);
88*0f4c859eSApple OSS Distributions static OSObject *buildOSOffset(object_t *);
89*0f4c859eSApple OSS Distributions static OSObject *buildOSBoolean(object_t *o);
90*0f4c859eSApple OSS Distributions
91*0f4c859eSApple OSS Distributions static void rememberObject(int, object_t *);
92*0f4c859eSApple OSS Distributions static OSObject *retrieveObject(int);
93*0f4c859eSApple OSS Distributions
94*0f4c859eSApple OSS Distributions // temp variable to use during parsing
95*0f4c859eSApple OSS Distributions static object_t *oo;
96*0f4c859eSApple OSS Distributions
97*0f4c859eSApple OSS Distributions // resultant object of parsed text
98*0f4c859eSApple OSS Distributions static OSObject *parsedObject;
99*0f4c859eSApple OSS Distributions
100*0f4c859eSApple OSS Distributions #define YYSTYPE object_t *
101*0f4c859eSApple OSS Distributions
102*0f4c859eSApple OSS Distributions __BEGIN_DECLS
103*0f4c859eSApple OSS Distributions #include <kern/kalloc.h>
104*0f4c859eSApple OSS Distributions __END_DECLS
105*0f4c859eSApple OSS Distributions
106*0f4c859eSApple OSS Distributions // Omit from static analysis.
107*0f4c859eSApple OSS Distributions #ifndef __clang_analyzer__
108*0f4c859eSApple OSS Distributions
109*0f4c859eSApple OSS Distributions #define malloc(size) malloc_impl(size)
110*0f4c859eSApple OSS Distributions #define malloc_type(type) kalloc_type(type, Z_SET_NOTSHARED)
111*0f4c859eSApple OSS Distributions static inline void *
malloc_impl(size_t size)112*0f4c859eSApple OSS Distributions malloc_impl(size_t size)
113*0f4c859eSApple OSS Distributions {
114*0f4c859eSApple OSS Distributions if (size == 0) {
115*0f4c859eSApple OSS Distributions return NULL;
116*0f4c859eSApple OSS Distributions }
117*0f4c859eSApple OSS Distributions return kalloc_data(size,
118*0f4c859eSApple OSS Distributions Z_VM_TAG_BT(Z_WAITOK_ZERO, VM_KERN_MEMORY_LIBKERN));
119*0f4c859eSApple OSS Distributions }
120*0f4c859eSApple OSS Distributions
121*0f4c859eSApple OSS Distributions #define free(addr) free_impl(addr)
122*0f4c859eSApple OSS Distributions #define free_type(type, addr) kfree_type(type, addr)
123*0f4c859eSApple OSS Distributions static inline void
free_impl(void * addr)124*0f4c859eSApple OSS Distributions free_impl(void *addr)
125*0f4c859eSApple OSS Distributions {
126*0f4c859eSApple OSS Distributions kfree_data_addr(addr);
127*0f4c859eSApple OSS Distributions }
128*0f4c859eSApple OSS Distributions static inline void
safe_free(void * addr,size_t size)129*0f4c859eSApple OSS Distributions safe_free(void *addr, size_t size)
130*0f4c859eSApple OSS Distributions {
131*0f4c859eSApple OSS Distributions kfree_data(addr, size);
132*0f4c859eSApple OSS Distributions }
133*0f4c859eSApple OSS Distributions
134*0f4c859eSApple OSS Distributions #define realloc(addr, osize, nsize) realloc_impl(addr, osize, nsize)
135*0f4c859eSApple OSS Distributions static inline void *
realloc_impl(void * addr,size_t osize,size_t nsize)136*0f4c859eSApple OSS Distributions realloc_impl(void *addr, size_t osize, size_t nsize)
137*0f4c859eSApple OSS Distributions {
138*0f4c859eSApple OSS Distributions return krealloc_data(addr, osize, nsize,
139*0f4c859eSApple OSS Distributions Z_VM_TAG_BT(Z_WAITOK_ZERO, VM_KERN_MEMORY_LIBKERN));
140*0f4c859eSApple OSS Distributions }
141*0f4c859eSApple OSS Distributions
142*0f4c859eSApple OSS Distributions %}
143*0f4c859eSApple OSS Distributions %token NUMBER
144*0f4c859eSApple OSS Distributions %token STRING
145*0f4c859eSApple OSS Distributions %token DATA
146*0f4c859eSApple OSS Distributions %token BOOLEAN
147*0f4c859eSApple OSS Distributions %token SYNTAX_ERROR
148*0f4c859eSApple OSS Distributions
149*0f4c859eSApple OSS Distributions %% /* Grammar rules and actions follow */
150*0f4c859eSApple OSS Distributions
151*0f4c859eSApple OSS Distributions input: /* empty */ { parsedObject = (OSObject *)NULL; YYACCEPT; }
152*0f4c859eSApple OSS Distributions | object { parsedObject = (OSObject *)$1; YYACCEPT; }
153*0f4c859eSApple OSS Distributions | SYNTAX_ERROR { yyerror("syntax error"); YYERROR; }
154*0f4c859eSApple OSS Distributions ;
155*0f4c859eSApple OSS Distributions
156*0f4c859eSApple OSS Distributions object: dict { $$ = (object_t *)buildOSDictionary($1); }
157*0f4c859eSApple OSS Distributions | array { $$ = (object_t *)buildOSArray($1); }
158*0f4c859eSApple OSS Distributions | set { $$ = (object_t *)buildOSSet($1); }
159*0f4c859eSApple OSS Distributions | string { $$ = (object_t *)buildOSString($1); }
160*0f4c859eSApple OSS Distributions | data { $$ = (object_t *)buildOSData($1); }
161*0f4c859eSApple OSS Distributions | offset { $$ = (object_t *)buildOSOffset($1); }
162*0f4c859eSApple OSS Distributions | boolean { $$ = (object_t *)buildOSBoolean($1); }
163*0f4c859eSApple OSS Distributions | '@' NUMBER { $$ = (object_t *)retrieveObject($2->u.offset);
164*0f4c859eSApple OSS Distributions if ($$) {
165*0f4c859eSApple OSS Distributions ((OSObject *)$$)->retain();
166*0f4c859eSApple OSS Distributions } else {
167*0f4c859eSApple OSS Distributions yyerror("forward reference detected");
168*0f4c859eSApple OSS Distributions YYERROR;
169*0f4c859eSApple OSS Distributions }
170*0f4c859eSApple OSS Distributions freeObject($2);
171*0f4c859eSApple OSS Distributions }
172*0f4c859eSApple OSS Distributions | object '@' NUMBER { $$ = $1;
173*0f4c859eSApple OSS Distributions rememberObject($3->u.offset, $1);
174*0f4c859eSApple OSS Distributions freeObject($3);
175*0f4c859eSApple OSS Distributions }
176*0f4c859eSApple OSS Distributions ;
177*0f4c859eSApple OSS Distributions
178*0f4c859eSApple OSS Distributions //------------------------------------------------------------------------------
179*0f4c859eSApple OSS Distributions
180*0f4c859eSApple OSS Distributions dict: '{' '}' { $$ = NULL; }
181*0f4c859eSApple OSS Distributions | '{' pairs '}' { $$ = $2; }
182*0f4c859eSApple OSS Distributions ;
183*0f4c859eSApple OSS Distributions
184*0f4c859eSApple OSS Distributions pairs: pair
185*0f4c859eSApple OSS Distributions | pairs pair { $2->next = $1; $1->prev = $2; $$ = $2; }
186*0f4c859eSApple OSS Distributions ;
187*0f4c859eSApple OSS Distributions
188*0f4c859eSApple OSS Distributions pair: object '=' object ';' { $$ = newObject();
189*0f4c859eSApple OSS Distributions $$->next = NULL;
190*0f4c859eSApple OSS Distributions $$->prev = NULL;
191*0f4c859eSApple OSS Distributions $$->u.key = $1;
192*0f4c859eSApple OSS Distributions $$->object = $3;
193*0f4c859eSApple OSS Distributions }
194*0f4c859eSApple OSS Distributions ;
195*0f4c859eSApple OSS Distributions
196*0f4c859eSApple OSS Distributions //------------------------------------------------------------------------------
197*0f4c859eSApple OSS Distributions
198*0f4c859eSApple OSS Distributions array: '(' ')' { $$ = NULL; }
199*0f4c859eSApple OSS Distributions | '(' elements ')' { $$ = $2; }
200*0f4c859eSApple OSS Distributions ;
201*0f4c859eSApple OSS Distributions
202*0f4c859eSApple OSS Distributions set: '[' ']' { $$ = NULL; }
203*0f4c859eSApple OSS Distributions | '[' elements ']' { $$ = $2; }
204*0f4c859eSApple OSS Distributions ;
205*0f4c859eSApple OSS Distributions
206*0f4c859eSApple OSS Distributions elements: object { $$ = newObject();
207*0f4c859eSApple OSS Distributions $$->object = $1;
208*0f4c859eSApple OSS Distributions $$->next = NULL;
209*0f4c859eSApple OSS Distributions $$->prev = NULL;
210*0f4c859eSApple OSS Distributions }
211*0f4c859eSApple OSS Distributions | elements ',' object { oo = newObject();
212*0f4c859eSApple OSS Distributions oo->object = $3;
213*0f4c859eSApple OSS Distributions oo->next = $1;
214*0f4c859eSApple OSS Distributions oo->prev = NULL;
215*0f4c859eSApple OSS Distributions $1->prev = oo;
216*0f4c859eSApple OSS Distributions $$ = oo;
217*0f4c859eSApple OSS Distributions }
218*0f4c859eSApple OSS Distributions ;
219*0f4c859eSApple OSS Distributions
220*0f4c859eSApple OSS Distributions //------------------------------------------------------------------------------
221*0f4c859eSApple OSS Distributions
222*0f4c859eSApple OSS Distributions offset: NUMBER ':' NUMBER { $$ = $1;
223*0f4c859eSApple OSS Distributions $$->size = $3->u.offset;
224*0f4c859eSApple OSS Distributions freeObject($3);
225*0f4c859eSApple OSS Distributions }
226*0f4c859eSApple OSS Distributions ;
227*0f4c859eSApple OSS Distributions
228*0f4c859eSApple OSS Distributions //------------------------------------------------------------------------------
229*0f4c859eSApple OSS Distributions
230*0f4c859eSApple OSS Distributions data: DATA
231*0f4c859eSApple OSS Distributions ;
232*0f4c859eSApple OSS Distributions
233*0f4c859eSApple OSS Distributions //------------------------------------------------------------------------------
234*0f4c859eSApple OSS Distributions
235*0f4c859eSApple OSS Distributions string: STRING
236*0f4c859eSApple OSS Distributions ;
237*0f4c859eSApple OSS Distributions
238*0f4c859eSApple OSS Distributions //------------------------------------------------------------------------------
239*0f4c859eSApple OSS Distributions
240*0f4c859eSApple OSS Distributions boolean: BOOLEAN
241*0f4c859eSApple OSS Distributions ;
242*0f4c859eSApple OSS Distributions
243*0f4c859eSApple OSS Distributions %%
244*0f4c859eSApple OSS Distributions
245*0f4c859eSApple OSS Distributions static int lineNumber = 0;
246*0f4c859eSApple OSS Distributions static const char *parseBuffer;
247*0f4c859eSApple OSS Distributions static int parseBufferIndex;
248*0f4c859eSApple OSS Distributions
249*0f4c859eSApple OSS Distributions #define currentChar() (parseBuffer[parseBufferIndex])
250*0f4c859eSApple OSS Distributions #define nextChar() (parseBuffer[++parseBufferIndex])
251*0f4c859eSApple OSS Distributions #define prevChar() (parseBuffer[parseBufferIndex - 1])
252*0f4c859eSApple OSS Distributions
253*0f4c859eSApple OSS Distributions #define isSpace(c) ((c) == ' ' || (c) == '\t')
254*0f4c859eSApple OSS Distributions #define isAlpha(c) (((c) >= 'A' && (c) <= 'Z') || ((c) >= 'a' && (c) <= 'z'))
255*0f4c859eSApple OSS Distributions #define isDigit(c) ((c) >= '0' && (c) <= '9')
256*0f4c859eSApple OSS Distributions #define isAlphaDigit(c) ((c) >= 'a' && (c) <= 'f')
257*0f4c859eSApple OSS Distributions #define isHexDigit(c) (isDigit(c) || isAlphaDigit(c))
258*0f4c859eSApple OSS Distributions #define isAlphaNumeric(c) (isAlpha(c) || isDigit(c) || ((c) == '-'))
259*0f4c859eSApple OSS Distributions
260*0f4c859eSApple OSS Distributions static char yyerror_message[128];
261*0f4c859eSApple OSS Distributions
262*0f4c859eSApple OSS Distributions int
yyerror(const char * s)263*0f4c859eSApple OSS Distributions yyerror(const char *s) /* Called by yyparse on error */
264*0f4c859eSApple OSS Distributions {
265*0f4c859eSApple OSS Distributions snprintf(yyerror_message, sizeof(yyerror_message), "OSUnserialize: %s near line %d\n", s, lineNumber);
266*0f4c859eSApple OSS Distributions return 0;
267*0f4c859eSApple OSS Distributions }
268*0f4c859eSApple OSS Distributions
269*0f4c859eSApple OSS Distributions int
yylex()270*0f4c859eSApple OSS Distributions yylex()
271*0f4c859eSApple OSS Distributions {
272*0f4c859eSApple OSS Distributions int c;
273*0f4c859eSApple OSS Distributions
274*0f4c859eSApple OSS Distributions if (parseBufferIndex == 0) lineNumber = 1;
275*0f4c859eSApple OSS Distributions
276*0f4c859eSApple OSS Distributions top:
277*0f4c859eSApple OSS Distributions c = currentChar();
278*0f4c859eSApple OSS Distributions
279*0f4c859eSApple OSS Distributions /* skip white space */
280*0f4c859eSApple OSS Distributions if (isSpace(c)) while ((c = nextChar()) != 0 && isSpace(c)) {};
281*0f4c859eSApple OSS Distributions
282*0f4c859eSApple OSS Distributions /* skip over comments */
283*0f4c859eSApple OSS Distributions if (c == '#') while ((c = nextChar()) != 0 && c != '\n') {};
284*0f4c859eSApple OSS Distributions
285*0f4c859eSApple OSS Distributions /* keep track of line number, don't return \n's */
286*0f4c859eSApple OSS Distributions if (c == '\n') {
287*0f4c859eSApple OSS Distributions lineNumber++;
288*0f4c859eSApple OSS Distributions (void)nextChar();
289*0f4c859eSApple OSS Distributions goto top;
290*0f4c859eSApple OSS Distributions }
291*0f4c859eSApple OSS Distributions
292*0f4c859eSApple OSS Distributions /* parse boolean */
293*0f4c859eSApple OSS Distributions if (c == '.') {
294*0f4c859eSApple OSS Distributions bool boolean = false;
295*0f4c859eSApple OSS Distributions if (nextChar() == 't') {
296*0f4c859eSApple OSS Distributions if (nextChar() != 'r') return SYNTAX_ERROR;
297*0f4c859eSApple OSS Distributions if (nextChar() != 'u') return SYNTAX_ERROR;
298*0f4c859eSApple OSS Distributions if (nextChar() != 'e') return SYNTAX_ERROR;
299*0f4c859eSApple OSS Distributions boolean = true;
300*0f4c859eSApple OSS Distributions } else {
301*0f4c859eSApple OSS Distributions if (currentChar() != 'f') return SYNTAX_ERROR;
302*0f4c859eSApple OSS Distributions if (nextChar() != 'a') return SYNTAX_ERROR;
303*0f4c859eSApple OSS Distributions if (nextChar() != 'l') return SYNTAX_ERROR;
304*0f4c859eSApple OSS Distributions if (nextChar() != 's') return SYNTAX_ERROR;
305*0f4c859eSApple OSS Distributions if (nextChar() != 'e') return SYNTAX_ERROR;
306*0f4c859eSApple OSS Distributions }
307*0f4c859eSApple OSS Distributions if (nextChar() != '.') return SYNTAX_ERROR;
308*0f4c859eSApple OSS Distributions /* skip over dot */
309*0f4c859eSApple OSS Distributions (void)nextChar();
310*0f4c859eSApple OSS Distributions
311*0f4c859eSApple OSS Distributions yylval = (object_t *)boolean;
312*0f4c859eSApple OSS Distributions return BOOLEAN;
313*0f4c859eSApple OSS Distributions }
314*0f4c859eSApple OSS Distributions
315*0f4c859eSApple OSS Distributions /* parse unquoted string */
316*0f4c859eSApple OSS Distributions if (isAlpha(c)) {
317*0f4c859eSApple OSS Distributions int start, length;
318*0f4c859eSApple OSS Distributions char * tempString;
319*0f4c859eSApple OSS Distributions
320*0f4c859eSApple OSS Distributions start = parseBufferIndex;
321*0f4c859eSApple OSS Distributions /* find end of string */
322*0f4c859eSApple OSS Distributions while (isAlphaNumeric(c)) {
323*0f4c859eSApple OSS Distributions c = nextChar();
324*0f4c859eSApple OSS Distributions }
325*0f4c859eSApple OSS Distributions length = parseBufferIndex - start;
326*0f4c859eSApple OSS Distributions
327*0f4c859eSApple OSS Distributions /* copy to null terminated buffer */
328*0f4c859eSApple OSS Distributions tempString = (char *)malloc(length + 1);
329*0f4c859eSApple OSS Distributions if (tempString == NULL) {
330*0f4c859eSApple OSS Distributions printf("OSUnserialize: can't alloc temp memory\n");
331*0f4c859eSApple OSS Distributions return 0;
332*0f4c859eSApple OSS Distributions }
333*0f4c859eSApple OSS Distributions bcopy(&parseBuffer[start], tempString, length);
334*0f4c859eSApple OSS Distributions tempString[length] = 0;
335*0f4c859eSApple OSS Distributions yylval = (object_t *)tempString;
336*0f4c859eSApple OSS Distributions return STRING;
337*0f4c859eSApple OSS Distributions }
338*0f4c859eSApple OSS Distributions
339*0f4c859eSApple OSS Distributions /* parse quoted string */
340*0f4c859eSApple OSS Distributions if (c == '"' || c == '\'') {
341*0f4c859eSApple OSS Distributions int start, length;
342*0f4c859eSApple OSS Distributions char * tempString;
343*0f4c859eSApple OSS Distributions char quoteChar = c;
344*0f4c859eSApple OSS Distributions
345*0f4c859eSApple OSS Distributions start = parseBufferIndex + 1; // skip quote
346*0f4c859eSApple OSS Distributions /* find end of string, line, buffer */
347*0f4c859eSApple OSS Distributions while ((c = nextChar()) != quoteChar) {
348*0f4c859eSApple OSS Distributions if (c == '\\') c = nextChar();
349*0f4c859eSApple OSS Distributions if (c == '\n') lineNumber++;
350*0f4c859eSApple OSS Distributions if (c == 0) return SYNTAX_ERROR;
351*0f4c859eSApple OSS Distributions }
352*0f4c859eSApple OSS Distributions length = parseBufferIndex - start;
353*0f4c859eSApple OSS Distributions /* skip over trailing quote */
354*0f4c859eSApple OSS Distributions (void)nextChar();
355*0f4c859eSApple OSS Distributions /* copy to null terminated buffer */
356*0f4c859eSApple OSS Distributions tempString = (char *)malloc(length + 1);
357*0f4c859eSApple OSS Distributions if (tempString == NULL) {
358*0f4c859eSApple OSS Distributions printf("OSUnserialize: can't alloc temp memory\n");
359*0f4c859eSApple OSS Distributions return 0;
360*0f4c859eSApple OSS Distributions }
361*0f4c859eSApple OSS Distributions
362*0f4c859eSApple OSS Distributions int to = 0;
363*0f4c859eSApple OSS Distributions for (int from=start; from < parseBufferIndex; from++) {
364*0f4c859eSApple OSS Distributions // hack - skip over backslashes
365*0f4c859eSApple OSS Distributions if (parseBuffer[from] == '\\') {
366*0f4c859eSApple OSS Distributions length--;
367*0f4c859eSApple OSS Distributions continue;
368*0f4c859eSApple OSS Distributions }
369*0f4c859eSApple OSS Distributions tempString[to] = parseBuffer[from];
370*0f4c859eSApple OSS Distributions to++;
371*0f4c859eSApple OSS Distributions }
372*0f4c859eSApple OSS Distributions tempString[length] = 0;
373*0f4c859eSApple OSS Distributions yylval = (object_t *)tempString;
374*0f4c859eSApple OSS Distributions return STRING;
375*0f4c859eSApple OSS Distributions }
376*0f4c859eSApple OSS Distributions
377*0f4c859eSApple OSS Distributions /* process numbers */
378*0f4c859eSApple OSS Distributions if (isDigit (c))
379*0f4c859eSApple OSS Distributions {
380*0f4c859eSApple OSS Distributions unsigned long long n = 0;
381*0f4c859eSApple OSS Distributions int base = 10;
382*0f4c859eSApple OSS Distributions
383*0f4c859eSApple OSS Distributions if (c == '0') {
384*0f4c859eSApple OSS Distributions c = nextChar();
385*0f4c859eSApple OSS Distributions if (c == 'x') {
386*0f4c859eSApple OSS Distributions base = 16;
387*0f4c859eSApple OSS Distributions c = nextChar();
388*0f4c859eSApple OSS Distributions }
389*0f4c859eSApple OSS Distributions }
390*0f4c859eSApple OSS Distributions if (base == 10) {
391*0f4c859eSApple OSS Distributions while(isDigit(c)) {
392*0f4c859eSApple OSS Distributions n = (n * base + c - '0');
393*0f4c859eSApple OSS Distributions c = nextChar();
394*0f4c859eSApple OSS Distributions }
395*0f4c859eSApple OSS Distributions } else {
396*0f4c859eSApple OSS Distributions while(isHexDigit(c)) {
397*0f4c859eSApple OSS Distributions if (isDigit(c)) {
398*0f4c859eSApple OSS Distributions n = (n * base + c - '0');
399*0f4c859eSApple OSS Distributions } else {
400*0f4c859eSApple OSS Distributions n = (n * base + 0xa + c - 'a');
401*0f4c859eSApple OSS Distributions }
402*0f4c859eSApple OSS Distributions c = nextChar();
403*0f4c859eSApple OSS Distributions }
404*0f4c859eSApple OSS Distributions }
405*0f4c859eSApple OSS Distributions
406*0f4c859eSApple OSS Distributions yylval = newObject();
407*0f4c859eSApple OSS Distributions yylval->u.offset = n;
408*0f4c859eSApple OSS Distributions
409*0f4c859eSApple OSS Distributions return NUMBER;
410*0f4c859eSApple OSS Distributions }
411*0f4c859eSApple OSS Distributions
412*0f4c859eSApple OSS Distributions #define OSDATA_ALLOC_SIZE 4096
413*0f4c859eSApple OSS Distributions
414*0f4c859eSApple OSS Distributions /* process data */
415*0f4c859eSApple OSS Distributions if (c == '<') {
416*0f4c859eSApple OSS Distributions unsigned char *d, *start, *lastStart;
417*0f4c859eSApple OSS Distributions
418*0f4c859eSApple OSS Distributions size_t buflen = OSDATA_ALLOC_SIZE;
419*0f4c859eSApple OSS Distributions start = lastStart = d = (unsigned char *)malloc(buflen);
420*0f4c859eSApple OSS Distributions c = nextChar(); // skip over '<'
421*0f4c859eSApple OSS Distributions while (c != 0 && c != '>') {
422*0f4c859eSApple OSS Distributions
423*0f4c859eSApple OSS Distributions if (isSpace(c)) while ((c = nextChar()) != 0 && isSpace(c)) {};
424*0f4c859eSApple OSS Distributions if (c == '#') while ((c = nextChar()) != 0 && c != '\n') {};
425*0f4c859eSApple OSS Distributions if (c == '\n') {
426*0f4c859eSApple OSS Distributions lineNumber++;
427*0f4c859eSApple OSS Distributions c = nextChar();
428*0f4c859eSApple OSS Distributions continue;
429*0f4c859eSApple OSS Distributions }
430*0f4c859eSApple OSS Distributions
431*0f4c859eSApple OSS Distributions // get high nibble
432*0f4c859eSApple OSS Distributions if (!isHexDigit(c)) break;
433*0f4c859eSApple OSS Distributions if (isDigit(c)) {
434*0f4c859eSApple OSS Distributions *d = (c - '0') << 4;
435*0f4c859eSApple OSS Distributions } else {
436*0f4c859eSApple OSS Distributions *d = (0xa + (c - 'a')) << 4;
437*0f4c859eSApple OSS Distributions }
438*0f4c859eSApple OSS Distributions
439*0f4c859eSApple OSS Distributions // get low nibble
440*0f4c859eSApple OSS Distributions c = nextChar();
441*0f4c859eSApple OSS Distributions if (!isHexDigit(c)) break;
442*0f4c859eSApple OSS Distributions if (isDigit(c)) {
443*0f4c859eSApple OSS Distributions *d |= c - '0';
444*0f4c859eSApple OSS Distributions } else {
445*0f4c859eSApple OSS Distributions *d |= 0xa + (c - 'a');
446*0f4c859eSApple OSS Distributions }
447*0f4c859eSApple OSS Distributions
448*0f4c859eSApple OSS Distributions d++;
449*0f4c859eSApple OSS Distributions if ((d - lastStart) >= OSDATA_ALLOC_SIZE) {
450*0f4c859eSApple OSS Distributions int oldsize = d - start;
451*0f4c859eSApple OSS Distributions assert(buflen == oldsize);
452*0f4c859eSApple OSS Distributions start = (unsigned char *)realloc(start, oldsize, buflen);
453*0f4c859eSApple OSS Distributions d = lastStart = start + oldsize;
454*0f4c859eSApple OSS Distributions }
455*0f4c859eSApple OSS Distributions c = nextChar();
456*0f4c859eSApple OSS Distributions }
457*0f4c859eSApple OSS Distributions if (c != '>' ) {
458*0f4c859eSApple OSS Distributions safe_free(start, buflen);
459*0f4c859eSApple OSS Distributions return SYNTAX_ERROR;
460*0f4c859eSApple OSS Distributions }
461*0f4c859eSApple OSS Distributions
462*0f4c859eSApple OSS Distributions // got it!
463*0f4c859eSApple OSS Distributions yylval = newObject();
464*0f4c859eSApple OSS Distributions yylval->object = start;
465*0f4c859eSApple OSS Distributions yylval->size = d - start;
466*0f4c859eSApple OSS Distributions
467*0f4c859eSApple OSS Distributions (void)nextChar(); // skip over '>'
468*0f4c859eSApple OSS Distributions return DATA;
469*0f4c859eSApple OSS Distributions }
470*0f4c859eSApple OSS Distributions
471*0f4c859eSApple OSS Distributions
472*0f4c859eSApple OSS Distributions /* return single chars, move pointer to next char */
473*0f4c859eSApple OSS Distributions (void)nextChar();
474*0f4c859eSApple OSS Distributions return c;
475*0f4c859eSApple OSS Distributions }
476*0f4c859eSApple OSS Distributions
477*0f4c859eSApple OSS Distributions // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
478*0f4c859eSApple OSS Distributions // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
479*0f4c859eSApple OSS Distributions // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
480*0f4c859eSApple OSS Distributions
481*0f4c859eSApple OSS Distributions #if DEBUG
482*0f4c859eSApple OSS Distributions int debugUnserializeAllocCount = 0;
483*0f4c859eSApple OSS Distributions #endif
484*0f4c859eSApple OSS Distributions
485*0f4c859eSApple OSS Distributions object_t *
newObject()486*0f4c859eSApple OSS Distributions newObject()
487*0f4c859eSApple OSS Distributions {
488*0f4c859eSApple OSS Distributions #if DEBUG
489*0f4c859eSApple OSS Distributions debugUnserializeAllocCount++;
490*0f4c859eSApple OSS Distributions #endif
491*0f4c859eSApple OSS Distributions return malloc_type(object_t);
492*0f4c859eSApple OSS Distributions }
493*0f4c859eSApple OSS Distributions
494*0f4c859eSApple OSS Distributions void
freeObject(object_t * o)495*0f4c859eSApple OSS Distributions freeObject(object_t *o)
496*0f4c859eSApple OSS Distributions {
497*0f4c859eSApple OSS Distributions #if DEBUG
498*0f4c859eSApple OSS Distributions debugUnserializeAllocCount--;
499*0f4c859eSApple OSS Distributions #endif
500*0f4c859eSApple OSS Distributions free_type(object_t, o);
501*0f4c859eSApple OSS Distributions }
502*0f4c859eSApple OSS Distributions
503*0f4c859eSApple OSS Distributions static OSDictionary *tags;
504*0f4c859eSApple OSS Distributions
505*0f4c859eSApple OSS Distributions static void
rememberObject(int tag,object_t * o)506*0f4c859eSApple OSS Distributions rememberObject(int tag, object_t *o)
507*0f4c859eSApple OSS Distributions {
508*0f4c859eSApple OSS Distributions char key[16];
509*0f4c859eSApple OSS Distributions snprintf(key, sizeof(key), "%u", tag);
510*0f4c859eSApple OSS Distributions
511*0f4c859eSApple OSS Distributions tags->setObject(key, (OSObject *)o);
512*0f4c859eSApple OSS Distributions }
513*0f4c859eSApple OSS Distributions
514*0f4c859eSApple OSS Distributions static OSObject *
retrieveObject(int tag)515*0f4c859eSApple OSS Distributions retrieveObject(int tag)
516*0f4c859eSApple OSS Distributions {
517*0f4c859eSApple OSS Distributions char key[16];
518*0f4c859eSApple OSS Distributions snprintf(key, sizeof(key), "%u", tag);
519*0f4c859eSApple OSS Distributions
520*0f4c859eSApple OSS Distributions return tags->getObject(key);
521*0f4c859eSApple OSS Distributions }
522*0f4c859eSApple OSS Distributions
523*0f4c859eSApple OSS Distributions OSObject *
buildOSDictionary(object_t * o)524*0f4c859eSApple OSS Distributions buildOSDictionary(object_t *o)
525*0f4c859eSApple OSS Distributions {
526*0f4c859eSApple OSS Distributions object_t *temp, *last = o;
527*0f4c859eSApple OSS Distributions int count = 0;
528*0f4c859eSApple OSS Distributions
529*0f4c859eSApple OSS Distributions // get count and last object
530*0f4c859eSApple OSS Distributions while (o) {
531*0f4c859eSApple OSS Distributions count++;
532*0f4c859eSApple OSS Distributions last = o;
533*0f4c859eSApple OSS Distributions o = o->next;
534*0f4c859eSApple OSS Distributions }
535*0f4c859eSApple OSS Distributions o = last;
536*0f4c859eSApple OSS Distributions
537*0f4c859eSApple OSS Distributions OSDictionary *d = OSDictionary::withCapacity(count);
538*0f4c859eSApple OSS Distributions
539*0f4c859eSApple OSS Distributions while (o) {
540*0f4c859eSApple OSS Distributions #ifdef metaclass_stuff_worksXXX
541*0f4c859eSApple OSS Distributions if (((OSObject *)o->u.key)->metaCast("OSSymbol")) {
542*0f4c859eSApple OSS Distributions // XXX the evil frontdoor
543*0f4c859eSApple OSS Distributions d->setObject((OSSymbol *)o->u.key, (OSObject *)o->object);
544*0f4c859eSApple OSS Distributions } else {
545*0f4c859eSApple OSS Distributions // If it isn't a symbol, I hope it's a string!
546*0f4c859eSApple OSS Distributions d->setObject((OSString *)o->u.key, (OSObject *)o->object);
547*0f4c859eSApple OSS Distributions }
548*0f4c859eSApple OSS Distributions #else
549*0f4c859eSApple OSS Distributions d->setObject((OSString *)o->u.key, (OSObject *)o->object);
550*0f4c859eSApple OSS Distributions #endif
551*0f4c859eSApple OSS Distributions ((OSObject *)o->object)->release();
552*0f4c859eSApple OSS Distributions ((OSObject *)o->u.key)->release();
553*0f4c859eSApple OSS Distributions temp = o;
554*0f4c859eSApple OSS Distributions o = o->prev;
555*0f4c859eSApple OSS Distributions freeObject(temp);
556*0f4c859eSApple OSS Distributions }
557*0f4c859eSApple OSS Distributions return d;
558*0f4c859eSApple OSS Distributions };
559*0f4c859eSApple OSS Distributions
560*0f4c859eSApple OSS Distributions OSObject *
buildOSArray(object_t * o)561*0f4c859eSApple OSS Distributions buildOSArray(object_t *o)
562*0f4c859eSApple OSS Distributions {
563*0f4c859eSApple OSS Distributions object_t *temp, *last = o;
564*0f4c859eSApple OSS Distributions int count = 0;
565*0f4c859eSApple OSS Distributions
566*0f4c859eSApple OSS Distributions // get count and last object
567*0f4c859eSApple OSS Distributions while (o) {
568*0f4c859eSApple OSS Distributions count++;
569*0f4c859eSApple OSS Distributions last = o;
570*0f4c859eSApple OSS Distributions o = o->next;
571*0f4c859eSApple OSS Distributions }
572*0f4c859eSApple OSS Distributions o = last;
573*0f4c859eSApple OSS Distributions
574*0f4c859eSApple OSS Distributions OSArray *a = OSArray::withCapacity(count);
575*0f4c859eSApple OSS Distributions
576*0f4c859eSApple OSS Distributions while (o) {
577*0f4c859eSApple OSS Distributions a->setObject((OSObject *)o->object);
578*0f4c859eSApple OSS Distributions ((OSObject *)o->object)->release();
579*0f4c859eSApple OSS Distributions temp = o;
580*0f4c859eSApple OSS Distributions o = o->prev;
581*0f4c859eSApple OSS Distributions freeObject(temp);
582*0f4c859eSApple OSS Distributions }
583*0f4c859eSApple OSS Distributions return a;
584*0f4c859eSApple OSS Distributions };
585*0f4c859eSApple OSS Distributions
586*0f4c859eSApple OSS Distributions OSObject *
buildOSSet(object_t * o)587*0f4c859eSApple OSS Distributions buildOSSet(object_t *o)
588*0f4c859eSApple OSS Distributions {
589*0f4c859eSApple OSS Distributions OSArray *a = (OSArray *)buildOSArray(o);
590*0f4c859eSApple OSS Distributions OSSet *s = OSSet::withArray(a, a->getCapacity());
591*0f4c859eSApple OSS Distributions
592*0f4c859eSApple OSS Distributions a->release();
593*0f4c859eSApple OSS Distributions return s;
594*0f4c859eSApple OSS Distributions };
595*0f4c859eSApple OSS Distributions
596*0f4c859eSApple OSS Distributions OSObject *
buildOSString(object_t * o)597*0f4c859eSApple OSS Distributions buildOSString(object_t *o)
598*0f4c859eSApple OSS Distributions {
599*0f4c859eSApple OSS Distributions OSString *s = OSString::withCString((char *)o);
600*0f4c859eSApple OSS Distributions
601*0f4c859eSApple OSS Distributions safe_free(o, strlen((char *)o) + 1);
602*0f4c859eSApple OSS Distributions
603*0f4c859eSApple OSS Distributions return s;
604*0f4c859eSApple OSS Distributions };
605*0f4c859eSApple OSS Distributions
606*0f4c859eSApple OSS Distributions OSObject *
buildOSData(object_t * o)607*0f4c859eSApple OSS Distributions buildOSData(object_t *o)
608*0f4c859eSApple OSS Distributions {
609*0f4c859eSApple OSS Distributions OSData *d;
610*0f4c859eSApple OSS Distributions
611*0f4c859eSApple OSS Distributions if (o->size) {
612*0f4c859eSApple OSS Distributions d = OSData::withBytes(o->object, o->size);
613*0f4c859eSApple OSS Distributions } else {
614*0f4c859eSApple OSS Distributions d = OSData::withCapacity(0);
615*0f4c859eSApple OSS Distributions }
616*0f4c859eSApple OSS Distributions safe_free(o->object, o->size);
617*0f4c859eSApple OSS Distributions freeObject(o);
618*0f4c859eSApple OSS Distributions return d;
619*0f4c859eSApple OSS Distributions };
620*0f4c859eSApple OSS Distributions
621*0f4c859eSApple OSS Distributions OSObject *
buildOSOffset(object_t * o)622*0f4c859eSApple OSS Distributions buildOSOffset(object_t *o)
623*0f4c859eSApple OSS Distributions {
624*0f4c859eSApple OSS Distributions OSNumber *off = OSNumber::withNumber(o->u.offset, o->size);
625*0f4c859eSApple OSS Distributions freeObject(o);
626*0f4c859eSApple OSS Distributions return off;
627*0f4c859eSApple OSS Distributions };
628*0f4c859eSApple OSS Distributions
629*0f4c859eSApple OSS Distributions OSObject *
buildOSBoolean(object_t * o)630*0f4c859eSApple OSS Distributions buildOSBoolean(object_t *o)
631*0f4c859eSApple OSS Distributions {
632*0f4c859eSApple OSS Distributions OSBoolean *b = OSBoolean::withBoolean((bool)o);
633*0f4c859eSApple OSS Distributions return b;
634*0f4c859eSApple OSS Distributions };
635*0f4c859eSApple OSS Distributions
636*0f4c859eSApple OSS Distributions __BEGIN_DECLS
637*0f4c859eSApple OSS Distributions #include <kern/locks.h>
638*0f4c859eSApple OSS Distributions __END_DECLS
639*0f4c859eSApple OSS Distributions
640*0f4c859eSApple OSS Distributions static lck_mtx_t *lock = 0;
641*0f4c859eSApple OSS Distributions extern lck_grp_t *IOLockGroup;
642*0f4c859eSApple OSS Distributions
643*0f4c859eSApple OSS Distributions OSObject*
OSUnserialize(const char * buffer,OSString ** errorString)644*0f4c859eSApple OSS Distributions OSUnserialize(const char *buffer, OSString **errorString)
645*0f4c859eSApple OSS Distributions {
646*0f4c859eSApple OSS Distributions OSObject *object;
647*0f4c859eSApple OSS Distributions
648*0f4c859eSApple OSS Distributions if (!lock) {
649*0f4c859eSApple OSS Distributions lock = lck_mtx_alloc_init(IOLockGroup, LCK_ATTR_NULL);
650*0f4c859eSApple OSS Distributions lck_mtx_lock(lock);
651*0f4c859eSApple OSS Distributions } else {
652*0f4c859eSApple OSS Distributions lck_mtx_lock(lock);
653*0f4c859eSApple OSS Distributions
654*0f4c859eSApple OSS Distributions }
655*0f4c859eSApple OSS Distributions
656*0f4c859eSApple OSS Distributions #if DEBUG
657*0f4c859eSApple OSS Distributions debugUnserializeAllocCount = 0;
658*0f4c859eSApple OSS Distributions #endif
659*0f4c859eSApple OSS Distributions yyerror_message[0] = 0; //just in case
660*0f4c859eSApple OSS Distributions parseBuffer = buffer;
661*0f4c859eSApple OSS Distributions parseBufferIndex = 0;
662*0f4c859eSApple OSS Distributions tags = OSDictionary::withCapacity(128);
663*0f4c859eSApple OSS Distributions if (yyparse() == 0) {
664*0f4c859eSApple OSS Distributions object = parsedObject;
665*0f4c859eSApple OSS Distributions if (errorString) *errorString = NULL;
666*0f4c859eSApple OSS Distributions } else {
667*0f4c859eSApple OSS Distributions object = NULL;
668*0f4c859eSApple OSS Distributions if (errorString)
669*0f4c859eSApple OSS Distributions *errorString = OSString::withCString(yyerror_message);
670*0f4c859eSApple OSS Distributions }
671*0f4c859eSApple OSS Distributions
672*0f4c859eSApple OSS Distributions tags->release();
673*0f4c859eSApple OSS Distributions #if DEBUG
674*0f4c859eSApple OSS Distributions if (debugUnserializeAllocCount) {
675*0f4c859eSApple OSS Distributions printf("OSUnserialize: allocation check failed, count = %d.\n",
676*0f4c859eSApple OSS Distributions debugUnserializeAllocCount);
677*0f4c859eSApple OSS Distributions }
678*0f4c859eSApple OSS Distributions #endif
679*0f4c859eSApple OSS Distributions lck_mtx_unlock(lock);
680*0f4c859eSApple OSS Distributions
681*0f4c859eSApple OSS Distributions return object;
682*0f4c859eSApple OSS Distributions }
683*0f4c859eSApple OSS Distributions
684*0f4c859eSApple OSS Distributions #endif // not __clang_analyzer__
685*0f4c859eSApple OSS Distributions
686*0f4c859eSApple OSS Distributions
687*0f4c859eSApple OSS Distributions //
688*0f4c859eSApple OSS Distributions //
689*0f4c859eSApple OSS Distributions //
690*0f4c859eSApple OSS Distributions //
691*0f4c859eSApple OSS Distributions //
692*0f4c859eSApple OSS Distributions // DO NOT EDIT OSUnserialize.cpp!
693*0f4c859eSApple OSS Distributions //
694*0f4c859eSApple OSS Distributions // this means you!
695*0f4c859eSApple OSS Distributions //
696*0f4c859eSApple OSS Distributions //
697*0f4c859eSApple OSS Distributions //
698*0f4c859eSApple OSS Distributions //
699*0f4c859eSApple OSS Distributions //
700