xref: /openbmc/linux/lib/build_OID_registry (revision 32c3c30a)
1a77ad6eaSDavid Howells#!/usr/bin/perl -w
2b4d0d230SThomas Gleixner# SPDX-License-Identifier: GPL-2.0-or-later
3a77ad6eaSDavid Howells#
4a77ad6eaSDavid Howells# Build a static ASN.1 Object Identified (OID) registry
5a77ad6eaSDavid Howells#
6a77ad6eaSDavid Howells# Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
7a77ad6eaSDavid Howells# Written by David Howells (dhowells@redhat.com)
8a77ad6eaSDavid Howells#
9a77ad6eaSDavid Howells
10a77ad6eaSDavid Howellsuse strict;
11*32c3c30aSUwe Kleine-Königuse Cwd qw(abs_path);
12a77ad6eaSDavid Howells
13a77ad6eaSDavid Howellsmy @names = ();
14a77ad6eaSDavid Howellsmy @oids = ();
15a77ad6eaSDavid Howells
16a77ad6eaSDavid Howellsif ($#ARGV != 1) {
17a77ad6eaSDavid Howells    print STDERR "Format: ", $0, " <in-h-file> <out-c-file>\n";
18a77ad6eaSDavid Howells    exit(2);
19a77ad6eaSDavid Howells}
20a77ad6eaSDavid Howells
21*32c3c30aSUwe Kleine-Königmy $abs_srctree = abs_path($ENV{'srctree'});
22*32c3c30aSUwe Kleine-König
23a77ad6eaSDavid Howells#
24a77ad6eaSDavid Howells# Open the file to read from
25a77ad6eaSDavid Howells#
26a77ad6eaSDavid Howellsopen IN_FILE, "<$ARGV[0]" || die;
27a77ad6eaSDavid Howellswhile (<IN_FILE>) {
28a77ad6eaSDavid Howells    chomp;
29a77ad6eaSDavid Howells    if (m!\s+OID_([a-zA-z][a-zA-Z0-9_]+),\s+/[*]\s+([012][.0-9]*)\s+[*]/!) {
30a77ad6eaSDavid Howells	push @names, $1;
31a77ad6eaSDavid Howells	push @oids, $2;
32a77ad6eaSDavid Howells    }
33a77ad6eaSDavid Howells}
34a77ad6eaSDavid Howellsclose IN_FILE || die;
35a77ad6eaSDavid Howells
36a77ad6eaSDavid Howells#
37a77ad6eaSDavid Howells# Open the files to write into
38a77ad6eaSDavid Howells#
39a77ad6eaSDavid Howellsopen C_FILE, ">$ARGV[1]" or die;
40a77ad6eaSDavid Howellsprint C_FILE "/*\n";
41*32c3c30aSUwe Kleine-Königprint C_FILE " * Automatically generated by ", $0 =~ s#^\Q$abs_srctree/\E##r, ".  Do not edit\n";
42a77ad6eaSDavid Howellsprint C_FILE " */\n";
43a77ad6eaSDavid Howells
44a77ad6eaSDavid Howells#
45a77ad6eaSDavid Howells# Split the data up into separate lists and also determine the lengths of the
46a77ad6eaSDavid Howells# encoded data arrays.
47a77ad6eaSDavid Howells#
48a77ad6eaSDavid Howellsmy @indices = ();
49a77ad6eaSDavid Howellsmy @lengths = ();
50a77ad6eaSDavid Howellsmy $total_length = 0;
51a77ad6eaSDavid Howells
52a77ad6eaSDavid Howellsfor (my $i = 0; $i <= $#names; $i++) {
53a77ad6eaSDavid Howells    my $name = $names[$i];
54a77ad6eaSDavid Howells    my $oid = $oids[$i];
55a77ad6eaSDavid Howells
56a77ad6eaSDavid Howells    my @components = split(/[.]/, $oid);
57a77ad6eaSDavid Howells
58a77ad6eaSDavid Howells    # Determine the encoded length of this OID
59a77ad6eaSDavid Howells    my $size = $#components;
60a77ad6eaSDavid Howells    for (my $loop = 2; $loop <= $#components; $loop++) {
61a77ad6eaSDavid Howells	my $c = $components[$loop];
62a77ad6eaSDavid Howells
63a77ad6eaSDavid Howells	# We will base128 encode the number
64a77ad6eaSDavid Howells	my $tmp = ($c == 0) ? 0 : int(log($c)/log(2));
65a77ad6eaSDavid Howells	$tmp = int($tmp / 7);
66a77ad6eaSDavid Howells	$size += $tmp;
67a77ad6eaSDavid Howells    }
68a77ad6eaSDavid Howells    push @lengths, $size;
69a77ad6eaSDavid Howells    push @indices, $total_length;
70a77ad6eaSDavid Howells    $total_length += $size;
71a77ad6eaSDavid Howells}
72a77ad6eaSDavid Howells
73a77ad6eaSDavid Howells#
74a77ad6eaSDavid Howells# Emit the look-up-by-OID index table
75a77ad6eaSDavid Howells#
76a77ad6eaSDavid Howellsprint C_FILE "\n";
77a77ad6eaSDavid Howellsif ($total_length <= 255) {
78a77ad6eaSDavid Howells    print C_FILE "static const unsigned char oid_index[OID__NR + 1] = {\n";
79a77ad6eaSDavid Howells} else {
80a77ad6eaSDavid Howells    print C_FILE "static const unsigned short oid_index[OID__NR + 1] = {\n";
81a77ad6eaSDavid Howells}
82a77ad6eaSDavid Howellsfor (my $i = 0; $i <= $#names; $i++) {
83a77ad6eaSDavid Howells    print C_FILE "\t[OID_", $names[$i], "] = ", $indices[$i], ",\n"
84a77ad6eaSDavid Howells}
85a77ad6eaSDavid Howellsprint C_FILE "\t[OID__NR] = ", $total_length, "\n";
86a77ad6eaSDavid Howellsprint C_FILE "};\n";
87a77ad6eaSDavid Howells
88a77ad6eaSDavid Howells#
89a77ad6eaSDavid Howells# Encode the OIDs
90a77ad6eaSDavid Howells#
91a77ad6eaSDavid Howellsmy @encoded_oids = ();
92a77ad6eaSDavid Howells
93a77ad6eaSDavid Howellsfor (my $i = 0; $i <= $#names; $i++) {
94a77ad6eaSDavid Howells    my @octets = ();
95a77ad6eaSDavid Howells
96a77ad6eaSDavid Howells    my @components = split(/[.]/, $oids[$i]);
97a77ad6eaSDavid Howells
98a77ad6eaSDavid Howells    push @octets, $components[0] * 40 + $components[1];
99a77ad6eaSDavid Howells
100a77ad6eaSDavid Howells    for (my $loop = 2; $loop <= $#components; $loop++) {
101a77ad6eaSDavid Howells	my $c = $components[$loop];
102a77ad6eaSDavid Howells
103a77ad6eaSDavid Howells	# Base128 encode the number
104a77ad6eaSDavid Howells	my $tmp = ($c == 0) ? 0 : int(log($c)/log(2));
105a77ad6eaSDavid Howells	$tmp = int($tmp / 7);
106a77ad6eaSDavid Howells
107a77ad6eaSDavid Howells	for (; $tmp > 0; $tmp--) {
108a77ad6eaSDavid Howells	    push @octets, (($c >> $tmp * 7) & 0x7f) | 0x80;
109a77ad6eaSDavid Howells	}
110a77ad6eaSDavid Howells	push @octets, $c & 0x7f;
111a77ad6eaSDavid Howells    }
112a77ad6eaSDavid Howells
113a77ad6eaSDavid Howells    push @encoded_oids, \@octets;
114a77ad6eaSDavid Howells}
115a77ad6eaSDavid Howells
116a77ad6eaSDavid Howells#
117a77ad6eaSDavid Howells# Create a hash value for each OID
118a77ad6eaSDavid Howells#
119a77ad6eaSDavid Howellsmy @hash_values = ();
120a77ad6eaSDavid Howellsfor (my $i = 0; $i <= $#names; $i++) {
121a77ad6eaSDavid Howells    my @octets = @{$encoded_oids[$i]};
122a77ad6eaSDavid Howells
123a77ad6eaSDavid Howells    my $hash = $#octets;
124a77ad6eaSDavid Howells    foreach (@octets) {
125a77ad6eaSDavid Howells	$hash += $_ * 33;
126a77ad6eaSDavid Howells    }
127a77ad6eaSDavid Howells
128a77ad6eaSDavid Howells    $hash = ($hash >> 24) ^ ($hash >> 16) ^ ($hash >> 8) ^ ($hash);
129a77ad6eaSDavid Howells
130a77ad6eaSDavid Howells    push @hash_values, $hash & 0xff;
131a77ad6eaSDavid Howells}
132a77ad6eaSDavid Howells
133a77ad6eaSDavid Howells#
134a77ad6eaSDavid Howells# Emit the OID data
135a77ad6eaSDavid Howells#
136a77ad6eaSDavid Howellsprint C_FILE "\n";
137a77ad6eaSDavid Howellsprint C_FILE "static const unsigned char oid_data[", $total_length, "] = {\n";
138a77ad6eaSDavid Howellsfor (my $i = 0; $i <= $#names; $i++) {
139a77ad6eaSDavid Howells    my @octets = @{$encoded_oids[$i]};
140a77ad6eaSDavid Howells    print C_FILE "\t";
141a77ad6eaSDavid Howells    print C_FILE $_, ", " foreach (@octets);
142a77ad6eaSDavid Howells    print C_FILE "\t// ", $names[$i];
143a77ad6eaSDavid Howells    print C_FILE "\n";
144a77ad6eaSDavid Howells}
145a77ad6eaSDavid Howellsprint C_FILE "};\n";
146a77ad6eaSDavid Howells
147a77ad6eaSDavid Howells#
148a77ad6eaSDavid Howells# Build the search index table (ordered by length then hash then content)
149a77ad6eaSDavid Howells#
150a77ad6eaSDavid Howellsmy @index_table = ( 0 .. $#names );
151a77ad6eaSDavid Howells
152a77ad6eaSDavid Howells@index_table = sort {
153a77ad6eaSDavid Howells    my @octets_a = @{$encoded_oids[$a]};
154a77ad6eaSDavid Howells    my @octets_b = @{$encoded_oids[$b]};
155a77ad6eaSDavid Howells
156a77ad6eaSDavid Howells    return $hash_values[$a] <=> $hash_values[$b]
157a77ad6eaSDavid Howells	if ($hash_values[$a] != $hash_values[$b]);
158a77ad6eaSDavid Howells    return $#octets_a <=> $#octets_b
159a77ad6eaSDavid Howells	if ($#octets_a != $#octets_b);
160a77ad6eaSDavid Howells    for (my $i = $#octets_a; $i >= 0; $i--) {
161a77ad6eaSDavid Howells	return $octets_a[$i] <=> $octets_b[$i]
162a77ad6eaSDavid Howells	    if ($octets_a[$i] != $octets_b[$i]);
163a77ad6eaSDavid Howells    }
164a77ad6eaSDavid Howells    return 0;
165a77ad6eaSDavid Howells
166a77ad6eaSDavid Howells} @index_table;
167a77ad6eaSDavid Howells
168a77ad6eaSDavid Howells#
169a77ad6eaSDavid Howells# Emit the search index and hash value table
170a77ad6eaSDavid Howells#
171a77ad6eaSDavid Howellsprint C_FILE "\n";
172a77ad6eaSDavid Howellsprint C_FILE "static const struct {\n";
173a77ad6eaSDavid Howellsprint C_FILE "\tunsigned char hash;\n";
174a77ad6eaSDavid Howellsif ($#names <= 255) {
175a77ad6eaSDavid Howells    print C_FILE "\tenum OID oid : 8;\n";
176a77ad6eaSDavid Howells} else {
177a77ad6eaSDavid Howells    print C_FILE "\tenum OID oid : 16;\n";
178a77ad6eaSDavid Howells}
179a77ad6eaSDavid Howellsprint C_FILE "} oid_search_table[OID__NR] = {\n";
180a77ad6eaSDavid Howellsfor (my $i = 0; $i <= $#names; $i++) {
181a77ad6eaSDavid Howells    my @octets = @{$encoded_oids[$index_table[$i]]};
182a77ad6eaSDavid Howells    printf(C_FILE "\t[%3u] = { %3u, OID_%-35s }, // ",
183a77ad6eaSDavid Howells	   $i,
184a77ad6eaSDavid Howells	   $hash_values[$index_table[$i]],
185a77ad6eaSDavid Howells	   $names[$index_table[$i]]);
186a77ad6eaSDavid Howells    printf C_FILE "%02x", $_ foreach (@octets);
187a77ad6eaSDavid Howells    print C_FILE "\n";
188a77ad6eaSDavid Howells}
189a77ad6eaSDavid Howellsprint C_FILE "};\n";
190a77ad6eaSDavid Howells
191a77ad6eaSDavid Howells#
192a77ad6eaSDavid Howells# Emit the OID debugging name table
193a77ad6eaSDavid Howells#
194a77ad6eaSDavid Howells#print C_FILE "\n";
195a77ad6eaSDavid Howells#print C_FILE "const char *const oid_name_table[OID__NR + 1] = {\n";
196a77ad6eaSDavid Howells#
197a77ad6eaSDavid Howells#for (my $i = 0; $i <= $#names; $i++) {
198a77ad6eaSDavid Howells#    print C_FILE "\t\"", $names[$i], "\",\n"
199a77ad6eaSDavid Howells#}
200a77ad6eaSDavid Howells#print C_FILE "\t\"Unknown-OID\"\n";
201a77ad6eaSDavid Howells#print C_FILE "};\n";
202a77ad6eaSDavid Howells
203a77ad6eaSDavid Howells#
204a77ad6eaSDavid Howells# Polish off
205a77ad6eaSDavid Howells#
206a77ad6eaSDavid Howellsclose C_FILE or die;
207