123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247 |
- #!/bin/sh
- #
- # Copyright (c) 1990, 1996
- # John Robert LoVerso. All rights reserved.
- # SMIv2 parsing copyright (c) 1999
- # William C. Fenner.
- #
- # Redistribution and use in source and binary forms, with or without
- # modification, are permitted provided that the following conditions
- # are met:
- #
- # 1. Redistributions of source code must retain the above copyright
- # notices, this list of conditions and the following disclaimer.
- #
- # 2. Redistributions in binary form must reproduce the above copyright
- # notices, this list of conditions and the following disclaimer in the
- # documentation and/or other materials provided with the distribution.
- #
- # THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR
- # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
- # OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
- # IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT,
- # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
- # NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
- # THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- #
- # 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
|