parse.c
上传用户:cxs890
上传日期:2021-05-22
资源大小:347k
文件大小:103k
源码类别:

SNMP编程

开发平台:

C/C++

  1. /*
  2.  * parse.c
  3.  *
  4.  * Update: 1998-09-22 <mslifcak@iss.net>
  5.  * Clear nbuckets in init_node_hash.
  6.  * New method xcalloc returns zeroed data structures.
  7.  * New method alloc_node encapsulates common node creation.
  8.  * New method to configure terminate comment at end of line.
  9.  * New method to configure accept underscore in labels.
  10.  *
  11.  * Update: 1998-10-10 <daves@csc.liv.ac.uk>
  12.  * fully qualified OID parsing patch
  13.  *
  14.  * Update: 1998-10-20 <daves@csc.liv.ac.uk>
  15.  * merge_anon_children patch
  16.  *
  17.  * Update: 1998-10-21 <mslifcak@iss.net>
  18.  * Merge_parse_objectid associates information with last node in chain.
  19.  */
  20. /******************************************************************
  21.         Copyright 1989, 1991, 1992 by Carnegie Mellon University
  22.                       All Rights Reserved
  23. Permission to use, copy, modify, and distribute this software and its
  24. documentation for any purpose and without fee is hereby granted,
  25. provided that the above copyright notice appear in all copies and that
  26. both that copyright notice and this permission notice appear in
  27. supporting documentation, and that the name of CMU not be
  28. used in advertising or publicity pertaining to distribution of the
  29. software without specific, written prior permission.
  30. CMU DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
  31. ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
  32. CMU BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
  33. ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
  34. WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
  35. ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
  36. SOFTWARE.
  37. ******************************************************************/
  38. #include <config.h>
  39. #include <snmp_logging.h>
  40. #include <stdio.h>
  41. #if HAVE_STDLIB_H
  42. #include <stdlib.h>
  43. #endif
  44. #if HAVE_STRING_H
  45. #include <string.h>
  46. #else
  47. #include <strings.h>
  48. #endif
  49. #include <ctype.h>
  50. #include <sys/types.h>
  51. #include <sys/stat.h>
  52. /* Wow.  This is ugly.  -- Wes */
  53. #if HAVE_DIRENT_H
  54. # include <dirent.h>
  55. # define NAMLEN(dirent) strlen((dirent)->d_name)
  56. #else
  57. # define dirent direct
  58. # define NAMLEN(dirent) (dirent)->d_namlen
  59. #endif
  60. #if HAVE_WINSOCK_H
  61. #include <ip/socket.h>
  62. #endif
  63. #if HAVE_NETINET_IN_H
  64. #include <netinet/in.h>
  65. #endif
  66. #if defined(HAVE_REGEX_H) && defined(HAVE_REGCOMP)
  67. #include <regex.h>
  68. #endif
  69. #if HAVE_DMALLOC_H
  70. #include <dmalloc.h>
  71. #endif
  72. #include "system.h"
  73. #include "parse.h"
  74. #include "asn1.h"
  75. #include "mib.h"
  76. #include "snmp_api.h"
  77. #include "snmp_debug.h"
  78. #include "snmp_logging.h"
  79. #include "default_store.h"
  80. #include "tools.h"
  81. /*
  82.  * This is one element of an object identifier with either an integer
  83.  * subidentifier, or a textual string label, or both.
  84.  * The subid is -1 if not present, and label is NULL if not present.
  85.  */
  86. struct subid_s {
  87.     int subid;
  88.     int modid;
  89.     char *label;
  90. };
  91. #define MAXTC   1024
  92. struct tc {     /* textual conventions */
  93.     int type;
  94.     int modid;
  95.     char *descriptor;
  96.     char *hint;
  97.     struct enum_list *enums;
  98.     struct range_list *ranges;
  99. } tclist[MAXTC];
  100. int Line = 0;
  101. const char *File = "(none)";
  102. static int anonymous = 0;
  103. #define SYNTAX_MASK     0x80
  104. /* types of tokens
  105.  Tokens wiht the SYNTAX_MASK bit set are syntax tokens */
  106. #define CONTINUE    -1
  107. #define ENDOFFILE   0
  108. #define LABEL       1
  109. #define SUBTREE     2
  110. #define SYNTAX      3
  111. #define OBJID       (4 | SYNTAX_MASK)
  112. #define OCTETSTR    (5 | SYNTAX_MASK)
  113. #define INTEGER     (6 | SYNTAX_MASK)
  114. #define INTEGER32   INTEGER
  115. #define NETADDR     (7 | SYNTAX_MASK)
  116. #define IPADDR      (8 | SYNTAX_MASK)
  117. #define COUNTER     (9 | SYNTAX_MASK)
  118. #define GAUGE       (10 | SYNTAX_MASK)
  119. #define TIMETICKS   (11 | SYNTAX_MASK)
  120. #define KW_OPAQUE   (12 | SYNTAX_MASK)
  121. #define NUL         (13 | SYNTAX_MASK)
  122. #define SEQUENCE    14
  123. #define OF          15  /* SEQUENCE OF */
  124. #define OBJTYPE     16
  125. #define ACCESS      17
  126. #define READONLY    18
  127. #define READWRITE   19
  128. #define WRITEONLY   20
  129. #define NOACCESS    21
  130. #undef  STATUS
  131. #define STATUS      22
  132. #define MANDATORY   23
  133. #define KW_OPTIONAL    24
  134. #define OBSOLETE    25
  135. /* #define RECOMMENDED 26 */
  136. #define PUNCT       27
  137. #define EQUALS      28
  138. #define NUMBER      29
  139. #define LEFTBRACKET 30
  140. #define RIGHTBRACKET 31
  141. #define LEFTPAREN   32
  142. #define RIGHTPAREN  33
  143. #define COMMA       34
  144. #define DESCRIPTION 35
  145. #define QUOTESTRING 36
  146. #define INDEX       37
  147. #define DEFVAL      38
  148. #define DEPRECATED  39
  149. #define SIZE        40
  150. #define BITSTRING   (41 | SYNTAX_MASK)
  151. #define NSAPADDRESS (42 | SYNTAX_MASK)
  152. #define COUNTER64   (43 | SYNTAX_MASK)
  153. #define OBJGROUP    44
  154. #define NOTIFTYPE   45
  155. #define AUGMENTS    46
  156. #define COMPLIANCE  47
  157. #define READCREATE  48
  158. #define UNITS       49
  159. #define REFERENCE   50
  160. #define NUM_ENTRIES 51
  161. #define MODULEIDENTITY 52
  162. #define LASTUPDATED 53
  163. #define ORGANIZATION 54
  164. #define CONTACTINFO 55
  165. #define UINTEGER32 (56 | SYNTAX_MASK)
  166. #define CURRENT     57
  167. #define DEFINITIONS 58
  168. #define END         59
  169. #define SEMI        60
  170. #define TRAPTYPE    61
  171. #define ENTERPRISE  62
  172. /* #define DISPLAYSTR (63 | SYNTAX_MASK) */
  173. #define BEGIN       64
  174. #define IMPORTS     65
  175. #define EXPORTS     66
  176. #define ACCNOTIFY   67
  177. #define BAR         68
  178. #define RANGE       69
  179. #define CONVENTION  70
  180. #define DISPLAYHINT 71
  181. #define FROM        72
  182. #define CAPABILITIES 73
  183. #define MACRO       74
  184. #define IMPLIED     75
  185. struct tree *
  186. read_module(const char *name);
  187. struct tok {
  188.     const char *name;                 /* token name */
  189.     int len;                    /* length not counting nul */
  190.     int token;                  /* value */
  191.     int hash;                   /* hash of name */
  192.     struct tok *next;           /* pointer to next in hash table */
  193. };
  194. /*
  195. static struct tok tokens[] = {
  196.     { "obsolete", sizeof ("obsolete")-1, OBSOLETE },
  197.     { "Opaque", sizeof ("Opaque")-1, KW_OPAQUE },
  198.     { "optional", sizeof ("optional")-1, KW_OPTIONAL },
  199.     { "LAST-UPDATED", sizeof ("LAST-UPDATED")-1, LASTUPDATED },
  200.     { "ORGANIZATION", sizeof ("ORGANIZATION")-1, ORGANIZATION },
  201.     { "CONTACT-INFO", sizeof ("CONTACT-INFO")-1, CONTACTINFO },
  202.     { "MODULE-IDENTITY", sizeof ("MODULE-IDENTITY")-1, MODULEIDENTITY },
  203.     { "MODULE-COMPLIANCE", sizeof ("MODULE-COMPLIANCE")-1, COMPLIANCE },
  204.     { "DEFINITIONS", sizeof("DEFINITIONS")-1, DEFINITIONS},
  205.     { "END", sizeof("END")-1, END},
  206.     { "AUGMENTS", sizeof ("AUGMENTS")-1, AUGMENTS },
  207.     { "not-accessible", sizeof ("not-accessible")-1, NOACCESS },
  208.     { "write-only", sizeof ("write-only")-1, WRITEONLY },
  209.     { "NsapAddress", sizeof("NsapAddress")-1, NSAPADDRESS},
  210.     { "UNITS", sizeof("Units")-1, UNITS},
  211.     { "REFERENCE", sizeof("REFERENCE")-1, REFERENCE},
  212.     { "NUM-ENTRIES", sizeof("NUM-ENTRIES")-1, NUM_ENTRIES},
  213.     { "BITSTRING", sizeof("BITSTRING")-1, BITSTRING},
  214.     { "BIT", sizeof("BIT")-1, CONTINUE},
  215.     { "BITS", sizeof("BITS")-1, BITSTRING},
  216.     { "Counter64", sizeof("Counter64")-1, COUNTER64},
  217.     { "TimeTicks", sizeof ("TimeTicks")-1, TIMETICKS },
  218.     { "NOTIFICATION-TYPE", sizeof ("NOTIFICATION-TYPE")-1, NOTIFTYPE },
  219.     { "OBJECT-GROUP", sizeof ("OBJECT-GROUP")-1, OBJGROUP },
  220.     { "OBJECT-IDENTITY", sizeof ("OBJECT-IDENTITY")-1, OBJGROUP },
  221.     { "OBJECTIDENTIFIER", sizeof ("OBJECTIDENTIFIER")-1, OBJID },
  222.     { "OBJECT", sizeof ("OBJECT")-1, CONTINUE },
  223.     { "NetworkAddress", sizeof ("NetworkAddress")-1, NETADDR },
  224.     { "Gauge", sizeof ("Gauge")-1, GAUGE },
  225.     { "Gauge32", sizeof ("Gauge32")-1, GAUGE },
  226.     { "Unsigned32", sizeof ("Unsigned32")-1, GAUGE },
  227.     { "read-write", sizeof ("read-write")-1, READWRITE },
  228.     { "read-create", sizeof ("read-create")-1, READCREATE },
  229.     { "OCTETSTRING", sizeof ("OCTETSTRING")-1, OCTETSTR },
  230.     { "OCTET", sizeof ("OCTET")-1, CONTINUE },
  231.     { "OF", sizeof ("OF")-1, OF },
  232.     { "SEQUENCE", sizeof ("SEQUENCE")-1, SEQUENCE },
  233.     { "NULL", sizeof ("NULL")-1, NUL },
  234.     { "IpAddress", sizeof ("IpAddress")-1, IPADDR },
  235.     { "UInteger32", sizeof ("UInteger32")-1, UINTEGER32 },
  236.     { "INTEGER", sizeof ("INTEGER")-1, INTEGER },
  237.     { "Integer32", sizeof ("Integer32")-1, INTEGER32 },
  238.     { "Counter", sizeof ("Counter")-1, COUNTER },
  239.     { "Counter32", sizeof ("Counter32")-1, COUNTER },
  240.     { "read-only", sizeof ("read-only")-1, READONLY },
  241.     { "DESCRIPTION", sizeof ("DESCRIPTION")-1, DESCRIPTION },
  242.     { "INDEX", sizeof ("INDEX")-1, INDEX },
  243.     { "DEFVAL", sizeof ("DEFVAL")-1, DEFVAL },
  244.     { "deprecated", sizeof ("deprecated")-1, DEPRECATED },
  245.     { "SIZE", sizeof ("SIZE")-1, SIZE },
  246.     { "MAX-ACCESS", sizeof ("MAX-ACCESS")-1, ACCESS },
  247.     { "ACCESS", sizeof ("ACCESS")-1, ACCESS },
  248.     { "mandatory", sizeof ("mandatory")-1, MANDATORY },
  249.     { "current", sizeof ("current")-1, CURRENT },
  250.     { "STATUS", sizeof ("STATUS")-1, STATUS },
  251.     { "SYNTAX", sizeof ("SYNTAX")-1, SYNTAX },
  252.     { "OBJECT-TYPE", sizeof ("OBJECT-TYPE")-1, OBJTYPE },
  253.     { "TRAP-TYPE", sizeof ("TRAP-TYPE")-1, TRAPTYPE },
  254.     { "ENTERPRISE", sizeof ("ENTERPRISE")-1, ENTERPRISE },
  255.     { "BEGIN", sizeof ("BEGIN")-1, BEGIN },
  256.     { "IMPORTS", sizeof ("IMPORTS")-1, IMPORTS },
  257.     { "EXPORTS", sizeof ("EXPORTS")-1, EXPORTS },
  258.     { "accessible-for-notify", sizeof ("accessible-for-notify")-1, ACCNOTIFY },
  259.     { "TEXTUAL-CONVENTION", sizeof ("TEXTUAL-CONVENTION")-1, CONVENTION },
  260.     { "NOTIFICATION-GROUP", sizeof ("NOTIFICATION-GROUP")-1, NOTIFTYPE },
  261.     { "DISPLAY-HINT", sizeof ("DISPLAY-HINT")-1, DISPLAYHINT },
  262.     { "FROM", sizeof ("FROM")-1, FROM },
  263.     { "AGENT-CAPABILITIES", sizeof ("AGENT-CAPABILITIES")-1, CAPABILITIES },
  264.     { "MACRO", sizeof ("MACRO")-1, MACRO },
  265.     { "IMPLIED", sizeof ("IMPLIED")-1, IMPLIED },
  266.     { NULL }
  267. }; */
  268. static struct module_compatability *module_map_head;
  269. /* static struct module_compatability module_map[] = {
  270. { "RFC1065-SMI", "RFC1155-SMI", NULL, 0},
  271. { "RFC1066-MIB", "RFC1156-MIB", NULL, 0},
  272.           /* 'mib' -> 'mib-2'  
  273. { "RFC1156-MIB", "RFC1158-MIB", NULL, 0},
  274.           /* 'snmpEnableAuthTraps' -> 'snmpEnableAuthenTraps'  
  275. { "RFC1158-MIB",       "RFC1213-MIB", NULL, 0},
  276.           /* 'nullOID' -> 'zeroDotZero'  
  277. { "RFC1155-SMI", "SNMPv2-SMI", NULL, 0},
  278. { "RFC1213-MIB", "SNMPv2-SMI", "mib-2", 0},
  279. { "RFC1213-MIB", "SNMPv2-MIB", "sys", 3},
  280. { "RFC1213-MIB", "IF-MIB", "if", 2},
  281. { "RFC1213-MIB", "IP-MIB", "ip", 2},
  282. { "RFC1213-MIB", "IP-MIB", "icmp", 4},
  283. { "RFC1213-MIB", "TCP-MIB", "tcp", 3},
  284. { "RFC1213-MIB", "UDP-MIB", "udp", 3},
  285. { "RFC1213-MIB", "SNMPv2-SMI", "transmission", 0},
  286. { "RFC1213-MIB", "SNMPv2-MIB", "snmp", 4},
  287. { "RFC1271-MIB", "RMON-MIB", NULL, 0},
  288. { "RFC1286-MIB", "SOURCE-ROUTING-MIB", "dot1dSr", 7},
  289. { "RFC1286-MIB", "BRIDGE-MIB", NULL, 0},
  290. { "RFC1315-MIB", "FRAME-RELAY-DTE-MIB", NULL, 0},
  291. { "RFC1316-MIB", "CHARACTER-MIB", NULL, 0},
  292. }; */
  293. #define MODULE_NOT_FOUND 0
  294. #define MODULE_LOADED_OK 1
  295. #define MODULE_ALREADY_LOADED 2
  296. /* #define MODULE_LOAD_FAILED 3  */
  297. #define MODULE_LOAD_FAILED MODULE_NOT_FOUND
  298. #define HASHSIZE        32
  299. #define BUCKET(x)       (x & (HASHSIZE-1))
  300. #define NHASHSIZE    128
  301. #define NBUCKET(x)   (x & (NHASHSIZE-1))
  302. static struct tok      *buckets[HASHSIZE];
  303. char * module_name (int modid,  char *cp);
  304. static struct node *nbuckets[NHASHSIZE];
  305. static struct tree *tbuckets[NHASHSIZE];
  306. static struct module *module_head = NULL;
  307. struct node *orphan_nodes = NULL;
  308. struct tree   *tree_head = NULL;
  309. extern struct tree tree[];
  310. #define NUMBER_OF_ROOT_NODES 3
  311. static struct module_import root_imports[NUMBER_OF_ROOT_NODES];
  312. static int current_module = 0;
  313. static int     max_module = 0;
  314. static char *last_err_module = 0; /* no repeats on "Cannot find module..." */
  315. #ifdef UCD_SNMP
  316. static void tree_from_node(struct tree *tp, struct node *np);
  317. static struct index_list * copy_indexes(struct index_list *);
  318. static void build_translation_table (void);
  319. static void init_tree_roots (void);
  320. static void unlink_tree(struct tree *);
  321. static void read_import_replacements (const char *, struct module_import *);
  322. #endif
  323. static void do_subtree (struct tree *, struct node **);
  324. static void do_linkup (struct module *, struct node *);
  325. static void dump_module_list (void);
  326. static int get_token (FILE *, char *, int);
  327. static int parseQuoteString (FILE *, char *, int);
  328. static int tossObjectIdentifier (FILE *);
  329. static int  name_hash (const char *);
  330. static void init_node_hash (struct node *);
  331. static void print_error (const char *, const char *, int);
  332. static void free_tree (struct tree *);
  333. static void free_partial_tree (struct tree *, int);
  334. static void free_node (struct node *);
  335. static void merge_anon_children (struct tree *, struct tree *);
  336. static void unlink_tbucket(struct tree *);
  337. static int getoid (FILE *, struct subid_s *, int);
  338. static struct node *parse_objectid (FILE *, char *);
  339. static int get_tc (const char *, int, int *, struct enum_list **, struct range_list **, char **);
  340. static int get_tc_index (const char *, int);
  341. static struct enum_list *parse_enumlist (FILE *, struct enum_list **);
  342. static struct range_list *parse_ranges(FILE *fp, struct range_list **);
  343. static struct node *parse_asntype (FILE *, char *, int *, char *);
  344. static struct node *parse_objecttype (FILE *, char *);
  345. static struct node *parse_objectgroup (FILE *, char *);
  346. static struct node *parse_notificationDefinition (FILE *, char *);
  347. static struct node *parse_trapDefinition (FILE *, char *);
  348. static struct node *parse_compliance (FILE *, char *);
  349. static struct node *parse_capabilities(FILE *, char *);
  350. static struct node *parse_moduleIdentity (FILE *, char *);
  351. static struct node *parse_macro(FILE *, char *);
  352. static        void  parse_imports (FILE *);
  353. static struct node *parse (FILE *, struct node *);
  354. static int read_module_internal (const char *);
  355. static void read_module_replacements (const char *);
  356. static void  new_module  (const char *, const char *);
  357. static struct node *merge_parse_objectid (struct node *, FILE *, char *);
  358. static struct index_list *getIndexes(FILE *fp, struct index_list **);
  359. static void free_indexes(struct index_list **);
  360. static void free_ranges(struct range_list **);
  361. static void free_enums(struct enum_list **);
  362. static struct range_list * copy_ranges(struct range_list *);
  363. static struct enum_list  * copy_enums(struct enum_list *);
  364. /* backwards compatibility wrappers */
  365. void snmp_set_mib_errors(int err)
  366. {
  367.   ds_set_boolean(DS_LIBRARY_ID, DS_LIB_MIB_ERRORS, err);
  368. }
  369. void snmp_set_mib_warnings(int warn)
  370. {
  371.   ds_set_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS, warn);
  372. }
  373. void snmp_set_save_descriptions(int save)
  374. {
  375.   ds_set_boolean(DS_LIBRARY_ID, DS_LIB_SAVE_MIB_DESCRS, save);
  376. }
  377. void snmp_set_mib_comment_term(int save)
  378. {
  379.   /* 0=strict, 1=EOL terminated */
  380.   ds_set_boolean(DS_LIBRARY_ID, DS_LIB_MIB_COMMENT_TERM, save);
  381. }
  382. void snmp_set_mib_parse_label(int save)
  383. {
  384.   /* 0=strict, 1=underscore OK in label */
  385.   ds_set_boolean(DS_LIBRARY_ID, DS_LIB_MIB_PARSE_LABEL, save);
  386. }
  387. /* end wrappers */
  388. void snmp_mib_toggle_options_usage(const char *lead, FILE *outf) {
  389.   fprintf(outf, "%sMIBOPTS values:n", lead);
  390.   fprintf(outf, "%s    u: %sallow the usage of underlines in mib symbols.n",
  391.           lead, ((ds_get_boolean(DS_LIBRARY_ID, DS_LIB_MIB_PARSE_LABEL))?"dis":""));
  392.   fprintf(outf, "%s    c: %sallow the usage of "--" to terminate comments.n",
  393.           lead, ((ds_get_boolean(DS_LIBRARY_ID, DS_LIB_MIB_COMMENT_TERM))?"":"dis"));
  394.   fprintf(outf, "%s    d: %ssave the descriptions of the mib objects.n",
  395.           lead, ((ds_get_boolean(DS_LIBRARY_ID, DS_LIB_SAVE_MIB_DESCRS))?"don't ":""));
  396.   fprintf(outf, "%s    e: Disable mib errors of MIB symbols conflictsn",
  397.           lead);
  398.   fprintf(outf, "%s    w: Enable mib warnings of MIB symbols conflictsn",
  399.           lead);
  400.   fprintf(outf, "%s    W: Enable detailed warnings of MIB symbols conflictsn",
  401.           lead);
  402.   fprintf(outf, "%s    R: Replace MIB symbols from latest modulen",
  403.           lead);
  404. }
  405. char *snmp_mib_toggle_options(char *options) {
  406.   if (options) {
  407.     while(*options) {
  408.       switch(*options) {
  409.         case 'u':
  410.           ds_set_boolean(DS_LIBRARY_ID, DS_LIB_MIB_PARSE_LABEL, !ds_get_boolean(DS_LIBRARY_ID, DS_LIB_MIB_PARSE_LABEL));
  411.           break;
  412.         case 'c':
  413.           ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_MIB_COMMENT_TERM);
  414.           break;
  415.         case 'e':
  416.           ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_MIB_ERRORS);
  417.           break;
  418.         case 'w':
  419.           ds_set_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS, 1);
  420.           break;
  421.         case 'W':
  422.           ds_set_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS, 2);
  423.           break;
  424.         case 'd':
  425.           ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_SAVE_MIB_DESCRS);
  426.           break;
  427.         case 'R':
  428.           ds_toggle_boolean(DS_LIBRARY_ID, DS_LIB_MIB_REPLACE);
  429.           break;
  430.         default:
  431.           /* return at the unknown option */
  432.           return options;
  433.       }
  434.       options++;
  435.     }
  436.   }
  437.   return NULL;
  438. }
  439. static int
  440. name_hash(const char* name)
  441. {
  442.     int hash = 0;
  443.     const char *cp;
  444.     if (name) {
  445.       for(cp = name; *cp; cp++) {
  446.         hash += tolower(*cp);
  447.       }
  448.     }
  449.     return(hash);
  450. }
  451. static void
  452. init_node_hash(struct node *nodes)
  453. {
  454.      register struct node *np, *nextp;
  455.      register int hash;
  456.      memset(nbuckets, 0, sizeof(nbuckets));
  457.      for(np = nodes; np;){
  458.          nextp = np->next;
  459.          hash = NBUCKET(name_hash(np->parent));
  460.          np->next = nbuckets[hash];
  461.          nbuckets[hash] = np;
  462.          np = nextp;
  463.      }
  464. }
  465. static int erroneousMibs = 0;
  466. int get_mib_parse_error_count(void)
  467. {
  468.     return erroneousMibs;
  469. }
  470. static void
  471. print_error(const char *string,
  472.     const char *token,
  473.     int type)
  474. {
  475.     erroneousMibs++;
  476.     DEBUGMSGTL(("parse-mibs", "n"));
  477.     if (type == ENDOFFILE)
  478.         snmp_log(LOG_ERR, "%s (EOF): At line %d in %sn", string, Line,
  479.                 File);
  480.     else if (token && *token)
  481.         snmp_log(LOG_ERR, "%s (%s): At line %d in %sn", string, token,
  482.                 Line, File);
  483.     else
  484.         snmp_log(LOG_ERR, "%s: At line %d in %sn", string, Line, File);
  485. }
  486. extern char *strdup (const char*);
  487. static void
  488. print_module_not_found(const char *cp)
  489. {
  490.     if (!last_err_module || strcmp(cp, last_err_module))
  491.         print_error("Cannot find module", cp, CONTINUE);
  492.     if (last_err_module) free(last_err_module);
  493.     last_err_module = strdup(cp);
  494. }
  495. static struct node *
  496. alloc_node(int modid)
  497. {
  498.     struct node *np;
  499.     np = (struct node *) calloc(1, sizeof(struct node));
  500.     if (np) {
  501.         np->tc_index = -1;
  502.         np->modid = modid;
  503.     }
  504.     return np;
  505. }
  506. static void unlink_tbucket(struct tree *tp)
  507. {
  508.     int hash = NBUCKET(name_hash(tp->label));
  509.     struct tree *otp = NULL, *ntp = tbuckets[hash];
  510.     while (ntp && ntp != tp) {
  511. otp = ntp; ntp = ntp->next;
  512.     }
  513.     if (!ntp) snmp_log(LOG_EMERG, "Can't find %s in tbucketsn", tp->label);
  514.     else if (otp) otp->next = ntp->next;
  515.     else tbuckets[hash] = tp->next;
  516. }
  517. #ifdef UCD_SNMP
  518. static void unlink_tree(struct tree *tp)
  519. {
  520. struct tree *otp = NULL, *ntp = tp->parent->child_list;
  521.     while (ntp && ntp != tp) {
  522. otp = ntp; ntp = ntp->next_peer;
  523.     }
  524.     if (!ntp) snmp_log(LOG_EMERG, "Can't find %s in %s's childrenn",
  525. tp->label, tp->parent->label);
  526.     else if (otp) otp->next_peer = ntp->next_peer;
  527.     else tp->parent->child_list = tp->next_peer;
  528. }
  529. #endif
  530. static void
  531. free_partial_tree(struct tree *tp, int keep_label)
  532. {
  533.     if ( !tp)
  534.         return;
  535.     /* remove the data from this tree node */
  536.     free_enums(&tp->enums);
  537.     if (!keep_label)
  538. SNMP_FREE(tp->label);
  539.     SNMP_FREE(tp->hint);
  540.     SNMP_FREE(tp->units);
  541.     SNMP_FREE(tp->description);
  542. }
  543. /*
  544.  * free a tree node. Note: the node must already have been unlinked
  545.  * from the tree when calling this routine
  546.  */
  547. static void
  548. free_tree(struct tree *Tree)
  549. {
  550.     if (!Tree)
  551.         return;
  552.     unlink_tbucket(Tree);
  553.     free_partial_tree (Tree, FALSE);
  554. #ifdef UCD_SNMP
  555.     if (Tree->number_modules > 1 )
  556.         free(Tree->module_list);
  557. #endif
  558.     free (Tree);
  559. }
  560. static void
  561. free_node(struct node *np)
  562. {
  563.     if ( !np) return;
  564.     free_enums(&np->enums);
  565.     free_ranges(&np->ranges);
  566.     free_indexes(&np->indexes);
  567.     if (np->label) free(np->label);
  568.     if (np->hint) free(np->hint);
  569.     if (np->units) free(np->units);
  570.     if (np->description) free(np->description);
  571.     if (np->parent) free(np->parent);
  572.     free(np);
  573. }
  574. #ifdef TEST
  575. static void
  576. print_nodes(FILE *fp,
  577.     struct node *root)
  578. {
  579. extern void xmalloc_stats (FILE *);
  580.     struct enum_list *ep;
  581.     struct index_list *ip;
  582.     struct range_list *rp;
  583.     struct node *np;
  584.     for(np = root; np; np = np->next){
  585.         fprintf(fp, "%s ::= { %s %ld } (%d)n", np->label, np->parent,
  586.                 np->subid, np->type);
  587.         if (np->tc_index >= 0)
  588.             fprintf(fp, "  TC = %sn", tclist[np->tc_index].descriptor);
  589.         if (np->enums){
  590.             fprintf(fp, "  Enums: n");
  591.             for(ep = np->enums; ep; ep = ep->next){
  592.                 fprintf(fp, "    %s(%d)n", ep->label, ep->value);
  593.             }
  594.         }
  595.         if (np->ranges){
  596.             fprintf(fp, "  Ranges: n");
  597.             for(rp = np->ranges; rp; rp = rp->next){
  598.                 fprintf(fp, "    %d..%dn", rp->low, rp->high);
  599.             }
  600.         }
  601.         if (np->indexes){
  602.             fprintf(fp, "  Indexes: n");
  603.             for(ip = np->indexes; ip; ip = ip->next){
  604.                 fprintf(fp, "    %sn", ip->ilabel);
  605.             }
  606.         }
  607.         if (np->hint)
  608.             fprintf(fp, "  Hint: %sn", np->hint);
  609.         if (np->units)
  610.             fprintf(fp, "  Units: %sn", np->units);
  611.     }
  612. }
  613. #endif
  614. void
  615. print_subtree(FILE *f,
  616.       struct tree *tree,
  617.       int count)
  618. {
  619. #ifdef UCD_SNMP
  620.     struct tree *tp;
  621.     int i;
  622.     char modbuf[256];
  623.     for(i = 0; i < count; i++)
  624.         fprintf(f, "  ");
  625.     fprintf(f, "Children of %s(%ld):n", tree->label, tree->subid);
  626.     count++;
  627.     for(tp = tree->child_list; tp; tp = tp->next_peer){
  628.         for(i = 0; i < count; i++)
  629.             fprintf(f, "  ");
  630.         fprintf(f, "%s:%s(%ld) type=%d",
  631.                 module_name(tp->module_list[0], modbuf),
  632.                 tp->label, tp->subid, tp->type);
  633.         if (tp->tc_index != -1) fprintf(f, " tc=%d", tp->tc_index);
  634.         if (tp->hint) fprintf(f, " hint=%s", tp->hint);
  635.         if (tp->units) fprintf(f, " units=%s", tp->units);
  636. if (tp->number_modules > 1) {
  637.     fprintf(f, " modules:");
  638.     for (i = 1; i < tp->number_modules; i++)
  639. fprintf(f, " %s", module_name(tp->module_list[i], modbuf));
  640. }
  641. fprintf(f, "n");
  642.     }
  643.     for(tp = tree->child_list; tp; tp = tp->next_peer){
  644.         if (tp->child_list)
  645.             print_subtree(f, tp, count);
  646.     }
  647. #endif
  648. }
  649. void
  650. print_ascii_dump_tree(FILE *f,
  651.       struct tree *tree,
  652.       int count)
  653. {
  654.     struct tree *tp;
  655.     count++;
  656.     for(tp = tree->child_list; tp; tp = tp->next_peer){
  657.           fprintf(f, "%s OBJECT IDENTIFIER ::= { %s %ld }n", tp->label, tree->label, tp->subid);
  658.     }
  659.     for(tp = tree->child_list; tp; tp = tp->next_peer){
  660.         if (tp->child_list)
  661.             print_ascii_dump_tree(f, tp, count);
  662.     }
  663. }
  664. #ifdef UCD_SNMP
  665. static int translation_table[256];
  666. static void
  667. build_translation_table()
  668. {
  669.     int count;
  670.     for(count = 0; count < 256; count++){
  671.         switch(count){
  672.             case OBJID:
  673.                 translation_table[count] = TYPE_OBJID;
  674.                 break;
  675.             case OCTETSTR:
  676.                 translation_table[count] = TYPE_OCTETSTR;
  677.                 break;
  678.             case INTEGER:
  679.                 translation_table[count] = TYPE_INTEGER;
  680.                 break;
  681.             case NETADDR:
  682.                 translation_table[count] = TYPE_IPADDR;
  683.                 break;
  684.             case IPADDR:
  685.                 translation_table[count] = TYPE_IPADDR;
  686.                 break;
  687.             case COUNTER:
  688.                 translation_table[count] = TYPE_COUNTER;
  689.                 break;
  690.             case GAUGE:
  691.                 translation_table[count] = TYPE_GAUGE;
  692.                 break;
  693.             case TIMETICKS:
  694.                 translation_table[count] = TYPE_TIMETICKS;
  695.                 break;
  696.             case KW_OPAQUE:
  697.                 translation_table[count] = TYPE_OPAQUE;
  698.                 break;
  699.             case NUL:
  700.                 translation_table[count] = TYPE_NULL;
  701.                 break;
  702.             case COUNTER64:
  703.                 translation_table[count] = TYPE_COUNTER64;
  704.                 break;
  705.             case BITSTRING:
  706.                 translation_table[count] = TYPE_BITSTRING;
  707.                 break;
  708.             case NSAPADDRESS:
  709.                 translation_table[count] = TYPE_NSAPADDRESS;
  710.                 break;
  711.             case UINTEGER32:
  712.                 translation_table[count] = TYPE_UINTEGER;
  713.                 break;
  714.             default:
  715.                 translation_table[count] = TYPE_OTHER;
  716.                 break;
  717.         }
  718.     }
  719. }
  720. #endif
  721. /*static void
  722. init_tree_roots()
  723. {
  724.     struct tree *tp, *lasttp;
  725.     int  base_modid;
  726.     int  hash;
  727.     base_modid = which_module("SNMPv2-SMI");
  728.     if (base_modid == -1 )
  729.         base_modid = which_module("RFC1155-SMI");
  730.     if (base_modid == -1 )
  731.         base_modid = which_module("RFC1213-MIB");
  732.     // build root node 
  733.     tp = (struct tree *) calloc(1, sizeof(struct tree));
  734.     if (tp == NULL) return;
  735.     tp->label = strdup("joint-iso-ccitt");
  736. #ifdef UCD_SNMP
  737.     tp->modid = base_modid;
  738.     tp->number_modules = 1;
  739.     tp->module_list = &(tp->modid);
  740.     tp->tc_index = -1;
  741. #endif
  742.     tp->subid = 2;
  743.     set_function(tp); // from mib.c 
  744.     hash = NBUCKET(name_hash(tp->label));
  745.     tp->next = tbuckets[hash];
  746.     tbuckets[hash] = tp;
  747.     lasttp = tp;
  748.     root_imports[0].label = strdup( tp->label );
  749.     root_imports[0].modid = base_modid;
  750.     // build root node 
  751.     tp = (struct tree *) calloc(1, sizeof(struct tree));
  752.     if (tp == NULL) return;
  753.     tp->next_peer = lasttp;
  754.     tp->label = strdup("ccitt");
  755. #ifdef UCD_SNMP
  756.     tp->modid = base_modid;
  757.     tp->number_modules = 1;
  758.     tp->module_list = &(tp->modid);
  759.     tp->tc_index = -1;
  760. #endif
  761.     tp->subid = 0;
  762.     set_function(tp); // from mib.c 
  763.     hash = NBUCKET(name_hash(tp->label));
  764.     tp->next = tbuckets[hash];
  765.     tbuckets[hash] = tp;
  766.     lasttp = tp;
  767.     root_imports[1].label = strdup( tp->label );
  768.     root_imports[1].modid = base_modid;
  769.     // build root node 
  770.     tp = (struct tree *) calloc(1, sizeof(struct tree));
  771.     if (tp == NULL) return;
  772.     tp->next_peer = lasttp;
  773.     tp->label = strdup("iso");
  774. #ifdef UCD_SNMP
  775.     tp->modid = base_modid;
  776.     tp->number_modules = 1;
  777.     tp->module_list = &(tp->modid);
  778.     tp->tc_index = -1;
  779. #endif
  780.     tp->subid = 1;
  781.     set_function(tp); // from mib.c
  782.     hash = NBUCKET(name_hash(tp->label));
  783.     tp->next = tbuckets[hash];
  784.     tbuckets[hash] = tp;
  785.     lasttp = tp;
  786.     root_imports[2].label = strdup( tp->label );
  787.     root_imports[2].modid = base_modid;
  788.     tree_head = tp;
  789. }
  790. */
  791. #ifdef STRICT_MIB_PARSEING
  792. #define label_compare strcasecmp
  793. #else
  794. #define label_compare strcmp
  795. #endif
  796. struct tree *
  797. find_tree_node(const char *name,
  798.        int modid)
  799. {
  800.     struct tree *tp, *headtp;
  801.   #ifdef UCD_SNMP
  802.     int count, *int_p, i;
  803.   #endif
  804.     if (!name || !*name)
  805. return(NULL);
  806.     headtp = tbuckets[NBUCKET(name_hash(name))];
  807.     for ( tp = headtp ; tp ; tp = tp->next ) {
  808.         if ( !label_compare(tp->label, name) ) {
  809.             if ( modid == -1 ) /* Any module */
  810.                 return(tp);
  811. #ifdef UCD_SNMP
  812.             for (int_p = tp->module_list, count=0 ;
  813.                        count < tp->number_modules ;
  814.                        ++count, ++int_p )
  815.                 if ( *int_p == modid )
  816. #endif
  817.                     return(tp);
  818.         }
  819.     }
  820.     return(NULL);
  821. }
  822. /* computes a value which represents how close name1 is to name2.
  823.  * high scores mean a worse match.
  824.  * (yes, the algorithm sucks!)
  825.  */
  826. #define MAX_BAD 0xffffff
  827. u_int
  828. compute_match(const char *search_base, const char *key) {
  829. #if defined(HAVE_REGEX_H) && defined(HAVE_REGCOMP)
  830.     int rc;
  831.     regex_t parsetree;
  832.     regmatch_t pmatch;
  833.     rc=regcomp(&parsetree, key, REG_ICASE | REG_EXTENDED);
  834.     if (rc == 0)
  835.         rc=regexec(&parsetree, search_base, 1, &pmatch, 0);
  836.     regfree(&parsetree);
  837.     if (rc == 0) {
  838.         /* found */
  839.         return pmatch.rm_so;
  840.     }
  841. #else /* use our own wildcard matcher */
  842.     /* first find the longest matching substring (ick) */
  843.     char *first = NULL, *result = NULL, *entry;
  844.     const char *position;
  845.     char *newkey = strdup(key);
  846.     entry = strtok( newkey, "*" );
  847.     position = search_base;
  848.     while ( entry ) {
  849.         result = strcasestr(position, entry);
  850.         if (result == NULL) {
  851.             free(newkey);
  852.             return MAX_BAD;
  853.         }
  854.         if (first == NULL)
  855.             first = result;
  856.         position = result + strlen(entry);
  857.         entry = strtok( NULL, "*" );
  858.     }
  859.     free(newkey);
  860.     if (result)
  861.         return(first-search_base);
  862. #endif
  863.     /* not found */
  864.     return MAX_BAD;
  865. }
  866. /*
  867.  * Find the tree node that best matches the pattern string.
  868.  * Use the "reported" flag such that only one match
  869.  * is attempted for every node.
  870.  *
  871.  * Warning! This function may recurse.
  872.  *
  873.  * Caller _must_ invoke clear_tree_flags before first call
  874.  * to this function.  This function may be called multiple times
  875.  * to ensure that the entire tree is traversed.
  876.  */
  877. struct tree *
  878. find_best_tree_node(const char *pattrn, struct tree *tree_top, u_int *match)
  879. {
  880.     struct tree *tp, *best_so_far = NULL, *retptr;
  881.     u_int old_match=MAX_BAD, new_match=MAX_BAD;
  882.     if (!pattrn || !*pattrn)
  883. return(NULL);
  884.     if (!tree_top)
  885.         tree_top = get_tree_head();
  886.     for ( tp = tree_top ; tp ; tp=tp->next_peer ) {
  887.         if (!tp->reported)
  888.             new_match = compute_match(tp->label, pattrn);
  889.         tp->reported = 1;
  890.         if (new_match < old_match) {
  891.             best_so_far = tp;
  892.             old_match = new_match;
  893.         }
  894.         if (new_match == 0)
  895.             break;  /* this is the best result we can get */
  896.         if (tp->child_list) {
  897.             retptr = find_best_tree_node(pattrn, tp->child_list, &new_match);
  898.             if (new_match < old_match) {
  899.                 best_so_far = retptr;
  900.                 old_match = new_match;
  901.             }
  902.             if (new_match == 0)
  903.                 break;  /* this is the best result we can get */
  904.         }
  905.     }
  906.     if (match)
  907.         *match = old_match;
  908.     return(best_so_far);
  909. }
  910. static void
  911. merge_anon_children(struct tree *tp1,
  912.     struct tree *tp2)
  913. /* NB: tp1 is the 'anonymous' node */
  914. {
  915.     struct tree *child1, *child2, *previous;
  916.     for ( child1 = tp1->child_list ; child1 ; ) {
  917.         for ( child2 = tp2->child_list, previous = NULL ;
  918.               child2 ; previous = child2, child2 = child2->next_peer ) {
  919.             if ( child1->subid == child2->subid ) {
  920. /*
  921.  * Found 'matching' children,
  922.  *  so merge them
  923.  */
  924. if ( !strncmp( child1->label, ANON, ANON_LEN)) {
  925.                     merge_anon_children( child1, child2 );
  926.                     child1->child_list = NULL;
  927.                     previous = child1; /* Finished with 'child1' */
  928.                     child1 = child1->next_peer;
  929.                     free_tree( previous );
  930.                     goto next;
  931.                 }
  932. else if ( !strncmp( child2->label, ANON, ANON_LEN)) {
  933.                     merge_anon_children( child2, child1 );
  934.                     if ( previous )
  935.                          previous->next_peer = child2->next_peer;
  936.                     else
  937.                          tp2->child_list = child2->next_peer;
  938.                     free_tree(child2);
  939.                     previous = child1; /* Move 'child1' to 'tp2' */
  940.                     child1 = child1->next_peer;
  941.                     previous->next_peer = tp2->child_list;
  942.                     tp2->child_list = previous;
  943.                     for ( previous = tp2->child_list ;
  944.                           previous ;
  945.                           previous = previous->next_peer )
  946.                                 previous->parent = tp2;
  947.                     goto next;
  948.                 }
  949. else if ( !label_compare( child1->label, child2->label) ) {
  950.             if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  951.         snmp_log(LOG_WARNING, "Warning: %s.%ld is both %s and %s (%s)n",
  952.         tp2->label, child1->subid,
  953.                                 child1->label, child2->label, File);
  954.                     continue;
  955.                 }
  956.                 else {
  957. /*
  958.  * Two copies of the same node.
  959.  * 'child2' adopts the children of 'child1'
  960.  */
  961.                     if ( child2->child_list ) {
  962.                         for ( previous = child2->child_list ;
  963.                               previous->next_peer ;
  964.                               previous = previous->next_peer )
  965.                                   ; /* Find the end of the list */
  966.                         previous->next_peer = child1->child_list;
  967.                     }
  968.                     else
  969.                         child2->child_list = child1->child_list;
  970.                     for ( previous = child1->child_list ;
  971.                           previous ;
  972.                           previous = previous->next_peer )
  973.                                   previous->parent = child2;
  974.                     child1->child_list = NULL;
  975.                     previous = child1; /* Finished with 'child1' */
  976.                     child1 = child1->next_peer;
  977.                     free_tree( previous );
  978.                     goto next;
  979.                 }
  980.             }
  981.         }
  982. /*
  983.  * If no match, move 'child1' to 'tp2' child_list
  984.  */
  985.         if ( child1 ) {
  986.             previous = child1;
  987.             child1 = child1->next_peer;
  988.             previous->parent = tp2;
  989.             previous->next_peer = tp2->child_list;
  990.             tp2->child_list = previous;
  991.         }
  992.       next:;
  993.     }
  994. }
  995. /*
  996.  * Find all the children of root in the list of nodes.  Link them into the
  997.  * tree and out of the nodes list.
  998.  */
  999. static void
  1000. do_subtree(struct tree *root,
  1001.    struct node **nodes)
  1002. {
  1003. #ifdef UCD_SNMP
  1004.     register struct tree *tp, *anon_tp=NULL;
  1005.     register struct node *np, **headp;
  1006.     struct node *oldnp = NULL, *child_list = NULL, *childp = NULL;
  1007.     int hash;
  1008.     int *int_p;
  1009.     tp = root;
  1010.     headp = &nbuckets[NBUCKET(name_hash(tp->label))];
  1011.     /*
  1012.      * Search each of the nodes for one whose parent is root, and
  1013.      * move each into a separate list.
  1014.      */
  1015.     for(np = *headp; np; np = np->next){
  1016.         if ( !label_compare(tp->label, np->parent)){
  1017.             /* take this node out of the node list */
  1018.             if (oldnp == NULL){
  1019.                 *headp = np->next;  /* fix root of node list */
  1020.             } else {
  1021.                 oldnp->next = np->next; /* link around this node */
  1022.             }
  1023.             if (child_list) childp->next = np;
  1024.             else child_list = np;
  1025.             childp = np;
  1026.         }
  1027.         else {
  1028.     oldnp = np;
  1029.         }
  1030.     }
  1031.     if (childp) childp->next = NULL;
  1032.     /*
  1033.      * Take each element in the child list and place it into the tree.
  1034.      */
  1035.     for(np = child_list; np; np = np->next){
  1036. anon_tp = NULL;
  1037.         tp = root->child_list;
  1038.         while (tp)
  1039.             if (tp->subid == np->subid) break;
  1040.             else tp = tp->next_peer;
  1041.         if (tp) {
  1042.     if (!label_compare (tp->label, np->label)) {
  1043.     /* Update list of modules */
  1044.                 int_p = (int *) malloc((tp->number_modules+1) * sizeof(int));
  1045.                 if (int_p == NULL) return;
  1046.                 memcpy(int_p, tp->module_list, tp->number_modules*sizeof(int));
  1047.                 int_p[tp->number_modules] = np->modid;
  1048.                 if (tp->number_modules > 1 )
  1049.                    free(tp->module_list);
  1050.                 ++tp->number_modules;
  1051.                 tp->module_list = int_p;
  1052. if ( ds_get_boolean(DS_LIBRARY_ID, DS_LIB_MIB_REPLACE) ) {
  1053.     /* Replace from node */
  1054.     tree_from_node(tp,np);
  1055. }
  1056.     /* Handle children */
  1057. do_subtree(tp, nodes);
  1058. continue;
  1059.             }
  1060.             if (!strncmp( np->label, ANON, ANON_LEN) ||
  1061.                 !strncmp( tp->label, ANON, ANON_LEN)) {
  1062.                 anon_tp = tp; /* Need to merge these two trees later */
  1063.             }
  1064.     else if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  1065. snmp_log(LOG_WARNING, "Warning: %s.%ld is both %s and %s (%s)n",
  1066. root->label, np->subid, tp->label, np->label, File);
  1067. }
  1068.         tp = (struct tree *) calloc(1, sizeof(struct tree));
  1069.         if (tp == NULL) return;
  1070.         tp->parent = root;
  1071.         tp->modid = np->modid;
  1072.         tp->number_modules = 1;
  1073.         tp->module_list = &(tp->modid);
  1074.         tree_from_node(tp, np);
  1075.         tp->next_peer = root->child_list;
  1076.         root->child_list = tp;
  1077.         hash = NBUCKET(name_hash(tp->label));
  1078.         tp->next = tbuckets[hash];
  1079.         tbuckets[hash] = tp;
  1080. /*      if (tp->type == TYPE_OTHER) */
  1081.             do_subtree(tp, nodes);      /* recurse on this child if it isn't
  1082.                                            an end node */
  1083.         if ( anon_tp ) {
  1084.             if (!strncmp( tp->label, ANON, ANON_LEN)) {
  1085. /*
  1086.  * The new node is anonymous,
  1087.  *  so merge it with the existing one.
  1088.  */
  1089.                 merge_anon_children( tp, anon_tp );
  1090. /* unlink and destroy tp */
  1091. unlink_tree(tp);
  1092. free_tree(tp);
  1093.             }
  1094.             else if (!strncmp( anon_tp->label, ANON, ANON_LEN)) {
  1095. struct tree *ntp;
  1096. /*
  1097.  * The old node was anonymous,
  1098.  *  so merge it with the existing one,
  1099.  *  and fill in the full information.
  1100.  */
  1101.                 merge_anon_children( anon_tp, tp );
  1102. /* unlink anon_tp from the hash */
  1103. unlink_tbucket(anon_tp);
  1104. /* get rid of old contents of anon_tp */
  1105.                 free_partial_tree(anon_tp, FALSE);
  1106. /* put in the current information */
  1107.                 anon_tp->label = tp->label;
  1108.                 anon_tp->child_list = tp->child_list;
  1109.                 anon_tp->modid = tp->modid;
  1110.                 anon_tp->tc_index = tp->tc_index;
  1111.                 anon_tp->type = tp->type;
  1112.                 anon_tp->enums = tp->enums;
  1113.                 anon_tp->indexes = tp->indexes;
  1114.                 anon_tp->ranges = tp->ranges;
  1115.                 anon_tp->hint = tp->hint;
  1116.                 anon_tp->units = tp->units;
  1117.                 anon_tp->description = tp->description;
  1118. anon_tp->parent = tp->parent;
  1119.                 set_function(anon_tp);
  1120. /* update parent pointer in moved children */
  1121. ntp = anon_tp->child_list;
  1122. while (ntp) {
  1123.     ntp->parent = anon_tp;
  1124.     ntp = ntp->next_peer;
  1125. }
  1126. /* hash in anon_tp in its new place */
  1127. hash = NBUCKET(name_hash(anon_tp->label));
  1128. anon_tp->next = tbuckets[hash];
  1129. tbuckets[hash] = anon_tp;
  1130. /* unlink and destroy tp */
  1131. unlink_tbucket(tp);
  1132. unlink_tree(tp);
  1133. free(tp);
  1134.             }
  1135.             else {
  1136.                 /* Uh?  One of these two should have been anonymous! */
  1137.         if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  1138.     snmp_log(LOG_WARNING,
  1139.                              "Warning: expected anonymous node (either %s or %s) in %sn",
  1140.                              tp->label, anon_tp->label, File);
  1141.             }
  1142.             anon_tp = NULL;
  1143.         }
  1144.     }
  1145.     /* free all nodes that were copied into tree */
  1146.     oldnp = NULL;
  1147.     for(np = child_list; np; np = np->next){
  1148.         if (oldnp)
  1149.             free_node(oldnp);
  1150.         oldnp = np;
  1151.     }
  1152.     if (oldnp)
  1153.         free_node(oldnp);
  1154. #endif
  1155. }
  1156. static void do_linkup(struct module *mp,
  1157.       struct node *np)
  1158. {
  1159.     struct module_import *mip;
  1160.     struct node *onp;
  1161.     struct tree *tp;
  1162.     int i;
  1163. /*
  1164.  * All modules implicitly import
  1165.  *   the roots of the tree
  1166.  */
  1167.     if (snmp_get_do_debugging() > 1) dump_module_list();
  1168.     DEBUGMSGTL(("parse-mibs", "Processing IMPORTS for module %d %sn", mp->modid, mp->name));
  1169.     if ( mp->no_imports == 0 ) {
  1170. mp->no_imports = NUMBER_OF_ROOT_NODES;
  1171. mp->imports = root_imports;
  1172.     }
  1173. /*
  1174.  * Build the tree
  1175.  */
  1176.     init_node_hash( np );
  1177.     for ( i=0, mip=mp->imports ; i < mp->no_imports ; ++i, ++mip ) {
  1178. char modbuf[256];
  1179. DEBUGMSGTL(("parse-mibs", "  Processing import: %sn", mip->label));
  1180. if (get_tc_index( mip->label, mip->modid ) != -1)
  1181.     continue;
  1182. tp = find_tree_node( mip->label, mip->modid );
  1183. if (!tp) {
  1184.     if (mip->modid != -1)
  1185. snmp_log(LOG_WARNING, "Did not find '%s' in module %s (%s)n",
  1186.                          mip->label, module_name(mip->modid, modbuf), File);
  1187.     continue;
  1188. }
  1189. do_subtree( tp, &np );
  1190.     }
  1191. /*
  1192.  * If any nodes left over,
  1193.  *   check that they're not the result of a "fully qualified"
  1194.  *   name, and then add them to the list of orphans
  1195.  */
  1196.     if (!np) return;
  1197.     for ( tp = tree_head ; tp ; tp=tp->next_peer )
  1198.         do_subtree( tp, &np );
  1199.     if (!np) return;
  1200.     for ( np = orphan_nodes ; np && np->next ; np = np->next )
  1201. ; /* find the end of the orphan list */
  1202.     for (i = 0; i < NHASHSIZE; i++)
  1203. if ( nbuckets[i] ) {
  1204.     if ( orphan_nodes )
  1205. onp = np->next = nbuckets[i];
  1206.     else
  1207. onp = orphan_nodes = nbuckets[i];
  1208.     nbuckets[i] = NULL;
  1209.     while (onp) {
  1210. if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  1211.     snmp_log(LOG_WARNING,
  1212.                              "Unlinked OID in %s: %s ::= { %s %ld }n",
  1213.                              (mp->name ? mp->name : "<no module>"),
  1214.                              (onp->label ? onp->label : "<no label>"),
  1215.                              (onp->parent ? onp->parent : "<no parent>"),
  1216.                              onp->subid);
  1217. np = onp;
  1218. onp = onp->next;
  1219.     }
  1220. }
  1221.     return;
  1222. }
  1223. /*
  1224.  * Takes a list of the form:
  1225.  * { iso org(3) dod(6) 1 }
  1226.  * and creates several nodes, one for each parent-child pair.
  1227.  * Returns 0 on error.
  1228.  */
  1229. static int
  1230. getoid(FILE *fp,
  1231.        struct subid_s *id, /* an array of subids */
  1232.        int length)  /* the length of the array */
  1233. {
  1234.     register int count;
  1235.     int type;
  1236.     char token[MAXTOKEN];
  1237.     if ((type = get_token(fp, token, MAXTOKEN)) != LEFTBRACKET){
  1238.         print_error("Expected "{"", token, type);
  1239.         return 0;
  1240.     }
  1241.     type = get_token(fp, token, MAXTOKEN);
  1242.     for(count = 0; count < length; count++, id++){
  1243.         id->label = NULL;
  1244.         id->modid = current_module;
  1245.         id->subid = -1;
  1246.         if (type == RIGHTBRACKET){
  1247.             return count;
  1248.         } else if (type != LABEL && type != NUMBER){
  1249.             print_error("Not valid for object identifier", token, type);
  1250.             return 0;
  1251.         }
  1252.         if (type == LABEL){
  1253.             /* this entry has a label */
  1254.             id->label = strdup(token);
  1255.             type = get_token(fp, token, MAXTOKEN);
  1256.             if (type == LEFTPAREN){
  1257.                 type = get_token(fp, token, MAXTOKEN);
  1258.                 if (type == NUMBER){
  1259.                     id->subid = atoi(token);
  1260.                     if ((type = get_token(fp, token, MAXTOKEN)) != RIGHTPAREN){
  1261.                         print_error("Expected a closing parenthesis",
  1262.                                     token, type);
  1263.                         return 0;
  1264.                     }
  1265.                 } else {
  1266.                     print_error("Expected a number", token, type);
  1267.                     return 0;
  1268.                 }
  1269.             } else {
  1270.                 continue;
  1271.             }
  1272.         } else if (type == NUMBER) {
  1273.             /* this entry  has just an integer sub-identifier */
  1274.             id->subid = atoi(token);
  1275.         }
  1276. else {
  1277.     print_error("Expected label or number", token, type);
  1278.     return 0;
  1279. }
  1280.         type = get_token(fp, token, MAXTOKEN);
  1281.     }
  1282.     print_error ("Too long OID", token, type);
  1283.     return 0;
  1284. }
  1285. /*
  1286.  * Parse a sequence of object subidentifiers for the given name.
  1287.  * The "label OBJECT IDENTIFIER ::=" portion has already been parsed.
  1288.  *
  1289.  * The majority of cases take this form :
  1290.  * label OBJECT IDENTIFIER ::= { parent 2 }
  1291.  * where a parent label and a child subidentifier number are specified.
  1292.  *
  1293.  * Variations on the theme include cases where a number appears with
  1294.  * the parent, or intermediate subidentifiers are specified by label,
  1295.  * by number, or both.
  1296.  *
  1297.  * Here are some representative samples :
  1298.  * internet        OBJECT IDENTIFIER ::= { iso org(3) dod(6) 1 }
  1299.  * mgmt            OBJECT IDENTIFIER ::= { internet 2 }
  1300.  * rptrInfoHealth  OBJECT IDENTIFIER ::= { snmpDot3RptrMgt 0 4 }
  1301.  *
  1302.  * Here is a very rare form :
  1303.  * iso             OBJECT IDENTIFIER ::= { 1 }
  1304.  *
  1305.  * Returns NULL on error.  When this happens, memory may be leaked.
  1306.  */
  1307. static struct node *
  1308. parse_objectid(FILE *fp,
  1309.        char *name)
  1310. {
  1311.     register int count;
  1312.     register struct subid_s *op, *nop;
  1313.     int length;
  1314.     struct subid_s loid[32];
  1315.     struct node *np, *root = NULL, *oldnp = NULL;
  1316.     struct tree *tp;
  1317.     if ((length = getoid(fp, loid, 32)) == 0){
  1318.         print_error("Bad object identifier", NULL, CONTINUE);
  1319.         return NULL;
  1320.     }
  1321.     /*
  1322.      * Handle numeric-only object identifiers,
  1323.      *  by labelling the first sub-identifier
  1324.      */
  1325.     op = loid;
  1326.     if ( !op->label )
  1327.       for ( tp = tree_head ; tp ; tp=tp->next_peer )
  1328.         if ( (int)tp->subid == op->subid ) {
  1329.             op->label = strdup(tp->label);
  1330.             break;
  1331.         }
  1332.     /*
  1333.      * Handle  "label OBJECT-IDENTIFIER ::= { subid }"
  1334.      */
  1335.     if (length == 1) {
  1336.         op = loid;
  1337.         np = alloc_node(op->modid);
  1338.         if (np == NULL) return(NULL);
  1339.         np->subid = op->subid;
  1340.         np->label = strdup(name);
  1341.         if (op->label) free(op->label);
  1342.         return np;
  1343.     }
  1344.     /*
  1345.      * For each parent-child subid pair in the subid array,
  1346.      * create a node and link it into the node list.
  1347.      */
  1348.     for(count = 0, op = loid, nop=loid+1; count < (length - 1);
  1349.       count++, op++, nop++){
  1350.         /* every node must have parent's name and child's name or number */
  1351. /* XX the next statement is always true -- does it matter ?? */
  1352.         if (op->label && (nop->label || (nop->subid != -1))){
  1353.             np = alloc_node(nop->modid);
  1354.             if (np == NULL) return(NULL);
  1355.             if (root == NULL) root = np;
  1356.             np->parent = strdup (op->label);
  1357.             if (count == (length - 2)) {
  1358.                 /* The name for this node is the label for this entry */
  1359.                 np->label = strdup (name);
  1360.             }
  1361.             else {
  1362.                 if (!nop->label) {
  1363.                     nop->label = (char *) malloc(20 + ANON_LEN);
  1364.                     if (nop->label == NULL) return(NULL);
  1365.                     sprintf(nop->label, "%s%d", ANON, anonymous++);
  1366.                 }
  1367.                 np->label = strdup (nop->label);
  1368.             }
  1369.             if (nop->subid != -1)
  1370.                 np->subid = nop->subid;
  1371.             else
  1372.                 print_error("Warning: This entry is pretty silly",
  1373.     np->label, CONTINUE);
  1374.             /* set up next entry */
  1375.             if (oldnp) oldnp->next = np;
  1376.             oldnp = np;
  1377.         } /* end if(op->label... */
  1378.     }
  1379.     /* free the loid array */
  1380.     for(count = 0, op = loid; count < length; count++, op++){
  1381.         if (op->label)
  1382.             free(op->label);
  1383.     }
  1384.     return root;
  1385. }
  1386. static int
  1387. get_tc(const char *descriptor,
  1388.        int modid,
  1389.        int *tc_index,
  1390.        struct enum_list **ep,
  1391.        struct range_list **rp,
  1392.        char **hint)
  1393. {
  1394.     int i;
  1395.     struct tc *tcp;
  1396.     i = get_tc_index(descriptor, modid);
  1397.     if (tc_index) *tc_index = i;
  1398.     if (i != -1)
  1399.       {
  1400.   tcp = &tclist[i];
  1401. if (ep) {
  1402.     free_enums(ep);
  1403.     *ep = copy_enums(tcp->enums);
  1404. }
  1405. if (rp) {
  1406.     free_ranges(rp);
  1407.     *rp = copy_ranges(tcp->ranges);
  1408. }
  1409. if (hint) {
  1410.     if (*hint) free(*hint);
  1411.     *hint = (tcp->hint ? strdup(tcp->hint) : NULL);
  1412. }
  1413. return tcp->type;
  1414.       }
  1415.     return LABEL;
  1416. }
  1417. /* return index into tclist of given TC descriptor
  1418.    return -1 if not found
  1419.  */
  1420. static int
  1421. get_tc_index(const char *descriptor,
  1422.      int modid)
  1423. {
  1424.     int i;
  1425.     struct tc *tcp;
  1426.     struct module *mp;
  1427.     struct module_import *mip;
  1428. /*
  1429.  * Check that the descriptor isn't imported
  1430.  *  by searching the import list
  1431.  */
  1432.     for ( mp = module_head ; mp ; mp = mp->next )
  1433.          if ( mp->modid == modid )
  1434.              break;
  1435.     if ( mp )
  1436.          for ( i=0, mip=mp->imports ; i < mp->no_imports ; ++i, ++mip ) {
  1437.              if ( !label_compare( mip->label, descriptor )) {
  1438. /* Found it - so amend the module ID */
  1439.                   modid = mip->modid;
  1440.                   break;
  1441.              }
  1442.          }
  1443.     for(i=0, tcp=tclist; i < MAXTC; i++, tcp++){
  1444.       if (tcp->type == 0)
  1445.           break;
  1446.       if (!label_compare(descriptor, tcp->descriptor) &&
  1447. ((modid == tcp->modid) || (modid == -1))){
  1448.           return i;
  1449.       }
  1450.     }
  1451.     return -1;
  1452. }
  1453. /* translate integer tc_index to string identifier from tclist
  1454.  *
  1455.  * Returns pointer to string in table (should not be modified) or NULL
  1456.  */
  1457. const char *
  1458. get_tc_descriptor(int tc_index)
  1459. {
  1460.   if (tc_index < 0 || tc_index >= MAXTC) return NULL;
  1461.   return (tclist[tc_index].descriptor);
  1462. }
  1463. /*
  1464.  * Parses an enumeration list of the form:
  1465.  *        { label(value) label(value) ... }
  1466.  * The initial { has already been parsed.
  1467.  * Returns NULL on error.
  1468.  */
  1469. static struct enum_list *
  1470. parse_enumlist(FILE *fp, struct enum_list **retp)
  1471. {
  1472.     register int type;
  1473.     char token [MAXTOKEN];
  1474.     struct enum_list *ep = NULL, **epp = &ep;
  1475.     free_enums(retp);
  1476.     while((type = get_token(fp, token, MAXTOKEN)) != ENDOFFILE){
  1477.         if (type == RIGHTBRACKET)
  1478.             break;
  1479.         if (type == LABEL){
  1480.             /* this is an enumerated label */
  1481.             *epp = (struct enum_list *) calloc(1, sizeof(struct enum_list));
  1482.             if (*epp == NULL) return(NULL);
  1483.             /* a reasonable approximation for the length */
  1484.             (*epp)->label = strdup(token);
  1485.             type = get_token(fp, token, MAXTOKEN);
  1486.             if (type != LEFTPAREN) {
  1487.                 print_error("Expected "("", token, type);
  1488.                 return NULL;
  1489.             }
  1490.             type = get_token(fp, token, MAXTOKEN);
  1491.             if (type != NUMBER) {
  1492.                 print_error("Expected integer", token, type);
  1493.                 return NULL;
  1494.             }
  1495.             (*epp)->value = atoi(token);
  1496.             type = get_token(fp, token, MAXTOKEN);
  1497.             if (type != RIGHTPAREN) {
  1498.                 print_error("Expected ")"", token, type);
  1499.                 return NULL;
  1500.             }
  1501.             epp = &(*epp)->next;
  1502.         }
  1503.     }
  1504.     if (type == ENDOFFILE){
  1505.         print_error("Expected "}"", token, type);
  1506.         return NULL;
  1507.     }
  1508.     *retp = ep;
  1509.     return ep;
  1510. }
  1511. static struct range_list *parse_ranges(FILE *fp, struct range_list **retp)
  1512. {   int low, high;
  1513.     char nexttoken[MAXTOKEN];
  1514.     int nexttype;
  1515.     struct range_list *rp = NULL, **rpp = &rp;
  1516.     int size = 0, taken = 1;
  1517.     free_ranges(retp);
  1518.     nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1519.     if (nexttype == SIZE) {
  1520. size = 1;
  1521. taken = 0;
  1522. nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1523. if (nexttype != LEFTPAREN)
  1524.     print_error("Expected "(" after SIZE", nexttoken, nexttype);
  1525.     }
  1526.     do {
  1527. if (!taken) nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1528. else taken = 0;
  1529. high = low = atol(nexttoken);
  1530. nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1531. if (nexttype == RANGE) {
  1532.     nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1533.     high = atol(nexttoken);
  1534.     nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1535. }
  1536. *rpp = (struct range_list *)calloc (1, sizeof(struct range_list));
  1537. if (*rpp == NULL) break;
  1538. (*rpp)->low = low;
  1539. (*rpp)->high = high;
  1540. rpp = &(*rpp)->next;
  1541.     } while (nexttype == BAR);
  1542.     if (size) {
  1543. if (nexttype != RIGHTPAREN)
  1544.     print_error ("Expected ")" after SIZE", nexttoken, nexttype);
  1545. nexttype = get_token(fp, nexttoken, nexttype);
  1546.     }
  1547.     if (nexttype != RIGHTPAREN)
  1548. print_error ("Expected ")"", nexttoken, nexttype);
  1549.     *retp = rp;
  1550.     return rp;
  1551. }
  1552. /*
  1553.  * Parses an asn type.  Structures are ignored by this parser.
  1554.  * Returns NULL on error.
  1555.  */
  1556. static struct node *
  1557. parse_asntype(FILE *fp,
  1558.       char *name,
  1559.       int *ntype,
  1560.       char *ntoken)
  1561. {
  1562.     int type, i;
  1563.     char token[MAXTOKEN];
  1564.     char quoted_string_buffer[MAXQUOTESTR];
  1565.     char *hint = NULL;
  1566.     struct tc *tcp;
  1567.     int level;
  1568.     type = get_token(fp, token, MAXTOKEN);
  1569.     if (type == SEQUENCE){
  1570.         level = 0;
  1571.         while((type = get_token(fp, token, MAXTOKEN)) != ENDOFFILE){
  1572.             if (type == LEFTBRACKET){
  1573.                 level++;
  1574.             }
  1575.             else if (type == RIGHTBRACKET && --level == 0){
  1576.                 *ntype = get_token(fp, ntoken, MAXTOKEN);
  1577.                 return NULL;
  1578.             }
  1579.         }
  1580.         print_error("Expected "}"", token, type);
  1581.         return NULL;
  1582.     } else if (type == LEFTBRACKET) {
  1583.         struct node *np;
  1584.         int ch_next = '{';
  1585.         ungetc(ch_next, fp);
  1586.         np = parse_objectid (fp, name);
  1587.         if (np != NULL) {
  1588.             *ntype = get_token(fp, ntoken, MAXTOKEN);
  1589.             return np;
  1590.         }
  1591.         return NULL;
  1592.     } else {
  1593.         if (type == CONVENTION) {
  1594.             while (type != SYNTAX && type != ENDOFFILE) {
  1595.                 if (type == DISPLAYHINT) {
  1596.                     type = get_token(fp, token, MAXTOKEN);
  1597.                     if (type != QUOTESTRING) print_error("DISPLAY-HINT must be string", token, type);
  1598.                     else hint = strdup (token);
  1599.                 }
  1600.                 else
  1601.     type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1602.             }
  1603.             type = get_token(fp, token, MAXTOKEN);
  1604.         }
  1605.         if (type == LABEL)
  1606.         {
  1607.             type = get_tc(token, current_module, NULL, NULL, NULL, NULL);
  1608.         }
  1609.         /* textual convention */
  1610.         for(i = 0; i < MAXTC; i++){
  1611.             if (tclist[i].type == 0)
  1612.                 break;
  1613.         }
  1614.         if (i == MAXTC){
  1615.             print_error("Too many textual conventions", token, type);
  1616.             SNMP_FREE(hint);
  1617.             return NULL;
  1618.         }
  1619.         if (!(type & SYNTAX_MASK)){
  1620.             print_error("Textual convention doesn't map to real type", token,
  1621.                         type);
  1622.             SNMP_FREE(hint);
  1623.             return NULL;
  1624.         }
  1625.         tcp = &tclist[i];
  1626.         tcp->modid = current_module;
  1627.         tcp->descriptor = strdup(name);
  1628.         tcp->hint = hint;
  1629.         tcp->type = type;
  1630.         *ntype = get_token(fp, ntoken, MAXTOKEN);
  1631.         if (*ntype == LEFTPAREN){
  1632.     tcp->ranges = parse_ranges(fp, &tcp->ranges);
  1633.             *ntype = get_token(fp, ntoken, MAXTOKEN);
  1634.         } else if (*ntype == LEFTBRACKET) {
  1635.             /* if there is an enumeration list, parse it */
  1636.             tcp->enums = parse_enumlist(fp, &tcp->enums);
  1637.             *ntype = get_token(fp, ntoken, MAXTOKEN);
  1638.         }
  1639.         return NULL;
  1640.     }
  1641. }
  1642. /*
  1643.  * Parses an OBJECT TYPE macro.
  1644.  * Returns 0 on error.
  1645.  */
  1646. static struct node *
  1647. parse_objecttype(FILE *fp,
  1648.  char *name)
  1649. {
  1650.     register int type;
  1651.     char token[MAXTOKEN];
  1652.     char nexttoken[MAXTOKEN];
  1653.     char quoted_string_buffer[MAXQUOTESTR];
  1654.     int nexttype, tctype;
  1655.     register struct node *np;
  1656.     type = get_token(fp, token, MAXTOKEN);
  1657.     if (type != SYNTAX){
  1658.         print_error("Bad format for OBJECT-TYPE", token, type);
  1659.         return NULL;
  1660.     }
  1661.     np = alloc_node(current_module);
  1662.     if (np == NULL) return(NULL);
  1663.     type = get_token(fp, token, MAXTOKEN);
  1664.     if (type == LABEL){
  1665.         int tmp_index;
  1666.         tctype = get_tc(token, current_module, &tmp_index,
  1667.                         &np->enums, &np->ranges, &np->hint);
  1668.         if (tctype == LABEL &&
  1669.             ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS) > 1){
  1670.             print_error("Warning: No known translation for type", token, type);
  1671.         }
  1672.         type = tctype;
  1673.         np->tc_index = tmp_index; /* store TC for later reference */
  1674.     }
  1675.     np->type = type;
  1676.     nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1677.     switch(type){
  1678.         case SEQUENCE:
  1679.             if (nexttype == OF){
  1680.                 nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1681.                 nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1682.             }
  1683.             break;
  1684.         case INTEGER:
  1685.         case UINTEGER32:
  1686.         case COUNTER:
  1687.         case GAUGE:
  1688.         case BITSTRING:
  1689.         case LABEL:
  1690.             if (nexttype == LEFTBRACKET) {
  1691.                 /* if there is an enumeration list, parse it */
  1692.                 np->enums = parse_enumlist(fp, &np->enums);
  1693.                 nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1694.             } else if (nexttype == LEFTPAREN){
  1695.                 /* if there is a range list, parse it */
  1696. np->ranges = parse_ranges(fp, &np->ranges);
  1697.                 nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1698.             }
  1699.             break;
  1700.         case OCTETSTR:
  1701.         case KW_OPAQUE:
  1702.             /* parse any SIZE specification */
  1703.             if (nexttype == LEFTPAREN) {
  1704.                 nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1705.                 if (nexttype == SIZE) {
  1706.                     nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1707.                     if (nexttype == LEFTPAREN) {
  1708. np->ranges = parse_ranges(fp, &np->ranges);
  1709.                         nexttype = get_token(fp, nexttoken, MAXTOKEN); /* ) */
  1710.                         if (nexttype == RIGHTPAREN)
  1711.                         {
  1712.                             nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1713.                             break;
  1714.                         }
  1715.                     }
  1716.                 }
  1717.                 print_error("Bad SIZE syntax", token, type);
  1718.                 free_node(np);
  1719.                 return NULL;
  1720.             }
  1721.             break;
  1722.         case OBJID:
  1723.         case NETADDR:
  1724.         case IPADDR:
  1725.         case TIMETICKS:
  1726.         case NUL:
  1727.         case NSAPADDRESS:
  1728.         case COUNTER64:
  1729.             break;
  1730.         default:
  1731.             print_error("Bad syntax", token, type);
  1732.             free_node(np);
  1733.             return NULL;
  1734.     }
  1735.     if (nexttype == UNITS){
  1736.         type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1737.         if (type != QUOTESTRING) {
  1738.             print_error("Bad UNITS", quoted_string_buffer, type);
  1739.             free_node(np);
  1740.             return NULL;
  1741.         }
  1742. np->units = strdup (quoted_string_buffer);
  1743.         nexttype = get_token(fp, nexttoken, MAXTOKEN);
  1744.     }
  1745.     if (nexttype != ACCESS){
  1746.         print_error("Should be ACCESS", nexttoken, nexttype);
  1747.         free_node(np);
  1748.         return NULL;
  1749.     }
  1750.     type = get_token(fp, token, MAXTOKEN);
  1751.     if (type != READONLY && type != READWRITE && type != WRITEONLY
  1752.         && type != NOACCESS && type != READCREATE && type != ACCNOTIFY){
  1753.         print_error("Bad ACCESS type", token, type);
  1754.         free_node(np);
  1755.         return NULL;
  1756.     }
  1757.     np->access = type;
  1758.     type = get_token(fp, token, MAXTOKEN);
  1759.     if (type != STATUS){
  1760.         print_error("Should be STATUS", token, type);
  1761.         free_node(np);
  1762.         return NULL;
  1763.     }
  1764.     type = get_token(fp, token, MAXTOKEN);
  1765.     if (type != MANDATORY && type != CURRENT && type != KW_OPTIONAL &&
  1766.         type != OBSOLETE && type != DEPRECATED){
  1767.         print_error("Bad STATUS", token, type);
  1768.         free_node(np);
  1769.         return NULL;
  1770.     }
  1771.     np->status = type;
  1772.     /*
  1773.      * Optional parts of the OBJECT-TYPE macro
  1774.      */
  1775.     type = get_token(fp, token, MAXTOKEN);
  1776.     while (type != EQUALS && type != ENDOFFILE) {
  1777.       switch (type) {
  1778.         case DESCRIPTION:
  1779.           type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1780.           if (type != QUOTESTRING) {
  1781.               print_error("Bad DESCRIPTION", quoted_string_buffer, type);
  1782.               free_node(np);
  1783.               return NULL;
  1784.           }
  1785.           if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_SAVE_MIB_DESCRS)) {
  1786.               np->description = strdup (quoted_string_buffer);
  1787.           }
  1788.           break;
  1789.         case REFERENCE:
  1790.           type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1791.           if (type != QUOTESTRING) {
  1792.               print_error("Bad REFERENCE", quoted_string_buffer, type);
  1793.               free_node(np);
  1794.               return NULL;
  1795.           }
  1796.           break;
  1797.         case INDEX:
  1798.           np->indexes = getIndexes(fp, &np->indexes);
  1799.           if (np->indexes == NULL) {
  1800.             print_error("Bad Index List",token,type);
  1801.             free_node(np);
  1802.             return NULL;
  1803.           }
  1804.           break;
  1805.         case DEFVAL:
  1806.         case AUGMENTS:
  1807.         case NUM_ENTRIES:
  1808.           if (tossObjectIdentifier(fp) != OBJID) {
  1809.               print_error("Bad Object Identifier", token, type);
  1810.               free_node(np);
  1811.               return NULL;
  1812.           }
  1813.           break;
  1814.         default:
  1815.           print_error("Bad format of optional clauses", token, type);
  1816.           free_node(np);
  1817.           return NULL;
  1818.       }
  1819.       type = get_token(fp, token, MAXTOKEN);
  1820.     }
  1821.     if (type != EQUALS){
  1822.         print_error("Bad format", token, type);
  1823.         free_node(np);
  1824.         return NULL;
  1825.     }
  1826.     return merge_parse_objectid(np, fp, name);
  1827. }
  1828. /*
  1829.  * Parses an OBJECT GROUP macro.
  1830.  * Returns 0 on error.
  1831.  *
  1832.  * Also parses object-identity, since they are similar (ignore STATUS).
  1833.  *   - WJH 10/96
  1834.  */
  1835. static struct node *
  1836. parse_objectgroup(FILE *fp,
  1837.   char *name)
  1838. {
  1839.     register int type;
  1840.     char token[MAXTOKEN];
  1841.     char quoted_string_buffer[MAXQUOTESTR];
  1842.     register struct node *np;
  1843.     np = alloc_node(current_module);
  1844.     if (np == NULL) return(NULL);
  1845.     type = get_token(fp, token, MAXTOKEN);
  1846.     while (type != EQUALS && type != ENDOFFILE) {
  1847.       switch (type) {
  1848.         case DESCRIPTION:
  1849.           type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1850.           if (type != QUOTESTRING) {
  1851.               print_error("Bad DESCRIPTION", quoted_string_buffer, type);
  1852.               free_node(np);
  1853.               return NULL;
  1854.           }
  1855.           if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_SAVE_MIB_DESCRS)) {
  1856.               np->description = strdup (quoted_string_buffer);
  1857.           }
  1858.           break;
  1859. case REFERENCE:
  1860.   type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1861.           if (type != QUOTESTRING) {
  1862.               print_error("Bad REFERENCE", quoted_string_buffer, type);
  1863.               free_node(np);
  1864.               return NULL;
  1865.           }
  1866.   break;
  1867.         default:
  1868.           /* NOTHING */
  1869.           break;
  1870.       }
  1871.       type = get_token(fp, token, MAXTOKEN);
  1872.     }
  1873.     return merge_parse_objectid(np, fp, name);
  1874. }
  1875. /*
  1876.  * Parses a NOTIFICATION-TYPE macro.
  1877.  * Returns 0 on error.
  1878.  */
  1879. static struct node *
  1880. parse_notificationDefinition(FILE *fp,
  1881.      char *name)
  1882. {
  1883.     register int type;
  1884.     char token[MAXTOKEN];
  1885.     char quoted_string_buffer[MAXQUOTESTR];
  1886.     register struct node *np;
  1887.     np = alloc_node(current_module);
  1888.     if (np == NULL) return(NULL);
  1889.     type = get_token(fp, token, MAXTOKEN);
  1890.     while (type != EQUALS && type != ENDOFFILE) {
  1891.       switch (type) {
  1892.         case DESCRIPTION:
  1893.           type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1894.           if (type != QUOTESTRING) {
  1895.               print_error("Bad DESCRIPTION", quoted_string_buffer, type);
  1896.               free_node(np);
  1897.               return NULL;
  1898.           }
  1899.           if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_SAVE_MIB_DESCRS)) {
  1900.               np->description = strdup (quoted_string_buffer);
  1901.           }
  1902.           break;
  1903.         default:
  1904.           /* NOTHING */
  1905.           break;
  1906.       }
  1907.       type = get_token(fp, token, MAXTOKEN);
  1908.     }
  1909.     return merge_parse_objectid(np, fp, name);
  1910. }
  1911. /*
  1912.  * Parses a TRAP-TYPE macro.
  1913.  * Returns 0 on error.
  1914.  */
  1915. static struct node *
  1916. parse_trapDefinition(FILE *fp,
  1917.      char *name)
  1918. {
  1919.     register int type;
  1920.     char token[MAXTOKEN];
  1921.     char quoted_string_buffer[MAXQUOTESTR];
  1922.     register struct node *np;
  1923.     np = alloc_node(current_module);
  1924.     if (np == NULL) return(NULL);
  1925.     type = get_token(fp, token, MAXTOKEN);
  1926.     while (type != EQUALS && type != ENDOFFILE) {
  1927.         switch (type) {
  1928.             case DESCRIPTION:
  1929.                 type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  1930.                 if (type != QUOTESTRING) {
  1931.                     print_error("Bad DESCRIPTION", quoted_string_buffer, type);
  1932.                     free_node(np);
  1933.                     return NULL;
  1934.                 }
  1935.                 if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_SAVE_MIB_DESCRS)) {
  1936.                     np->description = strdup (quoted_string_buffer);
  1937.                 }
  1938.                 break;
  1939.             case ENTERPRISE:
  1940.                 type = get_token(fp, token, MAXTOKEN);
  1941.                 if (type == LEFTBRACKET) {
  1942.                     type = get_token(fp, token, MAXTOKEN);
  1943.                     if (type != LABEL) {
  1944.                         print_error("Bad Trap Format", token, type);
  1945.                         free_node(np);
  1946.                         return NULL;
  1947.                     }
  1948.                     np->parent = strdup(token);
  1949.                     /* Get right bracket */
  1950.                     type = get_token(fp, token, MAXTOKEN);
  1951.                 }
  1952.                 else if (type == LABEL)
  1953.                     np->parent = strdup(token);
  1954.                 break;
  1955.             default:
  1956.                 /* NOTHING */
  1957.                 break;
  1958.         }
  1959.         type = get_token(fp, token, MAXTOKEN);
  1960.     }
  1961.     type = get_token(fp, token, MAXTOKEN);
  1962.     np->label = strdup(name);
  1963.     if (type != NUMBER) {
  1964.         print_error("Expected a Number", token, type);
  1965.         free_node(np);
  1966.         return NULL;
  1967.     }
  1968.     np->subid = atoi(token);
  1969.     np->next = alloc_node(current_module);
  1970.     if (np->next == NULL)  {
  1971.         free_node(np);
  1972.         return(NULL);
  1973.     }
  1974.     np->next->parent = np->parent;
  1975.     np->parent = (char *)malloc(strlen(np->parent)+2);
  1976.     if (np->parent == NULL) {
  1977.         free_node(np->next); free_node(np);
  1978.         return(NULL);
  1979.     }
  1980.     strcpy(np->parent, np->next->parent);
  1981.     strcat(np->parent, "#");
  1982.     np->next->label = strdup(np->parent);
  1983.     return np;
  1984. }
  1985. /*
  1986.  * Parses a compliance macro
  1987.  * Returns 0 on error.
  1988.  */
  1989. static struct node *
  1990. parse_compliance(FILE *fp,
  1991.  char *name)
  1992. {
  1993.     register int type;
  1994.     char token[MAXTOKEN];
  1995.     char quoted_string_buffer[MAXQUOTESTR];
  1996.     register struct node *np;
  1997.     np = alloc_node(current_module);
  1998.     if (np == NULL) return(NULL);
  1999.     type = get_token(fp, token, MAXTOKEN);
  2000.     while (type != EQUALS && type != ENDOFFILE) {
  2001.         type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  2002.     }
  2003.     return merge_parse_objectid(np, fp, name);
  2004. }
  2005. /*
  2006.  * Parses a capabilities macro
  2007.  * Returns 0 on error.
  2008.  */
  2009. static struct node *
  2010. parse_capabilities(FILE *fp,
  2011.    char *name)
  2012. {
  2013.     register int type;
  2014.     char token[MAXTOKEN];
  2015.     char quoted_string_buffer[MAXQUOTESTR];
  2016.     register struct node *np;
  2017.     np = alloc_node(current_module);
  2018.     if (np == NULL) return(NULL);
  2019.     type = get_token(fp, token, MAXTOKEN);
  2020.     while (type != EQUALS && type != ENDOFFILE) {
  2021.         type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  2022.     }
  2023.     return merge_parse_objectid(np, fp, name);
  2024. }
  2025. /*
  2026.  * Parses a module identity macro
  2027.  * Returns 0 on error.
  2028.  */
  2029. static struct node *
  2030. parse_moduleIdentity(FILE *fp,
  2031.      char *name)
  2032. {
  2033.     register int type;
  2034.     char token[MAXTOKEN];
  2035.     char quoted_string_buffer[MAXQUOTESTR];
  2036.     register struct node *np;
  2037.     np = alloc_node(current_module);
  2038.     if (np == NULL) return(NULL);
  2039.     type = get_token(fp, token, MAXTOKEN);
  2040.     while (type != EQUALS && type != ENDOFFILE) {
  2041.         type = get_token(fp, quoted_string_buffer, MAXQUOTESTR);
  2042.     }
  2043.     return merge_parse_objectid(np, fp, name);
  2044. }
  2045. /*
  2046.  * Parses a MACRO definition
  2047.  * Expect BEGIN, discard everything to end.
  2048.  * Returns 0 on error.
  2049.  */
  2050. static struct node *
  2051. parse_macro(FILE *fp,
  2052.     char *name)
  2053. {
  2054.     register int type;
  2055.     char token[MAXTOKEN];
  2056.     struct node *np;
  2057.     int iLine = Line;
  2058.     np = alloc_node(current_module);
  2059.     if (np == NULL) return(NULL);
  2060.     type = get_token(fp, token, sizeof(token));
  2061.     while (type != EQUALS && type != ENDOFFILE) {
  2062.         type = get_token(fp, token, sizeof(token));
  2063.     }
  2064.     if (type != EQUALS) return NULL;
  2065.     while (type != BEGIN && type != ENDOFFILE) {
  2066.         type = get_token(fp, token, sizeof(token));
  2067.     }
  2068.     if (type != BEGIN) return NULL;
  2069.     while (type != END && type != ENDOFFILE) {
  2070.         type = get_token(fp, token, sizeof(token));
  2071.     }
  2072.     if (type != END) return NULL;
  2073.     if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  2074. snmp_log(LOG_WARNING,
  2075.  "%s MACRO (lines %d..%d parsed and ignored).n", name, iLine, Line);
  2076.     return np;
  2077. }
  2078. /*
  2079.  * Parses a module import clause
  2080.  *   loading any modules referenced
  2081.  */
  2082. static void
  2083. parse_imports(FILE *fp)
  2084. {
  2085. #ifdef UCD_SNMP
  2086.     register int type;
  2087.     char token[MAXTOKEN];
  2088.     char modbuf[256];
  2089. #define MAX_IMPORTS 256
  2090.     struct module_import import_list[MAX_IMPORTS];
  2091.     int this_module, old_current_module;
  2092.     const char *old_File;
  2093.     int old_line;
  2094.     struct module *mp;
  2095.     int import_count=0; /* Total number of imported descriptors */
  2096.     int i=0, old_i; /* index of first import from each module */
  2097.     type = get_token(fp, token, MAXTOKEN);
  2098. /*
  2099.  * Parse the IMPORTS clause
  2100.  */
  2101.     while (type != SEMI && type != ENDOFFILE) {
  2102. if (type == LABEL ) {
  2103.     if (import_count == MAX_IMPORTS ) {
  2104. print_error("Too many imported symbols", token, type);
  2105. do {
  2106.     type = get_token(fp, token, MAXTOKEN);
  2107. } while (type != SEMI && type != ENDOFFILE);
  2108. return;
  2109.     }
  2110.     import_list[import_count++].label = strdup(token);
  2111. }
  2112. else if ( type == FROM ) {
  2113.     type = get_token(fp, token, MAXTOKEN);
  2114.             if ( import_count == i ) { /* All imports are handled internally */
  2115.        type = get_token(fp, token, MAXTOKEN);
  2116.                continue;
  2117.             }
  2118.     this_module = which_module(token);
  2119.     for ( old_i=i ; i<import_count ; ++i)
  2120. import_list[i].modid = this_module;
  2121.     old_current_module = current_module; /* Save state */
  2122.             old_File = File;
  2123.     old_line = Line;
  2124.     current_module = this_module;
  2125. /*
  2126.  * Recursively read any pre-requisite modules
  2127.  */
  2128.     if  (read_module_internal(token) == MODULE_NOT_FOUND ) {
  2129. for ( ; old_i<import_count ; ++old_i ) {
  2130.     read_import_replacements( token, &import_list[old_i]);
  2131. }
  2132.     }
  2133.     current_module = old_current_module; /* Restore state */
  2134.     File = old_File;
  2135.     Line = old_line;
  2136. }
  2137. type = get_token(fp, token, MAXTOKEN);
  2138.     }
  2139. /*
  2140.  * Save the import information
  2141.  *   in the global module table
  2142.  */
  2143.     for ( mp=module_head ; mp ; mp=mp->next )
  2144. if ( mp->modid == current_module) {
  2145.             if ( import_count == 0)
  2146. return;
  2147.             if (mp->imports && (mp->imports != root_imports))
  2148.     {
  2149. /* this can happen if all modules are in one source file. */
  2150. for ( i=0 ; i<mp->no_imports; ++i ) {
  2151. DEBUGMSGTL(("parse-mibs",  "#### freeing Module %d '%s' %dn",
  2152. mp->modid, mp->imports[i].label, mp->imports[i].modid));
  2153.     free(mp->imports[i].label);
  2154. }
  2155. free(mp->imports);
  2156.     }
  2157.             mp->imports = (struct module_import *)
  2158.               calloc(import_count, sizeof(struct module_import));
  2159.             if (mp->imports == NULL) return;
  2160.     for ( i=0 ; i<import_count ; ++i ) {
  2161. mp->imports[i].label = import_list[i].label;
  2162. mp->imports[i].modid = import_list[i].modid;
  2163. DEBUGMSGTL(("parse-mibs",  "#### adding Module %d '%s' %dn",
  2164. mp->modid, mp->imports[i].label, mp->imports[i].modid));
  2165.     }
  2166.     mp->no_imports = import_count;
  2167.     return;
  2168. }
  2169. /*
  2170.  * Shouldn't get this far
  2171.  */
  2172.     print_module_not_found(module_name(current_module,modbuf));
  2173.     return;
  2174. #endif
  2175. }
  2176. /*
  2177.  * MIB module handling routines
  2178.  */
  2179. static void dump_module_list (void)
  2180. {
  2181.     struct module *mp = module_head;
  2182.     DEBUGMSGTL(("parse-mibs", "Module list:n"));
  2183.     while (mp) {
  2184. DEBUGMSGTL(("parse-mibs", "  %s %d %s %dn", mp->name, mp->modid, mp->file, mp->no_imports));
  2185. mp = mp->next;
  2186.     }
  2187. }
  2188. int
  2189. which_module(const char *name)
  2190. {
  2191.     struct module *mp;
  2192.     for ( mp=module_head ; mp ; mp=mp->next )
  2193. if ( !label_compare(mp->name, name))
  2194.     return(mp->modid);
  2195.     DEBUGMSGTL(("parse-mibs", "Module %s not foundn", name));
  2196.     return(-1);
  2197. }
  2198. /*
  2199.  * module_name - copy module name to user buffer, return ptr to same.
  2200.  */
  2201. char *
  2202. module_name (int modid,
  2203.      char *cp)
  2204. {
  2205.     struct module *mp;
  2206.     for ( mp=module_head ; mp ; mp=mp->next )
  2207. if ( mp->modid == modid )
  2208. {
  2209.     strcpy(cp, mp->name);
  2210.     return(cp);
  2211. }
  2212.     DEBUGMSGTL(("parse-mibs", "Module %d not foundn", modid));
  2213.     sprintf(cp, "#%d", modid);
  2214.     return(cp);
  2215. }
  2216. /*
  2217.  *  Backwards compatability
  2218.  *  Read newer modules that replace the one specified:-
  2219.  * either all of them (read_module_replacements),
  2220.  * or those relating to a specified identifier (read_import_replacements)
  2221.  * plus an interface to add new replacement requirements
  2222.  */
  2223. void
  2224. add_module_replacement(const char *old_module,
  2225.        const char *new_module_name,
  2226.        const char *tag,
  2227.        int len)
  2228. {
  2229.     struct module_compatability *mcp;
  2230.     mcp =  (struct module_compatability *)
  2231.       calloc(1, sizeof( struct module_compatability));
  2232.     if (mcp == NULL) return;
  2233.     mcp->old_module = strdup( old_module );
  2234.     mcp->new_module = strdup( new_module_name );
  2235. if (tag)
  2236.     mcp->tag     = strdup( tag );
  2237.     mcp->tag_len = len;
  2238.     mcp->next    = module_map_head;
  2239.     module_map_head = mcp;
  2240. }
  2241. static void
  2242. read_module_replacements(const char *name)
  2243. {
  2244.     struct module_compatability *mcp;
  2245.     for ( mcp=module_map_head ; mcp; mcp=mcp->next ) {
  2246. if ( !label_compare( mcp->old_module, name )) {
  2247.     if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  2248. snmp_log(LOG_WARNING,
  2249.                          "Loading replacement module %s for %s (%s)n",
  2250.                          mcp->new_module, name, File);
  2251.     (void)read_module( mcp->new_module );
  2252.     return;
  2253. }
  2254.     }
  2255.     if (!ds_get_boolean(DS_LIBRARY_ID, DS_LIB_MIB_ERRORS))
  2256. print_module_not_found(name);
  2257. }
  2258. #ifdef UCD_SNMP
  2259. static void
  2260. read_import_replacements(const char *old_module_name,
  2261.  struct module_import *identifier)
  2262. {
  2263. struct module_compatability *mcp;
  2264. /*
  2265.  * Look for matches first
  2266.  */
  2267.     for ( mcp=module_map_head ; mcp; mcp=mcp->next ) {
  2268.       if ( !label_compare( mcp->old_module, old_module_name )) {
  2269. if ( /* exact match */
  2270.      ( mcp->tag_len==0 &&
  2271.     (mcp->tag == NULL ||
  2272.                      !label_compare( mcp->tag, identifier->label ))) ||
  2273. /* prefix match */
  2274.           ( mcp->tag_len!=0 &&
  2275.     !strncmp( mcp->tag, identifier->label, mcp->tag_len ))
  2276.    ) {
  2277.     if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  2278.         snmp_log(LOG_WARNING,
  2279.                          "Importing %s from replacement module %s instead of %s (%s)n",
  2280.                          identifier->label, mcp->new_module, old_module_name, File);
  2281.     (void)read_module( mcp->new_module );
  2282.     identifier->modid = which_module(mcp->new_module);
  2283.     return; /* finished! */
  2284.         }
  2285.       }
  2286.     }
  2287. /*
  2288.  * If no exact match, load everything relevant
  2289.  */
  2290.     read_module_replacements( old_module_name );
  2291. }
  2292. #endif
  2293. /*
  2294.  *  Read in the named module
  2295.  * Returns the root of the whole tree
  2296.  * (by analogy with 'read_mib')
  2297.  */
  2298. static int
  2299. read_module_internal (const char *name)
  2300. {
  2301.     struct module *mp;
  2302.     FILE *fp;
  2303.     struct node *np;
  2304.     if ( tree_head == NULL )
  2305. init_mib_internals(); /* was init_mib */
  2306.     for ( mp=module_head ; mp ; mp=mp->next )
  2307. if ( !label_compare(mp->name, name)) {
  2308.          const char *oldFile = File;
  2309.          int oldLine = Line;
  2310.     if ( mp->no_imports != -1 ) {
  2311. DEBUGMSGTL(("parse-mibs", "Module %s already loadedn", name));
  2312. return MODULE_ALREADY_LOADED;
  2313.     }
  2314.     if ((fp = fopen(mp->file, "r")) == NULL) {
  2315. snmp_log_perror(mp->file);
  2316. return MODULE_LOAD_FAILED;
  2317.     }
  2318.     mp->no_imports=0; /* Note that we've read the file */
  2319.     File = mp->file;
  2320.     Line = 1;
  2321. /*
  2322.  * Parse the file
  2323.  */
  2324.     np = parse( fp, NULL );
  2325.     fclose(fp);
  2326.     File = oldFile;
  2327.     Line = oldLine;
  2328.     return MODULE_LOADED_OK;
  2329. }
  2330.     if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS) > 1)
  2331. snmp_log(LOG_WARNING, "Module %s not foundn", name);
  2332.     return MODULE_NOT_FOUND;
  2333. }
  2334. void
  2335. adopt_orphans (void)
  2336. {
  2337.     struct node *np, *onp;
  2338.     struct tree *tp = NULL;
  2339.     int i, adopted;
  2340.     if ( !orphan_nodes )
  2341. return;
  2342.     init_node_hash(orphan_nodes);
  2343.     orphan_nodes = NULL;
  2344.     while (1) {
  2345. adopted = 0;
  2346. for ( i = 0; i < NHASHSIZE; i++)
  2347.     if ( nbuckets[i] ) {
  2348.         for ( np = nbuckets[i] ; np!= NULL ; np=np->next )
  2349.             tp = find_tree_node( np->parent, -1 );
  2350.             if ( tp ) {
  2351.         do_subtree( tp, &np );
  2352.         adopted = 1;
  2353.             }
  2354.     }
  2355. if ( adopted == 0 )
  2356.     break;
  2357.     }
  2358. /*
  2359.  * Report on outstanding orphans
  2360.  *    and link them back into the orphan list
  2361.  */
  2362.     for (i = 0; i < NHASHSIZE; i++)
  2363. if ( nbuckets[i] ) {
  2364.     if ( orphan_nodes )
  2365. onp = np->next = nbuckets[i];
  2366.     else
  2367. onp = orphan_nodes = nbuckets[i];
  2368.     nbuckets[i] = NULL;
  2369.     while (onp) {
  2370.          char modbuf[256];
  2371.          snmp_log (LOG_WARNING,
  2372.                           "Unlinked OID in %s: %s ::= { %s %ld }n",
  2373.                           module_name(onp->modid, modbuf),
  2374.                           (onp->label ? onp->label : "<no label>"),
  2375.                           (onp->parent ? onp->parent : "<no parent>"),
  2376.                           onp->subid);
  2377. np = onp;
  2378. onp = onp->next;
  2379.     }
  2380. }
  2381. }
  2382. struct tree *
  2383. read_module(const char *name)
  2384. {
  2385.     if ( read_module_internal(name) == MODULE_NOT_FOUND )
  2386. read_module_replacements( name );
  2387.     return tree_head;
  2388. }
  2389. void
  2390. unload_module_by_ID( int modID, struct tree *tree_top )
  2391. {
  2392. #ifdef UCD_SNMP
  2393.     struct tree *tp, *prev, *next;
  2394.     int i;
  2395.     prev = NULL;
  2396.     for ( tp=tree_top ; tp ; tp=next ) {
  2397. next = tp->next_peer;
  2398. /*
  2399.  * This next section looks rather complex.
  2400.  * Essentially, this is equivalent to the code fragment:
  2401.  *   if (tp->modID = modID)
  2402.  *     tp->number_modules--;
  2403.  * but handles one tree node being part of several modules.
  2404.  */
  2405. for ( i=0 ; i<tp->number_modules ; i++ ) {
  2406.     if ( tp->module_list[i] == modID ) {
  2407. tp->number_modules--;
  2408. switch ( tp->number_modules ) {
  2409.     case 0: /* That was the only module */
  2410. tp->modid = -1; /* Mark as unused */
  2411. break;
  2412.     case 1: /* We did have a list of two, but this is no
  2413.    longer needed.  Transfer the other entry
  2414.    ( i.e. module_list[1-i] - think about it! )
  2415.    to the 'single' slot tp->modid, and discard
  2416.    the list.
  2417.  */
  2418. tp->modid = tp->module_list[1-i];
  2419. free(tp->module_list);
  2420. tp->module_list = NULL; /* let's be tidy */
  2421. break;
  2422.     default: /* We still need the list, so shuffle down
  2423.    all following entries to close up the gap */
  2424. while ( i < tp->number_modules ) {
  2425.    tp->module_list[i] = tp->module_list[i+1];
  2426.    i++;
  2427. }
  2428. break;
  2429. }
  2430. break; /* Don't need to look through the rest of the list */
  2431.     }
  2432. }
  2433. /*
  2434.  *  OK - that's dealt with *this* node.
  2435.  * Now let's look at the children.
  2436.  * (Isn't recursion wonderful!)
  2437.  */
  2438. if ( tp->child_list )
  2439.     unload_module_by_ID( modID, tp->child_list );
  2440. if ( tp->number_modules == 0 ) {
  2441. /* This node isn't needed any more (except perhaps
  2442. for the sake of the children) */
  2443.       if ( tp->child_list == NULL ) {
  2444. if ( prev )
  2445.     prev->next_peer = tp->next_peer;
  2446. else
  2447.     tp->parent->child_list = tp->next_peer;
  2448. free_tree( tp );
  2449.    }
  2450.    else
  2451. free_partial_tree( tp, TRUE );
  2452. }
  2453. else
  2454.     prev = tp;
  2455.     }
  2456. #endif
  2457. }
  2458. int
  2459. unload_module(const char *name)
  2460. {
  2461.     struct module *mp;
  2462.     int modID = -1;
  2463.     for ( mp=module_head ; mp ; mp=mp->next )
  2464. if ( !label_compare(mp->name, name)) {
  2465.     modID = mp->modid;
  2466.     break;
  2467. }
  2468.     if ( modID == -1 ) {
  2469. DEBUGMSGTL(("unload-mib", "Module %s not found to unloadn", name));
  2470. return MODULE_NOT_FOUND;
  2471.     }
  2472.     unload_module_by_ID( modID, tree_head );
  2473.     mp->no_imports = -1; /* mark as unloaded */
  2474.     return MODULE_LOADED_OK; /* Well, you know what I mean! */
  2475. }
  2476. static void
  2477. new_module (const char *name,
  2478.     const char *file)
  2479. {
  2480.     struct module *mp;
  2481.     for ( mp=module_head ; mp ; mp=mp->next )
  2482. if ( !label_compare(mp->name, name)) {
  2483.     DEBUGMSGTL(("parse-mibs", "Module %s already notedn", name));
  2484. /* Not the same file */
  2485.     if (label_compare(mp->file, file)) {
  2486. if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  2487.     snmp_log(LOG_WARNING,
  2488.                              "Warning: Module %s was in %s now is %sn",
  2489.                              name, mp->file, file);
  2490. /* Use the new one in preference */
  2491. free(mp->file);
  2492.                 mp->file = strdup(file);
  2493.             }
  2494.     return;
  2495. }
  2496. /* Add this module to the list */
  2497.     DEBUGMSGTL(("parse-mibs", "  Module %d %s is in %sn", max_module, name, file));
  2498.     mp = (struct module *) calloc(1, sizeof(struct module));
  2499.     if (mp == NULL) return;
  2500.     mp->name = strdup(name);
  2501.     mp->file = strdup(file);
  2502.     mp->imports = NULL;
  2503.     mp->no_imports = -1; /* Not yet loaded */
  2504.     mp->modid = max_module;
  2505.     ++max_module;
  2506.     mp->next = module_head; /* Or add to the *end* of the list? */
  2507.     module_head = mp;
  2508. }
  2509. /*
  2510.  * Parses a mib file and returns a linked list of nodes found in the file.
  2511.  * Returns NULL on error.
  2512.  */
  2513. static struct node *
  2514. parse(FILE *fp,
  2515.       struct node *root)
  2516. {
  2517.     char token[MAXTOKEN];
  2518.     char name[MAXTOKEN];
  2519.     int type = LABEL;
  2520.     int lasttype = LABEL;
  2521. #define BETWEEN_MIBS          1
  2522. #define IN_MIB                2
  2523.     int state = BETWEEN_MIBS;
  2524.     struct node *np, *nnp;
  2525.     DEBUGMSGTL(("parse-file", "Parsing file:  %s...n", File));
  2526.     if (last_err_module) free(last_err_module); last_err_module = 0;
  2527.     np = root;
  2528.     if (np != NULL) {
  2529.         /* now find end of chain */
  2530.         while(np->next)
  2531.             np = np->next;
  2532.     }
  2533.     while (type != ENDOFFILE){
  2534.         if (lasttype == CONTINUE) lasttype = type;
  2535.         else type = lasttype = get_token(fp, token, MAXTOKEN);
  2536.         switch (type) {
  2537.         case END:
  2538.             if (state != IN_MIB){
  2539.                 print_error("Error, END before start of MIB", NULL, type);
  2540.                 return NULL;
  2541.             }
  2542.     else {
  2543. struct module *mp;
  2544. #ifdef TEST
  2545. printf("nNodes for Module %s:n", name);
  2546. print_nodes( stdout, np );
  2547. #endif
  2548. for (mp = module_head; mp; mp = mp->next)
  2549.     if (mp->modid == current_module) break;
  2550. do_linkup(mp, root);
  2551. np = root = NULL;
  2552.     }
  2553.             state = BETWEEN_MIBS;
  2554. #ifdef TEST
  2555.             if (ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS))
  2556.               xmalloc_stats (stderr);
  2557. #endif
  2558.             continue;
  2559.         case IMPORTS:
  2560.             parse_imports( fp );
  2561.             continue;
  2562.         case EXPORTS:
  2563.             while (type != SEMI && type != ENDOFFILE)
  2564.                 type = get_token(fp, token, MAXTOKEN);
  2565.             continue;
  2566.         case LABEL:
  2567.             break;
  2568.         case ENDOFFILE:
  2569.             continue;
  2570.         default:
  2571.     strcpy(name, token);
  2572.     type = get_token(fp, token, MAXTOKEN);
  2573.     nnp = NULL;
  2574.     if (type == MACRO) {
  2575. nnp = parse_macro(fp, name);
  2576. if (nnp == NULL){
  2577.     print_error("Bad parse of MACRO", NULL, type);
  2578.     /*return NULL;*/
  2579. }
  2580. free_node(nnp); /* IGNORE */
  2581. nnp = NULL;
  2582.     }
  2583.     else
  2584. print_error(name, "is a reserved word", lasttype);
  2585.             continue;         /* see if we can parse the rest of the file */
  2586.         }
  2587.         strcpy(name, token);
  2588.         type = get_token(fp, token, MAXTOKEN);
  2589.         nnp = NULL;
  2590. /* Handle obsolete method to assign an object identifier to a
  2591.    module*/
  2592. if (lasttype == LABEL && type == LEFTBRACKET) {
  2593.     while (type != RIGHTBRACKET && type != ENDOFFILE)
  2594. type = get_token(fp, token, MAXTOKEN);
  2595.     if (type == ENDOFFILE){
  2596. print_error("Expected "}"", token, type);
  2597. return NULL;
  2598.     }
  2599.     type = get_token(fp, token, MAXTOKEN);
  2600. }
  2601.         switch (type) {
  2602.         case DEFINITIONS:
  2603.             if (state != BETWEEN_MIBS){
  2604.                 print_error("Error, nested MIBS", NULL, type);
  2605.                 return NULL;
  2606.             }
  2607.             state = IN_MIB;
  2608.             current_module = which_module( name );
  2609.             if ( current_module == -1 ) {
  2610.                 new_module(name, File);
  2611.                 current_module = which_module(name);
  2612.             }
  2613.             DEBUGMSGTL(("parse-mibs", "Parsing MIB: %d %sn", current_module, name));
  2614.             while ((type = get_token (fp, token, MAXTOKEN)) != ENDOFFILE)
  2615.                 if (type == BEGIN) break;
  2616.             break;
  2617.         case OBJTYPE:
  2618.             nnp = parse_objecttype(fp, name);
  2619.             if (nnp == NULL){
  2620.                 print_error("Bad parse of OBJECT-TYPE", NULL, type);
  2621.                 return NULL;
  2622.             }
  2623.             break;
  2624.         case OBJGROUP:
  2625.             nnp = parse_objectgroup(fp, name);
  2626.             if (nnp == NULL){
  2627.                 print_error("Bad parse of OBJECT-GROUP", NULL, type);
  2628.                 return NULL;
  2629.             }
  2630.             break;
  2631.         case TRAPTYPE:
  2632.             nnp = parse_trapDefinition(fp, name);
  2633.             if (nnp == NULL){
  2634.                 print_error("Bad parse of TRAP-TYPE", NULL, type);
  2635.                 return NULL;
  2636.             }
  2637.             break;
  2638.         case NOTIFTYPE:
  2639.             nnp = parse_notificationDefinition(fp, name);
  2640.             if (nnp == NULL){
  2641.                 print_error("Bad parse of NOTIFICATION-TYPE", NULL, type);
  2642.                 return NULL;
  2643.             }
  2644.             break;
  2645.         case COMPLIANCE:
  2646.             nnp = parse_compliance(fp, name);
  2647.             if (nnp == NULL){
  2648.                 print_error("Bad parse of MODULE-COMPLIANCE", NULL, type);
  2649.                 return NULL;
  2650.             }
  2651.             break;
  2652.         case CAPABILITIES:
  2653.             nnp = parse_capabilities(fp, name);
  2654.             if (nnp == NULL){
  2655.                 print_error("Bad parse of AGENT-CAPABILITIES", NULL, type);
  2656.                 return NULL;
  2657.             }
  2658.             break;
  2659.         case MACRO:
  2660.             nnp = parse_macro(fp, name);
  2661.             if (nnp == NULL){
  2662.                 print_error("Bad parse of MACRO", NULL, type);
  2663.                 /*return NULL;*/
  2664.             }
  2665.             free_node(nnp); /* IGNORE */
  2666.     nnp = NULL;
  2667.             break;
  2668.         case MODULEIDENTITY:
  2669.             nnp = parse_moduleIdentity(fp, name);
  2670.             if (nnp == NULL){
  2671.                 print_error("Bad parse of MODULE-IDENTITY", NULL, type);
  2672.                 return NULL;
  2673.             }
  2674.             break;
  2675.         case OBJID:
  2676.             type = get_token(fp, token, MAXTOKEN);
  2677.             if (type != EQUALS){
  2678.                 print_error("Expected "::="", token, type);
  2679.                 return NULL;
  2680.             }
  2681.             nnp = parse_objectid(fp, name);
  2682.             if (nnp == NULL){
  2683.                 print_error("Bad parse of OBJECT IDENTIFIER", NULL, type);
  2684.                 return NULL;
  2685.             }
  2686.             break;
  2687.         case EQUALS:
  2688.             nnp = parse_asntype(fp, name, &type, token);
  2689.             lasttype = CONTINUE;
  2690.             break;
  2691.         case ENDOFFILE:
  2692.             break;
  2693.         default:
  2694.     print_error("Bad operator", token, type);
  2695.     return NULL;
  2696.         }
  2697.         if (nnp) {
  2698.             if (np) np->next = nnp;
  2699.             else np = root = nnp;
  2700.             while (np->next) np = np->next;
  2701.         }
  2702.     }
  2703.     DEBUGMSGTL(("parse-file", "End of file (%s)n", File));
  2704.     return root;
  2705. }
  2706. /* return zero if character is not a label character. */
  2707. static int
  2708. is_labelchar (int ich)
  2709. {
  2710.     if ((isalnum(ich)) || (ich == '-'))
  2711.     return 1;
  2712.     if (ich == '_' && ds_get_boolean(DS_LIBRARY_ID, DS_LIB_MIB_PARSE_LABEL))
  2713.     return 1;
  2714.     return 0;
  2715. }
  2716. /*
  2717.  * Parses a token from the file.  The type of the token parsed is returned,
  2718.  * and the text is placed in the string pointed to by token.
  2719.  * Warning: this method may recurse.
  2720.  */
  2721. static int
  2722. get_token(FILE *fp,
  2723.   char *token,
  2724.   int maxtlen)
  2725. {
  2726.     register int ch, ch_next;
  2727.     register char *cp = token;
  2728.     register int hash = 0;
  2729.     register struct tok *tp;
  2730.     int too_long = 0;
  2731.     /* skip all white space */
  2732.     do {
  2733.         ch = getc(fp);
  2734.         if (ch == 'n')
  2735.             Line++;
  2736.     }
  2737.     while(isspace(ch) && ch != EOF);
  2738.     *cp++ = (char)ch; *cp = '';
  2739.     switch (ch) {
  2740.     case EOF:
  2741.         return ENDOFFILE;
  2742.     case '"':
  2743.         return parseQuoteString(fp, token, maxtlen);
  2744.     case ''': /* binary or hex constant */
  2745. while ((ch = getc(fp)) != EOF && ch != ''' && cp-token < maxtlen-2)
  2746.     *cp++ = (char)ch;
  2747. if (ch == ''') {
  2748.     unsigned long val = 0;
  2749.     *cp++ = ''';
  2750. ch = getc(fp);
  2751.     *cp++ = (char)ch;
  2752.     *cp = 0;
  2753.     cp = token+1;
  2754.     switch (ch) {
  2755.     case EOF:
  2756. return ENDOFFILE;
  2757.     case 'b':
  2758.     case 'B':
  2759. while ((ch = *cp++) != ''')
  2760.     if (ch != '0' && ch != '1') return LABEL;
  2761.     else val = val * 2 + ch - '0';
  2762. break;
  2763.     case 'h':
  2764.     case 'H':
  2765. while ((ch = *cp++) != ''')
  2766.     if ('0' <= ch && ch <= '9') val = val*16+ch-'0';
  2767.     else if ('a' <= ch && ch <= 'f') val = val*16+ch-'a'+10;
  2768.     else if ('A' <= ch && ch <= 'F') val = val*16+ch-'A'+10;
  2769.     else return LABEL;
  2770. break;
  2771.     default:
  2772. return LABEL;
  2773.     }
  2774.     sprintf(token, "%ld", val);
  2775.     return NUMBER;
  2776. }
  2777. else return LABEL;
  2778.     case '(':
  2779. return LEFTPAREN;
  2780.     case ')':
  2781. return RIGHTPAREN;
  2782.     case '{':
  2783. return LEFTBRACKET;
  2784.     case '}':
  2785. return RIGHTBRACKET;
  2786.     case ';':
  2787. return SEMI;
  2788.     case ',':
  2789. return COMMA;
  2790.     case '|':
  2791. return BAR;
  2792.     case '.':
  2793. ch_next = getc(fp);
  2794. if (ch_next == '.') return RANGE;
  2795. ungetc(ch_next, fp);
  2796. return LABEL;
  2797.     case ':':
  2798. ch_next = getc(fp);
  2799. if (ch_next != ':') {
  2800.     ungetc(ch_next, fp);
  2801.     return LABEL;
  2802. }
  2803. ch_next = getc(fp);
  2804. if (ch_next != '=') {
  2805.     ungetc(ch_next, fp);
  2806.     return LABEL;
  2807. }
  2808. return EQUALS;
  2809.     case '-':
  2810. ch_next = getc(fp);
  2811. if (ch_next == '-') {
  2812.   if (ds_get_boolean(DS_LIBRARY_ID, DS_LIB_MIB_COMMENT_TERM)) {
  2813.     /* Treat the rest of this line as a comment. */
  2814.     while ((ch_next != EOF) && (ch_next != 'n'))
  2815.     ch_next = getc(fp);
  2816.   } else {
  2817.             /* Treat the rest of the line or until another '--' as a comment */
  2818.             /* (this is the "technically" correct way to parse comments) */
  2819.     ch = ' ';
  2820.     ch_next = getc(fp);
  2821.     while (ch_next != EOF && ch_next != 'n' &&
  2822. (ch != '-' || ch_next != '-')) {
  2823. ch = ch_next; ch_next = getc(fp);
  2824.     }
  2825.   }
  2826.     if (ch_next == EOF) return ENDOFFILE;
  2827.     if (ch_next == 'n') Line++;
  2828.     return get_token (fp, token, maxtlen);
  2829. }
  2830. ungetc(ch_next, fp);
  2831.     default:
  2832. /*
  2833.  * Accumulate characters until end of token is found.  Then attempt to
  2834.  * match this token as a reserved word.  If a match is found, return the
  2835.  * type.  Else it is a label.
  2836.  */
  2837. if (!is_labelchar(ch)) return LABEL;
  2838. hash += tolower(ch);
  2839.   more:
  2840. while (is_labelchar(ch_next = getc(fp))) {
  2841.     hash += tolower(ch_next);
  2842.     if (cp - token < maxtlen - 1) *cp++ = (char)ch_next;
  2843.     else too_long = 1;
  2844. }
  2845. ungetc(ch_next, fp);
  2846. *cp = '';
  2847. if (too_long)
  2848.     print_error("Warning: token too long", token, CONTINUE);
  2849. for (tp = buckets[BUCKET(hash)]; tp; tp = tp->next) {
  2850.     if ((tp->hash == hash) && (!label_compare(tp->name, token)))
  2851. break;
  2852. }
  2853. if (tp) {
  2854.     if (tp->token != CONTINUE) return (tp->token);
  2855.     while (isspace((ch_next = getc(fp))))
  2856. if (ch_next == 'n') Line++;
  2857.     if (ch_next == EOF) return ENDOFFILE;
  2858.     if (isalnum(ch_next)) {
  2859. *cp++ = (char)ch_next;
  2860. hash += tolower(ch_next);
  2861. goto more;
  2862.     }
  2863. }
  2864. if (token[0] == '-' || isdigit(token[0])) {
  2865.    for(cp = token+1; *cp; cp++)
  2866.       if (!isdigit(*cp))
  2867.   return LABEL;
  2868.    return NUMBER;
  2869. }
  2870. return LABEL;
  2871.     }
  2872. }
  2873. int
  2874. snmp_get_token(FILE *fp,
  2875.   char *token,
  2876.   int maxtlen)
  2877. {
  2878.     return get_token(fp, token, maxtlen);
  2879. }
  2880. /*
  2881.  * Returns the root of the whole tree
  2882.  *   (for backwards compatability)
  2883.  */
  2884. struct tree *
  2885. read_mib(const char *filename)
  2886. {
  2887.     FILE *fp;
  2888.     char token[MAXTOKEN];
  2889.     fp = fopen(filename, "r");
  2890.     if (fp == NULL) {
  2891.         snmp_log_perror(filename);
  2892.         return NULL;
  2893.     }
  2894.     Line = 1;
  2895.     File = filename;
  2896.     DEBUGMSGTL(("parse-mibs", "Parsing file: %s...n", filename));
  2897.     get_token( fp, token, MAXTOKEN);
  2898.     fclose(fp);
  2899.     new_module(token, filename);
  2900.     (void) read_module(token);
  2901.     return tree_head;
  2902. }
  2903. struct tree *
  2904. read_all_mibs(void)
  2905. {
  2906.     struct module *mp;
  2907.     for ( mp=module_head ; mp ; mp=mp->next )
  2908. if ( mp->no_imports == -1 )
  2909.             read_module( mp->name );
  2910.     adopt_orphans();
  2911.     return tree_head;
  2912. }
  2913. static int
  2914. parseQuoteString(FILE *fp,
  2915.  char *token,
  2916.  int maxtlen)
  2917. {
  2918.     register int ch;
  2919.     int count = 0;
  2920.     int too_long = 0;
  2921.     char *token_start = token;
  2922.     for (ch = getc(fp); ch != EOF; ch = getc(fp)) {
  2923.         if (ch == 'r') continue;
  2924.         if (ch == 'n') {
  2925.             Line++;
  2926.         }
  2927.         else if (ch == '"') {
  2928.             *token = '';
  2929.             if (too_long &&
  2930.                 ds_get_int(DS_LIBRARY_ID, DS_LIB_MIB_WARNINGS) > 1)
  2931.             {
  2932.                 /* show short form for brevity sake */
  2933.                 char ch_save = *(token_start + 50);
  2934.                 *(token_start + 50) = '';
  2935.                 print_error ("Warning: string too long",
  2936.                              token_start, QUOTESTRING);
  2937.                 *(token_start + 50) = ch_save;
  2938.             }
  2939.             return QUOTESTRING;
  2940.         }
  2941.         /* maximum description length check.  If greater, keep parsing
  2942.            but truncate the string */
  2943.         if (++count < maxtlen)
  2944.             *token++ = (char)ch;
  2945.         else too_long = 1;
  2946.     }
  2947.     return 0;
  2948. }
  2949. /*
  2950.  * struct index_list *
  2951.  * getIndexes(FILE *fp):
  2952.  *   This routine parses a string like  { blah blah blah } and returns a
  2953.  *   list of the strings enclosed within it.
  2954.  *
  2955.  */
  2956. static struct index_list *
  2957. getIndexes(FILE *fp, struct index_list **retp) {
  2958.   int type;
  2959.   char token[MAXTOKEN];
  2960.   char nextIsImplied = 0;
  2961.   struct index_list *mylist = NULL;
  2962.   struct index_list **mypp = &mylist;
  2963.   free_indexes(retp);
  2964.   type = get_token(fp, token, MAXTOKEN);
  2965.   if (type != LEFTBRACKET) {
  2966.     return NULL;
  2967.   }
  2968.   type = get_token(fp, token, MAXTOKEN);
  2969.   while (type != RIGHTBRACKET && type != ENDOFFILE) {
  2970.     if ((type == LABEL) || (type & SYNTAX_MASK)) {
  2971.       *mypp = (struct index_list *) calloc(1, sizeof(struct index_list));
  2972.       if (*mypp) {
  2973.         (*mypp)->ilabel = strdup(token);
  2974.         (*mypp)->isimplied = nextIsImplied;
  2975.         mypp = &(*mypp)->next;
  2976.         nextIsImplied = 0;
  2977.       }
  2978.     } else if (type == IMPLIED) {
  2979.         nextIsImplied = 1;
  2980.     }
  2981.     type = get_token(fp, token, MAXTOKEN);
  2982.   }
  2983.   *retp = mylist;
  2984.   return mylist;
  2985. }
  2986. static void
  2987. free_indexes(struct index_list **spp) {
  2988.   if (spp && *spp) {
  2989.   struct index_list *pp, *npp;
  2990.   pp = *spp; *spp = NULL;
  2991.   while(pp) {
  2992.     npp = pp->next;
  2993.     if (pp->ilabel) free(pp->ilabel);
  2994.     free(pp);
  2995.     pp = npp;
  2996.   }
  2997.   }
  2998. }
  2999. static void
  3000. free_ranges(struct range_list **spp) {
  3001.   if (spp && *spp) {
  3002.   struct range_list *pp, *npp;
  3003.   pp = *spp; *spp = NULL;
  3004.   while(pp) {
  3005.     npp = pp->next;
  3006.     free(pp);
  3007.     pp = npp;
  3008.   }
  3009.   }
  3010. }
  3011. static void
  3012. free_enums(struct enum_list **spp)
  3013. {
  3014.   if (spp && *spp) {
  3015.   struct enum_list *pp, *npp;
  3016.   pp = *spp; *spp = NULL;
  3017.   while(pp)
  3018.   {
  3019.     npp = pp->next;
  3020.     if (pp->label) free(pp->label);
  3021.     free(pp);
  3022.     pp = npp;
  3023.   }
  3024.   }
  3025. }
  3026. static struct enum_list *
  3027. copy_enums (struct enum_list *sp)
  3028. {
  3029.   struct enum_list *xp = NULL, **spp = &xp;
  3030.   while (sp) {
  3031.     *spp = (struct enum_list *) calloc(1, sizeof(struct enum_list));
  3032.     if (!*spp) break;
  3033.     (*spp)->label = strdup(sp->label);
  3034.     (*spp)->value = sp->value;
  3035.     spp = &(*spp)->next;
  3036.     sp = sp->next;
  3037.   }
  3038.   return (xp);
  3039. }
  3040. static struct range_list *
  3041. copy_ranges (struct range_list *sp)
  3042. {
  3043.   struct range_list *xp = NULL, **spp = &xp;
  3044.   while (sp) {
  3045.     *spp = (struct range_list *) calloc(1, sizeof(struct range_list));
  3046.     if (!*spp) break;
  3047.     (*spp)->low = sp->low;
  3048.     (*spp)->high = sp->high;
  3049.     spp = &(*spp)->next;
  3050.     sp = sp->next;
  3051.   }
  3052.   return (xp);
  3053. }
  3054. /*
  3055. static struct index_list *
  3056. copy_indexes (struct index_list *sp)
  3057. {
  3058.   struct index_list *xp = NULL, **spp = &xp;
  3059.   while (sp) {
  3060.     *spp = (struct index_list *) calloc(1, sizeof(struct index_list));
  3061.     if (!*spp) break;
  3062.     (*spp)->ilabel = strdup(sp->ilabel);
  3063.     spp = &(*spp)->next;
  3064.     sp = sp->next;
  3065.   }
  3066.   return (xp);
  3067. }
  3068. */
  3069. /*
  3070.  * This routine parses a string like  { blah blah blah } and returns OBJID if
  3071.  * it is well formed, and NULL if not.
  3072.  */
  3073. static int
  3074. tossObjectIdentifier(FILE *fp)
  3075. {
  3076.     int type;
  3077.     char token[MAXTOKEN];
  3078.     int bracketcount = 1;
  3079.     type = get_token(fp, token, MAXTOKEN);
  3080.     if (type != LEFTBRACKET)
  3081.         return 0;
  3082.     while ((type != RIGHTBRACKET || bracketcount > 0) && type != ENDOFFILE )
  3083.     {
  3084.         type = get_token(fp, token, MAXTOKEN);
  3085.         if (type == LEFTBRACKET)
  3086.           bracketcount++;
  3087.         else if (type == RIGHTBRACKET)
  3088.           bracketcount--;
  3089.     }
  3090.     if (type == RIGHTBRACKET)
  3091.         return OBJID;
  3092.     else
  3093.         return 0;
  3094. }
  3095. struct tree *
  3096. find_node(const char *name,
  3097.   struct tree *subtree)    /* Unused */
  3098. {
  3099.   return( find_tree_node( name, -1 ));
  3100. }
  3101. struct module *
  3102. find_module(int mid)
  3103. {
  3104.   struct module *mp;
  3105.   for(mp=module_head; mp!=NULL; mp = mp->next) {
  3106.     if (mp->modid == mid)
  3107.       break;
  3108.   }
  3109.   if (mp != 0)
  3110.     return mp;
  3111.   return NULL;
  3112. }
  3113. static char leave_indent[256];
  3114. static int leave_was_simple;
  3115. static void print_mib_leaves(FILE *f, struct tree *tp, int width)
  3116. #ifdef UCD_SNMP
  3117. struct tree *ntp;
  3118.   char *ip = leave_indent+strlen(leave_indent)-1;
  3119.   char last_ipch = *ip;
  3120.   *ip = '+';
  3121.   if (tp->type == 0)
  3122.     fprintf(f, "%s--%s(%ld)n", leave_indent, tp->label, tp->subid);
  3123.   else {
  3124.     const char *acc, *typ;
  3125.     int size = 0;
  3126.     switch (tp->access) {
  3127.     case MIB_ACCESS_NOACCESS: acc = "----"; break;
  3128.     case MIB_ACCESS_READONLY: acc = "-R--"; break;
  3129.     case MIB_ACCESS_WRITEONLY: acc = "--W-"; break;
  3130.     case MIB_ACCESS_READWRITE: acc = "-RW-"; break;
  3131.     case MIB_ACCESS_NOTIFY: acc = "---N"; break;
  3132.     case MIB_ACCESS_CREATE: acc = "CR--"; break;
  3133.     default: acc = "    "; break;
  3134.     }
  3135.     switch (tp->type) {
  3136.     case TYPE_OBJID: typ = "ObjID    "; break;
  3137.     case TYPE_OCTETSTR: typ = "String   "; size = 1; break;
  3138.     case TYPE_INTEGER:
  3139. if (tp->enums) typ = "EnumVal  ";
  3140. else typ = "Integer  "; break;
  3141.     case TYPE_NETADDR: typ = "NetAddr  "; break;
  3142.     case TYPE_IPADDR: typ = "IpAddr   "; break;
  3143.     case TYPE_COUNTER: typ = "Counter  "; break;
  3144.     case TYPE_GAUGE: typ = "Gauge    "; break;
  3145.     case TYPE_TIMETICKS: typ = "TimeTicks"; break;
  3146.     case TYPE_OPAQUE: typ = "Opaque   "; size = 1; break;
  3147.     case TYPE_NULL: typ = "Null     "; break;
  3148.     case TYPE_COUNTER64: typ = "Counter64"; break;
  3149.     case TYPE_BITSTRING: typ = "BitString"; break;
  3150.     case TYPE_NSAPADDRESS: typ = "NsapAddr "; break;
  3151.     case TYPE_UINTEGER: typ = "UInteger "; break;
  3152.     default: typ = "         "; break;
  3153.     }
  3154.     fprintf(f, "%s-- %s %s %s(%ld)n", leave_indent, acc, typ, tp->label, tp->subid);
  3155.     *ip = last_ipch;
  3156.     if (tp->tc_index >= 0)
  3157.       fprintf(f, "%s        Textual Convention: %sn", leave_indent,
  3158.       tclist[tp->tc_index].descriptor);
  3159.     if (tp->enums) {
  3160.       struct enum_list *ep = tp->enums;
  3161.       int cpos = 0, cmax = width - strlen(leave_indent) - 16;
  3162.       fprintf(f, "%s        Values: ", leave_indent);
  3163.       while (ep) {
  3164. char buf[80];
  3165. int bufw;
  3166. if (ep != tp->enums) fprintf(f, ", ");
  3167. sprintf(buf, "%s(%d)", ep->label, ep->value);
  3168. cpos += (bufw = strlen(buf) + 2);
  3169. if (cpos >= cmax) {
  3170.   fprintf(f, "n%s                ", leave_indent);
  3171.   cpos = bufw;
  3172. }
  3173. fprintf(f, "%s", buf);
  3174. ep = ep->next;
  3175.       }
  3176.       fprintf(f, "n");
  3177.     }
  3178.     if (tp->ranges) {
  3179.       struct range_list *rp = tp->ranges;
  3180.       if (size) fprintf(f, "%s        Size: ", leave_indent);
  3181.       else fprintf(f, "%s        Range: ", leave_indent);
  3182.       while (rp) {
  3183.         if (rp != tp->ranges) fprintf(f, " | ");
  3184. if (rp->low == rp->high) fprintf(f, "%d", rp->low);
  3185. else fprintf(f, "%d..%d", rp->low, rp->high);
  3186. rp = rp->next;
  3187.       }
  3188.       fprintf(f, "n");
  3189.     }
  3190.   }
  3191.   *ip = last_ipch;
  3192.   strcat(leave_indent, "  |");
  3193.   leave_was_simple = tp->type != 0;
  3194.   { int i, j, count = 0;
  3195.     struct leave {
  3196.       oid id;
  3197.       struct tree *tp;
  3198.     } *leaves, *lp;
  3199.     for (ntp = tp->child_list; ntp; ntp = ntp->next_peer) count++;
  3200.     if (count) {
  3201.       leaves = (struct leave *)calloc(count, sizeof(struct leave));
  3202.       if (!leaves) return;
  3203.       for (ntp = tp->child_list, count = 0; ntp; ntp = ntp->next_peer) {
  3204. for (i = 0, lp = leaves; i < count; i++, lp++)
  3205.   if (lp->id >= ntp->subid) break;
  3206. for (j = count; j > i; j--) leaves[j] = leaves[j-1];
  3207. lp->id = ntp->subid;
  3208. lp->tp = ntp;
  3209. count++;
  3210.       }
  3211.       for (i = 1, lp = leaves; i <= count; i++, lp++) {
  3212. if (!leave_was_simple || lp->tp->type == 0)
  3213.   fprintf(f, "%sn", leave_indent);
  3214. if (i == count) ip[3] = ' ';
  3215. print_mib_leaves(f, lp->tp, width);
  3216.       }
  3217.       free(leaves);
  3218.       leave_was_simple = 0;
  3219.     }
  3220.   }
  3221.   ip[1] = 0;
  3222. #endif
  3223. }
  3224. void print_mib_tree(FILE *f, struct tree *tp, int width)
  3225. {
  3226.   leave_indent[0] = ' ';
  3227.   leave_indent[1] = 0;
  3228.   leave_was_simple = 1;
  3229.   print_mib_leaves(f, tp, width);
  3230. }
  3231. /*
  3232.  * Merge the parsed object identifier with the existing node.
  3233.  * If there is a problem with the identifier, release the existing node.
  3234.  */
  3235. static struct node *
  3236. merge_parse_objectid(struct node *np,
  3237.      FILE *fp,
  3238.      char *name)
  3239. {
  3240.     struct node *nnp;
  3241.     nnp = parse_objectid(fp, name);
  3242.     if (nnp) {
  3243. /* apply last OID sub-identifier data to the information */
  3244. /* already collected for this node. */
  3245. struct node *headp, *nextp;
  3246. int ncount = 0;
  3247. nextp = headp = nnp;
  3248. while (nnp->next) {
  3249.     nextp = nnp;
  3250.     ncount++;
  3251.     nnp = nnp->next;
  3252. }
  3253. np->label = nnp->label;
  3254. np->subid = nnp->subid;
  3255. np->modid = nnp->modid;
  3256. np->parent = nnp->parent;
  3257. free(nnp);
  3258. if (ncount) {
  3259.     nextp->next = np;
  3260.     np = headp;
  3261. }
  3262.     }
  3263.     else {
  3264.         free_node(np); np = NULL;
  3265.     }
  3266.     return np;
  3267. }
  3268. /*
  3269.  * transfer data to tree from node
  3270.  *
  3271.  * move pointers for alloc'd data from np to tp.
  3272.  * this prevents them from being freed when np is released.
  3273.  * parent member is not moved.
  3274.  *
  3275.  * CAUTION: nodes may be repeats of existing tree nodes.
  3276.  * This can happen especially when resolving IMPORT clauses.
  3277.  *
  3278.  */
  3279. #ifdef UCD_SNMP
  3280. static void
  3281. tree_from_node(struct tree *tp, struct node *np)
  3282. {
  3283.    free_partial_tree(tp, FALSE);
  3284.     tp->label = np->label;  np->label = NULL;
  3285.     tp->enums = np->enums;  np->enums = NULL;
  3286.     tp->ranges = np->ranges;  np->ranges = NULL;
  3287.     tp->indexes = np->indexes;  np->indexes = NULL;
  3288.     tp->tc_index = np->tc_index;
  3289.     tp->access = np->access;
  3290.     tp->status = np->status;
  3291.     tp->hint = np->hint;  np->hint = NULL;
  3292.     tp->units = np->units;  np->units = NULL;
  3293.     tp->description = np->description;  np->description = NULL;
  3294.     tp->subid = np->subid;
  3295.     tp->type = translation_table[np->type];
  3296.     set_function(tp);
  3297. }
  3298.  #endif