2 * Copyright (c) 1999-2019 Apple Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
32 * OSUnserializeXML.y created by rsulack on Tue Oct 12 1999
35 // parser for unserializing OSContainer objects serialized to XML
38 // bison -p OSUnserializeXML OSUnserializeXML.y
39 // head -50 OSUnserializeXML.y > OSUnserializeXML.cpp
40 // sed -e "s/#include <stdio.h>//" < OSUnserializeXML.tab.c >> OSUnserializeXML.cpp
42 // when changing code check in both OSUnserializeXML.y and OSUnserializeXML.cpp
48 // DO NOT EDIT OSUnserializeXML.cpp!
51 /* A Bison parser, made by GNU Bison 2.3. */
53 /* Skeleton implementation for Bison's Yacc-like parsers in C
55 * Copyright (C) 1984, 1989, 1990, 2000, 2001, 2002, 2003, 2004, 2005, 2006
56 * Free Software Foundation, Inc.
58 * This program is free software; you can redistribute it and/or modify
59 * it under the terms of the GNU General Public License as published by
60 * the Free Software Foundation; either version 2, or (at your option)
63 * This program is distributed in the hope that it will be useful,
64 * but WITHOUT ANY WARRANTY; without even the implied warranty of
65 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
66 * GNU General Public License for more details.
68 * You should have received a copy of the GNU General Public License
69 * along with this program; if not, write to the Free Software
70 * Foundation, Inc., 51 Franklin Street, Fifth Floor,
71 * Boston, MA 02110-1301, USA. */
73 /* As a special exception, you may create a larger work that contains
74 * part or all of the Bison parser skeleton and distribute that work
75 * under terms of your choice, so long as that work isn't itself a
76 * parser generator using the skeleton or a modified version thereof
77 * as a parser skeleton. Alternatively, if you modify or redistribute
78 * the parser skeleton itself, you may (at your option) remove this
79 * special exception, which will cause the skeleton and the resulting
80 * Bison output files to be licensed under the GNU General Public
81 * License without this special exception.
83 * This special exception was added by the Free Software Foundation in
84 * version 2.2 of Bison. */
86 /* C LALR(1) parser skeleton written by Richard Stallman, by
87 * simplifying the original so-called "semantic" parser. */
89 /* All symbols defined below should begin with yy or YY, to avoid
90 * infringing on user name space. This should be done even for local
91 * variables, as they might otherwise be expanded by user macros.
92 * There are some unavoidable exceptions within include files to
93 * define necessary library symbols; they are noted "INFRINGES ON
94 * USER NAME SPACE" below. */
96 /* Identify Bison output. */
100 #define YYBISON_VERSION "2.3"
103 #define YYSKELETON_NAME "yacc.c"
108 /* Using locations. */
109 #define YYLSP_NEEDED 0
111 /* Substitute the variable and function names. */
112 #define yyparse OSUnserializeXMLparse
113 #define yylex OSUnserializeXMLlex
114 #define yyerror OSUnserializeXMLerror
115 #define yylval OSUnserializeXMLlval
116 #define yychar OSUnserializeXMLchar
117 #define yydebug OSUnserializeXMLdebug
118 #define yynerrs OSUnserializeXMLnerrs
124 /* Put the tokens into the symbol table, so that GDB and other debuggers
125 * know about them. */
143 #define DICTIONARY 261
149 #define SYNTAX_ERROR 267
154 /* Copy the first part of user declarations. */
155 #line 61 "OSUnserializeXML.y"
158 #include <libkern/c++/OSMetaClass.h>
159 #include <libkern/c++/OSContainers.h>
160 #include <libkern/c++/OSLib.h>
162 #define MAX_OBJECTS 131071
163 #define MAX_REFED_OBJECTS 65535
165 #define YYSTYPE object_t *
166 #define YYPARSE_PARAM state
167 #define YYLEX_PARAM (parser_state_t *)state
169 // this is the internal struct used to hold objects on parser stack
170 // it represents objects both before and after they have been created
171 typedef struct object
{
174 struct object
*elements
;
176 OSSymbol
*key
; // for dictionary
178 void *data
; // for data
179 char *string
; // for string & symbol
180 int string_alloc_length
;
181 long long number
; // for number
185 // this code is reentrant, this structure contains all
186 // state information for the parsing of a single buffer
187 typedef struct parser_state
{
188 const char *parseBuffer
; // start of text to be parsed
189 int parseBufferIndex
; // current index into text
190 int lineNumber
; // current line number
191 object_t
*objects
; // internal objects in use
192 object_t
*freeObjects
; // internal objects that are free
193 OSDictionary
*tags
; // used to remember "ID" tags
194 OSString
**errorString
; // parse error with line
195 OSObject
*parsedObject
; // resultant object of parsed text
196 int parsedObjectCount
;
197 int retrievedObjectCount
;
200 #define STATE ((parser_state_t *)state)
203 #define yyerror(s) OSUnserializeerror(STATE, (s))
204 static int OSUnserializeerror(parser_state_t
*state
, const char *s
);
206 static int yylex(YYSTYPE
*lvalp
, parser_state_t
*state
);
208 static object_t
*newObject(parser_state_t
*state
);
209 static void freeObject(parser_state_t
*state
, object_t
*o
);
210 static void rememberObject(parser_state_t
*state
, int tag
, OSObject
*o
);
211 static object_t
*retrieveObject(parser_state_t
*state
, int tag
);
212 static void cleanupObjects(parser_state_t
*state
);
214 static object_t
*buildDictionary(parser_state_t
*state
, object_t
*o
);
215 static object_t
*buildArray(parser_state_t
*state
, object_t
*o
);
216 static object_t
*buildSet(parser_state_t
*state
, object_t
*o
);
217 static object_t
*buildString(parser_state_t
*state
, object_t
*o
);
218 static object_t
*buildSymbol(parser_state_t
*state
, object_t
*o
);
219 static object_t
*buildData(parser_state_t
*state
, object_t
*o
);
220 static object_t
*buildNumber(parser_state_t
*state
, object_t
*o
);
221 static object_t
*buildBoolean(parser_state_t
*state
, object_t
*o
);
224 #include <kern/kalloc.h>
227 #define malloc(size) malloc_impl(size)
229 malloc_impl(size_t size
)
234 return kheap_alloc_tag_bt(KHEAP_DEFAULT
, size
,
235 (zalloc_flags_t
) (Z_WAITOK
| Z_ZERO
),
236 VM_KERN_MEMORY_LIBKERN
);
239 #define free(addr) free_impl(addr)
241 free_impl(void *addr
)
243 kheap_free_addr(KHEAP_DEFAULT
, addr
);
246 safe_free(void *addr
, size_t size
)
250 kheap_free(KHEAP_DEFAULT
, addr
, size
);
254 #define realloc(addr, osize, nsize) realloc_impl(addr, osize, nsize)
256 realloc_impl(void *addr
, size_t osize
, size_t nsize
)
259 return malloc(nsize
);
261 if (nsize
== osize
) {
264 void *nmem
= malloc(nsize
);
266 safe_free(addr
, osize
);
269 (void)memcpy(nmem
, addr
, (nsize
> osize
) ? osize
: nsize
);
270 safe_free(addr
, osize
);
277 /* Enabling traces. */
282 /* Enabling verbose error messages. */
283 #ifdef YYERROR_VERBOSE
284 # undef YYERROR_VERBOSE
285 # define YYERROR_VERBOSE 1
287 # define YYERROR_VERBOSE 0
290 /* Enabling the token table. */
291 #ifndef YYTOKEN_TABLE
292 # define YYTOKEN_TABLE 0
295 #if !defined YYSTYPE && !defined YYSTYPE_IS_DECLARED
297 # define yystype YYSTYPE /* obsolescent; will be withdrawn */
298 # define YYSTYPE_IS_DECLARED 1
299 # define YYSTYPE_IS_TRIVIAL 1
304 /* Copy the second part of user declarations. */
307 /* Line 216 of yacc.c. */
308 #line 258 "OSUnserializeXML.tab.c"
315 typedef YYTYPE_UINT8 yytype_uint8
;
317 typedef unsigned char yytype_uint8
;
321 typedef YYTYPE_INT8 yytype_int8
;
322 #elif (defined __STDC__ || defined __C99__FUNC__ \
323 || defined __cplusplus || defined _MSC_VER)
324 typedef signed char yytype_int8
;
326 typedef short int yytype_int8
;
330 typedef YYTYPE_UINT16 yytype_uint16
;
332 typedef unsigned short int yytype_uint16
;
336 typedef YYTYPE_INT16 yytype_int16
;
338 typedef short int yytype_int16
;
342 # ifdef __SIZE_TYPE__
343 # define YYSIZE_T __SIZE_TYPE__
344 # elif defined size_t
345 # define YYSIZE_T size_t
346 # elif !defined YYSIZE_T && (defined __STDC__ || defined __C99__FUNC__ \
347 || defined __cplusplus || defined _MSC_VER)
348 # include <stddef.h> /* INFRINGES ON USER NAME SPACE */
349 # define YYSIZE_T size_t
351 # define YYSIZE_T unsigned int
355 #define YYSIZE_MAXIMUM ((YYSIZE_T) -1)
358 # if defined YYENABLE_NLS && YYENABLE_NLS
360 # include <libintl.h> /* INFRINGES ON USER NAME SPACE */
361 # define YY_(msgid) dgettext ("bison-runtime", msgid)
365 # define YY_(msgid) msgid
369 /* Suppress unused-variable warnings by "using" E. */
370 #if !defined lint || defined __GNUC__
371 # define YYUSE(e) ((void) (e))
373 # define YYUSE(e) /* empty */
376 /* Identity function, used to suppress warnings about constant conditions. */
380 #if (defined __STDC__ || defined __C99__FUNC__ \
381 || defined __cplusplus || defined _MSC_VER)
394 #if !defined yyoverflow || YYERROR_VERBOSE
396 /* The parser invokes alloca or malloc; define the necessary symbols. */
398 # ifdef YYSTACK_USE_ALLOCA
399 # if YYSTACK_USE_ALLOCA
401 # define YYSTACK_ALLOC __builtin_alloca
402 # elif defined __BUILTIN_VA_ARG_INCR
403 # include <alloca.h> /* INFRINGES ON USER NAME SPACE */
405 # define YYSTACK_ALLOC __alloca
406 # elif defined _MSC_VER
407 # include <malloc.h> /* INFRINGES ON USER NAME SPACE */
408 # define alloca _alloca
410 # define YYSTACK_ALLOC alloca
411 # if !defined _ALLOCA_H && !defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \
412 || defined __cplusplus || defined _MSC_VER)
413 # include <stdlib.h> /* INFRINGES ON USER NAME SPACE */
422 # ifdef YYSTACK_ALLOC
423 /* Pacify GCC's `empty if-body' warning. */
424 # define YYSTACK_FREE(Ptr) do { /* empty */ ; } while (YYID (0))
425 # ifndef YYSTACK_ALLOC_MAXIMUM
426 /* The OS might guarantee only one guard page at the bottom of the stack,
427 * and a page size can be as small as 4096 bytes. So we cannot safely
428 * invoke alloca (N) if N exceeds 4096. Use a slightly smaller number
429 * to allow for a few compiler-allocated temporary stack slots. */
430 # define YYSTACK_ALLOC_MAXIMUM 4032 /* reasonable circa 2006 */
433 # define YYSTACK_ALLOC YYMALLOC
434 # define YYSTACK_FREE YYFREE
435 # ifndef YYSTACK_ALLOC_MAXIMUM
436 # define YYSTACK_ALLOC_MAXIMUM YYSIZE_MAXIMUM
438 # if (defined __cplusplus && !defined _STDLIB_H \
439 && !((defined YYMALLOC || defined malloc) \
440 && (defined YYFREE || defined free)))
441 # include <stdlib.h> /* INFRINGES ON USER NAME SPACE */
447 # define YYMALLOC malloc
448 # if !defined malloc && !defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \
449 || defined __cplusplus || defined _MSC_VER)
450 void *malloc(YYSIZE_T
); /* INFRINGES ON USER NAME SPACE */
455 # if !defined free && !defined _STDLIB_H && (defined __STDC__ || defined __C99__FUNC__ \
456 || defined __cplusplus || defined _MSC_VER)
457 void free(void *); /* INFRINGES ON USER NAME SPACE */
461 #endif /* ! defined yyoverflow || YYERROR_VERBOSE */
464 #if (!defined yyoverflow \
465 && (!defined __cplusplus \
466 || (defined YYSTYPE_IS_TRIVIAL && YYSTYPE_IS_TRIVIAL)))
468 /* A type that is properly aligned for any stack member. */
474 /* The size of the maximum gap between one aligned stack and the next. */
475 # define YYSTACK_GAP_MAXIMUM (sizeof (union yyalloc) - 1)
477 /* The size of an array large to enough to hold all stacks, each with
479 # define YYSTACK_BYTES(N) \
480 ((N) * (sizeof (yytype_int16) + sizeof (YYSTYPE)) \
481 + YYSTACK_GAP_MAXIMUM)
483 /* Copy COUNT objects from FROM to TO. The source and destination do
486 # if defined __GNUC__ && 1 < __GNUC__
487 # define YYCOPY(To, From, Count) \
488 __builtin_memcpy (To, From, (Count) * sizeof (*(From)))
490 # define YYCOPY(To, From, Count) \
494 for (yyi = 0; yyi < (Count); yyi++) \
495 (To)[yyi] = (From)[yyi]; \
501 /* Relocate STACK from its old location to the new one. The
502 * local variables YYSIZE and YYSTACKSIZE give the old and new number of
503 * elements in the stack, and YYPTR gives the new location of the
504 * stack. Advance YYPTR to a properly aligned location for the next
506 # define YYSTACK_RELOCATE(Stack) \
509 YYSIZE_T yynewbytes; \
510 YYCOPY (&yyptr->Stack, Stack, yysize); \
511 Stack = &yyptr->Stack; \
512 yynewbytes = yystacksize * sizeof (*Stack) + YYSTACK_GAP_MAXIMUM; \
513 yyptr += yynewbytes / sizeof (*yyptr); \
519 /* YYFINAL -- State number of the termination state. */
521 /* YYLAST -- Last index in YYTABLE. */
524 /* YYNTOKENS -- Number of terminals. */
526 /* YYNNTS -- Number of nonterminals. */
528 /* YYNRULES -- Number of rules. */
530 /* YYNRULES -- Number of states. */
533 /* YYTRANSLATE(YYLEX) -- Bison symbol number corresponding to YYLEX. */
535 #define YYMAXUTOK 267
537 #define YYTRANSLATE(YYX) \
538 ((unsigned int) (YYX) <= YYMAXUTOK ? yytranslate[YYX] : YYUNDEFTOK)
540 /* YYTRANSLATE[YYLEX] -- Bison symbol number corresponding to YYLEX. */
541 static const yytype_uint8 yytranslate
[] =
543 0, 2, 2, 2, 2, 2, 2, 2, 2, 2,
544 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
545 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
546 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
547 15, 16, 2, 2, 2, 2, 2, 2, 2, 2,
548 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
549 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
550 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
551 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
552 2, 17, 2, 18, 2, 2, 2, 2, 2, 2,
553 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
554 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
555 2, 2, 2, 13, 2, 14, 2, 2, 2, 2,
556 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
557 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
558 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
559 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
560 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
561 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
562 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
563 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
564 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
565 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
566 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
567 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,
568 2, 2, 2, 2, 2, 2, 1, 2, 3, 4,
569 5, 6, 7, 8, 9, 10, 11, 12
573 /* YYPRHS[YYN] -- Index of the first RHS symbol of rule number YYN in
575 static const yytype_uint8 yyprhs
[] =
577 0, 0, 3, 4, 6, 8, 10, 12, 14, 16,
578 18, 20, 22, 24, 27, 31, 33, 35, 38, 41,
579 43, 46, 50, 52, 55, 59, 61, 63, 66, 68,
583 /* YYRHS -- A `-1'-separated list of the rules' RHS. */
584 static const yytype_int8 yyrhs
[] =
586 20, 0, -1, -1, 21, -1, 12, -1, 22, -1,
587 26, -1, 27, -1, 33, -1, 30, -1, 32, -1,
588 29, -1, 31, -1, 13, 14, -1, 13, 23, 14,
589 -1, 6, -1, 24, -1, 23, 24, -1, 25, 21,
590 -1, 8, -1, 15, 16, -1, 15, 28, 16, -1,
591 3, -1, 17, 18, -1, 17, 28, 18, -1, 10,
592 -1, 21, -1, 28, 21, -1, 4, -1, 5, -1,
596 /* YYRLINE[YYN] -- source line where rule number YYN was defined. */
597 static const yytype_uint16 yyrline
[] =
599 0, 192, 192, 195, 200, 205, 217, 229, 241, 253,
600 265, 277, 289, 313, 316, 319, 322, 323, 338, 347,
601 359, 362, 365, 368, 371, 374, 377, 380, 387, 390,
606 #if YYDEBUG || YYERROR_VERBOSE || YYTOKEN_TABLE
607 /* YYTNAME[SYMBOL-NUM] -- String name of the symbol SYMBOL-NUM.
608 * First, the terminals, then, starting at YYNTOKENS, nonterminals. */
609 static const char *const yytname
[] =
611 "$end", "error", "$undefined", "ARRAY", "BOOLEAN", "DATA", "DICTIONARY",
612 "IDREF", "KEY", "NUMBER", "SET", "STRING", "SYNTAX_ERROR", "'{'", "'}'",
613 "'('", "')'", "'['", "']'", "$accept", "input", "object", "dict",
614 "pairs", "pair", "key", "array", "set", "elements", "boolean", "data",
615 "idref", "number", "string", 0
620 /* YYTOKNUM[YYLEX-NUM] -- Internal token number corresponding to
621 * token YYLEX-NUM. */
622 static const yytype_uint16 yytoknum
[] =
624 0, 256, 257, 258, 259, 260, 261, 262, 263, 264,
625 265, 266, 267, 123, 125, 40, 41, 91, 93
629 /* YYR1[YYN] -- Symbol number of symbol that rule YYN derives. */
630 static const yytype_uint8 yyr1
[] =
632 0, 19, 20, 20, 20, 21, 21, 21, 21, 21,
633 21, 21, 21, 22, 22, 22, 23, 23, 24, 25,
634 26, 26, 26, 27, 27, 27, 28, 28, 29, 30,
638 /* YYR2[YYN] -- Number of symbols composing right hand side of rule YYN. */
639 static const yytype_uint8 yyr2
[] =
641 0, 2, 0, 1, 1, 1, 1, 1, 1, 1,
642 1, 1, 1, 2, 3, 1, 1, 2, 2, 1,
643 2, 3, 1, 2, 3, 1, 1, 2, 1, 1,
647 /* YYDEFACT[STATE-NAME] -- Default rule to reduce with in state
648 * STATE-NUM when YYTABLE doesn't specify something else to do. Zero
649 * means the default is an error. */
650 static const yytype_uint8 yydefact
[] =
652 2, 22, 28, 29, 15, 30, 31, 25, 32, 4,
653 0, 0, 0, 0, 3, 5, 6, 7, 11, 9,
654 12, 10, 8, 19, 13, 0, 16, 0, 20, 26,
655 0, 23, 0, 1, 14, 17, 18, 21, 27, 24
658 /* YYDEFGOTO[NTERM-NUM]. */
659 static const yytype_int8 yydefgoto
[] =
661 -1, 13, 29, 15, 25, 26, 27, 16, 17, 30,
665 /* YYPACT[STATE-NUM] -- Index in YYTABLE of the portion describing
667 #define YYPACT_NINF -20
668 static const yytype_int8 yypact
[] =
670 46, -20, -20, -20, -20, -20, -20, -20, -20, -20,
671 4, 61, -2, 10, -20, -20, -20, -20, -20, -20,
672 -20, -20, -20, -20, -20, 6, -20, 91, -20, -20,
673 76, -20, 30, -20, -20, -20, -20, -20, -20, -20
676 /* YYPGOTO[NTERM-NUM]. */
677 static const yytype_int8 yypgoto
[] =
679 -20, -20, 0, -20, -20, -19, -20, -20, -20, 5,
680 -20, -20, -20, -20, -20
683 /* YYTABLE[YYPACT[STATE-NUM]]. What to do in state STATE-NUM. If
684 * positive, shift that token. If negative, reduce the rule which
685 * number is the opposite. If zero, do what YYDEFACT says.
686 * If YYTABLE_NINF, syntax error. */
687 #define YYTABLE_NINF -1
688 static const yytype_uint8 yytable
[] =
690 14, 1, 2, 3, 4, 5, 35, 6, 7, 8,
691 33, 10, 23, 11, 23, 12, 31, 32, 24, 0,
692 34, 0, 0, 0, 0, 0, 0, 36, 0, 0,
693 38, 0, 38, 1, 2, 3, 4, 5, 0, 6,
694 7, 8, 0, 10, 0, 11, 0, 12, 39, 1,
695 2, 3, 4, 5, 0, 6, 7, 8, 9, 10,
696 0, 11, 0, 12, 1, 2, 3, 4, 5, 0,
697 6, 7, 8, 0, 10, 0, 11, 28, 12, 1,
698 2, 3, 4, 5, 0, 6, 7, 8, 0, 10,
699 0, 11, 37, 12, 1, 2, 3, 4, 5, 0,
700 6, 7, 8, 0, 10, 0, 11, 0, 12
703 static const yytype_int8 yycheck
[] =
705 0, 3, 4, 5, 6, 7, 25, 9, 10, 11,
706 0, 13, 8, 15, 8, 17, 18, 12, 14, -1,
707 14, -1, -1, -1, -1, -1, -1, 27, -1, -1,
708 30, -1, 32, 3, 4, 5, 6, 7, -1, 9,
709 10, 11, -1, 13, -1, 15, -1, 17, 18, 3,
710 4, 5, 6, 7, -1, 9, 10, 11, 12, 13,
711 -1, 15, -1, 17, 3, 4, 5, 6, 7, -1,
712 9, 10, 11, -1, 13, -1, 15, 16, 17, 3,
713 4, 5, 6, 7, -1, 9, 10, 11, -1, 13,
714 -1, 15, 16, 17, 3, 4, 5, 6, 7, -1,
715 9, 10, 11, -1, 13, -1, 15, -1, 17
718 /* YYSTOS[STATE-NUM] -- The (internal number of the) accessing
719 * symbol of state STATE-NUM. */
720 static const yytype_uint8 yystos
[] =
722 0, 3, 4, 5, 6, 7, 9, 10, 11, 12,
723 13, 15, 17, 20, 21, 22, 26, 27, 29, 30,
724 31, 32, 33, 8, 14, 23, 24, 25, 16, 21,
725 28, 18, 28, 0, 14, 24, 21, 16, 21, 18
728 #define yyerrok (yyerrstatus = 0)
729 #define yyclearin (yychar = YYEMPTY)
733 #define YYACCEPT goto yyacceptlab
734 #define YYABORT goto yyabortlab
735 #define YYERROR goto yyerrorlab
738 /* Like YYERROR except do call yyerror. This remains here temporarily
739 * to ease the transition to the new meaning of YYERROR, for GCC.
740 * Once GCC version 2 has supplanted version 1, this can go. */
742 #define YYFAIL goto yyerrlab
744 #define YYRECOVERING() (!!yyerrstatus)
746 #define YYBACKUP(Token, Value) \
748 if (yychar == YYEMPTY && yylen == 1) \
752 yytoken = YYTRANSLATE (yychar); \
758 yyerror (YY_("syntax error: cannot back up")); \
765 #define YYERRCODE 256
768 /* YYLLOC_DEFAULT -- Set CURRENT to span from RHS[1] to RHS[N].
769 * If N is 0, then set CURRENT to the empty location which ends
770 * the previous symbol: RHS[0] (always defined). */
772 #define YYRHSLOC(Rhs, K) ((Rhs)[K])
773 #ifndef YYLLOC_DEFAULT
774 # define YYLLOC_DEFAULT(Current, Rhs, N) \
778 (Current).first_line = YYRHSLOC (Rhs, 1).first_line; \
779 (Current).first_column = YYRHSLOC (Rhs, 1).first_column; \
780 (Current).last_line = YYRHSLOC (Rhs, N).last_line; \
781 (Current).last_column = YYRHSLOC (Rhs, N).last_column; \
785 (Current).first_line = (Current).last_line = \
786 YYRHSLOC (Rhs, 0).last_line; \
787 (Current).first_column = (Current).last_column = \
788 YYRHSLOC (Rhs, 0).last_column; \
794 /* YY_LOCATION_PRINT -- Print the location on the stream.
795 * This macro was not mandated originally: define only if we know
796 * we won't break user code: when these are the locations we know. */
798 #ifndef YY_LOCATION_PRINT
799 # if defined YYLTYPE_IS_TRIVIAL && YYLTYPE_IS_TRIVIAL
800 # define YY_LOCATION_PRINT(File, Loc) \
801 fprintf (File, "%d.%d-%d.%d", \
802 (Loc).first_line, (Loc).first_column, \
803 (Loc).last_line, (Loc).last_column)
805 # define YY_LOCATION_PRINT(File, Loc) ((void) 0)
810 /* YYLEX -- calling `yylex' with the right arguments. */
813 # define YYLEX yylex (&yylval, YYLEX_PARAM)
815 # define YYLEX yylex (&yylval)
818 /* Enable debugging if requested. */
822 # include <stdio.h> /* INFRINGES ON USER NAME SPACE */
823 # define YYFPRINTF fprintf
826 # define YYDPRINTF(Args) \
832 # define YY_SYMBOL_PRINT(Title, Type, Value, Location) \
836 YYFPRINTF (stderr, "%s ", Title); \
837 yy_symbol_print (stderr, \
839 YYFPRINTF (stderr, "\n"); \
844 /*--------------------------------.
845 | Print this symbol on YYOUTPUT. |
846 | `--------------------------------*/
849 #if (defined __STDC__ || defined __C99__FUNC__ \
850 || defined __cplusplus || defined _MSC_VER)
852 yy_symbol_value_print(FILE *yyoutput
, int yytype
, YYSTYPE
const * const yyvaluep
)
855 yy_symbol_value_print(yyoutput
, yytype
, yyvaluep
)
858 YYSTYPE
const * const yyvaluep
;
865 if (yytype
< YYNTOKENS
) {
866 YYPRINT(yyoutput
, yytoknum
[yytype
], *yyvaluep
);
878 /*--------------------------------.
879 | Print this symbol on YYOUTPUT. |
880 | `--------------------------------*/
882 #if (defined __STDC__ || defined __C99__FUNC__ \
883 || defined __cplusplus || defined _MSC_VER)
885 yy_symbol_print(FILE *yyoutput
, int yytype
, YYSTYPE
const * const yyvaluep
)
888 yy_symbol_print(yyoutput
, yytype
, yyvaluep
)
891 YYSTYPE
const * const yyvaluep
;
894 if (yytype
< YYNTOKENS
) {
895 YYFPRINTF(yyoutput
, "token %s (", yytname
[yytype
]);
897 YYFPRINTF(yyoutput
, "nterm %s (", yytname
[yytype
]);
900 yy_symbol_value_print(yyoutput
, yytype
, yyvaluep
);
901 YYFPRINTF(yyoutput
, ")");
904 /*------------------------------------------------------------------.
905 | yy_stack_print -- Print the state stack from its BOTTOM up to its |
907 | `------------------------------------------------------------------*/
909 #if (defined __STDC__ || defined __C99__FUNC__ \
910 || defined __cplusplus || defined _MSC_VER)
912 yy_stack_print(yytype_int16
*bottom
, yytype_int16
*top
)
915 yy_stack_print(bottom
, top
)
916 yytype_int16
*bottom
;
920 YYFPRINTF(stderr
, "Stack now");
921 for (; bottom
<= top
; ++bottom
) {
922 YYFPRINTF(stderr
, " %d", *bottom
);
924 YYFPRINTF(stderr
, "\n");
927 # define YY_STACK_PRINT(Bottom, Top) \
930 yy_stack_print ((Bottom), (Top)); \
934 /*------------------------------------------------.
935 | Report that the YYRULE is going to be reduced. |
936 | `------------------------------------------------*/
938 #if (defined __STDC__ || defined __C99__FUNC__ \
939 || defined __cplusplus || defined _MSC_VER)
941 yy_reduce_print(YYSTYPE
*yyvsp
, int yyrule
)
944 yy_reduce_print(yyvsp
, yyrule
)
949 int yynrhs
= yyr2
[yyrule
];
951 unsigned long int yylno
= yyrline
[yyrule
];
952 YYFPRINTF(stderr
, "Reducing stack by rule %d (line %lu):\n",
954 /* The symbols being reduced. */
955 for (yyi
= 0; yyi
< yynrhs
; yyi
++) {
956 fprintf(stderr
, " $%d = ", yyi
+ 1);
957 yy_symbol_print(stderr
, yyrhs
[yyprhs
[yyrule
] + yyi
],
958 &(yyvsp
[(yyi
+ 1) - (yynrhs
)])
960 fprintf(stderr
, "\n");
964 # define YY_REDUCE_PRINT(Rule) \
967 yy_reduce_print (yyvsp, Rule); \
970 /* Nonzero means print parse trace. It is left uninitialized so that
971 * multiple parsers can coexist. */
974 # define YYDPRINTF(Args)
975 # define YY_SYMBOL_PRINT(Title, Type, Value, Location)
976 # define YY_STACK_PRINT(Bottom, Top)
977 # define YY_REDUCE_PRINT(Rule)
978 #endif /* !YYDEBUG */
981 /* YYINITDEPTH -- initial size of the parser's stacks. */
983 # define YYINITDEPTH 200
986 /* YYMAXDEPTH -- maximum size the stacks can grow to (effective only
987 * if the built-in stack extension method is used).
989 * Do not make this value too large; the results are undefined if
990 * YYSTACK_ALLOC_MAXIMUM < YYSTACK_BYTES (YYMAXDEPTH)
991 * evaluated with infinite-precision integer arithmetic. */
994 # define YYMAXDEPTH 10000
1002 # if defined __GLIBC__ && defined _STRING_H
1003 # define yystrlen strlen
1005 /* Return the length of YYSTR. */
1006 #if (defined __STDC__ || defined __C99__FUNC__ \
1007 || defined __cplusplus || defined _MSC_VER)
1009 yystrlen(const char *yystr
)
1017 for (yylen
= 0; yystr
[yylen
]; yylen
++) {
1026 # if defined __GLIBC__ && defined _STRING_H && defined _GNU_SOURCE
1027 # define yystpcpy stpcpy
1029 /* Copy YYSRC to YYDEST, returning the address of the terminating '\0' in
1031 #if (defined __STDC__ || defined __C99__FUNC__ \
1032 || defined __cplusplus || defined _MSC_VER)
1034 yystpcpy(char *yydest
, const char *yysrc
)
1037 yystpcpy(yydest
, yysrc
)
1043 const char *yys
= yysrc
;
1045 while ((*yyd
++ = *yys
++) != '\0') {
1055 /* Copy to YYRES the contents of YYSTR after stripping away unnecessary
1056 * quotes and backslashes, so that it's suitable for yyerror. The
1057 * heuristic is that double-quoting is unnecessary unless the string
1058 * contains an apostrophe, a comma, or backslash (other than
1059 * backslash-backslash). YYSTR is taken from yytname. If YYRES is
1060 * null, do not copy; instead, return the length of what the result
1061 * would have been. */
1063 yytnamerr(char *yyres
, const char *yystr
)
1065 if (*yystr
== '"') {
1067 char const *yyp
= yystr
;
1073 goto do_not_strip_quotes
;
1076 if (*++yyp
!= '\\') {
1077 goto do_not_strip_quotes
;
1094 do_not_strip_quotes
:;
1098 return yystrlen(yystr
);
1101 return yystpcpy(yyres
, yystr
) - yyres
;
1105 /* Copy into YYRESULT an error message about the unexpected token
1106 * YYCHAR while in state YYSTATE. Return the number of bytes copied,
1107 * including the terminating null byte. If YYRESULT is null, do not
1108 * copy anything; just return the number of bytes that would be
1109 * copied. As a special case, return 0 if an ordinary "syntax error"
1110 * message will do. Return YYSIZE_MAXIMUM if overflow occurs during
1111 * size calculation. */
1113 yysyntax_error(char *yyresult
, int yystate
, int yychar
)
1115 int yyn
= yypact
[yystate
];
1117 if (!(YYPACT_NINF
< yyn
&& yyn
<= YYLAST
)) {
1120 int yytype
= YYTRANSLATE(yychar
);
1121 YYSIZE_T yysize0
= yytnamerr(0, yytname
[yytype
]);
1122 YYSIZE_T yysize
= yysize0
;
1124 int yysize_overflow
= 0;
1125 enum { YYERROR_VERBOSE_ARGS_MAXIMUM
= 5 };
1126 char const *yyarg
[YYERROR_VERBOSE_ARGS_MAXIMUM
];
1130 /* This is so xgettext sees the translatable formats that are
1131 * constructed on the fly. */
1132 YY_("syntax error, unexpected %s");
1133 YY_("syntax error, unexpected %s, expecting %s");
1134 YY_("syntax error, unexpected %s, expecting %s or %s");
1135 YY_("syntax error, unexpected %s, expecting %s or %s or %s");
1136 YY_("syntax error, unexpected %s, expecting %s or %s or %s or %s");
1140 static char const yyunexpected
[] = "syntax error, unexpected %s";
1141 static char const yyexpecting
[] = ", expecting %s";
1142 static char const yyor
[] = " or %s";
1143 char yyformat
[sizeof yyunexpected
1144 + sizeof yyexpecting
- 1
1145 + ((YYERROR_VERBOSE_ARGS_MAXIMUM
- 2)
1146 * (sizeof yyor
- 1))];
1147 char const *yyprefix
= yyexpecting
;
1149 /* Start YYX at -YYN if negative to avoid negative indexes in
1151 int yyxbegin
= yyn
< 0 ? -yyn
: 0;
1153 /* Stay within bounds of both yycheck and yytname. */
1154 int yychecklim
= YYLAST
- yyn
+ 1;
1155 int yyxend
= yychecklim
< YYNTOKENS
? yychecklim
: YYNTOKENS
;
1158 yyarg
[0] = yytname
[yytype
];
1159 yyfmt
= yystpcpy(yyformat
, yyunexpected
);
1161 for (yyx
= yyxbegin
; yyx
< yyxend
; ++yyx
) {
1162 if (yycheck
[yyx
+ yyn
] == yyx
&& yyx
!= YYTERROR
) {
1163 if (yycount
== YYERROR_VERBOSE_ARGS_MAXIMUM
) {
1166 yyformat
[sizeof yyunexpected
- 1] = '\0';
1169 yyarg
[yycount
++] = yytname
[yyx
];
1170 yysize1
= yysize
+ yytnamerr(0, yytname
[yyx
]);
1171 yysize_overflow
|= (yysize1
< yysize
);
1173 yyfmt
= yystpcpy(yyfmt
, yyprefix
);
1178 yyf
= YY_(yyformat
);
1179 yysize1
= yysize
+ yystrlen(yyf
);
1180 yysize_overflow
|= (yysize1
< yysize
);
1183 if (yysize_overflow
) {
1184 return YYSIZE_MAXIMUM
;
1188 /* Avoid sprintf, as that infringes on the user's name space.
1189 * Don't have undefined behavior even if the translation
1190 * produced a string with the wrong number of "%s"s. */
1191 char *yyp
= yyresult
;
1193 while ((*yyp
= *yyf
) != '\0') {
1194 if (*yyp
== '%' && yyf
[1] == 's' && yyi
< yycount
) {
1195 yyp
+= yytnamerr(yyp
, yyarg
[yyi
++]);
1206 #endif /* YYERROR_VERBOSE */
1209 /*-----------------------------------------------.
1210 | Release the memory associated to this symbol. |
1211 | `-----------------------------------------------*/
1214 #if (defined __STDC__ || defined __C99__FUNC__ \
1215 || defined __cplusplus || defined _MSC_VER)
1217 yydestruct(const char *yymsg
, int yytype
, YYSTYPE
*yyvaluep
)
1220 yydestruct(yymsg
, yytype
, yyvaluep
)
1231 YY_SYMBOL_PRINT(yymsg
, yytype
, yyvaluep
, yylocationp
);
1240 /* Prevent warnings from -Wmissing-prototypes. */
1242 #ifdef YYPARSE_PARAM
1243 #if defined __STDC__ || defined __cplusplus
1244 int yyparse(void *YYPARSE_PARAM
);
1248 #else /* ! YYPARSE_PARAM */
1249 #if defined __STDC__ || defined __cplusplus
1254 #endif /* ! YYPARSE_PARAM */
1265 #ifdef YYPARSE_PARAM
1266 #if (defined __STDC__ || defined __C99__FUNC__ \
1267 || defined __cplusplus || defined _MSC_VER)
1269 yyparse(void *YYPARSE_PARAM
)
1272 yyparse(YYPARSE_PARAM
)
1273 void *YYPARSE_PARAM
;
1275 #else /* ! YYPARSE_PARAM */
1276 #if (defined __STDC__ || defined __C99__FUNC__ \
1277 || defined __cplusplus || defined _MSC_VER)
1287 /* The look-ahead symbol. */
1290 /* The semantic value of the look-ahead symbol. */
1293 /* Number of syntax errors so far. */
1299 /* Number of tokens to shift before error messages enabled. */
1301 /* Look-ahead token as an internal (translated) token number. */
1304 /* Buffer for error messages, and its allocated size. */
1306 char *yymsg
= yymsgbuf
;
1307 YYSIZE_T yymsg_alloc
= sizeof yymsgbuf
;
1310 /* Three stacks and their tools:
1311 * `yyss': related to states,
1312 * `yyvs': related to semantic values,
1313 * `yyls': related to locations.
1315 * Refer to the stacks thru separate pointers, to allow yyoverflow
1316 * to reallocate them elsewhere. */
1318 /* The state stack. */
1319 yytype_int16 yyssa
[YYINITDEPTH
];
1320 yytype_int16
*yyss
= yyssa
;
1321 yytype_int16
*yyssp
;
1323 /* The semantic value stack. */
1324 YYSTYPE yyvsa
[YYINITDEPTH
];
1325 YYSTYPE
*yyvs
= yyvsa
;
1330 #define YYPOPSTACK(N) (yyvsp -= (N), yyssp -= (N))
1332 YYSIZE_T yystacksize
= YYINITDEPTH
;
1334 /* The variables used to return semantic value and location from the
1335 * action routines. */
1339 /* The number of symbols on the RHS of the reduced rule.
1340 * Keep to zero when no symbol should be popped. */
1343 YYDPRINTF((stderr
, "Starting parse\n"));
1348 yychar
= YYEMPTY
; /* Cause a token to be read. */
1350 /* Initialize stack pointers.
1351 * Waste one element of value and location stack
1352 * so that they stay on the same level as the state stack.
1353 * The wasted elements are never initialized. */
1360 /*------------------------------------------------------------.
1361 | yynewstate -- Push a new state, which is found in yystate. |
1362 | `------------------------------------------------------------*/
1364 /* In all cases, when you get here, the value and location stacks
1365 * have just been pushed. So pushing a state here evens the stacks. */
1371 if (yyss
+ yystacksize
- 1 <= yyssp
) {
1372 /* Get the current used size of the three stacks, in elements. */
1373 YYSIZE_T yysize
= yyssp
- yyss
+ 1;
1377 /* Give user a chance to reallocate the stack. Use copies of
1378 * these so that the &'s don't force the real ones into
1380 YYSTYPE
*yyvs1
= yyvs
;
1381 yytype_int16
*yyss1
= yyss
;
1384 /* Each stack pointer address is followed by the size of the
1385 * data in use in that stack, in bytes. This used to be a
1386 * conditional around just the two extra args, but that might
1387 * be undefined if yyoverflow is a macro. */
1388 yyoverflow(YY_("memory exhausted"),
1389 &yyss1
, yysize
* sizeof(*yyssp
),
1390 &yyvs1
, yysize
* sizeof(*yyvsp
),
1397 #else /* no yyoverflow */
1398 # ifndef YYSTACK_RELOCATE
1399 goto yyexhaustedlab
;
1401 /* Extend the stack our own way. */
1402 if (YYMAXDEPTH
<= yystacksize
) {
1403 goto yyexhaustedlab
;
1406 if (YYMAXDEPTH
< yystacksize
) {
1407 yystacksize
= YYMAXDEPTH
;
1411 yytype_int16
*yyss1
= yyss
;
1412 union yyalloc
*yyptr
=
1413 (union yyalloc
*) YYSTACK_ALLOC(YYSTACK_BYTES(yystacksize
));
1415 goto yyexhaustedlab
;
1417 YYSTACK_RELOCATE(yyss
);
1418 YYSTACK_RELOCATE(yyvs
);
1420 # undef YYSTACK_RELOCATE
1421 if (yyss1
!= yyssa
) {
1422 YYSTACK_FREE(yyss1
);
1426 #endif /* no yyoverflow */
1428 yyssp
= yyss
+ yysize
- 1;
1429 yyvsp
= yyvs
+ yysize
- 1;
1432 YYDPRINTF((stderr
, "Stack size increased to %lu\n",
1433 (unsigned long int) yystacksize
));
1435 if (yyss
+ yystacksize
- 1 <= yyssp
) {
1440 YYDPRINTF((stderr
, "Entering state %d\n", yystate
));
1449 /* Do appropriate processing given the current state. Read a
1450 * look-ahead token if we need one and don't already have one. */
1452 /* First try to decide what to do without reference to look-ahead token. */
1453 yyn
= yypact
[yystate
];
1454 if (yyn
== YYPACT_NINF
) {
1458 /* Not known => get a look-ahead token if don't already have one. */
1460 /* YYCHAR is either YYEMPTY or YYEOF or a valid look-ahead symbol. */
1461 if (yychar
== YYEMPTY
) {
1462 YYDPRINTF((stderr
, "Reading a token: "));
1466 if (yychar
<= YYEOF
) {
1467 yychar
= yytoken
= YYEOF
;
1468 YYDPRINTF((stderr
, "Now at end of input.\n"));
1470 yytoken
= YYTRANSLATE(yychar
);
1471 YY_SYMBOL_PRINT("Next token is", yytoken
, &yylval
, &yylloc
);
1474 /* If the proper action on seeing token YYTOKEN is to reduce or to
1475 * detect an error, take that action. */
1477 if (yyn
< 0 || YYLAST
< yyn
|| yycheck
[yyn
] != yytoken
) {
1482 if (yyn
== 0 || yyn
== YYTABLE_NINF
) {
1489 if (yyn
== YYFINAL
) {
1493 /* Count tokens shifted since error; after three, turn off error
1499 /* Shift the look-ahead token. */
1500 YY_SYMBOL_PRINT("Shifting", yytoken
, &yylval
, &yylloc
);
1502 /* Discard the shifted token unless it is eof. */
1503 if (yychar
!= YYEOF
) {
1513 /*-----------------------------------------------------------.
1514 | yydefault -- do the default action for the current state. |
1515 | `-----------------------------------------------------------*/
1517 yyn
= yydefact
[yystate
];
1524 /*-----------------------------.
1525 | yyreduce -- Do a reduction. |
1526 | `-----------------------------*/
1528 /* yyn is the number of a rule to reduce with. */
1531 /* If YYLEN is nonzero, implement the default value of the action:
1534 * Otherwise, the following line sets YYVAL to garbage.
1535 * This behavior is undocumented and Bison
1536 * users should not rely upon it. Assigning to YYVAL
1537 * unconditionally makes the parser a bit smaller, and it avoids a
1538 * GCC warning that YYVAL may be used uninitialized. */
1539 yyval
= yyvsp
[1 - yylen
];
1542 YY_REDUCE_PRINT(yyn
);
1545 #line 192 "OSUnserializeXML.y"
1546 { yyerror("unexpected end of buffer");
1552 #line 195 "OSUnserializeXML.y"
1553 { STATE
->parsedObject
= (yyvsp
[(1) - (1)])->object
;
1554 (yyvsp
[(1) - (1)])->object
= 0;
1555 freeObject(STATE
, (yyvsp
[(1) - (1)]));
1561 #line 200 "OSUnserializeXML.y"
1562 { yyerror("syntax error");
1568 #line 205 "OSUnserializeXML.y"
1569 { (yyval
) = buildDictionary(STATE
, (yyvsp
[(1) - (1)]));
1571 if (!yyval
->object
) {
1572 yyerror("buildDictionary");
1575 STATE
->parsedObjectCount
++;
1576 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1577 yyerror("maximum object count");
1584 #line 217 "OSUnserializeXML.y"
1585 { (yyval
) = buildArray(STATE
, (yyvsp
[(1) - (1)]));
1587 if (!yyval
->object
) {
1588 yyerror("buildArray");
1591 STATE
->parsedObjectCount
++;
1592 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1593 yyerror("maximum object count");
1600 #line 229 "OSUnserializeXML.y"
1601 { (yyval
) = buildSet(STATE
, (yyvsp
[(1) - (1)]));
1603 if (!yyval
->object
) {
1604 yyerror("buildSet");
1607 STATE
->parsedObjectCount
++;
1608 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1609 yyerror("maximum object count");
1616 #line 241 "OSUnserializeXML.y"
1617 { (yyval
) = buildString(STATE
, (yyvsp
[(1) - (1)]));
1619 if (!yyval
->object
) {
1620 yyerror("buildString");
1623 STATE
->parsedObjectCount
++;
1624 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1625 yyerror("maximum object count");
1632 #line 253 "OSUnserializeXML.y"
1633 { (yyval
) = buildData(STATE
, (yyvsp
[(1) - (1)]));
1635 if (!yyval
->object
) {
1636 yyerror("buildData");
1639 STATE
->parsedObjectCount
++;
1640 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1641 yyerror("maximum object count");
1648 #line 265 "OSUnserializeXML.y"
1649 { (yyval
) = buildNumber(STATE
, (yyvsp
[(1) - (1)]));
1651 if (!yyval
->object
) {
1652 yyerror("buildNumber");
1655 STATE
->parsedObjectCount
++;
1656 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1657 yyerror("maximum object count");
1664 #line 277 "OSUnserializeXML.y"
1665 { (yyval
) = buildBoolean(STATE
, (yyvsp
[(1) - (1)]));
1667 if (!yyval
->object
) {
1668 yyerror("buildBoolean");
1671 STATE
->parsedObjectCount
++;
1672 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1673 yyerror("maximum object count");
1680 #line 289 "OSUnserializeXML.y"
1681 { (yyval
) = retrieveObject(STATE
, (yyvsp
[(1) - (1)])->idref
);
1683 STATE
->retrievedObjectCount
++;
1684 (yyval
)->object
->retain();
1685 if (STATE
->retrievedObjectCount
> MAX_REFED_OBJECTS
) {
1686 yyerror("maximum object reference count");
1690 yyerror("forward reference detected");
1693 freeObject(STATE
, (yyvsp
[(1) - (1)]));
1695 STATE
->parsedObjectCount
++;
1696 if (STATE
->parsedObjectCount
> MAX_OBJECTS
) {
1697 yyerror("maximum object count");
1704 #line 313 "OSUnserializeXML.y"
1705 { (yyval
) = (yyvsp
[(1) - (2)]);
1706 (yyval
)->elements
= NULL
;
1711 #line 316 "OSUnserializeXML.y"
1712 { (yyval
) = (yyvsp
[(1) - (3)]);
1713 (yyval
)->elements
= (yyvsp
[(2) - (3)]);
1718 #line 323 "OSUnserializeXML.y"
1719 { (yyval
) = (yyvsp
[(2) - (2)]);
1720 (yyval
)->next
= (yyvsp
[(1) - (2)]);
1725 if (o
->key
== (yyval
)->key
) {
1726 yyerror("duplicate dictionary key");
1735 #line 338 "OSUnserializeXML.y"
1736 { (yyval
) = (yyvsp
[(1) - (2)]);
1737 (yyval
)->key
= (OSSymbol
*)(yyval
)->object
;
1738 (yyval
)->object
= (yyvsp
[(2) - (2)])->object
;
1739 (yyval
)->next
= NULL
;
1740 (yyvsp
[(2) - (2)])->object
= 0;
1741 freeObject(STATE
, (yyvsp
[(2) - (2)]));
1746 #line 347 "OSUnserializeXML.y"
1747 { (yyval
) = buildSymbol(STATE
, (yyvsp
[(1) - (1)]));
1749 // STATE->parsedObjectCount++;
1750 // if (STATE->parsedObjectCount > MAX_OBJECTS) {
1751 // yyerror("maximum object count");
1758 #line 359 "OSUnserializeXML.y"
1759 { (yyval
) = (yyvsp
[(1) - (2)]);
1760 (yyval
)->elements
= NULL
;
1765 #line 362 "OSUnserializeXML.y"
1766 { (yyval
) = (yyvsp
[(1) - (3)]);
1767 (yyval
)->elements
= (yyvsp
[(2) - (3)]);
1772 #line 368 "OSUnserializeXML.y"
1773 { (yyval
) = (yyvsp
[(1) - (2)]);
1774 (yyval
)->elements
= NULL
;
1779 #line 371 "OSUnserializeXML.y"
1780 { (yyval
) = (yyvsp
[(1) - (3)]);
1781 (yyval
)->elements
= (yyvsp
[(2) - (3)]);
1786 #line 377 "OSUnserializeXML.y"
1787 { (yyval
) = (yyvsp
[(1) - (1)]);
1788 (yyval
)->next
= NULL
;
1793 #line 380 "OSUnserializeXML.y"
1794 { (yyval
) = (yyvsp
[(2) - (2)]);
1795 (yyval
)->next
= (yyvsp
[(1) - (2)]);
1800 /* Line 1267 of yacc.c. */
1801 #line 1747 "OSUnserializeXML.tab.c"
1804 YY_SYMBOL_PRINT("-> $$ =", yyr1
[yyn
], &yyval
, &yyloc
);
1808 YY_STACK_PRINT(yyss
, yyssp
);
1813 /* Now `shift' the result of the reduction. Determine what state
1814 * that goes to, based on the state we popped back to and the rule
1815 * number reduced by. */
1819 yystate
= yypgoto
[yyn
- YYNTOKENS
] + *yyssp
;
1820 if (0 <= yystate
&& yystate
<= YYLAST
&& yycheck
[yystate
] == *yyssp
) {
1821 yystate
= yytable
[yystate
];
1823 yystate
= yydefgoto
[yyn
- YYNTOKENS
];
1829 /*------------------------------------.
1830 | yyerrlab -- here on detecting error |
1831 | `------------------------------------*/
1833 /* If not already recovering from an error, report this error. */
1836 #if !YYERROR_VERBOSE
1837 yyerror(YY_("syntax error"));
1840 YYSIZE_T yysize
= yysyntax_error(0, yystate
, yychar
);
1841 if (yymsg_alloc
< yysize
&& yymsg_alloc
< YYSTACK_ALLOC_MAXIMUM
) {
1842 YYSIZE_T yyalloc
= 2 * yysize
;
1843 if (!(yysize
<= yyalloc
&& yyalloc
<= YYSTACK_ALLOC_MAXIMUM
)) {
1844 yyalloc
= YYSTACK_ALLOC_MAXIMUM
;
1846 if (yymsg
!= yymsgbuf
) {
1847 YYSTACK_FREE(yymsg
);
1849 yymsg
= (char *) YYSTACK_ALLOC(yyalloc
);
1851 yymsg_alloc
= yyalloc
;
1854 yymsg_alloc
= sizeof yymsgbuf
;
1858 if (0 < yysize
&& yysize
<= yymsg_alloc
) {
1859 (void) yysyntax_error(yymsg
, yystate
, yychar
);
1862 yyerror(YY_("syntax error"));
1864 goto yyexhaustedlab
;
1873 if (yyerrstatus
== 3) {
1874 /* If just tried and failed to reuse look-ahead token after an
1875 * error, discard it. */
1877 if (yychar
<= YYEOF
) {
1878 /* Return failure if at end of input. */
1879 if (yychar
== YYEOF
) {
1883 yydestruct("Error: discarding",
1889 /* Else will try to reuse look-ahead token after shifting the error
1894 /*---------------------------------------------------.
1895 | yyerrorlab -- error raised explicitly by YYERROR. |
1896 | `---------------------------------------------------*/
1899 /* Pacify compilers like GCC when the user code never invokes
1900 * YYERROR and the label yyerrorlab therefore never appears in user
1902 if (/*CONSTCOND*/ 0) {
1906 /* Do not reclaim the symbols of the rule which action triggered
1910 YY_STACK_PRINT(yyss
, yyssp
);
1915 /*-------------------------------------------------------------.
1916 | yyerrlab1 -- common code for both syntax error and YYERROR. |
1917 | `-------------------------------------------------------------*/
1919 yyerrstatus
= 3; /* Each real token shifted decrements this. */
1922 yyn
= yypact
[yystate
];
1923 if (yyn
!= YYPACT_NINF
) {
1925 if (0 <= yyn
&& yyn
<= YYLAST
&& yycheck
[yyn
] == YYTERROR
) {
1933 /* Pop the current state because it cannot handle the error token. */
1934 if (yyssp
== yyss
) {
1939 yydestruct("Error: popping",
1940 yystos
[yystate
], yyvsp
);
1943 YY_STACK_PRINT(yyss
, yyssp
);
1946 if (yyn
== YYFINAL
) {
1953 /* Shift the error token. */
1954 YY_SYMBOL_PRINT("Shifting", yystos
[yyn
], yyvsp
, yylsp
);
1960 /*-------------------------------------.
1961 | yyacceptlab -- YYACCEPT comes here. |
1962 | `-------------------------------------*/
1967 /*-----------------------------------.
1968 | yyabortlab -- YYABORT comes here. |
1969 | `-----------------------------------*/
1975 /*-------------------------------------------------.
1976 | yyexhaustedlab -- memory exhaustion comes here. |
1977 | `-------------------------------------------------*/
1979 yyerror(YY_("memory exhausted"));
1985 if (yychar
!= YYEOF
&& yychar
!= YYEMPTY
) {
1986 yydestruct("Cleanup: discarding lookahead",
1989 /* Do not reclaim the symbols of the rule which action triggered
1990 * this YYABORT or YYACCEPT. */
1992 YY_STACK_PRINT(yyss
, yyssp
);
1993 while (yyssp
!= yyss
) {
1994 yydestruct("Cleanup: popping",
1995 yystos
[*yyssp
], yyvsp
);
1999 if (yyss
!= yyssa
) {
2004 if (yymsg
!= yymsgbuf
) {
2005 YYSTACK_FREE(yymsg
);
2008 /* Make sure YYID is used. */
2009 return YYID(yyresult
);
2013 #line 402 "OSUnserializeXML.y"
2017 OSUnserializeerror(parser_state_t
* state
, const char *s
) /* Called by yyparse on errors */
2019 if (state
->errorString
) {
2020 char tempString
[128];
2021 snprintf(tempString
, 128, "OSUnserializeXML: %s near line %d\n", s
, state
->lineNumber
);
2022 *(state
->errorString
) = OSString::withCString(tempString
);
2028 #define TAG_MAX_LENGTH 32
2029 #define TAG_MAX_ATTRIBUTES 32
2034 #define TAG_IGNORE 4
2036 #define currentChar() (state->parseBuffer[state->parseBufferIndex])
2037 #define nextChar() (state->parseBuffer[++state->parseBufferIndex])
2038 #define prevChar() (state->parseBuffer[state->parseBufferIndex - 1])
2040 #define isSpace(c) ((c) == ' ' || (c) == '\t')
2041 #define isAlpha(c) (((c) >= 'A' && (c) <= 'Z') || ((c) >= 'a' && (c) <= 'z'))
2042 #define isDigit(c) ((c) >= '0' && (c) <= '9')
2043 #define isAlphaDigit(c) ((c) >= 'a' && (c) <= 'f')
2044 #define isHexDigit(c) (isDigit(c) || isAlphaDigit(c))
2045 #define isAlphaNumeric(c) (isAlpha(c) || isDigit(c) || ((c) == '-'))
2048 getTag(parser_state_t
*state
,
2049 char tag
[TAG_MAX_LENGTH
],
2050 int *attributeCount
,
2051 char attributes
[TAG_MAX_ATTRIBUTES
][TAG_MAX_LENGTH
],
2052 char values
[TAG_MAX_ATTRIBUTES
][TAG_MAX_LENGTH
] )
2055 int c
= currentChar();
2056 int tagType
= TAG_START
;
2058 *attributeCount
= 0;
2063 c
= nextChar(); // skip '<'
2066 // <!TAG declarations >
2067 // <!-- comments -->
2070 bool isComment
= (c
== '-') && ((c
= nextChar()) != 0) && (c
== '-');
2071 if (!isComment
&& !isAlpha(c
)) {
2072 return TAG_BAD
; // <!1, <!-A, <!eos
2074 while (c
&& (c
= nextChar()) != 0) {
2076 state
->lineNumber
++;
2098 // <? Processing Instructions ?>
2100 while ((c
= nextChar()) != 0) {
2102 state
->lineNumber
++;
2120 c
= nextChar(); // skip '/'
2127 /* find end of tag while copying it */
2128 while (isAlphaNumeric(c
)) {
2131 if (length
>= (TAG_MAX_LENGTH
- 1)) {
2138 // printf("tag %s, type %d\n", tag, tagType);
2140 // look for attributes of the form attribute = "value" ...
2141 while ((c
!= '>') && (c
!= '/')) {
2142 while (isSpace(c
)) {
2147 while (isAlphaNumeric(c
)) {
2148 attributes
[*attributeCount
][length
++] = c
;
2149 if (length
>= (TAG_MAX_LENGTH
- 1)) {
2154 attributes
[*attributeCount
][length
] = 0;
2156 while (isSpace(c
)) {
2165 while (isSpace(c
)) {
2175 values
[*attributeCount
][length
++] = c
;
2176 if (length
>= (TAG_MAX_LENGTH
- 1)) {
2184 values
[*attributeCount
][length
] = 0;
2186 c
= nextChar(); // skip closing quote
2188 // printf(" attribute '%s' = '%s', nextchar = '%c'\n",
2189 // attributes[*attributeCount], values[*attributeCount], c);
2191 (*attributeCount
)++;
2192 if (*attributeCount
>= TAG_MAX_ATTRIBUTES
) {
2198 c
= nextChar(); // skip '/'
2199 tagType
= TAG_EMPTY
;
2204 c
= nextChar(); // skip '>'
2210 getString(parser_state_t
*state
, int *alloc_lengthp
)
2212 int c
= currentChar();
2213 int start
, length
, i
, j
;
2216 start
= state
->parseBufferIndex
;
2217 /* find end of string */
2221 state
->lineNumber
++;
2233 length
= state
->parseBufferIndex
- start
;
2235 /* copy to null terminated buffer */
2236 tempString
= (char *)malloc(length
+ 1);
2237 if (tempString
== NULL
) {
2238 printf("OSUnserializeXML: can't alloc temp memory\n");
2241 if (alloc_lengthp
) {
2242 *alloc_lengthp
= length
+ 1;
2245 // copy out string in tempString
2246 // "&" -> '&', "<" -> '<', ">" -> '>'
2249 while (i
< length
) {
2250 c
= state
->parseBuffer
[start
+ i
++];
2252 tempString
[j
++] = c
;
2254 if ((i
+ 3) > length
) {
2257 c
= state
->parseBuffer
[start
+ i
++];
2259 if (state
->parseBuffer
[start
+ i
++] != 't') {
2262 if (state
->parseBuffer
[start
+ i
++] != ';') {
2265 tempString
[j
++] = '<';
2269 if (state
->parseBuffer
[start
+ i
++] != 't') {
2272 if (state
->parseBuffer
[start
+ i
++] != ';') {
2275 tempString
[j
++] = '>';
2278 if ((i
+ 3) > length
) {
2282 if (state
->parseBuffer
[start
+ i
++] != 'm') {
2285 if (state
->parseBuffer
[start
+ i
++] != 'p') {
2288 if (state
->parseBuffer
[start
+ i
++] != ';') {
2291 tempString
[j
++] = '&';
2299 // printf("string %s\n", tempString);
2305 safe_free(tempString
, length
+ 1);
2306 if (alloc_lengthp
) {
2314 getNumber(parser_state_t
*state
)
2316 unsigned long long n
= 0;
2318 bool negate
= false;
2319 int c
= currentChar();
2333 while (isDigit(c
)) {
2334 n
= (n
* base
+ c
- '0');
2338 n
= (unsigned long long)((long long)n
* (long long)-1);
2341 while (isHexDigit(c
)) {
2343 n
= (n
* base
+ c
- '0');
2345 n
= (n
* base
+ 0xa + c
- 'a');
2350 // printf("number 0x%x\n", (unsigned long)n);
2354 // taken from CFXMLParsing/CFPropertyList.c
2356 static const signed char __CFPLDataDecodeTable
[128] = {
2357 /* 000 */ -1, -1, -1, -1, -1, -1, -1, -1,
2358 /* 010 */ -1, -1, -1, -1, -1, -1, -1, -1,
2359 /* 020 */ -1, -1, -1, -1, -1, -1, -1, -1,
2360 /* 030 */ -1, -1, -1, -1, -1, -1, -1, -1,
2361 /* ' ' */ -1, -1, -1, -1, -1, -1, -1, -1,
2362 /* '(' */ -1, -1, -1, 62, -1, -1, -1, 63,
2363 /* '0' */ 52, 53, 54, 55, 56, 57, 58, 59,
2364 /* '8' */ 60, 61, -1, -1, -1, 0, -1, -1,
2365 /* '@' */ -1, 0, 1, 2, 3, 4, 5, 6,
2366 /* 'H' */ 7, 8, 9, 10, 11, 12, 13, 14,
2367 /* 'P' */ 15, 16, 17, 18, 19, 20, 21, 22,
2368 /* 'X' */ 23, 24, 25, -1, -1, -1, -1, -1,
2369 /* '`' */ -1, 26, 27, 28, 29, 30, 31, 32,
2370 /* 'h' */ 33, 34, 35, 36, 37, 38, 39, 40,
2371 /* 'p' */ 41, 42, 43, 44, 45, 46, 47, 48,
2372 /* 'x' */ 49, 50, 51, -1, -1, -1, -1, -1
2375 #define DATA_ALLOC_SIZE 4096
2378 getCFEncodedData(parser_state_t
*state
, unsigned int *size
)
2380 int numeq
= 0, cntr
= 0;
2381 unsigned int acc
= 0;
2383 size_t tmpbuflen
= DATA_ALLOC_SIZE
;
2384 unsigned char *tmpbuf
= (unsigned char *)malloc(tmpbuflen
);
2386 int c
= currentChar();
2392 safe_free(tmpbuf
, tmpbuflen
);
2401 state
->lineNumber
++;
2403 if (__CFPLDataDecodeTable
[c
] < 0) {
2409 acc
+= __CFPLDataDecodeTable
[c
];
2410 if (0 == (cntr
& 0x3)) {
2411 if (tmpbuflen
<= tmpbufpos
+ 2) {
2412 size_t oldsize
= tmpbuflen
;
2413 tmpbuflen
+= DATA_ALLOC_SIZE
;
2414 tmpbuf
= (unsigned char *)realloc(tmpbuf
, oldsize
, tmpbuflen
);
2416 tmpbuf
[tmpbufpos
++] = (acc
>> 16) & 0xff;
2418 tmpbuf
[tmpbufpos
++] = (acc
>> 8) & 0xff;
2421 tmpbuf
[tmpbufpos
++] = acc
& 0xff;
2428 safe_free(tmpbuf
, tmpbuflen
);
2435 getHexData(parser_state_t
*state
, unsigned int *size
)
2438 unsigned char *d
, *start
, *lastStart
;
2440 size_t buflen
= DATA_ALLOC_SIZE
;
2441 start
= lastStart
= d
= (unsigned char *)malloc(buflen
);
2446 while ((c
= nextChar()) != 0 && isSpace(c
)) {
2451 state
->lineNumber
++;
2458 *d
= (c
- '0') << 4;
2459 } else if (isAlphaDigit(c
)) {
2460 *d
= (0xa + (c
- 'a')) << 4;
2469 } else if (isAlphaDigit(c
)) {
2470 *d
|= 0xa + (c
- 'a');
2476 if ((d
- lastStart
) >= DATA_ALLOC_SIZE
) {
2477 int oldsize
= d
- start
;
2478 assert(oldsize
== buflen
);
2479 buflen
+= DATA_ALLOC_SIZE
;
2480 start
= (unsigned char *)realloc(start
, oldsize
, buflen
);
2481 d
= lastStart
= start
+ oldsize
;
2492 safe_free(start
, buflen
);
2497 yylex(YYSTYPE
*lvalp
, parser_state_t
*state
)
2501 char tag
[TAG_MAX_LENGTH
];
2503 char attributes
[TAG_MAX_ATTRIBUTES
][TAG_MAX_LENGTH
];
2504 char values
[TAG_MAX_ATTRIBUTES
][TAG_MAX_LENGTH
];
2510 /* skip white space */
2512 while ((c
= nextChar()) != 0 && isSpace(c
)) {
2517 /* keep track of line number, don't return \n's */
2519 STATE
->lineNumber
++;
2524 // end of the buffer?
2529 tagType
= getTag(STATE
, tag
, &attributeCount
, attributes
, values
);
2530 if (tagType
== TAG_BAD
) {
2531 return SYNTAX_ERROR
;
2533 if (tagType
== TAG_IGNORE
) {
2537 // handle allocation and check for "ID" and "IDREF" tags up front
2538 *lvalp
= object
= newObject(STATE
);
2540 for (i
= 0; i
< attributeCount
; i
++) {
2541 if (attributes
[i
][0] == 'I' && attributes
[i
][1] == 'D') {
2542 // check for idref's, note: we ignore the tag, for
2543 // this to work correctly, all idrefs must be unique
2544 // across the whole serialization
2545 if (attributes
[i
][2] == 'R' && attributes
[i
][3] == 'E' &&
2546 attributes
[i
][4] == 'F' && !attributes
[i
][5]) {
2547 if (tagType
!= TAG_EMPTY
) {
2548 return SYNTAX_ERROR
;
2550 object
->idref
= strtol(values
[i
], NULL
, 0);
2554 if (!attributes
[i
][2]) {
2555 object
->idref
= strtol(values
[i
], NULL
, 0);
2557 return SYNTAX_ERROR
;
2564 if (!strcmp(tag
, "array")) {
2565 if (tagType
== TAG_EMPTY
) {
2566 object
->elements
= NULL
;
2569 return (tagType
== TAG_START
) ? '(' : ')';
2573 if (!strcmp(tag
, "dict")) {
2574 if (tagType
== TAG_EMPTY
) {
2575 object
->elements
= NULL
;
2578 return (tagType
== TAG_START
) ? '{' : '}';
2580 if (!strcmp(tag
, "data")) {
2582 if (tagType
== TAG_EMPTY
) {
2583 object
->data
= NULL
;
2588 bool isHexFormat
= false;
2589 for (i
= 0; i
< attributeCount
; i
++) {
2590 if (!strcmp(attributes
[i
], "format") && !strcmp(values
[i
], "hex")) {
2595 // CF encoded is the default form
2597 object
->data
= getHexData(STATE
, &size
);
2599 object
->data
= getCFEncodedData(STATE
, &size
);
2601 object
->size
= size
;
2602 if ((getTag(STATE
, tag
, &attributeCount
, attributes
, values
) != TAG_END
) || strcmp(tag
, "data")) {
2603 return SYNTAX_ERROR
;
2609 if (!strcmp(tag
, "false")) {
2610 if (tagType
== TAG_EMPTY
) {
2617 if (!strcmp(tag
, "integer")) {
2618 object
->size
= 64; // default
2619 for (i
= 0; i
< attributeCount
; i
++) {
2620 if (!strcmp(attributes
[i
], "size")) {
2621 object
->size
= strtoul(values
[i
], NULL
, 0);
2624 if (tagType
== TAG_EMPTY
) {
2628 object
->number
= getNumber(STATE
);
2629 if ((getTag(STATE
, tag
, &attributeCount
, attributes
, values
) != TAG_END
) || strcmp(tag
, "integer")) {
2630 return SYNTAX_ERROR
;
2636 if (!strcmp(tag
, "key")) {
2637 if (tagType
== TAG_EMPTY
) {
2638 return SYNTAX_ERROR
;
2640 object
->string
= getString(STATE
, &alloc_length
);
2641 if (!object
->string
) {
2642 return SYNTAX_ERROR
;
2644 object
->string_alloc_length
= alloc_length
;
2645 if ((getTag(STATE
, tag
, &attributeCount
, attributes
, values
) != TAG_END
)
2646 || strcmp(tag
, "key")) {
2647 return SYNTAX_ERROR
;
2653 if (!strcmp(tag
, "plist")) {
2654 freeObject(STATE
, object
);
2659 if (!strcmp(tag
, "string")) {
2660 if (tagType
== TAG_EMPTY
) {
2661 object
->string
= (char *)malloc(1);
2662 object
->string_alloc_length
= 1;
2663 object
->string
[0] = 0;
2666 object
->string
= getString(STATE
, &alloc_length
);
2667 if (!object
->string
) {
2668 return SYNTAX_ERROR
;
2670 object
->string_alloc_length
= alloc_length
;
2671 if ((getTag(STATE
, tag
, &attributeCount
, attributes
, values
) != TAG_END
)
2672 || strcmp(tag
, "string")) {
2673 return SYNTAX_ERROR
;
2677 if (!strcmp(tag
, "set")) {
2678 if (tagType
== TAG_EMPTY
) {
2679 object
->elements
= NULL
;
2682 if (tagType
== TAG_START
) {
2690 if (!strcmp(tag
, "true")) {
2691 if (tagType
== TAG_EMPTY
) {
2699 return SYNTAX_ERROR
;
2702 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2703 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2704 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2706 // "java" like allocation, if this code hits a syntax error in the
2707 // the middle of the parsed string we just bail with pointers hanging
2708 // all over place, this code helps keeps it all together
2710 //static int object_count = 0;
2713 newObject(parser_state_t
*state
)
2717 if (state
->freeObjects
) {
2718 o
= state
->freeObjects
;
2719 state
->freeObjects
= state
->freeObjects
->next
;
2721 o
= (object_t
*)malloc(sizeof(object_t
));
2723 o
->free
= state
->objects
;
2731 freeObject(parser_state_t
* state
, object_t
*o
)
2733 o
->next
= state
->freeObjects
;
2734 state
->freeObjects
= o
;
2738 cleanupObjects(parser_state_t
*state
)
2740 object_t
*t
, *o
= state
->objects
;
2744 // printf("OSUnserializeXML: releasing object o=%x object=%x\n", (int)o, (int)o->object);
2745 o
->object
->release();
2748 // printf("OSUnserializeXML: freeing object o=%x data=%x\n", (int)o, (int)o->data);
2752 // printf("OSUnserializeXML: releasing object o=%x key=%x\n", (int)o, (int)o->key);
2756 // printf("OSUnserializeXML: freeing object o=%x string=%x\n", (int)o, (int)o->string);
2762 safe_free(t
, sizeof(object_t
));
2765 // printf("object_count = %d\n", object_count);
2768 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2769 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2770 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2773 rememberObject(parser_state_t
*state
, int tag
, OSObject
*o
)
2776 snprintf(key
, 16, "%u", tag
);
2778 // printf("remember key %s\n", key);
2780 state
->tags
->setObject(key
, o
);
2784 retrieveObject(parser_state_t
*state
, int tag
)
2789 snprintf(key
, 16, "%u", tag
);
2791 // printf("retrieve key '%s'\n", key);
2793 ref
= state
->tags
->getObject(key
);
2798 o
= newObject(state
);
2803 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2804 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2805 // !@$&)(^Q$&*^!$(*!@$_(^%_(*Q#$(_*&!$_(*&!$_(*&!#$(*!@&^!@#%!_!#
2808 buildDictionary(parser_state_t
*state
, object_t
* header
)
2814 // get count and reverse order
2815 o
= header
->elements
;
2816 header
->elements
= 0;
2822 t
->next
= header
->elements
;
2823 header
->elements
= t
;
2826 dict
= OSDictionary::withCapacity(count
);
2827 if (header
->idref
>= 0) {
2828 rememberObject(state
, header
->idref
, dict
);
2831 o
= header
->elements
;
2833 dict
->setObject(o
->key
, o
->object
);
2836 o
->object
->release();
2842 freeObject(state
, t
);
2850 buildArray(parser_state_t
*state
, object_t
* header
)
2856 // get count and reverse order
2857 o
= header
->elements
;
2858 header
->elements
= 0;
2864 t
->next
= header
->elements
;
2865 header
->elements
= t
;
2868 array
= OSArray::withCapacity(count
);
2869 if (header
->idref
>= 0) {
2870 rememberObject(state
, header
->idref
, array
);
2873 o
= header
->elements
;
2875 array
->setObject(o
->object
);
2877 o
->object
->release();
2882 freeObject(state
, t
);
2890 buildSet(parser_state_t
*state
, object_t
*header
)
2892 object_t
*o
= buildArray(state
, header
);
2894 OSArray
*array
= (OSArray
*)o
->object
;
2895 OSSet
*set
= OSSet::withArray(array
, array
->getCapacity());
2897 // write over the reference created in buildArray
2898 if (header
->idref
>= 0) {
2899 rememberObject(state
, header
->idref
, set
);
2908 buildString(parser_state_t
*state
, object_t
*o
)
2912 string
= OSString::withCString(o
->string
);
2913 if (o
->idref
>= 0) {
2914 rememberObject(state
, o
->idref
, string
);
2925 buildSymbol(parser_state_t
*state
, object_t
*o
)
2929 symbol
= const_cast < OSSymbol
* > (OSSymbol::withCString(o
->string
));
2930 if (o
->idref
>= 0) {
2931 rememberObject(state
, o
->idref
, symbol
);
2934 safe_free(o
->string
, o
->string_alloc_length
);
2942 buildData(parser_state_t
*state
, object_t
*o
)
2947 data
= OSData::withBytes(o
->data
, o
->size
);
2949 data
= OSData::withCapacity(0);
2951 if (o
->idref
>= 0) {
2952 rememberObject(state
, o
->idref
, data
);
2964 buildNumber(parser_state_t
*state
, object_t
*o
)
2966 OSNumber
*number
= OSNumber::withNumber(o
->number
, o
->size
);
2968 if (o
->idref
>= 0) {
2969 rememberObject(state
, o
->idref
, number
);
2977 buildBoolean(parser_state_t
*state __unused
, object_t
*o
)
2979 o
->object
= ((o
->number
== 0) ? kOSBooleanFalse
: kOSBooleanTrue
);
2980 o
->object
->retain();
2985 OSUnserializeXML(const char *buffer
, OSString
**errorString
)
2992 parser_state_t
*state
= (parser_state_t
*)malloc(sizeof(parser_state_t
));
2999 *errorString
= NULL
;
3002 state
->parseBuffer
= buffer
;
3003 state
->parseBufferIndex
= 0;
3004 state
->lineNumber
= 1;
3006 state
->freeObjects
= 0;
3007 state
->tags
= OSDictionary::withCapacity(128);
3008 state
->errorString
= errorString
;
3009 state
->parsedObject
= 0;
3010 state
->parsedObjectCount
= 0;
3011 state
->retrievedObjectCount
= 0;
3013 (void)yyparse((void *)state
);
3015 object
= state
->parsedObject
;
3017 cleanupObjects(state
);
3018 state
->tags
->release();
3019 safe_free(state
, sizeof(parser_state_t
));
3024 #include <libkern/OSSerializeBinary.h>
3027 OSUnserializeXML(const char *buffer
, size_t bufferSize
, OSString
**errorString
)
3032 if (bufferSize
< sizeof(kOSSerializeBinarySignature
)) {
3036 if (!strcmp(kOSSerializeBinarySignature
, buffer
)
3037 || (kOSSerializeIndexedBinarySignature
== (uint8_t)buffer
[0])) {
3038 return OSUnserializeBinary(buffer
, bufferSize
, errorString
);
3041 // XML must be null terminated
3042 if (buffer
[bufferSize
- 1]) {
3046 return OSUnserializeXML(buffer
, errorString
);
3055 // DO NOT EDIT OSUnserializeXML.cpp!