1*05b00f60SXin Li#!/bin/sh 2*05b00f60SXin Li# 3*05b00f60SXin Li# Copyright (c) 1990, 1996 4*05b00f60SXin Li# John Robert LoVerso. All rights reserved. 5*05b00f60SXin Li# SMIv2 parsing copyright (c) 1999 6*05b00f60SXin Li# William C. Fenner. 7*05b00f60SXin Li# 8*05b00f60SXin Li# Redistribution and use in source and binary forms, with or without 9*05b00f60SXin Li# modification, are permitted provided that the following conditions 10*05b00f60SXin Li# are met: 11*05b00f60SXin Li# 12*05b00f60SXin Li# 1. Redistributions of source code must retain the above copyright 13*05b00f60SXin Li# notices, this list of conditions and the following disclaimer. 14*05b00f60SXin Li# 15*05b00f60SXin Li# 2. Redistributions in binary form must reproduce the above copyright 16*05b00f60SXin Li# notices, this list of conditions and the following disclaimer in the 17*05b00f60SXin Li# documentation and/or other materials provided with the distribution. 18*05b00f60SXin Li# 19*05b00f60SXin Li# THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR 20*05b00f60SXin Li# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 21*05b00f60SXin Li# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 22*05b00f60SXin Li# IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT, 23*05b00f60SXin Li# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 24*05b00f60SXin Li# NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 25*05b00f60SXin Li# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 26*05b00f60SXin Li# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 27*05b00f60SXin Li# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 28*05b00f60SXin Li# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 29*05b00f60SXin Li 30*05b00f60SXin Li# 31*05b00f60SXin Li# This script will read either ASN.1-style MIB files or the ".defs" files 32*05b00f60SXin Li# created by the ISODE "mosy" program on such files. 33*05b00f60SXin Li# 34*05b00f60SXin Li# The output of this script is the "mib.h" file used by tcpdumps' ASN.1/SNMP 35*05b00f60SXin Li# decoding code. 36*05b00f60SXin Li# 37*05b00f60SXin Li# This script needs to be run by "gawk" (GNU awk). "nawk" will work, but 38*05b00f60SXin Li# dump will get a recursion error if you process LARGE mibs. While it would 39*05b00f60SXin Li# by farily easy to rewrite this not to use recursion (and also easy to 40*05b00f60SXin Li# eliminate use of gsub and functions to use classic "awk"), you have to 41*05b00f60SXin Li# order the structure declarations in defined-first order for the compiler 42*05b00f60SXin Li# not to barf; too bad tsort doesn't take arguments. 43*05b00f60SXin Li# 44*05b00f60SXin Li 45*05b00f60SXin Licat << EOF 46*05b00f60SXin Li/* 47*05b00f60SXin Li * This file was generated by tcpdump/makemib on `date` 48*05b00f60SXin Li * You probably don't want to edit this by hand! 49*05b00f60SXin Li * 50*05b00f60SXin Li * struct mib somename = { desc, oid-octet, type, child-pointer, next-pointer 51*05b00f60SXin Li}; 52*05b00f60SXin Li */ 53*05b00f60SXin Li 54*05b00f60SXin LiEOF 55*05b00f60SXin Li 56*05b00f60SXin Liawk ' 57*05b00f60SXin LiBEGIN { 58*05b00f60SXin Li debug=0; 59*05b00f60SXin Li # for sanity, we prep the namespace with objects from RFC-1155 60*05b00f60SXin Li # (we manually establish the root) 61*05b00f60SXin Li oid["iso"]=1 62*05b00f60SXin Li oidadd("org", "iso", 3) 63*05b00f60SXin Li oidadd("dod", "org", 6) 64*05b00f60SXin Li oidadd("internet", "dod", 1) 65*05b00f60SXin Li oidadd("directory", "internet", 1) 66*05b00f60SXin Li oidadd("mgmt", "internet", 2) 67*05b00f60SXin Li#XXX oidadd("mib", "mgmt", 1) 68*05b00f60SXin Li oidadd("mib-2", "mgmt", 1) 69*05b00f60SXin Li oidadd("experimental", "internet", 3) 70*05b00f60SXin Li oidadd("private", "internet", 4) 71*05b00f60SXin Li oidadd("enterprises", "private", 1) 72*05b00f60SXin Li oidadd("ip", "mib-2", 4) 73*05b00f60SXin Li oidadd("transmission", "mib-2", 10) 74*05b00f60SXin Li 75*05b00f60SXin Li holddesc="none" 76*05b00f60SXin Li} 77*05b00f60SXin Li 78*05b00f60SXin Li# 79*05b00f60SXin Li# Read mosy "*.defs" file. mosy does all the parsing work; we just read 80*05b00f60SXin Li# its simple and straightforward output. It would not be too hard to make 81*05b00f60SXin Li# tcpdump directly read mosy output, but... 82*05b00f60SXin Li# 83*05b00f60SXin Li# Ignore these unless the current file is called something.defs; false 84*05b00f60SXin Li# positives are too common in DESCRIPTIONs. 85*05b00f60SXin Li 86*05b00f60SXin LiNF > 1 && index($2,".")>0 && FILENAME ~ /\.defs/ { 87*05b00f60SXin Li # currently ignore items of the form "{ iso.3.6.1 }" 88*05b00f60SXin Li if (split($2, p, ".") == 2) { 89*05b00f60SXin Li oidadd($1, p[1], p[2]) 90*05b00f60SXin Li } 91*05b00f60SXin Li next 92*05b00f60SXin Li} 93*05b00f60SXin Li 94*05b00f60SXin Li# 95*05b00f60SXin Li# Must be a MIB file 96*05b00f60SXin Li# Make it easier to parse - used to be done by sed 97*05b00f60SXin Li{ sub(/--\*.*\*--/, ""); sub(/--.*/, ""); gsub(/[{}]/, " & "); } 98*05b00f60SXin Li 99*05b00f60SXin Li# 100*05b00f60SXin Li# this next section is simple and naive, but does the job ok 101*05b00f60SXin Li# 102*05b00f60SXin Li 103*05b00f60SXin Li# foo OBJECT IDENTIFIER ::= { baz 17 } 104*05b00f60SXin Li# or 105*05b00f60SXin Li# foo OBJECT IDENTIFIER ::= 106*05b00f60SXin Li# { baz 17 } 107*05b00f60SXin Li$2$3$4 == "OBJECTIDENTIFIER::=" { 108*05b00f60SXin Li holddesc="none" 109*05b00f60SXin Li if (NF == 8) 110*05b00f60SXin Li oidadd($1, $6, $7) 111*05b00f60SXin Li if (NF == 4) 112*05b00f60SXin Li holddesc=$1 113*05b00f60SXin Li next 114*05b00f60SXin Li} 115*05b00f60SXin Li$1 == "{" && holddesc != "none" && NF == 4 { 116*05b00f60SXin Li oidadd(holddesc, $2, $3) 117*05b00f60SXin Li holddesc="none" 118*05b00f60SXin Li} 119*05b00f60SXin Li# 120*05b00f60SXin Li# foo OBJECT IDENTIFIER 121*05b00f60SXin Li# ::= { bar 1 } 122*05b00f60SXin Li$2$3 == "OBJECTIDENTIFIER" && $1 != "SYNTAX" && NF == 3 { 123*05b00f60SXin Li holddesc=$1 124*05b00f60SXin Li} 125*05b00f60SXin Li# 126*05b00f60SXin Li# foo 127*05b00f60SXin Li# OBJECT IDENTIFIER ::= { bar 1 } 128*05b00f60SXin Li# a couple of heuristics to exclude single words in e.g. long 129*05b00f60SXin Li# DESCRIPTION clauses 130*05b00f60SXin LiNF == 1 && $1 ~ "[a-z][a-z]*[A-Z]" && $1 !~ /[(){}.,]/ && holddesc == "none" { 131*05b00f60SXin Li holddesc=$1 132*05b00f60SXin Li} 133*05b00f60SXin Li$1$2$3 == "OBJECTIDENTIFIER::=" && holddesc != "none" { 134*05b00f60SXin Li oidadd(holddesc, $5, $6) 135*05b00f60SXin Li holddesc="none" 136*05b00f60SXin Li} 137*05b00f60SXin Li# 138*05b00f60SXin Li# "normal" style 139*05b00f60SXin Li# foo OBJECT-TYPE ... 140*05b00f60SXin Li# ... 141*05b00f60SXin Li# ::= { baz 5 } 142*05b00f60SXin Li$2 == "MODULE-IDENTITY" || $2 == "MODULE-COMPLIANCE" || 143*05b00f60SXin Li $2 == "OBJECT-IDENTITY" || $2 == "OBJECT-TYPE" || 144*05b00f60SXin Li $2 == "OBJECT-GROUP" || 145*05b00f60SXin Li $2 == "NOTIFICATION-TYPE" || $2 == "NOTIFICATION-GROUP" { 146*05b00f60SXin Li holddesc=$1 147*05b00f60SXin Li} 148*05b00f60SXin Li$1 == "::=" && holddesc != "none" && NF == 5 { 149*05b00f60SXin Li oidadd(holddesc, $3, $4) 150*05b00f60SXin Li holddesc="none" 151*05b00f60SXin Li} 152*05b00f60SXin Li# 153*05b00f60SXin Li# foo ::= { baz 17 } 154*05b00f60SXin Li$2$3 == "::={" { 155*05b00f60SXin Li oidadd($1,$4,$5) 156*05b00f60SXin Li holddesc="none" 157*05b00f60SXin Li} 158*05b00f60SXin Li 159*05b00f60SXin Li 160*05b00f60SXin Li# 161*05b00f60SXin Li# End of the road - output the data. 162*05b00f60SXin Li# 163*05b00f60SXin Li 164*05b00f60SXin LiEND { 165*05b00f60SXin Li print "struct obj" 166*05b00f60SXin Li dump("iso") 167*05b00f60SXin Li print "*mibroot = &_iso_obj;" 168*05b00f60SXin Li} 169*05b00f60SXin Li 170*05b00f60SXin Lifunction inn(file) { 171*05b00f60SXin Li if (file == "" || file == "-") 172*05b00f60SXin Li return "" 173*05b00f60SXin Li return " in " file 174*05b00f60SXin Li} 175*05b00f60SXin Li 176*05b00f60SXin Li# 177*05b00f60SXin Li# add a new object to the tree 178*05b00f60SXin Li# 179*05b00f60SXin Li# new OBJECT IDENTIFIER ::= { parent value } 180*05b00f60SXin Li# 181*05b00f60SXin Li 182*05b00f60SXin Lifunction oidadd(new, parent, value) { 183*05b00f60SXin Li # Ignore 0.0 184*05b00f60SXin Li if (parent == "0" && value == 0) 185*05b00f60SXin Li return 186*05b00f60SXin Li if (debug) 187*05b00f60SXin Li print "/* oidadd" inn(FILENAME) ":", new, "in", parent, "as", value, "line", $0, "*/" 188*05b00f60SXin Li # use safe C identifiers 189*05b00f60SXin Li gsub(/[-&\/]/,"",new) 190*05b00f60SXin Li gsub(/[-&\/]/,"",parent) 191*05b00f60SXin Li # check if parent missing 192*05b00f60SXin Li if (oid[parent] == "") { 193*05b00f60SXin Li printf "/* parse problem%s: no parent for %s.%s(%d) */\n", \ 194*05b00f60SXin Li inn(FILENAME), parent, new, value 195*05b00f60SXin Li return 196*05b00f60SXin Li } 197*05b00f60SXin Li # check if parent.value already exists 198*05b00f60SXin Li if (oid[new] > 0 && oid[new] != value) { 199*05b00f60SXin Li printf "/* parse problem%s: dup %s.%s(%d) != old (%d) */\n", \ 200*05b00f60SXin Li inn(FILENAME), parent, new, value, oid[new] 201*05b00f60SXin Li return 202*05b00f60SXin Li } 203*05b00f60SXin Li # check for new name for parent.value 204*05b00f60SXin Li if (child[parent] != "") { 205*05b00f60SXin Li for (sib = child[parent]; sib != ""; sib = sibling[sib]) 206*05b00f60SXin Li if (oid[sib] == value) { 207*05b00f60SXin Li if (new != sib) 208*05b00f60SXin Li printf "/* parse problem%s: new name" \ 209*05b00f60SXin Li " \"%s\"" \ 210*05b00f60SXin Li " for %s.%s(%d) ignored */\n", \ 211*05b00f60SXin Li inn(FILENAME), new, parent, \ 212*05b00f60SXin Li sib, value 213*05b00f60SXin Li return 214*05b00f60SXin Li } 215*05b00f60SXin Li } 216*05b00f60SXin Li 217*05b00f60SXin Li oid[new]=value 218*05b00f60SXin Li if (child[parent] == "") { 219*05b00f60SXin Li child[parent] = new 220*05b00f60SXin Li } else { 221*05b00f60SXin Li sibling[new] = child[parent] 222*05b00f60SXin Li child[parent] = new 223*05b00f60SXin Li } 224*05b00f60SXin Li} 225*05b00f60SXin Li 226*05b00f60SXin Li# 227*05b00f60SXin Li# old(?) routine to recurse down the tree (in postfix order for convenience) 228*05b00f60SXin Li# 229*05b00f60SXin Li 230*05b00f60SXin Lifunction dump(item, c, s) { 231*05b00f60SXin Li# newitem=sofar"."item"("oid[item]")" 232*05b00f60SXin Li# printf "/* %s c=%s s=%s */\n", newitem, child[item], sibling[item] 233*05b00f60SXin Li c="NULL" 234*05b00f60SXin Li if (child[item] != "") { 235*05b00f60SXin Li dump(child[item]) 236*05b00f60SXin Li c = "&_"child[item]"_obj" 237*05b00f60SXin Li } 238*05b00f60SXin Li s="NULL" 239*05b00f60SXin Li if (sibling[item] != "") { 240*05b00f60SXin Li dump(sibling[item]) 241*05b00f60SXin Li s = "&_"sibling[item]"_obj" 242*05b00f60SXin Li } 243*05b00f60SXin Li printf "_%s_obj = {\n\t\"%s\", %d, 0,\n\t%s, %s\n},\n", \ 244*05b00f60SXin Li item, item, oid[item], c, s 245*05b00f60SXin Li} 246*05b00f60SXin Li' $@ 247*05b00f60SXin Liexit 0 248