#!/bin/sh # # Copyright (c) 1990, 1996, by John Robert LoVerso. # All rights reserved. # SMIv2 parsing copyright (c) 1999 by William C. Fenner. # # Redistribution and use in source and binary forms are permitted # provided that the above copyright notice and this paragraph are # duplicated in all such forms and that any documentation, # advertising materials, and other materials related to such # distribution and use acknowledge that the software was developed # by John Robert LoVerso. # THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. # # @(#) $Id: makemib,v 1.2 1999-11-21 17:24:15 fenner Exp $ (jlv) # # This script will read either ASN.1-style MIB files or the ".defs" files # created by the ISODE "mosy" program on such files. # # The output of this script is the "mib.h" file used by tcpdumps' ASN.1/SNMP # decoding code. # # This script needs to be run by "gawk" (GNU awk). "nawk" will work, but # dump will get a recursion error if you process LARGE mibs. While it would # by farily easy to rewrite this not to use recursion (and also easy to # eliminate use of gsub and functions to use classic "awk"), you have to # order the structure declarations in defined-first order for the compiler # not to barf; too bad tsort doesn't take arguments. # cat << EOF /* * This file was generated by tcpdump/makemib on `date` * You probably don't want to edit this by hand! * * struct mib somename = { desc, oid-octet, type, child-pointer, next-pointer }; */ EOF awk ' BEGIN { debug=0; # for sanity, we prep the namespace with objects from RFC-1155 # (we manually establish the root) oid["iso"]=1 oidadd("org", "iso", 3) oidadd("dod", "org", 6) oidadd("internet", "dod", 1) oidadd("directory", "internet", 1) oidadd("mgmt", "internet", 2) #XXX oidadd("mib", "mgmt", 1) oidadd("mib-2", "mgmt", 1) oidadd("experimental", "internet", 3) oidadd("private", "internet", 4) oidadd("enterprises", "private", 1) oidadd("ip", "mib-2", 4) oidadd("transmission", "mib-2", 10) holddesc="none" } # # Read mosy "*.defs" file. mosy does all the parsing work; we just read # its simple and straightforward output. It would not be too hard to make # tcpdump directly read mosy output, but... # # Ignore these unless the current file is called something.defs; false # positives are too common in DESCRIPTIONs. NF > 1 && index($2,".")>0 && FILENAME ~ /\.defs/ { # currently ignore items of the form "{ iso.3.6.1 }" if (split($2, p, ".") == 2) { oidadd($1, p[1], p[2]) } next } # # Must be a MIB file # Make it easier to parse - used to be done by sed { sub(/--\*.*\*--/, ""); sub(/--.*/, ""); gsub(/[{}]/, " & "); } # # this next section is simple and naive, but does the job ok # # foo OBJECT IDENTIFIER ::= { baz 17 } # or # foo OBJECT IDENTIFIER ::= # { baz 17 } $2$3$4 == "OBJECTIDENTIFIER::=" { holddesc="none" if (NF == 8) oidadd($1, $6, $7) if (NF == 4) holddesc=$1 next } $1 == "{" && holddesc != "none" && NF == 4 { oidadd(holddesc, $2, $3) holddesc="none" } # # foo OBJECT IDENTIFIER # ::= { bar 1 } $2$3 == "OBJECTIDENTIFIER" && $1 != "SYNTAX" && NF == 3 { holddesc=$1 } # # foo # OBJECT IDENTIFIER ::= { bar 1 } # a couple of heuristics to exclude single words in e.g. long # DESCRIPTION clauses NF == 1 && $1 ~ "[a-z][a-z]*[A-Z]" && $1 !~ /[(){}.,]/ && holddesc == "none" { holddesc=$1 } $1$2$3 == "OBJECTIDENTIFIER::=" && holddesc != "none" { oidadd(holddesc, $5, $6) holddesc="none" } # # "normal" style # foo OBJECT-TYPE ... # ... # ::= { baz 5 } $2 == "MODULE-IDENTITY" || $2 == "MODULE-COMPLIANCE" || $2 == "OBJECT-IDENTITY" || $2 == "OBJECT-TYPE" || $2 == "OBJECT-GROUP" || $2 == "NOTIFICATION-TYPE" || $2 == "NOTIFICATION-GROUP" { holddesc=$1 } $1 == "::=" && holddesc != "none" && NF == 5 { oidadd(holddesc, $3, $4) holddesc="none" } # # foo ::= { baz 17 } $2$3 == "::={" { oidadd($1,$4,$5) holddesc="none" } # # End of the road - output the data. # END { print "struct obj" dump("iso") print "*mibroot = &_iso_obj;" } function inn(file) { if (file == "" || file == "-") return "" return " in " file } # # add a new object to the tree # # new OBJECT IDENTIFIER ::= { parent value } # function oidadd(new, parent, value) { # Ignore 0.0 if (parent == "0" && value == 0) return if (debug) print "/* oidadd" inn(FILENAME) ":", new, "in", parent, "as", value, "line", $0, "*/" # use safe C identifiers gsub(/[-&\/]/,"",new) gsub(/[-&\/]/,"",parent) # check if parent missing if (oid[parent] == "") { printf "/* parse problem%s: no parent for %s.%s(%d) */\n", \ inn(FILENAME), parent, new, value return } # check if parent.value already exists if (oid[new] > 0 && oid[new] != value) { printf "/* parse problem%s: dup %s.%s(%d) != old (%d) */\n", \ inn(FILENAME), parent, new, value, oid[new] return } # check for new name for parent.value if (child[parent] != "") { for (sib = child[parent]; sib != ""; sib = sibling[sib]) if (oid[sib] == value) { if (new != sib) printf "/* parse problem%s: new name" \ " \"%s\"" \ " for %s.%s(%d) ignored */\n", \ inn(FILENAME), new, parent, \ sib, value return } } oid[new]=value if (child[parent] == "") { child[parent] = new } else { sibling[new] = child[parent] child[parent] = new } } # # old(?) routine to recurse down the tree (in postfix order for convenience) # function dump(item, c, s) { # newitem=sofar"."item"("oid[item]")" # printf "/* %s c=%s s=%s */\n", newitem, child[item], sibling[item] c="NULL" if (child[item] != "") { dump(child[item]) c = "&_"child[item]"_obj" } s="NULL" if (sibling[item] != "") { dump(sibling[item]) s = "&_"sibling[item]"_obj" } printf "_%s_obj = {\n\t\"%s\", %d, 0,\n\t%s, %s\n},\n", \ item, item, oid[item], c, s } ' $@ exit 0