xref: /aosp_15_r20/external/tcpdump/makemib (revision 05b00f6010a2396e3db2409989fc67270046269f)
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