2 * Central Regulatory Domain Agent for Linux
4 * Userspace helper which sends regulatory domains to Linux via nl80211
13 #include <arpa/inet.h>
15 #include <netlink/genl/genl.h>
16 #include <netlink/genl/family.h>
17 #include <netlink/genl/ctrl.h>
18 #include <netlink/msg.h>
19 #include <netlink/attr.h>
20 #include <linux/nl80211.h>
25 #include <openssl/objects.h>
26 #include <openssl/bn.h>
27 #include <openssl/rsa.h>
28 #include <openssl/sha.h>
36 #include "keys-gcrypt.c"
39 struct nl80211_state {
40 struct nl_handle *nl_handle;
41 struct nl_cache *nl_cache;
42 struct genl_family *nl80211;
45 static int nl80211_init(struct nl80211_state *state)
49 state->nl_handle = nl_handle_alloc();
50 if (!state->nl_handle) {
51 fprintf(stderr, "Failed to allocate netlink handle.\n");
55 if (genl_connect(state->nl_handle)) {
56 fprintf(stderr, "Failed to connect to generic netlink.\n");
58 goto out_handle_destroy;
61 state->nl_cache = genl_ctrl_alloc_cache(state->nl_handle);
62 if (!state->nl_cache) {
63 fprintf(stderr, "Failed to allocate generic netlink cache.\n");
65 goto out_handle_destroy;
68 state->nl80211 = genl_ctrl_search_by_name(state->nl_cache, "nl80211");
69 if (!state->nl80211) {
70 fprintf(stderr, "nl80211 not found.\n");
78 nl_cache_free(state->nl_cache);
80 nl_handle_destroy(state->nl_handle);
84 static void nl80211_cleanup(struct nl80211_state *state)
86 genl_family_put(state->nl80211);
87 nl_cache_free(state->nl_cache);
88 nl_handle_destroy(state->nl_handle);
91 static int reg_handler(struct nl_msg *msg, void *arg)
93 printf("=== reg_handler() called\n");
97 static int wait_handler(struct nl_msg *msg, void *arg)
105 static int error_handler(struct sockaddr_nl *nla, struct nlmsgerr *err, void *arg)
107 fprintf(stderr, "nl80211 error %d\n", err->error);
111 int isalpha_upper(char letter)
113 if (letter >= 65 && letter <= 90)
118 static int is_alpha2(char *alpha2)
120 if (isalpha_upper(alpha2[0]) && isalpha_upper(alpha2[1]))
125 static int is_world_regdom(char *alpha2)
128 if (alpha2[0] == 48 && alpha2[1] == 48)
133 static void *get_file_ptr(__u8 *db, int dblen, int structlen, __be32 ptr)
135 __u32 p = ntohl(ptr);
137 if (p > dblen - structlen) {
138 fprintf(stderr, "Invalid database file, bad pointer!\n");
142 return (void *)(db + p);
145 static int put_reg_rule(__u8 *db, int dblen, __be32 ruleptr, struct nl_msg *msg)
147 struct regdb_file_reg_rule *rule;
148 struct regdb_file_freq_range *freq;
149 struct regdb_file_power_rule *power;
151 rule = get_file_ptr(db, dblen, sizeof(*rule), ruleptr);
152 freq = get_file_ptr(db, dblen, sizeof(*freq), rule->freq_range_ptr);
153 power = get_file_ptr(db, dblen, sizeof(*power), rule->power_rule_ptr);
155 NLA_PUT_U32(msg, NL80211_ATTR_REG_RULE_FLAGS, ntohl(rule->flags));
156 NLA_PUT_U32(msg, NL80211_ATTR_FREQ_RANGE_START, ntohl(freq->start_freq));
157 NLA_PUT_U32(msg, NL80211_ATTR_FREQ_RANGE_END, ntohl(freq->end_freq));
158 NLA_PUT_U32(msg, NL80211_ATTR_FREQ_RANGE_MAX_BW, ntohl(freq->max_bandwidth));
159 NLA_PUT_U32(msg, NL80211_ATTR_POWER_RULE_MAX_ANT_GAIN, ntohl(power->max_antenna_gain));
160 NLA_PUT_U32(msg, NL80211_ATTR_POWER_RULE_MAX_EIRP, ntohl(power->max_eirp));
168 int main(int argc, char **argv)
173 struct regdb_file_header *header;
174 struct regdb_file_reg_country *countries;
175 int dblen, siglen, num_countries, i, j, r;
178 struct nl80211_state nlstate;
179 struct nl_cb *cb = NULL;
181 int found_country = 0;
184 struct regdb_file_reg_rules_collection *rcoll;
185 struct regdb_file_reg_country *country;
186 struct nlattr *nl_reg_rules;
191 __u8 hash[SHA_DIGEST_LENGTH];
195 gcry_mpi_t mpi_e, mpi_n;
196 gcry_sexp_t rsa, signature, data;
201 char *regdb = "/usr/lib/crda/regulatory.bin";
204 fprintf(stderr, "Usage: %s\n", argv[0]);
208 env_country = getenv("COUNTRY");
210 fprintf(stderr, "COUNTRY environment variable not set.\n");
215 if (!is_alpha2(env_country) && !is_world_regdom(env_country)) {
216 fprintf(stderr, "Invalid alpha2 set in COUNTRY\n");
220 memcpy(alpha2, env_country, 2);
222 r = nl80211_init(&nlstate);
226 fd = open(regdb, O_RDONLY);
228 perror("failed to open db file");
233 if (fstat(fd, &stat)) {
234 perror("failed to fstat db file");
239 dblen = stat.st_size;
241 db = mmap(NULL, dblen, PROT_READ, MAP_PRIVATE, fd, 0);
242 if (db == MAP_FAILED) {
243 perror("failed to mmap db file");
248 header = get_file_ptr(db, dblen, sizeof(*header), 0);
250 if (ntohl(header->magic) != REGDB_MAGIC) {
251 fprintf(stderr, "Invalid database magic\n");
256 if (ntohl(header->version) != REGDB_VERSION) {
257 fprintf(stderr, "Invalid database version\n");
262 siglen = ntohl(header->signature_length);
263 /* adjust dblen so later sanity checks don't run into the signature */
266 if (dblen <= sizeof(*header)) {
267 fprintf(stderr, "Invalid signature length %d\n", siglen);
272 /* verify signature */
276 fprintf(stderr, "Failed to create RSA key\n");
281 if (SHA1(db, dblen, hash) != hash) {
282 fprintf(stderr, "Failed to calculate SHA sum\n");
287 for (i = 0; i < sizeof(keys)/sizeof(keys[0]); i++) {
291 if (RSA_size(rsa) != siglen)
294 ok = RSA_verify(NID_sha1, hash, SHA_DIGEST_LENGTH,
295 db + dblen, siglen, rsa) == 1;
301 fprintf(stderr, "Database signature wrong\n");
310 BN_print_fp(stdout, &keys[0].n);
316 gcry_check_version(NULL);
319 gcry_md_hash_buffer(GCRY_MD_SHA1, hash, db, dblen);
321 if (gcry_sexp_build(&data, NULL, "(data (flags pkcs1) (hash sha1 %b))",
323 fprintf(stderr, "failed to build data expression\n");
327 if (gcry_sexp_build(&signature, NULL, "(sig-val (rsa (s %b)))",
328 siglen, db + dblen)) {
329 fprintf(stderr, "failed to build signature expression\n");
333 for (i = 0; i < sizeof(keys)/sizeof(keys[0]); i++) {
334 if (gcry_mpi_scan(&mpi_e, GCRYMPI_FMT_USG,
335 keys[0].e, keys[0].len_e, NULL) ||
336 gcry_mpi_scan(&mpi_n, GCRYMPI_FMT_USG,
337 keys[0].n, keys[0].len_n, NULL)) {
338 fprintf(stderr, "failed to convert numbers\n");
342 if (gcry_sexp_build(&rsa, NULL,
343 "(public-key (rsa (n %m) (e %m)))",
345 fprintf(stderr, "failed to build rsa key\n");
349 if (!gcry_pk_verify(signature, data, rsa)) {
356 fprintf(stderr, "Database signature wrong\n");
362 num_countries = ntohl(header->reg_country_num);
363 countries = get_file_ptr(db, dblen,
364 sizeof(struct regdb_file_reg_country) * num_countries,
365 header->reg_country_ptr);
367 for (i = 0; i < num_countries; i++) {
368 country = countries + i;
369 if (memcmp(country->alpha2, alpha2, 2) == 0) {
375 if (!found_country) {
376 fprintf(stderr, "failed to find a country match in regulatory database\n");
382 fprintf(stderr, "failed to allocate netlink msg\n");
386 genlmsg_put(msg, 0, 0, genl_family_get_id(nlstate.nl80211), 0,
387 0, NL80211_CMD_SET_REG, 0);
390 rcoll = get_file_ptr(db, dblen, sizeof(*rcoll), country->reg_collection_ptr);
391 num_rules = ntohl(rcoll->reg_rule_num);
392 /* re-get pointer with sanity checking for num_rules */
393 rcoll = get_file_ptr(db, dblen,
394 sizeof(*rcoll) + num_rules * sizeof(__be32),
395 country->reg_collection_ptr);
397 NLA_PUT_STRING(msg, NL80211_ATTR_REG_ALPHA2, (char *) country->alpha2);
399 nl_reg_rules = nla_nest_start(msg, NL80211_ATTR_REG_RULES);
402 goto nla_put_failure;
405 for (j = 0; j < num_rules; j++) {
406 struct nlattr *nl_reg_rule;
407 nl_reg_rule = nla_nest_start(msg, i);
409 goto nla_put_failure;
411 r = put_reg_rule(db, dblen, rcoll->reg_rule_ptrs[j], msg);
413 goto nla_put_failure;
415 nla_nest_end(msg, nl_reg_rule);
418 nla_nest_end(msg, nl_reg_rules);
420 cb = nl_cb_alloc(NL_CB_CUSTOM);
424 r = nl_send_auto_complete(nlstate.nl_handle, msg);
427 fprintf(stderr, "failed to send regulatory request: %d\n", r);
431 nl_cb_set(cb, NL_CB_VALID, NL_CB_CUSTOM, reg_handler, NULL);
432 nl_cb_set(cb, NL_CB_ACK, NL_CB_CUSTOM, wait_handler, &finished);
433 nl_cb_err(cb, NL_CB_CUSTOM, error_handler, NULL);
436 r = nl_wait_for_ack(nlstate.nl_handle);
438 fprintf(stderr, "failed to set regulatory domain: %d\n", r);
448 nl80211_cleanup(&nlstate);