2 /* Include this before everything else, for various large-file definitions */
8 * We want a reentrant scanner.
13 * We don't use input, so don't generate code for it.
18 * We don't use unput, so don't generate code for it.
23 * We don't read interactively from the terminal.
25 %option never-interactive
28 * The language we're scanning is case-insensitive.
33 * We use start condition stacks.
38 * We want to stop processing when we get to the end of the input.
43 * The type for the state we keep for a scanner.
45 %option extra-type="DiamDict_scanner_state_t *"
48 * Prefix scanner routines with "DiamDict_" rather than "yy", so this scanner
49 * can coexist with other scanners.
51 %option prefix="DiamDict_"
54 * We have to override the memory allocators so that we don't get
55 * "unused argument" warnings from the yyscanner argument (which
56 * we don't use, as we have a global memory allocator).
58 * We provide, as macros, our own versions of the routines generated by Flex,
59 * which just call malloc()/realloc()/free() (as the Flex versions do),
60 * discarding the extra argument.
69 ** Diameter Dictionary Import Routines
71 ** (c) 2007, Luis E. Garcia Ontanon <luis@ontanon.org>
73 ** This library is free software; you can redistribute it and/or
74 ** modify it under the terms of the GNU Library General Public
75 ** License as published by the Free Software Foundation; either
76 ** version 2 of the License, or (at your option) any later version.
78 ** This library is distributed in the hope that it will be useful,
79 ** but WITHOUT ANY WARRANTY; without even the implied warranty of
80 ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
81 ** Library General Public License for more details.
83 ** You should have received a copy of the GNU Library General Public
84 ** License along with this library; if not, write to the Free Software
85 ** Foundation, Inc., 51 Franklin Street, Fifth Floor,
86 ** Boston, MA 02110-1301, USA.
88 ** See draft-frascone-xml-dictionary for the syntax of the
98 #include "diam_dict.h"
99 #include <epan/to_str.h>
100 #include <wsutil/file_util.h>
103 * Disable diagnostics in the code generated by Flex.
107 typedef struct entity_t {
110 struct entity_t* next;
113 #define ATTR_UINT(cont) do { D(("attr_uint " #cont "\t" )); yyextra->attr_uint = &(cont); yy_push_state(GET_UINT_ATTR, yyscanner); } while(0)
114 #define ATTR_STR(cont) do { D(("attr_str " #cont "\t" )); yyextra->attr_str = &(cont); yy_push_state(GET_ATTR, yyscanner); } while(0)
115 #define IGNORE() do { D(("ignore: %s\t",yytext)); yy_push_state(IGNORE_ATTR, yyscanner); } while(0)
117 #define D(args) ddict_debug args
119 #define MAX_INCLUDE_DEPTH 10
120 #define YY_INPUT(buf,result,max_size) { result = yyextra->current_yyinput(buf,max_size,yyscanner); }
121 #define YY_USER_INIT { \
122 DiamDict_scanner_state_t *scanner_state = DiamDict_get_extra(yyscanner); \
123 BEGIN(scanner_state->start_state); \
126 #define APPEND(txt,len) append_to_buffer(txt,len,yyextra)
135 unsigned size_strbuf;
140 ddict_application_t* appl;
142 ddict_enum_t* enumitem;
144 ddict_typedefn_t* typedefn;
147 ddict_xmlpi_t* xmlpi;
149 ddict_application_t* last_appl;
150 ddict_avp_t* last_avp;
151 ddict_enum_t* last_enumitem;
152 ddict_gavp_t* last_gavp;
153 ddict_typedefn_t* last_typedefn;
154 ddict_cmd_t* last_cmd;
155 ddict_vendor_t* last_vnd;
156 ddict_xmlpi_t* last_xmlpi;
163 size_t (*current_yyinput)(char*,size_t,yyscan_t);
164 int (*current_close)(FILE *fh);
166 YY_BUFFER_STATE include_stack[MAX_INCLUDE_DEPTH];
167 int include_stack_ptr;
170 } DiamDict_scanner_state_t;
172 static void ddict_debug(const char* fmt, ...) G_GNUC_PRINTF(1, 2);
173 static void append_to_buffer(const char* txt, unsigned len, DiamDict_scanner_state_t *statep);
174 static FILE* ddict_open(const char*, const char*);
177 * Sleazy hack to suppress compiler warnings in yy_fatal_error().
179 #define YY_EXIT_FAILURE ((void)yyscanner, 2)
182 * Macros for the allocators, to discard the extra argument.
184 #define DiamDict_alloc(size, yyscanner) (void *)malloc(size)
185 #define DiamDict_realloc(ptr, size, yyscanner) (void *)realloc((char *)(ptr), (size))
186 #define DiamDict_free(ptr, yyscanner) free((char *)ptr)
191 xmlpi_start [[:blank:] \r\n]*<\?[[:blank:] \r\n]*
192 xmlpi_end [[:blank:] \r\n]*\?>[[:blank:] \r\n]*
193 xmlpi_key_attr [[:blank:] \r\n]*key[[:blank:] \r\n]*=[[:blank:] \r\n]*\042
194 xmlpi_value_attr [[:blank:] \r\n]*value[[:blank:] \r\n]*=[[:blank:] \r\n]*\042
196 comment_start [[:blank:] \r\n]*<!--[[:blank:] \r\n]*
197 comment_end [[:blank:] \r\n]*-->[[:blank:] \r\n]*
198 open_tag [[:blank:] \r\n]*<[[:blank:] \r\n]*
199 end_tag [[:blank:] \r\n]*\/>[[:blank:] \r\n]*
200 close_tag [[:blank:] \r\n]*>[[:blank:] \r\n]*
201 open_closetag [[:blank:] \r\n]*<\/[[:blank:] \r\n]*
202 equals [[:blank:] \r\n]*=[[:blank:] \r\n]*
203 whitespace [[:blank:] \r\n]*
204 dquoted \042[^\042]*\042
206 doctype [[:blank:] \r\n]*<!DOCTYPE[^\[]*\[[[:blank:] \r\n]*
207 doctype_end [[:blank:] \r\n]*\][[:blank:] \r\n]*>[[:blank:] \r\n]*
209 start_entity [[:blank:] \r\n]*<\!ENTITY[[:blank:] \r\n]*
210 system [[:blank:] \r\n]*SYSTEM[[:blank:] \r\n]*\042
211 entityname [a-z0-9-]+
213 end_entity \042[[:blank:] \r\n]*>[[:blank:] \r\n]*
227 dictionary_start <dictionary>
228 dictionary_end <\/dictionary>
230 base_start <base[^>*]*>
233 application_start <application
234 application_end<\/application>
236 command_start <command
237 command_end<\/command>
239 typedefn_start <typedefn
247 grouped_start <grouped>
248 grouped_end <\/grouped>
255 ignored_attr [a-z0-9-]+=
256 ignored_quoted \042[^\042]*\042
261 vendor_attr vendor-id=\042
262 typename_attr type-name=\042
263 typeparent_attr type-parent=\042
264 description_attr description=\042
268 %S LOADING LOADING_COMMENT LOADING_XMLPI ENTITY GET_SYSTEM GET_FILE END_ENTITY
269 %S GET_ATTR GET_UINT_ATTR END_ATTR OUTSIDE IN_DICT IN_APPL IN_AVP APPL_ATTRS IGNORE_ATTR
270 %S TYPE_ATTRS GAVP_ATTRS ENUM_ATTRS AVP_ATTRS VENDOR_ATTRS COMMAND_ATTRS TYPEDEFN_ATTRS
271 %S XMLPI_ATTRS XMLPI_GETKEY XMLPI_GETVAL XMLPI_ENDATTR
274 <LOADING>{doctype_end} ;
276 <LOADING>{comment_start} BEGIN LOADING_COMMENT;
278 <LOADING_COMMENT>{comment_end} BEGIN LOADING;
280 <LOADING>{xmlpi_start} BEGIN LOADING_XMLPI;
281 <LOADING_XMLPI>{whitespace} ;
282 <LOADING_XMLPI>{entityname} {
283 yyextra->xmlpi = g_new(ddict_xmlpi_t,1);
284 yyextra->xmlpi->name = g_strdup(yytext);
285 yyextra->xmlpi->key = NULL;
286 yyextra->xmlpi->value = NULL;
287 yyextra->xmlpi->next = NULL;
289 if (!yyextra->dict->xmlpis)
290 yyextra->last_xmlpi = yyextra->dict->xmlpis = yyextra->xmlpi;
292 yyextra->last_xmlpi = yyextra->last_xmlpi->next = yyextra->xmlpi;
297 <XMLPI_ATTRS>{xmlpi_key_attr} BEGIN XMLPI_GETKEY;
298 <XMLPI_GETKEY>{ndquot} { yyextra->xmlpi->key = g_strdup(yytext); BEGIN XMLPI_ATTRS; }
300 <XMLPI_ATTRS>{xmlpi_value_attr} BEGIN XMLPI_GETVAL;
301 <XMLPI_GETVAL>{ndquot} { yyextra->xmlpi->value = g_strdup(yytext); BEGIN XMLPI_ATTRS; }
304 <XMLPI_ATTRS>{xmlpi_end} BEGIN LOADING;
307 <LOADING>{start_entity} BEGIN ENTITY;
308 <ENTITY>{entityname} {
309 entity_t* e = g_new(entity_t,1);
310 e->name = g_strdup(yytext);
311 e->next = yyextra->ents;
315 <GET_SYSTEM>{system} BEGIN GET_FILE;
317 yyextra->ents->file = g_strdup(yytext);
320 <END_ENTITY>{end_entity} BEGIN LOADING;
322 <LOADING>{open_tag} APPEND("<",1);
324 <LOADING>{close_tag} APPEND(">",1);
326 <LOADING>{end_tag} APPEND("/>",2);
328 <LOADING>{open_closetag} APPEND("</",2);
330 <LOADING>{whitespace} APPEND(" ",1);
332 <LOADING>{dquoted} APPEND(yytext, (unsigned) yyleng);
334 <LOADING>{equals} APPEND("=",1);
336 <LOADING>{any} APPEND(yytext, (unsigned) yyleng);
342 while(*p != ';') p++;
346 D(("looking for entity: %s\n",yytext));
348 if ( yyextra->include_stack_ptr >= MAX_INCLUDE_DEPTH ) {
349 fprintf(stderr, "included files nested to deeply\n");
353 for (e = yyextra->ents; e; e = e->next) {
354 if (strcmp(e->name,yytext) == 0) {
355 yyin = ddict_open(yyextra->sys_dir,e->file);
356 D(("entity: %s filename: %s yyin: %p\n",e->name,e->file,(void*)yyin));
359 fprintf(stderr, "Could not open file: '%s', error: %s\n", e->file, g_strerror(errno) );
361 fprintf(stderr, "Could not open file: '%s', error unknown (errno == 0)\n", e->file );
364 yyextra->include_stack[yyextra->include_stack_ptr++] = YY_CURRENT_BUFFER;
365 yy_switch_to_buffer(yy_create_buffer(yyin, YY_BUF_SIZE, yyscanner), yyscanner);
373 fprintf(stderr, "Could not find entity: '%s'\n", yytext );
380 if (!yyin) yyterminate();
382 yyextra->current_close(yyin);
383 D(("closing: %p %i\n",(void*)yyin,yyextra->include_stack_ptr));
385 if ( --yyextra->include_stack_ptr < 0 ) {
386 D(("DONE READING\n"));
390 yy_delete_buffer( YY_CURRENT_BUFFER, yyscanner);
391 yy_switch_to_buffer(yyextra->include_stack[yyextra->include_stack_ptr], yyscanner);
398 *(yyextra->attr_str) = g_strdup(yytext);
400 yyextra->attr_str = NULL;
404 <GET_UINT_ATTR>{number} {
405 *(yyextra->attr_uint) = (unsigned) strtoul(yytext,NULL,10);
407 yyextra->attr_uint = NULL;
411 <END_ATTR>{dquot} { yy_pop_state(yyscanner); }
418 <IGNORE_ATTR>{ignored_quoted} {
419 D(("=>%s<=\n",yytext));
420 yy_pop_state(yyscanner);
423 <OUTSIDE>{dictionary_start} {
424 D(("dictionary_start\n"));
429 <IN_DICT>{base_start} {
434 <IN_DICT>{application_start} {
435 D(("application_start\n"));
437 yyextra->appl = g_new(ddict_application_t,1);
438 yyextra->appl->name = NULL;
439 yyextra->appl->code = 0;
440 yyextra->appl->next = NULL;
442 if (!yyextra->dict->applications)
443 yyextra->last_appl = yyextra->dict->applications = yyextra->appl;
445 yyextra->last_appl = yyextra->last_appl->next = yyextra->appl;
450 <APPL_ATTRS>{name_attr} { ATTR_STR(yyextra->appl->name); }
451 <APPL_ATTRS>{id_attr} { ATTR_UINT(yyextra->appl->code); }
453 <APPL_ATTRS>{stop} BEGIN IN_APPL;
454 <APPL_ATTRS>{stop_end} BEGIN IN_DICT;
456 <IN_APPL>{command_end} ;
458 <IN_APPL>{command_start} {
459 D(("command_start\n"));
461 yyextra->cmd = g_new(ddict_cmd_t,1);
462 yyextra->cmd->name = NULL;
463 yyextra->cmd->vendor = NULL;
464 yyextra->cmd->code = 0;
465 yyextra->cmd->next = NULL;
467 if (!yyextra->dict->cmds)
468 yyextra->last_cmd = yyextra->dict->cmds = yyextra->cmd;
470 yyextra->last_cmd = yyextra->last_cmd->next = yyextra->cmd;
475 <COMMAND_ATTRS>{name_attr} { ATTR_STR(yyextra->cmd->name); }
476 <COMMAND_ATTRS>{vendor_attr} { ATTR_STR(yyextra->cmd->vendor); }
477 <COMMAND_ATTRS>{code_attr} { ATTR_UINT(yyextra->cmd->code); }
478 <COMMAND_ATTRS>{stop} |
479 <COMMAND_ATTRS>{stop_end} { BEGIN IN_APPL; }
481 <IN_DICT>{vendor_start} {
482 D(("vendor_start\n"));
484 yyextra->vnd = g_new(ddict_vendor_t,1);
485 yyextra->vnd->name = NULL;
486 yyextra->vnd->code = 0;
487 yyextra->vnd->next = NULL;
489 if (!yyextra->dict->vendors)
490 yyextra->last_vnd = yyextra->dict->vendors = yyextra->vnd;
492 yyextra->last_vnd = yyextra->last_vnd->next = yyextra->vnd;
497 <VENDOR_ATTRS>{name_attr} { ATTR_STR(yyextra->vnd->desc); }
498 <VENDOR_ATTRS>{vendor_attr} { ATTR_STR(yyextra->vnd->name); }
499 <VENDOR_ATTRS>{code_attr} { ATTR_UINT(yyextra->vnd->code); }
500 <VENDOR_ATTRS>{stop} { BEGIN IN_APPL; }
501 <VENDOR_ATTRS>{stop_end} { BEGIN IN_DICT; }
503 <IN_APPL>{typedefn_start} {
504 D(("typedefn_start\n"));
506 yyextra->typedefn = g_new(ddict_typedefn_t,1);
507 yyextra->typedefn->name = NULL;
508 yyextra->typedefn->parent = NULL;
509 yyextra->typedefn->next = NULL;
511 if (!yyextra->dict->typedefns)
512 yyextra->last_typedefn = yyextra->dict->typedefns = yyextra->typedefn;
514 yyextra->last_typedefn = yyextra->last_typedefn->next = yyextra->typedefn;
516 BEGIN TYPEDEFN_ATTRS;
519 <TYPEDEFN_ATTRS>{typename_attr} { ATTR_STR(yyextra->typedefn->name); }
520 <TYPEDEFN_ATTRS>{typeparent_attr} { ATTR_STR(yyextra->typedefn->parent); }
521 <TYPEDEFN_ATTRS>{stop} |
522 <TYPEDEFN_ATTRS>{stop_end} { BEGIN IN_APPL; }
525 <IN_APPL>{avp_start} {
528 yyextra->avp = g_new(ddict_avp_t,1);
529 yyextra->avp->name = NULL;
530 yyextra->avp->description = NULL;
531 yyextra->avp->vendor = NULL;
532 yyextra->avp->code = 0;
533 yyextra->avp->type = NULL;
534 yyextra->avp->enums = NULL;
535 yyextra->avp->gavps = NULL;
536 yyextra->avp->next = NULL;
538 if (! yyextra->dict->avps )
539 yyextra->last_avp = yyextra->dict->avps = yyextra->avp;
541 yyextra->last_avp = yyextra->last_avp->next = yyextra->avp;
546 <AVP_ATTRS>{name_attr} { ATTR_STR(yyextra->avp->name); }
547 <AVP_ATTRS>{description_attr} { ATTR_STR(yyextra->avp->description); }
548 <AVP_ATTRS>{vendor_attr} { ATTR_STR(yyextra->avp->vendor); }
549 <AVP_ATTRS>{code_attr} { ATTR_UINT(yyextra->avp->code); }
550 <AVP_ATTRS>{stop} { BEGIN IN_AVP; }
551 <AVP_ATTRS>{stop_end} { BEGIN IN_APPL; }
554 <IN_AVP>{grouped_start} { yyextra->avp->type = g_strdup("Grouped"); };
555 <IN_AVP>{grouped_end} ;
557 <IN_AVP>{type_start} { BEGIN TYPE_ATTRS; }
558 <TYPE_ATTRS>{typename_attr} { ATTR_STR(yyextra->avp->type); }
560 <IN_AVP>{gavp_start} {
563 yyextra->gavp = g_new(ddict_gavp_t,1);
564 yyextra->gavp->name = NULL;
565 yyextra->gavp->code = 0;
566 yyextra->gavp->next = NULL;
568 if (!yyextra->avp->gavps)
569 yyextra->last_gavp = yyextra->avp->gavps = yyextra->gavp;
571 yyextra->last_gavp = yyextra->last_gavp->next = yyextra->gavp;
577 <GAVP_ATTRS>{name_attr} { ATTR_STR(yyextra->gavp->name); }
580 <IN_AVP>{enum_start} {
583 yyextra->enumitem = g_new(ddict_enum_t,1);
584 yyextra->enumitem->name = NULL;
585 yyextra->enumitem->code = 0;
586 yyextra->enumitem->next = NULL;
588 if (!yyextra->avp->enums)
589 yyextra->last_enumitem = yyextra->avp->enums = yyextra->enumitem;
591 yyextra->last_enumitem = yyextra->last_enumitem->next = yyextra->enumitem;
597 <ENUM_ATTRS>{name_attr} { ATTR_STR(yyextra->enumitem->name); }
598 <ENUM_ATTRS>{code_attr} { ATTR_UINT(yyextra->enumitem->code); }
600 <TYPE_ATTRS,GAVP_ATTRS,ENUM_ATTRS>{stop} { BEGIN IN_AVP; }
601 <TYPE_ATTRS,GAVP_ATTRS,ENUM_ATTRS>{stop_end} { BEGIN IN_AVP; }
603 <IN_AVP>{avp_end} { D(("avp_end\n")); BEGIN IN_APPL; }
605 <IN_APPL>{application_end} {
606 D(("application_end\n")); BEGIN IN_DICT;
608 <IN_APPL>{stop_end} {
609 D(("application_stop_end\n")); BEGIN IN_DICT;
611 <IN_APPL>{vendor_end} {
612 D(("vendor_end\n")); BEGIN IN_DICT;
614 <IN_APPL>{base_end} {
615 D(("base_end\n")); BEGIN IN_DICT;
618 <IN_DICT>{dictionary_end} {
622 <AVP_ATTRS,ENUM_ATTRS,GAVP_ATTRS,TYPE_ATTRS,TYPEDEFN_ATTRS,VENDOR_ATTRS,APPL_ATTRS,COMMAND_ATTRS>{ignored_attr} IGNORE();
629 * Turn diagnostics back on, so we check the code that we've written.
633 static int debugging = 0;
635 static void ddict_debug(const char* fmt, ...) {
639 if (debugging) vfprintf(stderr, fmt, ap);
646 * Sleazy hack to avoid unused function warnings for yy_top_state.
648 extern void ddict_unused(yyscan_t yyscanner);
651 ddict_unused(yyscan_t yyscanner)
653 yy_top_state(yyscanner);
657 append_to_buffer(const char* txt, unsigned len, DiamDict_scanner_state_t *statep)
660 if (statep->strbuf == NULL) {
661 statep->strbuf = (char*)g_malloc(statep->size_strbuf);
662 statep->read_ptr = statep->strbuf;
663 statep->write_ptr = statep->strbuf;
666 if (statep->len_strbuf + len >= statep->size_strbuf) {
667 statep->strbuf = (char*)g_realloc(statep->strbuf,statep->size_strbuf *= 2);
668 statep->read_ptr = statep->strbuf;
671 statep->write_ptr = statep->strbuf + statep->len_strbuf;
672 memcpy(statep->write_ptr, txt, len + 1);
673 statep->len_strbuf += len;
677 file_input(char* buf, size_t max, yyscan_t scanner)
679 FILE *in = yyget_in(scanner);
682 read_cnt = fread(buf,1,max,in);
684 if ( read_cnt == max ) {
686 } else if (read_cnt > 0) {
695 string_input(char* buf, size_t max, yyscan_t scanner)
697 DiamDict_scanner_state_t *statep = yyget_extra(scanner);
699 if (statep->read_ptr >= statep->write_ptr ) {
701 } else if ( statep->read_ptr + max > statep->write_ptr ) {
702 max = statep->write_ptr - statep->read_ptr;
705 memcpy(buf,statep->read_ptr,max);
706 statep->read_ptr += max;
712 * If we're reading from a string, yyin is set to stdin, and we don't
713 * want to close that.
716 string_close(FILE *fh _U_)
722 ddict_open(const char* system_directory, const char* filename)
726 if (system_directory) {
727 fname = ws_strdup_printf("%s" G_DIR_SEPARATOR_S "%s",
728 system_directory,filename);
730 fname = g_strdup(filename);
733 fh = ws_fopen(fname,"r");
735 D(("fname: %s fh: %p\n",fname,(void*)fh));
744 ddict_scan(const char* system_directory, const char* filename, int dbg)
746 DiamDict_scanner_state_t state;
752 state.sys_dir = system_directory;
754 state.write_ptr = NULL;
755 state.read_ptr = NULL;
758 state.size_strbuf = 8192;
759 state.len_strbuf = 0;
761 state.dict = g_new(ddict_t,1);
762 state.dict->applications = NULL;
763 state.dict->vendors = NULL;
764 state.dict->cmds = NULL;
765 state.dict->typedefns = NULL;
766 state.dict->avps = NULL;
767 state.dict->xmlpis = NULL;
771 state.enumitem = NULL;
773 state.typedefn = NULL;
778 state.last_appl = NULL;
779 state.last_avp = NULL;
780 state.last_enumitem = NULL;
781 state.last_gavp = NULL;
782 state.last_typedefn = NULL;
783 state.last_cmd = NULL;
784 state.last_vnd = NULL;
785 state.last_xmlpi = NULL;
789 state.attr_str = NULL;
790 state.attr_uint = NULL;
795 * Reads the file, does some work, and stores a modified version
796 * of the file contents in memory.
798 state.current_yyinput = file_input;
799 state.current_close = fclose;
800 state.include_stack_ptr = 0;
802 in = ddict_open(system_directory,filename);
805 D(("unable to open %s: %s\n", filename, g_strerror(errno)));
810 if (DiamDict_lex_init(&scanner) != 0) {
811 /* Note: cannot be reached since memory allocation failure terminates early */
812 D(("Can't initialize scanner: %s\n", g_strerror(errno)));
818 DiamDict_set_in(in, scanner);
820 /* Associate the state with the scanner */
821 DiamDict_set_extra(&state, scanner);
823 state.start_state = LOADING;
824 DiamDict_lex(scanner);
826 DiamDict_lex_destroy(scanner);
828 * XXX - can the lexical analyzer terminate without closing
829 * all open input files?
832 D(("\n---------------\n%s\n------- %u -------\n",state.strbuf,state.len_strbuf));
837 * Reads the modified version of the file contents and does the
840 state.current_yyinput = string_input;
841 state.current_close = string_close;
843 if (DiamDict_lex_init(&scanner) != 0) {
844 /* Note: cannot be reached since memory allocation failure terminates early */
845 D(("Can't initialize scanner: %s\n", g_strerror(errno)));
847 g_free(state.strbuf);
851 /* Associate the state with the scanner */
852 DiamDict_set_extra(&state, scanner);
854 state.start_state = OUTSIDE;
855 DiamDict_lex(scanner);
857 DiamDict_lex_destroy(scanner);
861 for (e = state.ents; e; e = en) {
868 g_free(state.strbuf);
874 ddict_free(ddict_t* d)
876 ddict_application_t *p, *pn;
877 ddict_vendor_t *v, *vn;
879 ddict_typedefn_t *t, *tn;
881 ddict_xmlpi_t *x, *xn;
883 #define FREE_NAMEANDOBJ(n) do { g_free(n->name); g_free(n); } while(0)
885 for (p = d->applications; p; p = pn ) {
890 for (v = d->vendors; v; v = vn) {
896 for (c = d->cmds; c; c = cn ) {
902 for (t = d->typedefns; t; t = tn) {
908 for (a = d->avps; a; a = an) {
909 ddict_gavp_t* g, *gn;
910 ddict_enum_t* e, *en;
913 for (g = a->gavps; g; g = gn) {
918 for (e = a->enums; e; e = en) {
925 g_free(a->description);
929 for (x = d->xmlpis; x; x = xn) {
940 ddict_print(FILE* fh, ddict_t* d)
942 ddict_application_t* p;
949 for (p = d->applications; p; p = p->next) {
950 fprintf(fh,"Application: %s[%u]:\n",
951 p->name ? p->name : "-",
955 for (v = d->vendors; v; v = v->next) {
956 fprintf(fh,"Vendor: %s[%u]:\n",
957 v->name ? v->name : "-",
961 for (c = d->cmds; c; c = c->next) {
962 fprintf(fh,"Command: %s[%u] \n",
963 c->name ? c->name : "-",
967 for (t = d->typedefns; t; t = t->next) {
968 fprintf(fh,"Type: %s -> %s \n",
969 t->name ? t->name : "-",
970 t->parent ? t->parent : "" );
973 for (a = d->avps; a; a = a->next) {
976 fprintf(fh,"AVP: %s[%u:%s] %s %s\n",
977 a->name ? a->name : "-",
979 a->vendor ? a->vendor : "None",
980 a->type ? a->type : "-",
981 a->description ? a->description : "");
983 for (g = a->gavps; g; g = g->next) {
984 fprintf(fh,"\tGAVP: %s\n",
985 g->name ? g->name : "-" );
988 for (e = a->enums; e; e = e->next) {
989 fprintf(fh,"\tEnum: %s[%u]\n",
990 e->name ? e->name : "-",
996 #ifdef TEST_DIAM_DICT_STANDALONE
998 main(int argc, char** argv)
1012 fprintf(stderr,"%s: usage [dictionary_dir] dictionary_filename\n",argv[0]);
1016 d = ddict_scan(dname,fname,1);
1018 fprintf(stderr, "Can't open dictionary\n");
1022 ddict_print(stdout, d);