8 #include <arpa/inet.h> /* ntohl */
13 #include <openssl/objects.h>
14 #include <openssl/bn.h>
15 #include <openssl/rsa.h>
16 #include <openssl/sha.h>
24 #include "keys-gcrypt.c"
27 static void *get_file_ptr(__u8 *db, int dblen, int structlen, __be32 ptr)
31 if (p > dblen - structlen) {
32 fprintf(stderr, "Invalid database file, bad pointer!\n");
36 return (void *)(db + p);
39 static void print_reg_rule(__u8 *db, int dblen, __be32 ruleptr)
41 struct regdb_file_reg_rule *rule;
42 struct regdb_file_freq_range *freq;
43 struct regdb_file_power_rule *power;
47 rule = get_file_ptr(db, dblen, sizeof(*rule), ruleptr);
48 freq = get_file_ptr(db, dblen, sizeof(*freq), rule->freq_range_ptr);
49 power = get_file_ptr(db, dblen, sizeof(*power), rule->power_rule_ptr);
51 printf("\t(%.3f - %.3f @ %.3f), ",
52 ((float)ntohl(freq->start_freq))/1000.0,
53 ((float)ntohl(freq->end_freq))/1000.0,
54 ((float)ntohl(freq->max_bandwidth))/1000.0);
58 if (power->max_antenna_gain)
59 printf("%.2f, ", ((float)ntohl(power->max_antenna_gain)/100.0));
64 printf("%.2f)", ((float)ntohl(power->max_eirp)/100.0));
68 flags = ntohl(rule->flags);
69 if (flags & RRF_NO_OFDM)
71 if (flags & RRF_NO_CCK)
73 if (flags & RRF_NO_INDOOR)
74 printf(", NO-INDOOR");
75 if (flags & RRF_NO_OUTDOOR)
76 printf(", NO-OUTDOOR");
79 if (flags & RRF_PTP_ONLY)
81 if (flags & RRF_PTMP_ONLY)
82 printf(", PTMP-ONLY");
83 if (flags & RRF_PASSIVE_SCAN)
84 printf(", PASSIVE-SCAN");
85 if (flags & RRF_NO_IBSS)
87 if (flags & RRF_NO_HT40)
90 ep = (flags & RRF_EDGE_POWER_MASK) >> EDGE_POWER_SHIFT;
92 printf(", EDGE-POWER-%d", ep);
97 int main(int argc, char **argv)
102 struct regdb_file_header *header;
103 struct regdb_file_reg_country *countries;
104 int dblen, siglen, num_countries, i, j;
107 __u8 hash[SHA_DIGEST_LENGTH];
111 gcry_mpi_t mpi_e, mpi_n;
112 gcry_sexp_t rsa, signature, data;
118 fprintf(stderr, "Usage: %s <filename>\n", argv[0]);
122 fd = open(argv[1], O_RDONLY);
124 perror("failed to open db file");
128 if (fstat(fd, &stat)) {
129 perror("failed to fstat db file");
133 dblen = stat.st_size;
135 db = mmap(NULL, dblen, PROT_READ, MAP_PRIVATE, fd, 0);
136 if (db == MAP_FAILED) {
137 perror("failed to mmap db file");
141 header = get_file_ptr(db, dblen, sizeof(*header), 0);
143 if (ntohl(header->magic) != REGDB_MAGIC) {
144 fprintf(stderr, "Invalid database magic\n");
148 if (ntohl(header->version) != REGDB_VERSION) {
149 fprintf(stderr, "Invalid database version\n");
153 siglen = ntohl(header->signature_length);
154 /* adjust dblen so later sanity checks don't run into the signature */
157 if (dblen <= sizeof(*header)) {
158 fprintf(stderr, "Invalid signature length %d\n", siglen);
162 /* verify signature */
166 fprintf(stderr, "Failed to create RSA key\n");
170 if (SHA1(db, dblen, hash) != hash) {
171 fprintf(stderr, "Failed to calculate SHA sum\n");
175 for (i = 0; i < sizeof(keys)/sizeof(keys[0]); i++) {
179 if (RSA_size(rsa) != siglen)
182 ok = RSA_verify(NID_sha1, hash, SHA_DIGEST_LENGTH,
183 db + dblen, siglen, rsa) == 1;
189 fprintf(stderr, "Database signature wrong\n");
197 BN_print_fp(stdout, &keys[0].n);
204 gcry_md_hash_buffer(GCRY_MD_SHA1, hash, db, dblen);
206 if (gcry_sexp_build(&data, NULL, "(data (flags pkcs1) (hash sha1 %b))",
208 fprintf(stderr, "failed to build data expression\n");
212 if (gcry_sexp_build(&signature, NULL, "(sig-val (rsa (s %b)))",
213 siglen, db + dblen)) {
214 fprintf(stderr, "failed to build signature expression\n");
218 for (i = 0; i < sizeof(keys)/sizeof(keys[0]); i++) {
219 if (gcry_mpi_scan(&mpi_e, GCRYMPI_FMT_USG,
220 keys[0].e, keys[0].len_e, NULL) ||
221 gcry_mpi_scan(&mpi_n, GCRYMPI_FMT_USG,
222 keys[0].n, keys[0].len_n, NULL)) {
223 fprintf(stderr, "failed to convert numbers\n");
227 if (gcry_sexp_build(&rsa, NULL,
228 "(public-key (rsa (n %m) (e %m)))",
230 fprintf(stderr, "failed to build rsa key\n");
234 if (!gcry_pk_verify(signature, data, rsa)) {
241 fprintf(stderr, "Database signature wrong\n");
247 num_countries = ntohl(header->reg_country_num);
248 countries = get_file_ptr(db, dblen,
249 sizeof(struct regdb_file_reg_country) * num_countries,
250 header->reg_country_ptr);
252 for (i = 0; i < num_countries; i++) {
253 struct regdb_file_reg_rules_collection *rcoll;
254 struct regdb_file_reg_country *country = countries + i;
257 printf("country %.2s:\n", country->alpha2);
258 rcoll = get_file_ptr(db, dblen, sizeof(*rcoll), country->reg_collection_ptr);
259 num_rules = ntohl(rcoll->reg_rule_num);
260 /* re-get pointer with sanity checking for num_rules */
261 rcoll = get_file_ptr(db, dblen,
262 sizeof(*rcoll) + num_rules * sizeof(__be32),
263 country->reg_collection_ptr);
264 for (j = 0; j < num_rules; j++)
265 print_reg_rule(db, dblen, rcoll->reg_rule_ptrs[j]);