summaryrefslogtreecommitdiffstats
path: root/modsign-post-KS-jwb.patch
diff options
context:
space:
mode:
Diffstat (limited to 'modsign-post-KS-jwb.patch')
-rw-r--r--modsign-post-KS-jwb.patch9153
1 files changed, 9153 insertions, 0 deletions
diff --git a/modsign-post-KS-jwb.patch b/modsign-post-KS-jwb.patch
new file mode 100644
index 000000000..6dd81ffa0
--- /dev/null
+++ b/modsign-post-KS-jwb.patch
@@ -0,0 +1,9153 @@
+From 2cdfd353ac1a5b9f62398ef59b4a08b5b55ac089 Mon Sep 17 00:00:00 2001
+From: Rusty Russell <rusty@rustcorp.com.au>
+Date: Wed, 5 Sep 2012 12:32:17 +0930
+Subject: [PATCH 01/26] module: signature checking hook
+
+We do a very simple search for a particular string appended to the module
+(which is cache-hot and about to be SHA'd anyway). There's both a config
+option and a boot parameter which control whether we accept (and taint) or
+fail with unsigned modules.
+
+Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
+---
+ Documentation/kernel-parameters.txt | 6 +++
+ include/linux/module.h | 8 ++++
+ init/Kconfig | 14 ++++++
+ kernel/module.c | 88 ++++++++++++++++++++++++++++++++++++-
+ 4 files changed, 115 insertions(+), 1 deletion(-)
+
+diff --git a/Documentation/kernel-parameters.txt b/Documentation/kernel-parameters.txt
+index ad7e2e5..9b2b8d3 100644
+--- a/Documentation/kernel-parameters.txt
++++ b/Documentation/kernel-parameters.txt
+@@ -1582,6 +1582,12 @@ bytes respectively. Such letter suffixes can also be entirely omitted.
+ log everything. Information is printed at KERN_DEBUG
+ so loglevel=8 may also need to be specified.
+
++ module.sig_enforce
++ [KNL] When CONFIG_MODULE_SIG is set, this means that
++ modules without (valid) signatures will fail to load.
++ Note that if CONFIG_MODULE_SIG_ENFORCE is set, that
++ is always true, so this option does nothing.
++
+ mousedev.tap_time=
+ [MOUSE] Maximum time between finger touching and
+ leaving touchpad surface for touch to be considered
+diff --git a/include/linux/module.h b/include/linux/module.h
+index fbcafe2..7760c6d 100644
+--- a/include/linux/module.h
++++ b/include/linux/module.h
+@@ -21,6 +21,9 @@
+ #include <linux/percpu.h>
+ #include <asm/module.h>
+
++/* In stripped ARM and x86-64 modules, ~ is surprisingly rare. */
++#define MODULE_SIG_STRING "~Module signature appended~\n"
++
+ /* Not Yet Implemented */
+ #define MODULE_SUPPORTED_DEVICE(name)
+
+@@ -260,6 +263,11 @@ struct module
+ const unsigned long *unused_gpl_crcs;
+ #endif
+
++#ifdef CONFIG_MODULE_SIG
++ /* Signature was verified. */
++ bool sig_ok;
++#endif
++
+ /* symbols that will be GPL-only in the near future. */
+ const struct kernel_symbol *gpl_future_syms;
+ const unsigned long *gpl_future_crcs;
+diff --git a/init/Kconfig b/init/Kconfig
+index af6c7f8..7452e19 100644
+--- a/init/Kconfig
++++ b/init/Kconfig
+@@ -1585,6 +1585,20 @@ config MODULE_SRCVERSION_ALL
+ the version). With this option, such a "srcversion" field
+ will be created for all modules. If unsure, say N.
+
++config MODULE_SIG
++ bool "Module signature verification"
++ depends on MODULES
++ help
++ Check modules for valid signatures upon load: the signature
++ is simply appended to the module. For more information see
++ Documentation/module-signing.txt.
++
++config MODULE_SIG_FORCE
++ bool "Require modules to be validly signed"
++ depends on MODULE_SIG
++ help
++ Reject unsigned modules or signed modules for which we don't have a
++ key. Without this, such modules will simply taint the kernel.
+ endif # MODULES
+
+ config INIT_ALL_POSSIBLE
+diff --git a/kernel/module.c b/kernel/module.c
+index 4edbd9c..5c6f65c 100644
+--- a/kernel/module.c
++++ b/kernel/module.c
+@@ -102,6 +102,43 @@ static LIST_HEAD(modules);
+ struct list_head *kdb_modules = &modules; /* kdb needs the list of modules */
+ #endif /* CONFIG_KGDB_KDB */
+
++#ifdef CONFIG_MODULE_SIG
++#ifdef CONFIG_MODULE_SIG_FORCE
++static bool sig_enforce = true;
++#else
++static bool sig_enforce = false;
++
++static int param_set_bool_enable_only(const char *val,
++ const struct kernel_param *kp)
++{
++ int err;
++ bool test;
++ struct kernel_param dummy_kp = *kp;
++
++ dummy_kp.arg = &test;
++
++ err = param_set_bool(val, &dummy_kp);
++ if (err)
++ return err;
++
++ /* Don't let them unset it once it's set! */
++ if (!test && sig_enforce)
++ return -EROFS;
++
++ if (test)
++ sig_enforce = true;
++ return 0;
++}
++
++static const struct kernel_param_ops param_ops_bool_enable_only = {
++ .set = param_set_bool_enable_only,
++ .get = param_get_bool,
++};
++#define param_check_bool_enable_only param_check_bool
++
++module_param(sig_enforce, bool_enable_only, 0644);
++#endif /* !CONFIG_MODULE_SIG_FORCE */
++#endif /* CONFIG_MODULE_SIG */
+
+ /* Block module loading/unloading? */
+ int modules_disabled = 0;
+@@ -136,6 +173,7 @@ struct load_info {
+ unsigned long symoffs, stroffs;
+ struct _ddebug *debug;
+ unsigned int num_debug;
++ bool sig_ok;
+ struct {
+ unsigned int sym, str, mod, vers, info, pcpu;
+ } index;
+@@ -2399,7 +2437,45 @@ static inline void kmemleak_load_module(const struct module *mod,
+ }
+ #endif
+
+-/* Sets info->hdr and info->len. */
++#ifdef CONFIG_MODULE_SIG
++static int module_sig_check(struct load_info *info,
++ const void *mod, unsigned long *len)
++{
++ int err = 0;
++ const unsigned long markerlen = strlen(MODULE_SIG_STRING);
++ const void *p = mod, *end = mod + *len;
++
++ /* Poor man's memmem. */
++ while ((p = memchr(p, MODULE_SIG_STRING[0], end - p))) {
++ if (p + markerlen > end)
++ break;
++
++ if (memcmp(p, MODULE_SIG_STRING, markerlen) == 0) {
++ const void *sig = p + markerlen;
++ /* Truncate module up to signature. */
++ *len = p - mod;
++ err = mod_verify_sig(mod, *len,
++ sig, end - sig,
++ &info->sig_ok);
++ break;
++ }
++ p++;
++ }
++
++ /* Not having a signature is only an error if we're strict. */
++ if (!err && !info->sig_ok && sig_enforce)
++ err = -EKEYREJECTED;
++ return err;
++}
++#else /* !CONFIG_MODULE_SIG */
++static int module_sig_check(struct load_info *info,
++ void *mod, unsigned long *len)
++{
++ return 0;
++}
++#endif /* !CONFIG_MODULE_SIG */
++
++/* Sets info->hdr, info->len and info->sig_ok. */
+ static int copy_and_check(struct load_info *info,
+ const void __user *umod, unsigned long len,
+ const char __user *uargs)
+@@ -2419,6 +2495,10 @@ static int copy_and_check(struct load_info *info,
+ goto free_hdr;
+ }
+
++ err = module_sig_check(info, hdr, &len);
++ if (err)
++ goto free_hdr;
++
+ /* Sanity checks against insmoding binaries or wrong arch,
+ weird elf version */
+ if (memcmp(hdr->e_ident, ELFMAG, SELFMAG) != 0
+@@ -2886,6 +2966,12 @@ static struct module *load_module(void __user *umod,
+ goto free_copy;
+ }
+
++#ifdef CONFIG_MODULE_SIG
++ mod->sig_ok = info.sig_ok;
++ if (!mod->sig_ok)
++ add_taint_module(mod, TAINT_FORCED_MODULE);
++#endif
++
+ /* Now module is in final location, initialize linked lists, etc. */
+ err = module_unload_init(mod);
+ if (err)
+--
+1.7.11.4
+
+
+From d4f65b5d2497b2fd9c45f06b71deb4ab084a5b66 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Thu, 13 Sep 2012 13:06:29 +0100
+Subject: [PATCH 02/26] KEYS: Add payload preparsing opportunity prior to key
+ instantiate or update
+
+Give the key type the opportunity to preparse the payload prior to the
+instantiation and update routines being called. This is done with the
+provision of two new key type operations:
+
+ int (*preparse)(struct key_preparsed_payload *prep);
+ void (*free_preparse)(struct key_preparsed_payload *prep);
+
+If the first operation is present, then it is called before key creation (in
+the add/update case) or before the key semaphore is taken (in the update and
+instantiate cases). The second operation is called to clean up if the first
+was called.
+
+preparse() is given the opportunity to fill in the following structure:
+
+ struct key_preparsed_payload {
+ char *description;
+ void *type_data[2];
+ void *payload;
+ const void *data;
+ size_t datalen;
+ size_t quotalen;
+ };
+
+Before the preparser is called, the first three fields will have been cleared,
+the payload pointer and size will be stored in data and datalen and the default
+quota size from the key_type struct will be stored into quotalen.
+
+The preparser may parse the payload in any way it likes and may store data in
+the type_data[] and payload fields for use by the instantiate() and update()
+ops.
+
+The preparser may also propose a description for the key by attaching it as a
+string to the description field. This can be used by passing a NULL or ""
+description to the add_key() system call or the key_create_or_update()
+function. This cannot work with request_key() as that required the description
+to tell the upcall about the key to be created.
+
+This, for example permits keys that store PGP public keys to generate their own
+name from the user ID and public key fingerprint in the key.
+
+The instantiate() and update() operations are then modified to look like this:
+
+ int (*instantiate)(struct key *key, struct key_preparsed_payload *prep);
+ int (*update)(struct key *key, struct key_preparsed_payload *prep);
+
+and the new payload data is passed in *prep, whether or not it was preparsed.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ Documentation/security/keys.txt | 50 +++++++++++++-
+ fs/cifs/cifs_spnego.c | 6 +-
+ fs/cifs/cifsacl.c | 8 +--
+ include/keys/user-type.h | 6 +-
+ include/linux/key-type.h | 35 +++++++++-
+ net/ceph/crypto.c | 9 +--
+ net/dns_resolver/dns_key.c | 6 +-
+ net/rxrpc/ar-key.c | 40 +++++------
+ security/keys/encrypted-keys/encrypted.c | 16 +++--
+ security/keys/key.c | 114 ++++++++++++++++++++++---------
+ security/keys/keyctl.c | 18 +++--
+ security/keys/keyring.c | 6 +-
+ security/keys/request_key_auth.c | 8 +--
+ security/keys/trusted.c | 16 +++--
+ security/keys/user_defined.c | 14 ++--
+ 15 files changed, 250 insertions(+), 102 deletions(-)
+
+diff --git a/Documentation/security/keys.txt b/Documentation/security/keys.txt
+index aa0dbd7..7d9ca92 100644
+--- a/Documentation/security/keys.txt
++++ b/Documentation/security/keys.txt
+@@ -412,6 +412,10 @@ The main syscalls are:
+ to the keyring. In this case, an error will be generated if the process
+ does not have permission to write to the keyring.
+
++ If the key type supports it, if the description is NULL or an empty
++ string, the key type will try and generate a description from the content
++ of the payload.
++
+ The payload is optional, and the pointer can be NULL if not required by
+ the type. The payload is plen in size, and plen can be zero for an empty
+ payload.
+@@ -1114,12 +1118,53 @@ The structure has a number of fields, some of which are mandatory:
+ it should return 0.
+
+
+- (*) int (*instantiate)(struct key *key, const void *data, size_t datalen);
++ (*) int (*preparse)(struct key_preparsed_payload *prep);
++
++ This optional method permits the key type to attempt to parse payload
++ before a key is created (add key) or the key semaphore is taken (update or
++ instantiate key). The structure pointed to by prep looks like:
++
++ struct key_preparsed_payload {
++ char *description;
++ void *type_data[2];
++ void *payload;
++ const void *data;
++ size_t datalen;
++ size_t quotalen;
++ };
++
++ Before calling the method, the caller will fill in data and datalen with
++ the payload blob parameters; quotalen will be filled in with the default
++ quota size from the key type and the rest will be cleared.
++
++ If a description can be proposed from the payload contents, that should be
++ attached as a string to the description field. This will be used for the
++ key description if the caller of add_key() passes NULL or "".
++
++ The method can attach anything it likes to type_data[] and payload. These
++ are merely passed along to the instantiate() or update() operations.
++
++ The method should return 0 if success ful or a negative error code
++ otherwise.
++
++
++ (*) void (*free_preparse)(struct key_preparsed_payload *prep);
++
++ This method is only required if the preparse() method is provided,
++ otherwise it is unused. It cleans up anything attached to the
++ description, type_data and payload fields of the key_preparsed_payload
++ struct as filled in by the preparse() method.
++
++
++ (*) int (*instantiate)(struct key *key, struct key_preparsed_payload *prep);
+
+ This method is called to attach a payload to a key during construction.
+ The payload attached need not bear any relation to the data passed to this
+ function.
+
++ The prep->data and prep->datalen fields will define the original payload
++ blob. If preparse() was supplied then other fields may be filled in also.
++
+ If the amount of data attached to the key differs from the size in
+ keytype->def_datalen, then key_payload_reserve() should be called.
+
+@@ -1135,6 +1180,9 @@ The structure has a number of fields, some of which are mandatory:
+ If this type of key can be updated, then this method should be provided.
+ It is called to update a key's payload from the blob of data provided.
+
++ The prep->data and prep->datalen fields will define the original payload
++ blob. If preparse() was supplied then other fields may be filled in also.
++
+ key_payload_reserve() should be called if the data length might change
+ before any changes are actually made. Note that if this succeeds, the type
+ is committed to changing the key because it's already been altered, so all
+diff --git a/fs/cifs/cifs_spnego.c b/fs/cifs/cifs_spnego.c
+index e622863..086f381 100644
+--- a/fs/cifs/cifs_spnego.c
++++ b/fs/cifs/cifs_spnego.c
+@@ -31,18 +31,18 @@
+
+ /* create a new cifs key */
+ static int
+-cifs_spnego_key_instantiate(struct key *key, const void *data, size_t datalen)
++cifs_spnego_key_instantiate(struct key *key, struct key_preparsed_payload *prep)
+ {
+ char *payload;
+ int ret;
+
+ ret = -ENOMEM;
+- payload = kmalloc(datalen, GFP_KERNEL);
++ payload = kmalloc(prep->datalen, GFP_KERNEL);
+ if (!payload)
+ goto error;
+
+ /* attach the data */
+- memcpy(payload, data, datalen);
++ memcpy(payload, prep->data, prep->datalen);
+ key->payload.data = payload;
+ ret = 0;
+
+diff --git a/fs/cifs/cifsacl.c b/fs/cifs/cifsacl.c
+index 05f4dc2..f3c60e2 100644
+--- a/fs/cifs/cifsacl.c
++++ b/fs/cifs/cifsacl.c
+@@ -167,17 +167,17 @@ static struct shrinker cifs_shrinker = {
+ };
+
+ static int
+-cifs_idmap_key_instantiate(struct key *key, const void *data, size_t datalen)
++cifs_idmap_key_instantiate(struct key *key, struct key_preparsed_payload *prep)
+ {
+ char *payload;
+
+- payload = kmalloc(datalen, GFP_KERNEL);
++ payload = kmalloc(prep->datalen, GFP_KERNEL);
+ if (!payload)
+ return -ENOMEM;
+
+- memcpy(payload, data, datalen);
++ memcpy(payload, prep->data, prep->datalen);
+ key->payload.data = payload;
+- key->datalen = datalen;
++ key->datalen = prep->datalen;
+ return 0;
+ }
+
+diff --git a/include/keys/user-type.h b/include/keys/user-type.h
+index bc9ec1d..5e452c8 100644
+--- a/include/keys/user-type.h
++++ b/include/keys/user-type.h
+@@ -35,8 +35,10 @@ struct user_key_payload {
+ extern struct key_type key_type_user;
+ extern struct key_type key_type_logon;
+
+-extern int user_instantiate(struct key *key, const void *data, size_t datalen);
+-extern int user_update(struct key *key, const void *data, size_t datalen);
++struct key_preparsed_payload;
++
++extern int user_instantiate(struct key *key, struct key_preparsed_payload *prep);
++extern int user_update(struct key *key, struct key_preparsed_payload *prep);
+ extern int user_match(const struct key *key, const void *criterion);
+ extern void user_revoke(struct key *key);
+ extern void user_destroy(struct key *key);
+diff --git a/include/linux/key-type.h b/include/linux/key-type.h
+index f0c651c..518a53a 100644
+--- a/include/linux/key-type.h
++++ b/include/linux/key-type.h
+@@ -26,6 +26,27 @@ struct key_construction {
+ struct key *authkey;/* authorisation for key being constructed */
+ };
+
++/*
++ * Pre-parsed payload, used by key add, update and instantiate.
++ *
++ * This struct will be cleared and data and datalen will be set with the data
++ * and length parameters from the caller and quotalen will be set from
++ * def_datalen from the key type. Then if the preparse() op is provided by the
++ * key type, that will be called. Then the struct will be passed to the
++ * instantiate() or the update() op.
++ *
++ * If the preparse() op is given, the free_preparse() op will be called to
++ * clear the contents.
++ */
++struct key_preparsed_payload {
++ char *description; /* Proposed key description (or NULL) */
++ void *type_data[2]; /* Private key-type data */
++ void *payload; /* Proposed payload */
++ const void *data; /* Raw data */
++ size_t datalen; /* Raw datalen */
++ size_t quotalen; /* Quota length for proposed payload */
++};
++
+ typedef int (*request_key_actor_t)(struct key_construction *key,
+ const char *op, void *aux);
+
+@@ -45,18 +66,28 @@ struct key_type {
+ /* vet a description */
+ int (*vet_description)(const char *description);
+
++ /* Preparse the data blob from userspace that is to be the payload,
++ * generating a proposed description and payload that will be handed to
++ * the instantiate() and update() ops.
++ */
++ int (*preparse)(struct key_preparsed_payload *prep);
++
++ /* Free a preparse data structure.
++ */
++ void (*free_preparse)(struct key_preparsed_payload *prep);
++
+ /* instantiate a key of this type
+ * - this method should call key_payload_reserve() to determine if the
+ * user's quota will hold the payload
+ */
+- int (*instantiate)(struct key *key, const void *data, size_t datalen);
++ int (*instantiate)(struct key *key, struct key_preparsed_payload *prep);
+
+ /* update a key of this type (optional)
+ * - this method should call key_payload_reserve() to recalculate the
+ * quota consumption
+ * - the key must be locked against read when modifying
+ */
+- int (*update)(struct key *key, const void *data, size_t datalen);
++ int (*update)(struct key *key, struct key_preparsed_payload *prep);
+
+ /* match a key against a description */
+ int (*match)(const struct key *key, const void *desc);
+diff --git a/net/ceph/crypto.c b/net/ceph/crypto.c
+index 9da7fdd..af14cb4 100644
+--- a/net/ceph/crypto.c
++++ b/net/ceph/crypto.c
+@@ -423,14 +423,15 @@ int ceph_encrypt2(struct ceph_crypto_key *secret, void *dst, size_t *dst_len,
+ }
+ }
+
+-int ceph_key_instantiate(struct key *key, const void *data, size_t datalen)
++int ceph_key_instantiate(struct key *key, struct key_preparsed_payload *prep)
+ {
+ struct ceph_crypto_key *ckey;
++ size_t datalen = prep->datalen;
+ int ret;
+ void *p;
+
+ ret = -EINVAL;
+- if (datalen <= 0 || datalen > 32767 || !data)
++ if (datalen <= 0 || datalen > 32767 || !prep->data)
+ goto err;
+
+ ret = key_payload_reserve(key, datalen);
+@@ -443,8 +444,8 @@ int ceph_key_instantiate(struct key *key, const void *data, size_t datalen)
+ goto err;
+
+ /* TODO ceph_crypto_key_decode should really take const input */
+- p = (void *)data;
+- ret = ceph_crypto_key_decode(ckey, &p, (char*)data+datalen);
++ p = (void *)prep->data;
++ ret = ceph_crypto_key_decode(ckey, &p, (char*)prep->data+datalen);
+ if (ret < 0)
+ goto err_ckey;
+
+diff --git a/net/dns_resolver/dns_key.c b/net/dns_resolver/dns_key.c
+index d9507dd..859ab8b 100644
+--- a/net/dns_resolver/dns_key.c
++++ b/net/dns_resolver/dns_key.c
+@@ -59,13 +59,13 @@ const struct cred *dns_resolver_cache;
+ * "ip1,ip2,...#foo=bar"
+ */
+ static int
+-dns_resolver_instantiate(struct key *key, const void *_data, size_t datalen)
++dns_resolver_instantiate(struct key *key, struct key_preparsed_payload *prep)
+ {
+ struct user_key_payload *upayload;
+ unsigned long derrno;
+ int ret;
+- size_t result_len = 0;
+- const char *data = _data, *end, *opt;
++ size_t datalen = prep->datalen, result_len = 0;
++ const char *data = prep->data, *end, *opt;
+
+ kenter("%%%d,%s,'%*.*s',%zu",
+ key->serial, key->description,
+diff --git a/net/rxrpc/ar-key.c b/net/rxrpc/ar-key.c
+index 8b1f9f4..106c5a6 100644
+--- a/net/rxrpc/ar-key.c
++++ b/net/rxrpc/ar-key.c
+@@ -26,8 +26,8 @@
+ #include "ar-internal.h"
+
+ static int rxrpc_vet_description_s(const char *);
+-static int rxrpc_instantiate(struct key *, const void *, size_t);
+-static int rxrpc_instantiate_s(struct key *, const void *, size_t);
++static int rxrpc_instantiate(struct key *, struct key_preparsed_payload *);
++static int rxrpc_instantiate_s(struct key *, struct key_preparsed_payload *);
+ static void rxrpc_destroy(struct key *);
+ static void rxrpc_destroy_s(struct key *);
+ static void rxrpc_describe(const struct key *, struct seq_file *);
+@@ -678,7 +678,7 @@ error:
+ *
+ * if no data is provided, then a no-security key is made
+ */
+-static int rxrpc_instantiate(struct key *key, const void *data, size_t datalen)
++static int rxrpc_instantiate(struct key *key, struct key_preparsed_payload *prep)
+ {
+ const struct rxrpc_key_data_v1 *v1;
+ struct rxrpc_key_token *token, **pp;
+@@ -686,26 +686,26 @@ static int rxrpc_instantiate(struct key *key, const void *data, size_t datalen)
+ u32 kver;
+ int ret;
+
+- _enter("{%x},,%zu", key_serial(key), datalen);
++ _enter("{%x},,%zu", key_serial(key), prep->datalen);
+
+ /* handle a no-security key */
+- if (!data && datalen == 0)
++ if (!prep->data && prep->datalen == 0)
+ return 0;
+
+ /* determine if the XDR payload format is being used */
+- if (datalen > 7 * 4) {
+- ret = rxrpc_instantiate_xdr(key, data, datalen);
++ if (prep->datalen > 7 * 4) {
++ ret = rxrpc_instantiate_xdr(key, prep->data, prep->datalen);
+ if (ret != -EPROTO)
+ return ret;
+ }
+
+ /* get the key interface version number */
+ ret = -EINVAL;
+- if (datalen <= 4 || !data)
++ if (prep->datalen <= 4 || !prep->data)
+ goto error;
+- memcpy(&kver, data, sizeof(kver));
+- data += sizeof(kver);
+- datalen -= sizeof(kver);
++ memcpy(&kver, prep->data, sizeof(kver));
++ prep->data += sizeof(kver);
++ prep->datalen -= sizeof(kver);
+
+ _debug("KEY I/F VERSION: %u", kver);
+
+@@ -715,11 +715,11 @@ static int rxrpc_instantiate(struct key *key, const void *data, size_t datalen)
+
+ /* deal with a version 1 key */
+ ret = -EINVAL;
+- if (datalen < sizeof(*v1))
++ if (prep->datalen < sizeof(*v1))
+ goto error;
+
+- v1 = data;
+- if (datalen != sizeof(*v1) + v1->ticket_length)
++ v1 = prep->data;
++ if (prep->datalen != sizeof(*v1) + v1->ticket_length)
+ goto error;
+
+ _debug("SCIX: %u", v1->security_index);
+@@ -784,17 +784,17 @@ error:
+ * instantiate a server secret key
+ * data should be a pointer to the 8-byte secret key
+ */
+-static int rxrpc_instantiate_s(struct key *key, const void *data,
+- size_t datalen)
++static int rxrpc_instantiate_s(struct key *key,
++ struct key_preparsed_payload *prep)
+ {
+ struct crypto_blkcipher *ci;
+
+- _enter("{%x},,%zu", key_serial(key), datalen);
++ _enter("{%x},,%zu", key_serial(key), prep->datalen);
+
+- if (datalen != 8)
++ if (prep->datalen != 8)
+ return -EINVAL;
+
+- memcpy(&key->type_data, data, 8);
++ memcpy(&key->type_data, prep->data, 8);
+
+ ci = crypto_alloc_blkcipher("pcbc(des)", 0, CRYPTO_ALG_ASYNC);
+ if (IS_ERR(ci)) {
+@@ -802,7 +802,7 @@ static int rxrpc_instantiate_s(struct key *key, const void *data,
+ return PTR_ERR(ci);
+ }
+
+- if (crypto_blkcipher_setkey(ci, data, 8) < 0)
++ if (crypto_blkcipher_setkey(ci, prep->data, 8) < 0)
+ BUG();
+
+ key->payload.data = ci;
+diff --git a/security/keys/encrypted-keys/encrypted.c b/security/keys/encrypted-keys/encrypted.c
+index 2d1bb8a..9e1e005 100644
+--- a/security/keys/encrypted-keys/encrypted.c
++++ b/security/keys/encrypted-keys/encrypted.c
+@@ -773,8 +773,8 @@ static int encrypted_init(struct encrypted_key_payload *epayload,
+ *
+ * On success, return 0. Otherwise return errno.
+ */
+-static int encrypted_instantiate(struct key *key, const void *data,
+- size_t datalen)
++static int encrypted_instantiate(struct key *key,
++ struct key_preparsed_payload *prep)
+ {
+ struct encrypted_key_payload *epayload = NULL;
+ char *datablob = NULL;
+@@ -782,16 +782,17 @@ static int encrypted_instantiate(struct key *key, const void *data,
+ char *master_desc = NULL;
+ char *decrypted_datalen = NULL;
+ char *hex_encoded_iv = NULL;
++ size_t datalen = prep->datalen;
+ int ret;
+
+- if (datalen <= 0 || datalen > 32767 || !data)
++ if (datalen <= 0 || datalen > 32767 || !prep->data)
+ return -EINVAL;
+
+ datablob = kmalloc(datalen + 1, GFP_KERNEL);
+ if (!datablob)
+ return -ENOMEM;
+ datablob[datalen] = 0;
+- memcpy(datablob, data, datalen);
++ memcpy(datablob, prep->data, datalen);
+ ret = datablob_parse(datablob, &format, &master_desc,
+ &decrypted_datalen, &hex_encoded_iv);
+ if (ret < 0)
+@@ -834,16 +835,17 @@ static void encrypted_rcu_free(struct rcu_head *rcu)
+ *
+ * On success, return 0. Otherwise return errno.
+ */
+-static int encrypted_update(struct key *key, const void *data, size_t datalen)
++static int encrypted_update(struct key *key, struct key_preparsed_payload *prep)
+ {
+ struct encrypted_key_payload *epayload = key->payload.data;
+ struct encrypted_key_payload *new_epayload;
+ char *buf;
+ char *new_master_desc = NULL;
+ const char *format = NULL;
++ size_t datalen = prep->datalen;
+ int ret = 0;
+
+- if (datalen <= 0 || datalen > 32767 || !data)
++ if (datalen <= 0 || datalen > 32767 || !prep->data)
+ return -EINVAL;
+
+ buf = kmalloc(datalen + 1, GFP_KERNEL);
+@@ -851,7 +853,7 @@ static int encrypted_update(struct key *key, const void *data, size_t datalen)
+ return -ENOMEM;
+
+ buf[datalen] = 0;
+- memcpy(buf, data, datalen);
++ memcpy(buf, prep->data, datalen);
+ ret = datablob_parse(buf, &format, &new_master_desc, NULL, NULL);
+ if (ret < 0)
+ goto out;
+diff --git a/security/keys/key.c b/security/keys/key.c
+index 50d96d4..1d039af 100644
+--- a/security/keys/key.c
++++ b/security/keys/key.c
+@@ -412,8 +412,7 @@ EXPORT_SYMBOL(key_payload_reserve);
+ * key_construction_mutex.
+ */
+ static int __key_instantiate_and_link(struct key *key,
+- const void *data,
+- size_t datalen,
++ struct key_preparsed_payload *prep,
+ struct key *keyring,
+ struct key *authkey,
+ unsigned long *_prealloc)
+@@ -431,7 +430,7 @@ static int __key_instantiate_and_link(struct key *key,
+ /* can't instantiate twice */
+ if (!test_bit(KEY_FLAG_INSTANTIATED, &key->flags)) {
+ /* instantiate the key */
+- ret = key->type->instantiate(key, data, datalen);
++ ret = key->type->instantiate(key, prep);
+
+ if (ret == 0) {
+ /* mark the key as being instantiated */
+@@ -482,22 +481,37 @@ int key_instantiate_and_link(struct key *key,
+ struct key *keyring,
+ struct key *authkey)
+ {
++ struct key_preparsed_payload prep;
+ unsigned long prealloc;
+ int ret;
+
++ memset(&prep, 0, sizeof(prep));
++ prep.data = data;
++ prep.datalen = datalen;
++ prep.quotalen = key->type->def_datalen;
++ if (key->type->preparse) {
++ ret = key->type->preparse(&prep);
++ if (ret < 0)
++ goto error;
++ }
++
+ if (keyring) {
+ ret = __key_link_begin(keyring, key->type, key->description,
+ &prealloc);
+ if (ret < 0)
+- return ret;
++ goto error_free_preparse;
+ }
+
+- ret = __key_instantiate_and_link(key, data, datalen, keyring, authkey,
++ ret = __key_instantiate_and_link(key, &prep, keyring, authkey,
+ &prealloc);
+
+ if (keyring)
+ __key_link_end(keyring, key->type, prealloc);
+
++error_free_preparse:
++ if (key->type->preparse)
++ key->type->free_preparse(&prep);
++error:
+ return ret;
+ }
+
+@@ -706,7 +720,7 @@ void key_type_put(struct key_type *ktype)
+ * if we get an error.
+ */
+ static inline key_ref_t __key_update(key_ref_t key_ref,
+- const void *payload, size_t plen)
++ struct key_preparsed_payload *prep)
+ {
+ struct key *key = key_ref_to_ptr(key_ref);
+ int ret;
+@@ -722,7 +736,7 @@ static inline key_ref_t __key_update(key_ref_t key_ref,
+
+ down_write(&key->sem);
+
+- ret = key->type->update(key, payload, plen);
++ ret = key->type->update(key, prep);
+ if (ret == 0)
+ /* updating a negative key instantiates it */
+ clear_bit(KEY_FLAG_NEGATIVE, &key->flags);
+@@ -774,6 +788,7 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
+ unsigned long flags)
+ {
+ unsigned long prealloc;
++ struct key_preparsed_payload prep;
+ const struct cred *cred = current_cred();
+ struct key_type *ktype;
+ struct key *keyring, *key = NULL;
+@@ -789,8 +804,9 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
+ }
+
+ key_ref = ERR_PTR(-EINVAL);
+- if (!ktype->match || !ktype->instantiate)
+- goto error_2;
++ if (!ktype->match || !ktype->instantiate ||
++ (!description && !ktype->preparse))
++ goto error_put_type;
+
+ keyring = key_ref_to_ptr(keyring_ref);
+
+@@ -798,18 +814,37 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
+
+ key_ref = ERR_PTR(-ENOTDIR);
+ if (keyring->type != &key_type_keyring)
+- goto error_2;
++ goto error_put_type;
++
++ memset(&prep, 0, sizeof(prep));
++ prep.data = payload;
++ prep.datalen = plen;
++ prep.quotalen = ktype->def_datalen;
++ if (ktype->preparse) {
++ ret = ktype->preparse(&prep);
++ if (ret < 0) {
++ key_ref = ERR_PTR(ret);
++ goto error_put_type;
++ }
++ if (!description)
++ description = prep.description;
++ key_ref = ERR_PTR(-EINVAL);
++ if (!description)
++ goto error_free_prep;
++ }
+
+ ret = __key_link_begin(keyring, ktype, description, &prealloc);
+- if (ret < 0)
+- goto error_2;
++ if (ret < 0) {
++ key_ref = ERR_PTR(ret);
++ goto error_free_prep;
++ }
+
+ /* if we're going to allocate a new key, we're going to have
+ * to modify the keyring */
+ ret = key_permission(keyring_ref, KEY_WRITE);
+ if (ret < 0) {
+ key_ref = ERR_PTR(ret);
+- goto error_3;
++ goto error_link_end;
+ }
+
+ /* if it's possible to update this type of key, search for an existing
+@@ -840,25 +875,27 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
+ perm, flags);
+ if (IS_ERR(key)) {
+ key_ref = ERR_CAST(key);
+- goto error_3;
++ goto error_link_end;
+ }
+
+ /* instantiate it and link it into the target keyring */
+- ret = __key_instantiate_and_link(key, payload, plen, keyring, NULL,
+- &prealloc);
++ ret = __key_instantiate_and_link(key, &prep, keyring, NULL, &prealloc);
+ if (ret < 0) {
+ key_put(key);
+ key_ref = ERR_PTR(ret);
+- goto error_3;
++ goto error_link_end;
+ }
+
+ key_ref = make_key_ref(key, is_key_possessed(keyring_ref));
+
+- error_3:
++error_link_end:
+ __key_link_end(keyring, ktype, prealloc);
+- error_2:
++error_free_prep:
++ if (ktype->preparse)
++ ktype->free_preparse(&prep);
++error_put_type:
+ key_type_put(ktype);
+- error:
++error:
+ return key_ref;
+
+ found_matching_key:
+@@ -866,10 +903,9 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
+ * - we can drop the locks first as we have the key pinned
+ */
+ __key_link_end(keyring, ktype, prealloc);
+- key_type_put(ktype);
+
+- key_ref = __key_update(key_ref, payload, plen);
+- goto error;
++ key_ref = __key_update(key_ref, &prep);
++ goto error_free_prep;
+ }
+ EXPORT_SYMBOL(key_create_or_update);
+
+@@ -888,6 +924,7 @@ EXPORT_SYMBOL(key_create_or_update);
+ */
+ int key_update(key_ref_t key_ref, const void *payload, size_t plen)
+ {
++ struct key_preparsed_payload prep;
+ struct key *key = key_ref_to_ptr(key_ref);
+ int ret;
+
+@@ -900,18 +937,31 @@ int key_update(key_ref_t key_ref, const void *payload, size_t plen)
+
+ /* attempt to update it if supported */
+ ret = -EOPNOTSUPP;
+- if (key->type->update) {
+- down_write(&key->sem);
+-
+- ret = key->type->update(key, payload, plen);
+- if (ret == 0)
+- /* updating a negative key instantiates it */
+- clear_bit(KEY_FLAG_NEGATIVE, &key->flags);
++ if (!key->type->update)
++ goto error;
+
+- up_write(&key->sem);
++ memset(&prep, 0, sizeof(prep));
++ prep.data = payload;
++ prep.datalen = plen;
++ prep.quotalen = key->type->def_datalen;
++ if (key->type->preparse) {
++ ret = key->type->preparse(&prep);
++ if (ret < 0)
++ goto error;
+ }
+
+- error:
++ down_write(&key->sem);
++
++ ret = key->type->update(key, &prep);
++ if (ret == 0)
++ /* updating a negative key instantiates it */
++ clear_bit(KEY_FLAG_NEGATIVE, &key->flags);
++
++ up_write(&key->sem);
++
++ if (key->type->preparse)
++ key->type->free_preparse(&prep);
++error:
+ return ret;
+ }
+ EXPORT_SYMBOL(key_update);
+diff --git a/security/keys/keyctl.c b/security/keys/keyctl.c
+index 3364fbf..505d40b 100644
+--- a/security/keys/keyctl.c
++++ b/security/keys/keyctl.c
+@@ -46,6 +46,9 @@ static int key_get_type_from_user(char *type,
+ * Extract the description of a new key from userspace and either add it as a
+ * new key to the specified keyring or update a matching key in that keyring.
+ *
++ * If the description is NULL or an empty string, the key type is asked to
++ * generate one from the payload.
++ *
+ * The keyring must be writable so that we can attach the key to it.
+ *
+ * If successful, the new key's serial number is returned, otherwise an error
+@@ -72,10 +75,17 @@ SYSCALL_DEFINE5(add_key, const char __user *, _type,
+ if (ret < 0)
+ goto error;
+
+- description = strndup_user(_description, PAGE_SIZE);
+- if (IS_ERR(description)) {
+- ret = PTR_ERR(description);
+- goto error;
++ description = NULL;
++ if (_description) {
++ description = strndup_user(_description, PAGE_SIZE);
++ if (IS_ERR(description)) {
++ ret = PTR_ERR(description);
++ goto error;
++ }
++ if (!*description) {
++ kfree(description);
++ description = NULL;
++ }
+ }
+
+ /* pull the payload in if one was supplied */
+diff --git a/security/keys/keyring.c b/security/keys/keyring.c
+index 81e7852..f04d8cf 100644
+--- a/security/keys/keyring.c
++++ b/security/keys/keyring.c
+@@ -66,7 +66,7 @@ static inline unsigned keyring_hash(const char *desc)
+ * operations.
+ */
+ static int keyring_instantiate(struct key *keyring,
+- const void *data, size_t datalen);
++ struct key_preparsed_payload *prep);
+ static int keyring_match(const struct key *keyring, const void *criterion);
+ static void keyring_revoke(struct key *keyring);
+ static void keyring_destroy(struct key *keyring);
+@@ -121,12 +121,12 @@ static void keyring_publish_name(struct key *keyring)
+ * Returns 0 on success, -EINVAL if given any data.
+ */
+ static int keyring_instantiate(struct key *keyring,
+- const void *data, size_t datalen)
++ struct key_preparsed_payload *prep)
+ {
+ int ret;
+
+ ret = -EINVAL;
+- if (datalen == 0) {
++ if (prep->datalen == 0) {
+ /* make the keyring available by name if it has one */
+ keyring_publish_name(keyring);
+ ret = 0;
+diff --git a/security/keys/request_key_auth.c b/security/keys/request_key_auth.c
+index 60d4e3f..85730d5 100644
+--- a/security/keys/request_key_auth.c
++++ b/security/keys/request_key_auth.c
+@@ -19,7 +19,8 @@
+ #include <asm/uaccess.h>
+ #include "internal.h"
+
+-static int request_key_auth_instantiate(struct key *, const void *, size_t);
++static int request_key_auth_instantiate(struct key *,
++ struct key_preparsed_payload *);
+ static void request_key_auth_describe(const struct key *, struct seq_file *);
+ static void request_key_auth_revoke(struct key *);
+ static void request_key_auth_destroy(struct key *);
+@@ -42,10 +43,9 @@ struct key_type key_type_request_key_auth = {
+ * Instantiate a request-key authorisation key.
+ */
+ static int request_key_auth_instantiate(struct key *key,
+- const void *data,
+- size_t datalen)
++ struct key_preparsed_payload *prep)
+ {
+- key->payload.data = (struct request_key_auth *) data;
++ key->payload.data = (struct request_key_auth *)prep->data;
+ return 0;
+ }
+
+diff --git a/security/keys/trusted.c b/security/keys/trusted.c
+index 2d5d041..42036c7 100644
+--- a/security/keys/trusted.c
++++ b/security/keys/trusted.c
+@@ -927,22 +927,23 @@ static struct trusted_key_payload *trusted_payload_alloc(struct key *key)
+ *
+ * On success, return 0. Otherwise return errno.
+ */
+-static int trusted_instantiate(struct key *key, const void *data,
+- size_t datalen)
++static int trusted_instantiate(struct key *key,
++ struct key_preparsed_payload *prep)
+ {
+ struct trusted_key_payload *payload = NULL;
+ struct trusted_key_options *options = NULL;
++ size_t datalen = prep->datalen;
+ char *datablob;
+ int ret = 0;
+ int key_cmd;
+
+- if (datalen <= 0 || datalen > 32767 || !data)
++ if (datalen <= 0 || datalen > 32767 || !prep->data)
+ return -EINVAL;
+
+ datablob = kmalloc(datalen + 1, GFP_KERNEL);
+ if (!datablob)
+ return -ENOMEM;
+- memcpy(datablob, data, datalen);
++ memcpy(datablob, prep->data, datalen);
+ datablob[datalen] = '\0';
+
+ options = trusted_options_alloc();
+@@ -1011,17 +1012,18 @@ static void trusted_rcu_free(struct rcu_head *rcu)
+ /*
+ * trusted_update - reseal an existing key with new PCR values
+ */
+-static int trusted_update(struct key *key, const void *data, size_t datalen)
++static int trusted_update(struct key *key, struct key_preparsed_payload *prep)
+ {
+ struct trusted_key_payload *p = key->payload.data;
+ struct trusted_key_payload *new_p;
+ struct trusted_key_options *new_o;
++ size_t datalen = prep->datalen;
+ char *datablob;
+ int ret = 0;
+
+ if (!p->migratable)
+ return -EPERM;
+- if (datalen <= 0 || datalen > 32767 || !data)
++ if (datalen <= 0 || datalen > 32767 || !prep->data)
+ return -EINVAL;
+
+ datablob = kmalloc(datalen + 1, GFP_KERNEL);
+@@ -1038,7 +1040,7 @@ static int trusted_update(struct key *key, const void *data, size_t datalen)
+ goto out;
+ }
+
+- memcpy(datablob, data, datalen);
++ memcpy(datablob, prep->data, datalen);
+ datablob[datalen] = '\0';
+ ret = datablob_parse(datablob, new_p, new_o);
+ if (ret != Opt_update) {
+diff --git a/security/keys/user_defined.c b/security/keys/user_defined.c
+index c7660a2..55dc889 100644
+--- a/security/keys/user_defined.c
++++ b/security/keys/user_defined.c
+@@ -58,13 +58,14 @@ EXPORT_SYMBOL_GPL(key_type_logon);
+ /*
+ * instantiate a user defined key
+ */
+-int user_instantiate(struct key *key, const void *data, size_t datalen)
++int user_instantiate(struct key *key, struct key_preparsed_payload *prep)
+ {
+ struct user_key_payload *upayload;
++ size_t datalen = prep->datalen;
+ int ret;
+
+ ret = -EINVAL;
+- if (datalen <= 0 || datalen > 32767 || !data)
++ if (datalen <= 0 || datalen > 32767 || !prep->data)
+ goto error;
+
+ ret = key_payload_reserve(key, datalen);
+@@ -78,7 +79,7 @@ int user_instantiate(struct key *key, const void *data, size_t datalen)
+
+ /* attach the data */
+ upayload->datalen = datalen;
+- memcpy(upayload->data, data, datalen);
++ memcpy(upayload->data, prep->data, datalen);
+ rcu_assign_keypointer(key, upayload);
+ ret = 0;
+
+@@ -92,13 +93,14 @@ EXPORT_SYMBOL_GPL(user_instantiate);
+ * update a user defined key
+ * - the key's semaphore is write-locked
+ */
+-int user_update(struct key *key, const void *data, size_t datalen)
++int user_update(struct key *key, struct key_preparsed_payload *prep)
+ {
+ struct user_key_payload *upayload, *zap;
++ size_t datalen = prep->datalen;
+ int ret;
+
+ ret = -EINVAL;
+- if (datalen <= 0 || datalen > 32767 || !data)
++ if (datalen <= 0 || datalen > 32767 || !prep->data)
+ goto error;
+
+ /* construct a replacement payload */
+@@ -108,7 +110,7 @@ int user_update(struct key *key, const void *data, size_t datalen)
+ goto error;
+
+ upayload->datalen = datalen;
+- memcpy(upayload->data, data, datalen);
++ memcpy(upayload->data, prep->data, datalen);
+
+ /* check the quota and attach the new data */
+ zap = upayload;
+--
+1.7.11.4
+
+
+From a3f7dff6cbc2eb6be871a58fef34cacf7f78abf8 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Thu, 13 Sep 2012 13:09:33 +0100
+Subject: [PATCH 03/26] MPILIB: Provide count_leading/trailing_zeros() based
+ on arch functions
+
+Provide count_leading/trailing_zeros() macros based on extant arch bit scanning
+functions rather than reimplementing from scratch in MPILIB.
+
+Whilst we're at it, turn count_foo_zeros(n, x) into n = count_foo_zeros(x).
+
+Also move the definition to asm-generic as other people may be interested in
+using it.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+Cc: David S. Miller <davem@davemloft.net>
+Cc: Dmitry Kasatkin <dmitry.kasatkin@intel.com>
+Cc: Arnd Bergmann <arnd@arndb.com>
+---
+ include/asm-generic/bitops/count_zeros.h | 57 +++++++++++++
+ lib/mpi/longlong.h | 138 +------------------------------
+ lib/mpi/mpi-bit.c | 2 +-
+ lib/mpi/mpi-pow.c | 4 +-
+ 4 files changed, 62 insertions(+), 139 deletions(-)
+ create mode 100644 include/asm-generic/bitops/count_zeros.h
+
+diff --git a/include/asm-generic/bitops/count_zeros.h b/include/asm-generic/bitops/count_zeros.h
+new file mode 100644
+index 0000000..97520d2
+--- /dev/null
++++ b/include/asm-generic/bitops/count_zeros.h
+@@ -0,0 +1,57 @@
++/* Count leading and trailing zeros functions
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _ASM_GENERIC_BITOPS_COUNT_ZEROS_H_
++#define _ASM_GENERIC_BITOPS_COUNT_ZEROS_H_
++
++#include <asm/bitops.h>
++
++/**
++ * count_leading_zeros - Count the number of zeros from the MSB back
++ * @x: The value
++ *
++ * Count the number of leading zeros from the MSB going towards the LSB in @x.
++ *
++ * If the MSB of @x is set, the result is 0.
++ * If only the LSB of @x is set, then the result is BITS_PER_LONG-1.
++ * If @x is 0 then the result is COUNT_LEADING_ZEROS_0.
++ */
++static inline int count_leading_zeros(unsigned long x)
++{
++ if (sizeof(x) == 4)
++ return BITS_PER_LONG - fls(x);
++ else
++ return BITS_PER_LONG - fls64(x);
++}
++
++#define COUNT_LEADING_ZEROS_0 BITS_PER_LONG
++
++/**
++ * count_trailing_zeros - Count the number of zeros from the LSB forwards
++ * @x: The value
++ *
++ * Count the number of trailing zeros from the LSB going towards the MSB in @x.
++ *
++ * If the LSB of @x is set, the result is 0.
++ * If only the MSB of @x is set, then the result is BITS_PER_LONG-1.
++ * If @x is 0 then the result is COUNT_TRAILING_ZEROS_0.
++ */
++static inline int count_trailing_zeros(unsigned long x)
++{
++#define COUNT_TRAILING_ZEROS_0 (-1)
++
++ if (sizeof(x) == 4)
++ return ffs(x);
++ else
++ return (x != 0) ? __ffs(x) : COUNT_TRAILING_ZEROS_0;
++}
++
++#endif /* _ASM_GENERIC_BITOPS_COUNT_ZEROS_H_ */
+diff --git a/lib/mpi/longlong.h b/lib/mpi/longlong.h
+index 29f9862..678ce4f 100644
+--- a/lib/mpi/longlong.h
++++ b/lib/mpi/longlong.h
+@@ -19,6 +19,8 @@
+ * the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
+ * MA 02111-1307, USA. */
+
++#include <asm-generic/bitops/count_zeros.h>
++
+ /* You have to define the following before including this file:
+ *
+ * UWtype -- An unsigned type, default type for operations (typically a "word")
+@@ -146,12 +148,6 @@ do { \
+ : "1" ((USItype)(n1)), \
+ "r" ((USItype)(n0)), \
+ "r" ((USItype)(d)))
+-
+-#define count_leading_zeros(count, x) \
+- __asm__ ("clz %0,%1" \
+- : "=r" ((USItype)(count)) \
+- : "r" ((USItype)(x)))
+-#define COUNT_LEADING_ZEROS_0 32
+ #endif /* __a29k__ */
+
+ #if defined(__alpha) && W_TYPE_SIZE == 64
+@@ -298,11 +294,6 @@ extern UDItype __udiv_qrnnd();
+ : "1" ((USItype)(nh)), \
+ "0" ((USItype)(nl)), \
+ "g" ((USItype)(d)))
+-#define count_leading_zeros(count, x) \
+- __asm__ ("bsch/1 %1,%0" \
+- : "=g" (count) \
+- : "g" ((USItype)(x)), \
+- "0" ((USItype)0))
+ #endif
+
+ /***************************************
+@@ -354,27 +345,6 @@ do { USItype __r; \
+ } while (0)
+ extern USItype __udiv_qrnnd();
+ #endif /* LONGLONG_STANDALONE */
+-#define count_leading_zeros(count, x) \
+-do { \
+- USItype __tmp; \
+- __asm__ ( \
+- "ldi 1,%0\n" \
+- "extru,= %1,15,16,%%r0 ; Bits 31..16 zero?\n" \
+- "extru,tr %1,15,16,%1 ; No. Shift down, skip add.\n" \
+- "ldo 16(%0),%0 ; Yes. Perform add.\n" \
+- "extru,= %1,23,8,%%r0 ; Bits 15..8 zero?\n" \
+- "extru,tr %1,23,8,%1 ; No. Shift down, skip add.\n" \
+- "ldo 8(%0),%0 ; Yes. Perform add.\n" \
+- "extru,= %1,27,4,%%r0 ; Bits 7..4 zero?\n" \
+- "extru,tr %1,27,4,%1 ; No. Shift down, skip add.\n" \
+- "ldo 4(%0),%0 ; Yes. Perform add.\n" \
+- "extru,= %1,29,2,%%r0 ; Bits 3..2 zero?\n" \
+- "extru,tr %1,29,2,%1 ; No. Shift down, skip add.\n" \
+- "ldo 2(%0),%0 ; Yes. Perform add.\n" \
+- "extru %1,30,1,%1 ; Extract bit 1.\n" \
+- "sub %0,%1,%0 ; Subtract it. " \
+- : "=r" (count), "=r" (__tmp) : "1" (x)); \
+-} while (0)
+ #endif /* hppa */
+
+ /***************************************
+@@ -457,15 +427,6 @@ do { \
+ : "0" ((USItype)(n0)), \
+ "1" ((USItype)(n1)), \
+ "rm" ((USItype)(d)))
+-#define count_leading_zeros(count, x) \
+-do { \
+- USItype __cbtmp; \
+- __asm__ ("bsrl %1,%0" \
+- : "=r" (__cbtmp) : "rm" ((USItype)(x))); \
+- (count) = __cbtmp ^ 31; \
+-} while (0)
+-#define count_trailing_zeros(count, x) \
+- __asm__ ("bsfl %1,%0" : "=r" (count) : "rm" ((USItype)(x)))
+ #ifndef UMUL_TIME
+ #define UMUL_TIME 40
+ #endif
+@@ -536,15 +497,6 @@ do { \
+ "dI" ((USItype)(d))); \
+ (r) = __rq.__i.__l; (q) = __rq.__i.__h; \
+ } while (0)
+-#define count_leading_zeros(count, x) \
+-do { \
+- USItype __cbtmp; \
+- __asm__ ("scanbit %1,%0" \
+- : "=r" (__cbtmp) \
+- : "r" ((USItype)(x))); \
+- (count) = __cbtmp ^ 31; \
+-} while (0)
+-#define COUNT_LEADING_ZEROS_0 (-32) /* sic */
+ #if defined(__i960mx) /* what is the proper symbol to test??? */
+ #define rshift_rhlc(r, h, l, c) \
+ do { \
+@@ -603,11 +555,6 @@ do { \
+ : "0" ((USItype)(n0)), \
+ "1" ((USItype)(n1)), \
+ "dmi" ((USItype)(d)))
+-#define count_leading_zeros(count, x) \
+- __asm__ ("bfffo %1{%b2:%b2},%0" \
+- : "=d" ((USItype)(count)) \
+- : "od" ((USItype)(x)), "n" (0))
+-#define COUNT_LEADING_ZEROS_0 32
+ #else /* not mc68020 */
+ #define umul_ppmm(xh, xl, a, b) \
+ do { USItype __umul_tmp1, __umul_tmp2; \
+@@ -664,15 +611,6 @@ do { USItype __umul_tmp1, __umul_tmp2; \
+ "rJ" ((USItype)(bh)), \
+ "rJ" ((USItype)(al)), \
+ "rJ" ((USItype)(bl)))
+-#define count_leading_zeros(count, x) \
+-do { \
+- USItype __cbtmp; \
+- __asm__ ("ff1 %0,%1" \
+- : "=r" (__cbtmp) \
+- : "r" ((USItype)(x))); \
+- (count) = __cbtmp ^ 31; \
+-} while (0)
+-#define COUNT_LEADING_ZEROS_0 63 /* sic */
+ #if defined(__m88110__)
+ #define umul_ppmm(wh, wl, u, v) \
+ do { \
+@@ -779,12 +717,6 @@ do { \
+ : "0" (__xx.__ll), \
+ "g" ((USItype)(d))); \
+ (r) = __xx.__i.__l; (q) = __xx.__i.__h; })
+-#define count_trailing_zeros(count, x) \
+-do { \
+- __asm__("ffsd %2,%0" \
+- : "=r"((USItype) (count)) \
+- : "0"((USItype) 0), "r"((USItype) (x))); \
+- } while (0)
+ #endif /* __ns32000__ */
+
+ /***************************************
+@@ -855,11 +787,6 @@ do { \
+ "rI" ((USItype)(al)), \
+ "r" ((USItype)(bl))); \
+ } while (0)
+-#define count_leading_zeros(count, x) \
+- __asm__ ("{cntlz|cntlzw} %0,%1" \
+- : "=r" ((USItype)(count)) \
+- : "r" ((USItype)(x)))
+-#define COUNT_LEADING_ZEROS_0 32
+ #if defined(_ARCH_PPC)
+ #define umul_ppmm(ph, pl, m0, m1) \
+ do { \
+@@ -1001,19 +928,6 @@ do { \
+ } while (0)
+ #define UMUL_TIME 20
+ #define UDIV_TIME 200
+-#define count_leading_zeros(count, x) \
+-do { \
+- if ((x) >= 0x10000) \
+- __asm__ ("clz %0,%1" \
+- : "=r" ((USItype)(count)) \
+- : "r" ((USItype)(x) >> 16)); \
+- else { \
+- __asm__ ("clz %0,%1" \
+- : "=r" ((USItype)(count)) \
+- : "r" ((USItype)(x))); \
+- (count) += 16; \
+- } \
+-} while (0)
+ #endif /* RT/ROMP */
+
+ /***************************************
+@@ -1142,13 +1056,6 @@ do { \
+ "rI" ((USItype)(d)) \
+ : "%g1" __AND_CLOBBER_CC)
+ #define UDIV_TIME 37
+-#define count_leading_zeros(count, x) \
+- __asm__ ("scan %1,0,%0" \
+- : "=r" ((USItype)(x)) \
+- : "r" ((USItype)(count)))
+-/* Early sparclites return 63 for an argument of 0, but they warn that future
+- implementations might change this. Therefore, leave COUNT_LEADING_ZEROS_0
+- undefined. */
+ #endif /* __sparclite__ */
+ #endif /* __sparc_v8__ */
+ /* Default to sparc v7 versions of umul_ppmm and udiv_qrnnd. */
+@@ -1454,47 +1361,6 @@ do { \
+ #define udiv_qrnnd __udiv_qrnnd_c
+ #endif
+
+-#undef count_leading_zeros
+-#if !defined(count_leading_zeros)
+- extern
+-#ifdef __STDC__
+- const
+-#endif
+- unsigned char __clz_tab[];
+-#define count_leading_zeros(count, x) \
+-do { \
+- UWtype __xr = (x); \
+- UWtype __a; \
+- \
+- if (W_TYPE_SIZE <= 32) { \
+- __a = __xr < ((UWtype) 1 << 2*__BITS4) \
+- ? (__xr < ((UWtype) 1 << __BITS4) ? 0 : __BITS4) \
+- : (__xr < ((UWtype) 1 << 3*__BITS4) ? 2*__BITS4 : 3*__BITS4); \
+- } \
+- else { \
+- for (__a = W_TYPE_SIZE - 8; __a > 0; __a -= 8) \
+- if (((__xr >> __a) & 0xff) != 0) \
+- break; \
+- } \
+- \
+- (count) = W_TYPE_SIZE - (__clz_tab[__xr >> __a] + __a); \
+-} while (0)
+- /* This version gives a well-defined value for zero. */
+-#define COUNT_LEADING_ZEROS_0 W_TYPE_SIZE
+-#endif
+-
+-#if !defined(count_trailing_zeros)
+-/* Define count_trailing_zeros using count_leading_zeros. The latter might be
+- defined in asm, but if it is not, the C version above is good enough. */
+-#define count_trailing_zeros(count, x) \
+-do { \
+- UWtype __ctz_x = (x); \
+- UWtype __ctz_c; \
+- count_leading_zeros(__ctz_c, __ctz_x & -__ctz_x); \
+- (count) = W_TYPE_SIZE - 1 - __ctz_c; \
+-} while (0)
+-#endif
+-
+ #ifndef UDIV_NEEDS_NORMALIZATION
+ #define UDIV_NEEDS_NORMALIZATION 0
+ #endif
+diff --git a/lib/mpi/mpi-bit.c b/lib/mpi/mpi-bit.c
+index 5687248..503537e 100644
+--- a/lib/mpi/mpi-bit.c
++++ b/lib/mpi/mpi-bit.c
+@@ -45,7 +45,7 @@ unsigned mpi_get_nbits(MPI a)
+ if (a->nlimbs) {
+ mpi_limb_t alimb = a->d[a->nlimbs - 1];
+ if (alimb)
+- count_leading_zeros(n, alimb);
++ n = count_leading_zeros(alimb);
+ else
+ n = BITS_PER_MPI_LIMB;
+ n = BITS_PER_MPI_LIMB - n + (a->nlimbs - 1) * BITS_PER_MPI_LIMB;
+diff --git a/lib/mpi/mpi-pow.c b/lib/mpi/mpi-pow.c
+index 67f3e79..5464c87 100644
+--- a/lib/mpi/mpi-pow.c
++++ b/lib/mpi/mpi-pow.c
+@@ -77,7 +77,7 @@ int mpi_powm(MPI res, MPI base, MPI exp, MPI mod)
+ mp = mp_marker = mpi_alloc_limb_space(msize);
+ if (!mp)
+ goto enomem;
+- count_leading_zeros(mod_shift_cnt, mod->d[msize - 1]);
++ mod_shift_cnt = count_leading_zeros(mod->d[msize - 1]);
+ if (mod_shift_cnt)
+ mpihelp_lshift(mp, mod->d, msize, mod_shift_cnt);
+ else
+@@ -169,7 +169,7 @@ int mpi_powm(MPI res, MPI base, MPI exp, MPI mod)
+
+ i = esize - 1;
+ e = ep[i];
+- count_leading_zeros(c, e);
++ c = count_leading_zeros(e);
+ e = (e << c) << 1; /* shift the exp bits to the left, lose msb */
+ c = BITS_PER_MPI_LIMB - 1 - c;
+
+--
+1.7.11.4
+
+
+From 79e4c942f35117b405402acf0f075ff79260e546 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Thu, 13 Sep 2012 15:17:21 +0100
+Subject: [PATCH 04/26] KEYS: Document asymmetric key type
+
+In-source documentation for the asymmetric key type. This will be located in:
+
+ Documentation/crypto/asymmetric-keys.txt
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ Documentation/crypto/asymmetric-keys.txt | 312 +++++++++++++++++++++++++++++++
+ 1 file changed, 312 insertions(+)
+ create mode 100644 Documentation/crypto/asymmetric-keys.txt
+
+diff --git a/Documentation/crypto/asymmetric-keys.txt b/Documentation/crypto/asymmetric-keys.txt
+new file mode 100644
+index 0000000..b767590
+--- /dev/null
++++ b/Documentation/crypto/asymmetric-keys.txt
+@@ -0,0 +1,312 @@
++ =============================================
++ ASYMMETRIC / PUBLIC-KEY CRYPTOGRAPHY KEY TYPE
++ =============================================
++
++Contents:
++
++ - Overview.
++ - Key identification.
++ - Accessing asymmetric keys.
++ - Signature verification.
++ - Asymmetric key subtypes.
++ - Instantiation data parsers.
++
++
++========
++OVERVIEW
++========
++
++The "asymmetric" key type is designed to be a container for the keys used in
++public-key cryptography, without imposing any particular restrictions on the
++form or mechanism of the cryptography or form of the key.
++
++The asymmetric key is given a subtype that defines what sort of data is
++associated with the key and provides operations to describe and destroy it.
++However, no requirement is made that the key data actually be stored in the
++key.
++
++A completely in-kernel key retention and operation subtype can be defined, but
++it would also be possible to provide access to cryptographic hardware (such as
++a TPM) that might be used to both retain the relevant key and perform
++operations using that key. In such a case, the asymmetric key would then
++merely be an interface to the TPM driver.
++
++Also provided is the concept of a data parser. Data parsers are responsible
++for extracting information from the blobs of data passed to the instantiation
++function. The first data parser that recognises the blob gets to set the
++subtype of the key and define the operations that can be done on that key.
++
++A data parser may interpret the data blob as containing the bits representing a
++key, or it may interpret it as a reference to a key held somewhere else in the
++system (for example, a TPM).
++
++
++==================
++KEY IDENTIFICATION
++==================
++
++If a key is added with an empty name, the instantiation data parsers are given
++the opportunity to pre-parse a key and to determine the description the key
++should be given from the content of the key.
++
++This can then be used to refer to the key, either by complete match or by
++partial match. The key type may also use other criteria to refer to a key.
++
++The asymmetric key type's match function can then perform a wider range of
++comparisons than just the straightforward comparison of the description with
++the criterion string:
++
++ (1) If the criterion string is of the form "id:<hexdigits>" then the match
++ function will examine a key's fingerprint to see if the hex digits given
++ after the "id:" match the tail. For instance:
++
++ keyctl search @s asymmetric id:5acc2142
++
++ will match a key with fingerprint:
++
++ 1A00 2040 7601 7889 DE11 882C 3823 04AD 5ACC 2142
++
++ (2) If the criterion string is of the form "<subtype>:<hexdigits>" then the
++ match will match the ID as in (1), but with the added restriction that
++ only keys of the specified subtype (e.g. tpm) will be matched. For
++ instance:
++
++ keyctl search @s asymmetric tpm:5acc2142
++
++Looking in /proc/keys, the last 8 hex digits of the key fingerprint are
++displayed, along with the subtype:
++
++ 1a39e171 I----- 1 perm 3f010000 0 0 asymmetri modsign.0: DSA 5acc2142 []
++
++
++=========================
++ACCESSING ASYMMETRIC KEYS
++=========================
++
++For general access to asymmetric keys from within the kernel, the following
++inclusion is required:
++
++ #include <crypto/public_key.h>
++
++This gives access to functions for dealing with asymmetric / public keys.
++Three enums are defined there for representing public-key cryptography
++algorithms:
++
++ enum pkey_algo
++
++digest algorithms used by those:
++
++ enum pkey_hash_algo
++
++and key identifier representations:
++
++ enum pkey_id_type
++
++Note that the key type representation types are required because key
++identifiers from different standards aren't necessarily compatible. For
++instance, PGP generates key identifiers by hashing the key data plus some
++PGP-specific metadata, whereas X.509 has arbitrary certificate identifiers.
++
++The operations defined upon a key are:
++
++ (1) Signature verification.
++
++Other operations are possible (such as encryption) with the same key data
++required for verification, but not currently supported, and others
++(eg. decryption and signature generation) require extra key data.
++
++
++SIGNATURE VERIFICATION
++----------------------
++
++An operation is provided to perform cryptographic signature verification, using
++an asymmetric key to provide or to provide access to the public key.
++
++ int verify_signature(const struct key *key,
++ const struct public_key_signature *sig);
++
++The caller must have already obtained the key from some source and can then use
++it to check the signature. The caller must have parsed the signature and
++transferred the relevant bits to the structure pointed to by sig.
++
++ struct public_key_signature {
++ u8 *digest;
++ u8 digest_size;
++ enum pkey_hash_algo pkey_hash_algo : 8;
++ u8 nr_mpi;
++ union {
++ MPI mpi[2];
++ ...
++ };
++ };
++
++The algorithm used must be noted in sig->pkey_hash_algo, and all the MPIs that
++make up the actual signature must be stored in sig->mpi[] and the count of MPIs
++placed in sig->nr_mpi.
++
++In addition, the data must have been digested by the caller and the resulting
++hash must be pointed to by sig->digest and the size of the hash be placed in
++sig->digest_size.
++
++The function will return 0 upon success or -EKEYREJECTED if the signature
++doesn't match.
++
++The function may also return -ENOTSUPP if an unsupported public-key algorithm
++or public-key/hash algorithm combination is specified or the key doesn't
++support the operation; -EBADMSG or -ERANGE if some of the parameters have weird
++data; or -ENOMEM if an allocation can't be performed. -EINVAL can be returned
++if the key argument is the wrong type or is incompletely set up.
++
++
++=======================
++ASYMMETRIC KEY SUBTYPES
++=======================
++
++Asymmetric keys have a subtype that defines the set of operations that can be
++performed on that key and that determines what data is attached as the key
++payload. The payload format is entirely at the whim of the subtype.
++
++The subtype is selected by the key data parser and the parser must initialise
++the data required for it. The asymmetric key retains a reference on the
++subtype module.
++
++The subtype definition structure can be found in:
++
++ #include <keys/asymmetric-subtype.h>
++
++and looks like the following:
++
++ struct asymmetric_key_subtype {
++ struct module *owner;
++ const char *name;
++
++ void (*describe)(const struct key *key, struct seq_file *m);
++ void (*destroy)(void *payload);
++ int (*verify_signature)(const struct key *key,
++ const struct public_key_signature *sig);
++ };
++
++Asymmetric keys point to this with their type_data[0] member.
++
++The owner and name fields should be set to the owning module and the name of
++the subtype. Currently, the name is only used for print statements.
++
++There are a number of operations defined by the subtype:
++
++ (1) describe().
++
++ Mandatory. This allows the subtype to display something in /proc/keys
++ against the key. For instance the name of the public key algorithm type
++ could be displayed. The key type will display the tail of the key
++ identity string after this.
++
++ (2) destroy().
++
++ Mandatory. This should free the memory associated with the key. The
++ asymmetric key will look after freeing the fingerprint and releasing the
++ reference on the subtype module.
++
++ (3) verify_signature().
++
++ Optional. These are the entry points for the key usage operations.
++ Currently there is only the one defined. If not set, the caller will be
++ given -ENOTSUPP. The subtype may do anything it likes to implement an
++ operation, including offloading to hardware.
++
++
++==========================
++INSTANTIATION DATA PARSERS
++==========================
++
++The asymmetric key type doesn't generally want to store or to deal with a raw
++blob of data that holds the key data. It would have to parse it and error
++check it each time it wanted to use it. Further, the contents of the blob may
++have various checks that can be performed on it (eg. self-signatures, validity
++dates) and may contain useful data about the key (identifiers, capabilities).
++
++Also, the blob may represent a pointer to some hardware containing the key
++rather than the key itself.
++
++Examples of blob formats for which parsers could be implemented include:
++
++ - OpenPGP packet stream [RFC 4880].
++ - X.509 ASN.1 stream.
++ - Pointer to TPM key.
++ - Pointer to UEFI key.
++
++During key instantiation each parser in the list is tried until one doesn't
++return -EBADMSG.
++
++The parser definition structure can be found in:
++
++ #include <keys/asymmetric-parser.h>
++
++and looks like the following:
++
++ struct asymmetric_key_parser {
++ struct module *owner;
++ const char *name;
++
++ int (*parse)(struct key_preparsed_payload *prep);
++ };
++
++The owner and name fields should be set to the owning module and the name of
++the parser.
++
++There is currently only a single operation defined by the parser, and it is
++mandatory:
++
++ (1) parse().
++
++ This is called to preparse the key from the key creation and update paths.
++ In particular, it is called during the key creation _before_ a key is
++ allocated, and as such, is permitted to provide the key's description in
++ the case that the caller declines to do so.
++
++ The caller passes a pointer to the following struct with all of the fields
++ cleared, except for data, datalen and quotalen [see
++ Documentation/security/keys.txt].
++
++ struct key_preparsed_payload {
++ char *description;
++ void *type_data[2];
++ void *payload;
++ const void *data;
++ size_t datalen;
++ size_t quotalen;
++ };
++
++ The instantiation data is in a blob pointed to by data and is datalen in
++ size. The parse() function is not permitted to change these two values at
++ all, and shouldn't change any of the other values _unless_ they are
++ recognise the blob format and will not return -EBADMSG to indicate it is
++ not theirs.
++
++ If the parser is happy with the blob, it should propose a description for
++ the key and attach it to ->description, ->type_data[0] should be set to
++ point to the subtype to be used, ->payload should be set to point to the
++ initialised data for that subtype, ->type_data[1] should point to a hex
++ fingerprint and quotalen should be updated to indicate how much quota this
++ key should account for.
++
++ When clearing up, the data attached to ->type_data[1] and ->description
++ will be kfree()'d and the data attached to ->payload will be passed to the
++ subtype's ->destroy() method to be disposed of. A module reference for
++ the subtype pointed to by ->type_data[0] will be put.
++
++
++ If the data format is not recognised, -EBADMSG should be returned. If it
++ is recognised, but the key cannot for some reason be set up, some other
++ negative error code should be returned. On success, 0 should be returned.
++
++ The key's fingerprint string may be partially matched upon. For a
++ public-key algorithm such as RSA and DSA this will likely be a printable
++ hex version of the key's fingerprint.
++
++Functions are provided to register and unregister parsers:
++
++ int register_asymmetric_key_parser(struct asymmetric_key_parser *parser);
++ void unregister_asymmetric_key_parser(struct asymmetric_key_parser *subtype);
++
++Parsers may not have the same name. The names are otherwise only used for
++displaying in debugging messages.
+--
+1.7.11.4
+
+
+From 2fb78e0d337ac41f2cddad18fc5c34374e5298ab Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Thu, 13 Sep 2012 15:17:21 +0100
+Subject: [PATCH 05/26] KEYS: Implement asymmetric key type
+
+Create a key type that can be used to represent an asymmetric key type for use
+in appropriate cryptographic operations, such as encryption, decryption,
+signature generation and signature verification.
+
+The key type is "asymmetric" and can provide access to a variety of
+cryptographic algorithms.
+
+Possibly, this would be better as "public_key" - but that has the disadvantage
+that "public key" is an overloaded term.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ crypto/Kconfig | 1 +
+ crypto/Makefile | 1 +
+ crypto/asymmetric_keys/Kconfig | 13 +++
+ crypto/asymmetric_keys/Makefile | 7 ++
+ crypto/asymmetric_keys/asymmetric_keys.h | 15 +++
+ crypto/asymmetric_keys/asymmetric_type.c | 156 +++++++++++++++++++++++++++++++
+ include/keys/asymmetric-subtype.h | 55 +++++++++++
+ include/keys/asymmetric-type.h | 25 +++++
+ 8 files changed, 273 insertions(+)
+ create mode 100644 crypto/asymmetric_keys/Kconfig
+ create mode 100644 crypto/asymmetric_keys/Makefile
+ create mode 100644 crypto/asymmetric_keys/asymmetric_keys.h
+ create mode 100644 crypto/asymmetric_keys/asymmetric_type.c
+ create mode 100644 include/keys/asymmetric-subtype.h
+ create mode 100644 include/keys/asymmetric-type.h
+
+diff --git a/crypto/Kconfig b/crypto/Kconfig
+index a323805..1ca0b24 100644
+--- a/crypto/Kconfig
++++ b/crypto/Kconfig
+@@ -1043,5 +1043,6 @@ config CRYPTO_USER_API_SKCIPHER
+ key cipher algorithms.
+
+ source "drivers/crypto/Kconfig"
++source crypto/asymmetric_keys/Kconfig
+
+ endif # if CRYPTO
+diff --git a/crypto/Makefile b/crypto/Makefile
+index 30f33d6..ced472e 100644
+--- a/crypto/Makefile
++++ b/crypto/Makefile
+@@ -96,3 +96,4 @@ obj-$(CONFIG_CRYPTO_USER_API_SKCIPHER) += algif_skcipher.o
+ #
+ obj-$(CONFIG_XOR_BLOCKS) += xor.o
+ obj-$(CONFIG_ASYNC_CORE) += async_tx/
++obj-$(CONFIG_ASYMMETRIC_KEY_TYPE) += asymmetric_keys/
+diff --git a/crypto/asymmetric_keys/Kconfig b/crypto/asymmetric_keys/Kconfig
+new file mode 100644
+index 0000000..cad29b3
+--- /dev/null
++++ b/crypto/asymmetric_keys/Kconfig
+@@ -0,0 +1,13 @@
++menuconfig ASYMMETRIC_KEY_TYPE
++ tristate "Asymmetric (public-key cryptographic) key type"
++ depends on KEYS
++ help
++ This option provides support for a key type that holds the data for
++ the asymmetric keys used for public key cryptographic operations such
++ as encryption, decryption, signature generation and signature
++ verification.
++
++if ASYMMETRIC_KEY_TYPE
++
++
++endif # ASYMMETRIC_KEY_TYPE
+diff --git a/crypto/asymmetric_keys/Makefile b/crypto/asymmetric_keys/Makefile
+new file mode 100644
+index 0000000..b725bcc
+--- /dev/null
++++ b/crypto/asymmetric_keys/Makefile
+@@ -0,0 +1,7 @@
++#
++# Makefile for asymmetric cryptographic keys
++#
++
++obj-$(CONFIG_ASYMMETRIC_KEY_TYPE) += asymmetric_keys.o
++
++asymmetric_keys-y := asymmetric_type.o
+diff --git a/crypto/asymmetric_keys/asymmetric_keys.h b/crypto/asymmetric_keys/asymmetric_keys.h
+new file mode 100644
+index 0000000..515b634
+--- /dev/null
++++ b/crypto/asymmetric_keys/asymmetric_keys.h
+@@ -0,0 +1,15 @@
++/* Internal definitions for asymmetric key type
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++static inline const char *asymmetric_key_id(const struct key *key)
++{
++ return key->type_data.p[1];
++}
+diff --git a/crypto/asymmetric_keys/asymmetric_type.c b/crypto/asymmetric_keys/asymmetric_type.c
+new file mode 100644
+index 0000000..bfb0424
+--- /dev/null
++++ b/crypto/asymmetric_keys/asymmetric_type.c
+@@ -0,0 +1,156 @@
++/* Asymmetric public-key cryptography key type
++ *
++ * See Documentation/security/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++#include <keys/asymmetric-subtype.h>
++#include <linux/seq_file.h>
++#include <linux/module.h>
++#include <linux/slab.h>
++#include "asymmetric_keys.h"
++
++MODULE_LICENSE("GPL");
++
++/*
++ * Match asymmetric keys on (part of) their name
++ * We have some shorthand methods for matching keys. We allow:
++ *
++ * "<desc>" - request a key by description
++ * "id:<id>" - request a key matching the ID
++ * "<subtype>:<id>" - request a key of a subtype
++ */
++static int asymmetric_key_match(const struct key *key, const void *description)
++{
++ const struct asymmetric_key_subtype *subtype = asymmetric_key_subtype(key);
++ const char *spec = description;
++ const char *id, *kid;
++ ptrdiff_t speclen;
++ size_t idlen, kidlen;
++
++ if (!subtype || !spec || !*spec)
++ return 0;
++
++ /* See if the full key description matches as is */
++ if (key->description && strcmp(key->description, description) == 0)
++ return 1;
++
++ /* All tests from here on break the criterion description into a
++ * specifier, a colon and then an identifier.
++ */
++ id = strchr(spec, ':');
++ if (!id)
++ return 0;
++
++ speclen = id - spec;
++ id++;
++
++ /* Anything after here requires a partial match on the ID string */
++ kid = asymmetric_key_id(key);
++ if (!kid)
++ return 0;
++
++ idlen = strlen(id);
++ kidlen = strlen(kid);
++ if (idlen > kidlen)
++ return 0;
++
++ kid += kidlen - idlen;
++ if (strcasecmp(id, kid) != 0)
++ return 0;
++
++ if (speclen == 2 &&
++ memcmp(spec, "id", 2) == 0)
++ return 1;
++
++ if (speclen == subtype->name_len &&
++ memcmp(spec, subtype->name, speclen) == 0)
++ return 1;
++
++ return 0;
++}
++
++/*
++ * Describe the asymmetric key
++ */
++static void asymmetric_key_describe(const struct key *key, struct seq_file *m)
++{
++ const struct asymmetric_key_subtype *subtype = asymmetric_key_subtype(key);
++ const char *kid = asymmetric_key_id(key);
++ size_t n;
++
++ seq_puts(m, key->description);
++
++ if (subtype) {
++ seq_puts(m, ": ");
++ subtype->describe(key, m);
++
++ if (kid) {
++ seq_putc(m, ' ');
++ n = strlen(kid);
++ if (n <= 8)
++ seq_puts(m, kid);
++ else
++ seq_puts(m, kid + n - 8);
++ }
++
++ seq_puts(m, " [");
++ /* put something here to indicate the key's capabilities */
++ seq_putc(m, ']');
++ }
++}
++
++/*
++ * Instantiate a asymmetric_key defined key. The key was preparsed, so we just
++ * have to transfer the data here.
++ */
++static int asymmetric_key_instantiate(struct key *key, struct key_preparsed_payload *prep)
++{
++ return -EOPNOTSUPP;
++}
++
++/*
++ * dispose of the data dangling from the corpse of a asymmetric key
++ */
++static void asymmetric_key_destroy(struct key *key)
++{
++ struct asymmetric_key_subtype *subtype = asymmetric_key_subtype(key);
++ if (subtype) {
++ subtype->destroy(key->payload.data);
++ module_put(subtype->owner);
++ key->type_data.p[0] = NULL;
++ }
++ kfree(key->type_data.p[1]);
++ key->type_data.p[1] = NULL;
++}
++
++struct key_type key_type_asymmetric = {
++ .name = "asymmetric",
++ .instantiate = asymmetric_key_instantiate,
++ .match = asymmetric_key_match,
++ .destroy = asymmetric_key_destroy,
++ .describe = asymmetric_key_describe,
++};
++EXPORT_SYMBOL_GPL(key_type_asymmetric);
++
++/*
++ * Module stuff
++ */
++static int __init asymmetric_key_init(void)
++{
++ return register_key_type(&key_type_asymmetric);
++}
++
++static void __exit asymmetric_key_cleanup(void)
++{
++ unregister_key_type(&key_type_asymmetric);
++}
++
++module_init(asymmetric_key_init);
++module_exit(asymmetric_key_cleanup);
+diff --git a/include/keys/asymmetric-subtype.h b/include/keys/asymmetric-subtype.h
+new file mode 100644
+index 0000000..4b840e8
+--- /dev/null
++++ b/include/keys/asymmetric-subtype.h
+@@ -0,0 +1,55 @@
++/* Asymmetric public-key cryptography key subtype
++ *
++ * See Documentation/security/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _KEYS_ASYMMETRIC_SUBTYPE_H
++#define _KEYS_ASYMMETRIC_SUBTYPE_H
++
++#include <linux/seq_file.h>
++#include <keys/asymmetric-type.h>
++
++struct public_key_signature;
++
++/*
++ * Keys of this type declare a subtype that indicates the handlers and
++ * capabilities.
++ */
++struct asymmetric_key_subtype {
++ struct module *owner;
++ const char *name;
++ unsigned short name_len; /* length of name */
++
++ /* Describe a key of this subtype for /proc/keys */
++ void (*describe)(const struct key *key, struct seq_file *m);
++
++ /* Destroy a key of this subtype */
++ void (*destroy)(void *payload);
++
++ /* Verify the signature on a key of this subtype (optional) */
++ int (*verify_signature)(const struct key *key,
++ const struct public_key_signature *sig);
++};
++
++/**
++ * asymmetric_key_subtype - Get the subtype from an asymmetric key
++ * @key: The key of interest.
++ *
++ * Retrieves and returns the subtype pointer of the asymmetric key from the
++ * type-specific data attached to the key.
++ */
++static inline
++struct asymmetric_key_subtype *asymmetric_key_subtype(const struct key *key)
++{
++ return key->type_data.p[0];
++}
++
++#endif /* _KEYS_ASYMMETRIC_SUBTYPE_H */
+diff --git a/include/keys/asymmetric-type.h b/include/keys/asymmetric-type.h
+new file mode 100644
+index 0000000..7dd4734
+--- /dev/null
++++ b/include/keys/asymmetric-type.h
+@@ -0,0 +1,25 @@
++/* Asymmetric Public-key cryptography key type interface
++ *
++ * See Documentation/security/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _KEYS_ASYMMETRIC_TYPE_H
++#define _KEYS_ASYMMETRIC_TYPE_H
++
++#include <linux/key-type.h>
++
++extern struct key_type key_type_asymmetric;
++
++/*
++ * The payload is at the discretion of the subtype.
++ */
++
++#endif /* _KEYS_ASYMMETRIC_TYPE_H */
+--
+1.7.11.4
+
+
+From d28ffd9e0c8987e5af59ae38bb48779b0d221f00 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Thu, 13 Sep 2012 15:17:32 +0100
+Subject: [PATCH 06/26] KEYS: Asymmetric key pluggable data parsers
+
+The instantiation data passed to the asymmetric key type are expected to be
+formatted in some way, and there are several possible standard ways to format
+the data.
+
+The two obvious standards are OpenPGP keys and X.509 certificates. The latter
+is especially useful when dealing with UEFI, and the former might be useful
+when dealing with, say, eCryptfs.
+
+Further, it might be desirable to provide formatted blobs that indicate
+hardware is to be accessed to retrieve the keys or that the keys live
+unretrievably in a hardware store, but that the keys can be used by means of
+the hardware.
+
+From userspace, the keys can be loaded using the keyctl command, for example,
+an X.509 binary certificate:
+
+ keyctl padd asymmetric foo @s <dhowells.pem
+
+or a PGP key:
+
+ keyctl padd asymmetric bar @s <dhowells.pub
+
+or a pointer into the contents of the TPM:
+
+ keyctl add asymmetric zebra "TPM:04982390582905f8" @s
+
+Inside the kernel, pluggable parsers register themselves and then get to
+examine the payload data to see if they can handle it. If they can, they get
+to:
+
+ (1) Propose a name for the key, to be used it the name is "" or NULL.
+
+ (2) Specify the key subtype.
+
+ (3) Provide the data for the subtype.
+
+The key type asks the parser to do its stuff before a key is allocated and thus
+before the name is set. If successful, the parser stores the suggested data
+into the key_preparsed_payload struct, which will be either used (if the key is
+successfully created and instantiated or updated) or discarded.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ crypto/asymmetric_keys/asymmetric_type.c | 120 ++++++++++++++++++++++++++++++-
+ include/keys/asymmetric-parser.h | 37 ++++++++++
+ 2 files changed, 156 insertions(+), 1 deletion(-)
+ create mode 100644 include/keys/asymmetric-parser.h
+
+diff --git a/crypto/asymmetric_keys/asymmetric_type.c b/crypto/asymmetric_keys/asymmetric_type.c
+index bfb0424..cf80765 100644
+--- a/crypto/asymmetric_keys/asymmetric_type.c
++++ b/crypto/asymmetric_keys/asymmetric_type.c
+@@ -11,6 +11,7 @@
+ * 2 of the Licence, or (at your option) any later version.
+ */
+ #include <keys/asymmetric-subtype.h>
++#include <keys/asymmetric-parser.h>
+ #include <linux/seq_file.h>
+ #include <linux/module.h>
+ #include <linux/slab.h>
+@@ -18,6 +19,9 @@
+
+ MODULE_LICENSE("GPL");
+
++static LIST_HEAD(asymmetric_key_parsers);
++static DECLARE_RWSEM(asymmetric_key_parsers_sem);
++
+ /*
+ * Match asymmetric keys on (part of) their name
+ * We have some shorthand methods for matching keys. We allow:
+@@ -107,12 +111,79 @@ static void asymmetric_key_describe(const struct key *key, struct seq_file *m)
+ }
+
+ /*
++ * Preparse a asymmetric payload to get format the contents appropriately for the
++ * internal payload to cut down on the number of scans of the data performed.
++ *
++ * We also generate a proposed description from the contents of the key that
++ * can be used to name the key if the user doesn't want to provide one.
++ */
++static int asymmetric_key_preparse(struct key_preparsed_payload *prep)
++{
++ struct asymmetric_key_parser *parser;
++ int ret;
++
++ pr_devel("==>%s()\n", __func__);
++
++ if (prep->datalen == 0)
++ return -EINVAL;
++
++ down_read(&asymmetric_key_parsers_sem);
++
++ ret = -EBADMSG;
++ list_for_each_entry(parser, &asymmetric_key_parsers, link) {
++ pr_debug("Trying parser '%s'\n", parser->name);
++
++ ret = parser->parse(prep);
++ if (ret != -EBADMSG) {
++ pr_debug("Parser recognised the format (ret %d)\n",
++ ret);
++ break;
++ }
++ }
++
++ up_read(&asymmetric_key_parsers_sem);
++ pr_devel("<==%s() = %d\n", __func__, ret);
++ return ret;
++}
++
++/*
++ * Clean up the preparse data
++ */
++static void asymmetric_key_free_preparse(struct key_preparsed_payload *prep)
++{
++ struct asymmetric_key_subtype *subtype = prep->type_data[0];
++
++ pr_devel("==>%s()\n", __func__);
++
++ if (subtype) {
++ subtype->destroy(prep->payload);
++ module_put(subtype->owner);
++ }
++ kfree(prep->type_data[1]);
++ kfree(prep->description);
++}
++
++/*
+ * Instantiate a asymmetric_key defined key. The key was preparsed, so we just
+ * have to transfer the data here.
+ */
+ static int asymmetric_key_instantiate(struct key *key, struct key_preparsed_payload *prep)
+ {
+- return -EOPNOTSUPP;
++ int ret;
++
++ pr_devel("==>%s()\n", __func__);
++
++ ret = key_payload_reserve(key, prep->quotalen);
++ if (ret == 0) {
++ key->type_data.p[0] = prep->type_data[0];
++ key->type_data.p[1] = prep->type_data[1];
++ key->payload.data = prep->payload;
++ prep->type_data[0] = NULL;
++ prep->type_data[1] = NULL;
++ prep->payload = NULL;
++ }
++ pr_devel("<==%s() = %d\n", __func__, ret);
++ return ret;
+ }
+
+ /*
+@@ -132,6 +203,8 @@ static void asymmetric_key_destroy(struct key *key)
+
+ struct key_type key_type_asymmetric = {
+ .name = "asymmetric",
++ .preparse = asymmetric_key_preparse,
++ .free_preparse = asymmetric_key_free_preparse,
+ .instantiate = asymmetric_key_instantiate,
+ .match = asymmetric_key_match,
+ .destroy = asymmetric_key_destroy,
+@@ -139,6 +212,51 @@ struct key_type key_type_asymmetric = {
+ };
+ EXPORT_SYMBOL_GPL(key_type_asymmetric);
+
++/**
++ * register_asymmetric_key_parser - Register a asymmetric key blob parser
++ * @parser: The parser to register
++ */
++int register_asymmetric_key_parser(struct asymmetric_key_parser *parser)
++{
++ struct asymmetric_key_parser *cursor;
++ int ret;
++
++ down_write(&asymmetric_key_parsers_sem);
++
++ list_for_each_entry(cursor, &asymmetric_key_parsers, link) {
++ if (strcmp(cursor->name, parser->name) == 0) {
++ pr_err("Asymmetric key parser '%s' already registered\n",
++ parser->name);
++ ret = -EEXIST;
++ goto out;
++ }
++ }
++
++ list_add_tail(&parser->link, &asymmetric_key_parsers);
++
++ pr_notice("Asymmetric key parser '%s' registered\n", parser->name);
++ ret = 0;
++
++out:
++ up_write(&asymmetric_key_parsers_sem);
++ return ret;
++}
++EXPORT_SYMBOL_GPL(register_asymmetric_key_parser);
++
++/**
++ * unregister_asymmetric_key_parser - Unregister a asymmetric key blob parser
++ * @parser: The parser to unregister
++ */
++void unregister_asymmetric_key_parser(struct asymmetric_key_parser *parser)
++{
++ down_write(&asymmetric_key_parsers_sem);
++ list_del(&parser->link);
++ up_write(&asymmetric_key_parsers_sem);
++
++ pr_notice("Asymmetric key parser '%s' unregistered\n", parser->name);
++}
++EXPORT_SYMBOL_GPL(unregister_asymmetric_key_parser);
++
+ /*
+ * Module stuff
+ */
+diff --git a/include/keys/asymmetric-parser.h b/include/keys/asymmetric-parser.h
+new file mode 100644
+index 0000000..09b3b48
+--- /dev/null
++++ b/include/keys/asymmetric-parser.h
+@@ -0,0 +1,37 @@
++/* Asymmetric public-key cryptography data parser
++ *
++ * See Documentation/crypto/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _KEYS_ASYMMETRIC_PARSER_H
++#define _KEYS_ASYMMETRIC_PARSER_H
++
++/*
++ * Key data parser. Called during key instantiation.
++ */
++struct asymmetric_key_parser {
++ struct list_head link;
++ struct module *owner;
++ const char *name;
++
++ /* Attempt to parse a key from the data blob passed to add_key() or
++ * keyctl_instantiate(). Should also generate a proposed description
++ * that the caller can optionally use for the key.
++ *
++ * Return EBADMSG if not recognised.
++ */
++ int (*parse)(struct key_preparsed_payload *prep);
++};
++
++extern int register_asymmetric_key_parser(struct asymmetric_key_parser *);
++extern void unregister_asymmetric_key_parser(struct asymmetric_key_parser *);
++
++#endif /* _KEYS_ASYMMETRIC_PARSER_H */
+--
+1.7.11.4
+
+
+From 171881bb3693176db4d0f85f78066fcdb6266525 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:24:55 +0100
+Subject: [PATCH 07/26] KEYS: Asymmetric public-key algorithm crypto key
+ subtype
+
+Add a subtype for supporting asymmetric public-key encryption algorithms such
+as DSA (FIPS-186) and RSA (PKCS#1 / RFC1337).
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ crypto/asymmetric_keys/Kconfig | 8 +++
+ crypto/asymmetric_keys/Makefile | 2 +
+ crypto/asymmetric_keys/public_key.c | 108 ++++++++++++++++++++++++++++++++++++
+ crypto/asymmetric_keys/public_key.h | 28 ++++++++++
+ include/crypto/public_key.h | 104 ++++++++++++++++++++++++++++++++++
+ 5 files changed, 250 insertions(+)
+ create mode 100644 crypto/asymmetric_keys/public_key.c
+ create mode 100644 crypto/asymmetric_keys/public_key.h
+ create mode 100644 include/crypto/public_key.h
+
+diff --git a/crypto/asymmetric_keys/Kconfig b/crypto/asymmetric_keys/Kconfig
+index cad29b3..bbfccaa 100644
+--- a/crypto/asymmetric_keys/Kconfig
++++ b/crypto/asymmetric_keys/Kconfig
+@@ -9,5 +9,13 @@ menuconfig ASYMMETRIC_KEY_TYPE
+
+ if ASYMMETRIC_KEY_TYPE
+
++config ASYMMETRIC_PUBLIC_KEY_SUBTYPE
++ tristate "Asymmetric public-key crypto algorithm subtype"
++ select MPILIB
++ help
++ This option provides support for asymmetric public key type handling.
++ If signature generation and/or verification are to be used,
++ appropriate hash algorithms (such as SHA-1) must be available.
++ ENOPKG will be reported if the requisite algorithm is unavailable.
+
+ endif # ASYMMETRIC_KEY_TYPE
+diff --git a/crypto/asymmetric_keys/Makefile b/crypto/asymmetric_keys/Makefile
+index b725bcc..5ed46ee 100644
+--- a/crypto/asymmetric_keys/Makefile
++++ b/crypto/asymmetric_keys/Makefile
+@@ -5,3 +5,5 @@
+ obj-$(CONFIG_ASYMMETRIC_KEY_TYPE) += asymmetric_keys.o
+
+ asymmetric_keys-y := asymmetric_type.o
++
++obj-$(CONFIG_ASYMMETRIC_PUBLIC_KEY_SUBTYPE) += public_key.o
+diff --git a/crypto/asymmetric_keys/public_key.c b/crypto/asymmetric_keys/public_key.c
+new file mode 100644
+index 0000000..cb2e291
+--- /dev/null
++++ b/crypto/asymmetric_keys/public_key.c
+@@ -0,0 +1,108 @@
++/* In-software asymmetric public-key crypto subtype
++ *
++ * See Documentation/crypto/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#define pr_fmt(fmt) "PKEY: "fmt
++#include <linux/module.h>
++#include <linux/export.h>
++#include <linux/kernel.h>
++#include <linux/slab.h>
++#include <linux/seq_file.h>
++#include <keys/asymmetric-subtype.h>
++#include "public_key.h"
++
++MODULE_LICENSE("GPL");
++
++const char *const pkey_algo[PKEY_ALGO__LAST] = {
++ [PKEY_ALGO_DSA] = "DSA",
++ [PKEY_ALGO_RSA] = "RSA",
++};
++EXPORT_SYMBOL_GPL(pkey_algo);
++
++const char *const pkey_hash_algo[PKEY_HASH__LAST] = {
++ [PKEY_HASH_MD4] = "md4",
++ [PKEY_HASH_MD5] = "md5",
++ [PKEY_HASH_SHA1] = "sha1",
++ [PKEY_HASH_RIPE_MD_160] = "rmd160",
++ [PKEY_HASH_SHA256] = "sha256",
++ [PKEY_HASH_SHA384] = "sha384",
++ [PKEY_HASH_SHA512] = "sha512",
++ [PKEY_HASH_SHA224] = "sha224",
++};
++EXPORT_SYMBOL_GPL(pkey_hash_algo);
++
++const char *const pkey_id_type[PKEY_ID_TYPE__LAST] = {
++ [PKEY_ID_PGP] = "PGP",
++ [PKEY_ID_X509] = "X509",
++};
++EXPORT_SYMBOL_GPL(pkey_id_type);
++
++/*
++ * Provide a part of a description of the key for /proc/keys.
++ */
++static void public_key_describe(const struct key *asymmetric_key,
++ struct seq_file *m)
++{
++ struct public_key *key = asymmetric_key->payload.data;
++
++ if (key)
++ seq_printf(m, "%s.%s",
++ pkey_id_type[key->id_type], key->algo->name);
++}
++
++/*
++ * Destroy a public key algorithm key.
++ */
++void public_key_destroy(void *payload)
++{
++ struct public_key *key = payload;
++ int i;
++
++ if (key) {
++ for (i = 0; i < ARRAY_SIZE(key->mpi); i++)
++ mpi_free(key->mpi[i]);
++ kfree(key);
++ }
++}
++EXPORT_SYMBOL_GPL(public_key_destroy);
++
++/*
++ * Verify a signature using a public key.
++ */
++static int public_key_verify_signature(const struct key *key,
++ const struct public_key_signature *sig)
++{
++ const struct public_key *pk = key->payload.data;
++
++ if (!pk->algo->verify_signature)
++ return -ENOTSUPP;
++
++ if (sig->nr_mpi != pk->algo->n_sig_mpi) {
++ pr_debug("Signature has %u MPI not %u\n",
++ sig->nr_mpi, pk->algo->n_sig_mpi);
++ return -EINVAL;
++ }
++
++ return pk->algo->verify_signature(pk, sig);
++}
++
++/*
++ * Public key algorithm asymmetric key subtype
++ */
++struct asymmetric_key_subtype public_key_subtype = {
++ .owner = THIS_MODULE,
++ .name = "public_key",
++ .describe = public_key_describe,
++ .destroy = public_key_destroy,
++ .verify_signature = public_key_verify_signature,
++};
++EXPORT_SYMBOL_GPL(public_key_subtype);
+diff --git a/crypto/asymmetric_keys/public_key.h b/crypto/asymmetric_keys/public_key.h
+new file mode 100644
+index 0000000..1f86aad
+--- /dev/null
++++ b/crypto/asymmetric_keys/public_key.h
+@@ -0,0 +1,28 @@
++/* Public key algorithm internals
++ *
++ * See Documentation/crypto/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <crypto/public_key.h>
++
++extern struct asymmetric_key_subtype public_key_subtype;
++
++/*
++ * Public key algorithm definition.
++ */
++struct public_key_algorithm {
++ const char *name;
++ u8 n_pub_mpi; /* Number of MPIs in public key */
++ u8 n_sec_mpi; /* Number of MPIs in secret key */
++ u8 n_sig_mpi; /* Number of MPIs in a signature */
++ int (*verify_signature)(const struct public_key *key,
++ const struct public_key_signature *sig);
++};
+diff --git a/include/crypto/public_key.h b/include/crypto/public_key.h
+new file mode 100644
+index 0000000..4b8b6c1
+--- /dev/null
++++ b/include/crypto/public_key.h
+@@ -0,0 +1,104 @@
++/* Asymmetric public-key algorithm definitions
++ *
++ * See Documentation/crypto/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _LINUX_PUBLIC_KEY_H
++#define _LINUX_PUBLIC_KEY_H
++
++#include <linux/mpi.h>
++
++enum pkey_algo {
++ PKEY_ALGO_DSA,
++ PKEY_ALGO_RSA,
++ PKEY_ALGO__LAST
++};
++
++extern const char *const pkey_algo[PKEY_ALGO__LAST];
++
++enum pkey_hash_algo {
++ PKEY_HASH_MD4,
++ PKEY_HASH_MD5,
++ PKEY_HASH_SHA1,
++ PKEY_HASH_RIPE_MD_160,
++ PKEY_HASH_SHA256,
++ PKEY_HASH_SHA384,
++ PKEY_HASH_SHA512,
++ PKEY_HASH_SHA224,
++ PKEY_HASH__LAST
++};
++
++extern const char *const pkey_hash_algo[PKEY_HASH__LAST];
++
++enum pkey_id_type {
++ PKEY_ID_PGP, /* OpenPGP generated key ID */
++ PKEY_ID_X509, /* X.509 arbitrary subjectKeyIdentifier */
++ PKEY_ID_TYPE__LAST
++};
++
++extern const char *const pkey_id_type[PKEY_ID_TYPE__LAST];
++
++/*
++ * Cryptographic data for the public-key subtype of the asymmetric key type.
++ *
++ * Note that this may include private part of the key as well as the public
++ * part.
++ */
++struct public_key {
++ const struct public_key_algorithm *algo;
++ u8 capabilities;
++#define PKEY_CAN_ENCRYPT 0x01
++#define PKEY_CAN_DECRYPT 0x02
++#define PKEY_CAN_SIGN 0x04
++#define PKEY_CAN_VERIFY 0x08
++ enum pkey_id_type id_type : 8;
++ union {
++ MPI mpi[5];
++ struct {
++ MPI p; /* DSA prime */
++ MPI q; /* DSA group order */
++ MPI g; /* DSA group generator */
++ MPI y; /* DSA public-key value = g^x mod p */
++ MPI x; /* DSA secret exponent (if present) */
++ } dsa;
++ struct {
++ MPI n; /* RSA public modulus */
++ MPI e; /* RSA public encryption exponent */
++ MPI d; /* RSA secret encryption exponent (if present) */
++ MPI p; /* RSA secret prime (if present) */
++ MPI q; /* RSA secret prime (if present) */
++ } rsa;
++ };
++};
++
++extern void public_key_destroy(void *payload);
++
++/*
++ * Public key cryptography signature data
++ */
++struct public_key_signature {
++ u8 *digest;
++ u8 digest_size; /* Number of bytes in digest */
++ u8 nr_mpi; /* Occupancy of mpi[] */
++ enum pkey_hash_algo pkey_hash_algo : 8;
++ union {
++ MPI mpi[2];
++ struct {
++ MPI s; /* m^d mod n */
++ } rsa;
++ struct {
++ MPI r;
++ MPI s;
++ } dsa;
++ };
++};
++
++#endif /* _LINUX_PUBLIC_KEY_H */
+--
+1.7.11.4
+
+
+From 8e45e274bb3f8bb90306e0102a2c48ea1ef179fb Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:25:04 +0100
+Subject: [PATCH 08/26] KEYS: Provide signature verification with an
+ asymmetric key
+
+Provide signature verification using an asymmetric-type key to indicate the
+public key to be used.
+
+The API is a single function that can be found in crypto/public_key.h:
+
+ int verify_signature(const struct key *key,
+ const struct public_key_signature *sig)
+
+The first argument is the appropriate key to be used and the second argument
+is the parsed signature data:
+
+ struct public_key_signature {
+ u8 *digest;
+ u16 digest_size;
+ enum pkey_hash_algo pkey_hash_algo : 8;
+ union {
+ MPI mpi[2];
+ struct {
+ MPI s; /* m^d mod n */
+ } rsa;
+ struct {
+ MPI r;
+ MPI s;
+ } dsa;
+ };
+ };
+
+This should be filled in prior to calling the function. The hash algorithm
+should already have been called and the hash finalised and the output should
+be in a buffer pointed to by the 'digest' member.
+
+Any extra data to be added to the hash by the hash format (eg. PGP) should
+have been added by the caller prior to finalising the hash.
+
+It is assumed that the signature is made up of a number of MPI values. If an
+algorithm becomes available for which this is not the case, the above structure
+will have to change.
+
+It is also assumed that it will have been checked that the signature algorithm
+matches the key algorithm.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ crypto/asymmetric_keys/Makefile | 2 +-
+ crypto/asymmetric_keys/signature.c | 49 ++++++++++++++++++++++++++++++++++++++
+ include/crypto/public_key.h | 4 ++++
+ 3 files changed, 54 insertions(+), 1 deletion(-)
+ create mode 100644 crypto/asymmetric_keys/signature.c
+
+diff --git a/crypto/asymmetric_keys/Makefile b/crypto/asymmetric_keys/Makefile
+index 5ed46ee..8dcdf0c 100644
+--- a/crypto/asymmetric_keys/Makefile
++++ b/crypto/asymmetric_keys/Makefile
+@@ -4,6 +4,6 @@
+
+ obj-$(CONFIG_ASYMMETRIC_KEY_TYPE) += asymmetric_keys.o
+
+-asymmetric_keys-y := asymmetric_type.o
++asymmetric_keys-y := asymmetric_type.o signature.o
+
+ obj-$(CONFIG_ASYMMETRIC_PUBLIC_KEY_SUBTYPE) += public_key.o
+diff --git a/crypto/asymmetric_keys/signature.c b/crypto/asymmetric_keys/signature.c
+new file mode 100644
+index 0000000..50b3f88
+--- /dev/null
++++ b/crypto/asymmetric_keys/signature.c
+@@ -0,0 +1,49 @@
++/* Signature verification with an asymmetric key
++ *
++ * See Documentation/security/asymmetric-keys.txt
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <keys/asymmetric-subtype.h>
++#include <linux/module.h>
++#include <linux/err.h>
++#include <crypto/public_key.h>
++#include "asymmetric_keys.h"
++
++/**
++ * verify_signature - Initiate the use of an asymmetric key to verify a signature
++ * @key: The asymmetric key to verify against
++ * @sig: The signature to check
++ *
++ * Returns 0 if successful or else an error.
++ */
++int verify_signature(const struct key *key,
++ const struct public_key_signature *sig)
++{
++ const struct asymmetric_key_subtype *subtype;
++ int ret;
++
++ pr_devel("==>%s()\n", __func__);
++
++ if (key->type != &key_type_asymmetric)
++ return -EINVAL;
++ subtype = asymmetric_key_subtype(key);
++ if (!subtype ||
++ !key->payload.data)
++ return -EINVAL;
++ if (!subtype->verify_signature)
++ return -ENOTSUPP;
++
++ ret = subtype->verify_signature(key, sig);
++
++ pr_devel("<==%s() = %d\n", __func__, ret);
++ return ret;
++}
++EXPORT_SYMBOL_GPL(verify_signature);
+diff --git a/include/crypto/public_key.h b/include/crypto/public_key.h
+index 4b8b6c1..f5b0224 100644
+--- a/include/crypto/public_key.h
++++ b/include/crypto/public_key.h
+@@ -101,4 +101,8 @@ struct public_key_signature {
+ };
+ };
+
++struct key;
++extern int verify_signature(const struct key *key,
++ const struct public_key_signature *sig);
++
+ #endif /* _LINUX_PUBLIC_KEY_H */
+--
+1.7.11.4
+
+
+From 941e72448fc68f41a56798112a6f20df6bcd0ad0 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:25:22 +0100
+Subject: [PATCH 09/26] MPILIB: Reinstate mpi_cmp[_ui]() and export for RSA
+ signature verification
+
+Reinstate and export mpi_cmp() and mpi_cmp_ui() from the MPI library for use by
+RSA signature verification as per RFC3447 section 5.2.2 step 1.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ lib/mpi/Makefile | 1 +
+ lib/mpi/mpi-cmp.c | 70 +++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ 2 files changed, 71 insertions(+)
+ create mode 100644 lib/mpi/mpi-cmp.c
+
+diff --git a/lib/mpi/Makefile b/lib/mpi/Makefile
+index 45ca90a..019a68c 100644
+--- a/lib/mpi/Makefile
++++ b/lib/mpi/Makefile
+@@ -14,6 +14,7 @@ mpi-y = \
+ generic_mpih-add1.o \
+ mpicoder.o \
+ mpi-bit.o \
++ mpi-cmp.o \
+ mpih-cmp.o \
+ mpih-div.o \
+ mpih-mul.o \
+diff --git a/lib/mpi/mpi-cmp.c b/lib/mpi/mpi-cmp.c
+new file mode 100644
+index 0000000..1871e7b
+--- /dev/null
++++ b/lib/mpi/mpi-cmp.c
+@@ -0,0 +1,70 @@
++/* mpi-cmp.c - MPI functions
++ * Copyright (C) 1998, 1999 Free Software Foundation, Inc.
++ *
++ * This file is part of GnuPG.
++ *
++ * GnuPG is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * GnuPG is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
++ */
++
++#include "mpi-internal.h"
++
++int mpi_cmp_ui(MPI u, unsigned long v)
++{
++ mpi_limb_t limb = v;
++
++ mpi_normalize(u);
++ if (!u->nlimbs && !limb)
++ return 0;
++ if (u->sign)
++ return -1;
++ if (u->nlimbs > 1)
++ return 1;
++
++ if (u->d[0] == limb)
++ return 0;
++ else if (u->d[0] > limb)
++ return 1;
++ else
++ return -1;
++}
++EXPORT_SYMBOL_GPL(mpi_cmp_ui);
++
++int mpi_cmp(MPI u, MPI v)
++{
++ mpi_size_t usize, vsize;
++ int cmp;
++
++ mpi_normalize(u);
++ mpi_normalize(v);
++ usize = u->nlimbs;
++ vsize = v->nlimbs;
++ if (!u->sign && v->sign)
++ return 1;
++ if (u->sign && !v->sign)
++ return -1;
++ if (usize != vsize && !u->sign && !v->sign)
++ return usize - vsize;
++ if (usize != vsize && u->sign && v->sign)
++ return vsize + usize;
++ if (!usize)
++ return 0;
++ cmp = mpihelp_cmp(u->d, v->d, usize);
++ if (!cmp)
++ return 0;
++ if ((cmp < 0 ? 1 : 0) == (u->sign ? 1 : 0))
++ return 1;
++ return -1;
++}
++EXPORT_SYMBOL_GPL(mpi_cmp);
+--
+1.7.11.4
+
+
+From ca876ff8fde3c6febb8a8578ca0e1608576071bf Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:25:40 +0100
+Subject: [PATCH 10/26] RSA: Implement signature verification algorithm
+ [PKCS#1 / RFC3447]
+
+Implement RSA public key cryptography [PKCS#1 / RFC3447]. At this time, only
+the signature verification algorithm is supported. This uses the asymmetric
+public key subtype to hold its key data.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ crypto/asymmetric_keys/Kconfig | 7 +
+ crypto/asymmetric_keys/Makefile | 1 +
+ crypto/asymmetric_keys/public_key.h | 2 +
+ crypto/asymmetric_keys/rsa.c | 269 ++++++++++++++++++++++++++++++++++++
+ 4 files changed, 279 insertions(+)
+ create mode 100644 crypto/asymmetric_keys/rsa.c
+
+diff --git a/crypto/asymmetric_keys/Kconfig b/crypto/asymmetric_keys/Kconfig
+index bbfccaa..561759d 100644
+--- a/crypto/asymmetric_keys/Kconfig
++++ b/crypto/asymmetric_keys/Kconfig
+@@ -18,4 +18,11 @@ config ASYMMETRIC_PUBLIC_KEY_SUBTYPE
+ appropriate hash algorithms (such as SHA-1) must be available.
+ ENOPKG will be reported if the requisite algorithm is unavailable.
+
++config PUBLIC_KEY_ALGO_RSA
++ tristate "RSA public-key algorithm"
++ depends on ASYMMETRIC_PUBLIC_KEY_SUBTYPE
++ select MPILIB_EXTRA
++ help
++ This option enables support for the RSA algorithm (PKCS#1, RFC3447).
++
+ endif # ASYMMETRIC_KEY_TYPE
+diff --git a/crypto/asymmetric_keys/Makefile b/crypto/asymmetric_keys/Makefile
+index 8dcdf0c..7c92691 100644
+--- a/crypto/asymmetric_keys/Makefile
++++ b/crypto/asymmetric_keys/Makefile
+@@ -7,3 +7,4 @@ obj-$(CONFIG_ASYMMETRIC_KEY_TYPE) += asymmetric_keys.o
+ asymmetric_keys-y := asymmetric_type.o signature.o
+
+ obj-$(CONFIG_ASYMMETRIC_PUBLIC_KEY_SUBTYPE) += public_key.o
++obj-$(CONFIG_PUBLIC_KEY_ALGO_RSA) += rsa.o
+diff --git a/crypto/asymmetric_keys/public_key.h b/crypto/asymmetric_keys/public_key.h
+index 1f86aad..5e5e356 100644
+--- a/crypto/asymmetric_keys/public_key.h
++++ b/crypto/asymmetric_keys/public_key.h
+@@ -26,3 +26,5 @@ struct public_key_algorithm {
+ int (*verify_signature)(const struct public_key *key,
+ const struct public_key_signature *sig);
+ };
++
++extern const struct public_key_algorithm RSA_public_key_algorithm;
+diff --git a/crypto/asymmetric_keys/rsa.c b/crypto/asymmetric_keys/rsa.c
+new file mode 100644
+index 0000000..9b31ee2
+--- /dev/null
++++ b/crypto/asymmetric_keys/rsa.c
+@@ -0,0 +1,269 @@
++/* RSA asymmetric public-key algorithm [RFC3447]
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#define pr_fmt(fmt) "RSA: "fmt
++#include <linux/module.h>
++#include <linux/kernel.h>
++#include <linux/slab.h>
++#include "public_key.h"
++
++MODULE_LICENSE("GPL");
++MODULE_DESCRIPTION("RSA Public Key Algorithm");
++
++#define kenter(FMT, ...) \
++ pr_devel("==> %s("FMT")\n", __func__, ##__VA_ARGS__)
++#define kleave(FMT, ...) \
++ pr_devel("<== %s()"FMT"\n", __func__, ##__VA_ARGS__)
++
++/*
++ * Hash algorithm OIDs plus ASN.1 DER wrappings [RFC4880 sec 5.2.2].
++ */
++static const u8 RSA_digest_info_MD5[] = {
++ 0x30, 0x20, 0x30, 0x0C, 0x06, 0x08,
++ 0x2A, 0x86, 0x48, 0x86, 0xF7, 0x0D, 0x02, 0x05, /* OID */
++ 0x05, 0x00, 0x04, 0x10
++};
++
++static const u8 RSA_digest_info_SHA1[] = {
++ 0x30, 0x21, 0x30, 0x09, 0x06, 0x05,
++ 0x2B, 0x0E, 0x03, 0x02, 0x1A,
++ 0x05, 0x00, 0x04, 0x14
++};
++
++static const u8 RSA_digest_info_RIPE_MD_160[] = {
++ 0x30, 0x21, 0x30, 0x09, 0x06, 0x05,
++ 0x2B, 0x24, 0x03, 0x02, 0x01,
++ 0x05, 0x00, 0x04, 0x14
++};
++
++static const u8 RSA_digest_info_SHA224[] = {
++ 0x30, 0x2d, 0x30, 0x0d, 0x06, 0x09,
++ 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x04,
++ 0x05, 0x00, 0x04, 0x1C
++};
++
++static const u8 RSA_digest_info_SHA256[] = {
++ 0x30, 0x31, 0x30, 0x0d, 0x06, 0x09,
++ 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01,
++ 0x05, 0x00, 0x04, 0x20
++};
++
++static const u8 RSA_digest_info_SHA384[] = {
++ 0x30, 0x41, 0x30, 0x0d, 0x06, 0x09,
++ 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x02,
++ 0x05, 0x00, 0x04, 0x30
++};
++
++static const u8 RSA_digest_info_SHA512[] = {
++ 0x30, 0x51, 0x30, 0x0d, 0x06, 0x09,
++ 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x03,
++ 0x05, 0x00, 0x04, 0x40
++};
++
++static const struct {
++ const u8 *data;
++ size_t size;
++} RSA_ASN1_templates[PKEY_HASH__LAST] = {
++#define _(X) { RSA_digest_info_##X, sizeof(RSA_digest_info_##X) }
++ [PKEY_HASH_MD5] = _(MD5),
++ [PKEY_HASH_SHA1] = _(SHA1),
++ [PKEY_HASH_RIPE_MD_160] = _(RIPE_MD_160),
++ [PKEY_HASH_SHA256] = _(SHA256),
++ [PKEY_HASH_SHA384] = _(SHA384),
++ [PKEY_HASH_SHA512] = _(SHA512),
++ [PKEY_HASH_SHA224] = _(SHA224),
++#undef _
++};
++
++/*
++ * RSAVP1() function [RFC3447 sec 5.2.2]
++ */
++static int RSAVP1(const struct public_key *key, MPI s, MPI *_m)
++{
++ MPI m;
++ int ret;
++
++ /* (1) Validate 0 <= s < n */
++ if (mpi_cmp_ui(s, 0) < 0) {
++ kleave(" = -EBADMSG [s < 0]");
++ return -EBADMSG;
++ }
++ if (mpi_cmp(s, key->rsa.n) >= 0) {
++ kleave(" = -EBADMSG [s >= n]");
++ return -EBADMSG;
++ }
++
++ m = mpi_alloc(0);
++ if (!m)
++ return -ENOMEM;
++
++ /* (2) m = s^e mod n */
++ ret = mpi_powm(m, s, key->rsa.e, key->rsa.n);
++ if (ret < 0) {
++ mpi_free(m);
++ return ret;
++ }
++
++ *_m = m;
++ return 0;
++}
++
++/*
++ * Integer to Octet String conversion [RFC3447 sec 4.1]
++ */
++static int RSA_I2OSP(MPI x, size_t xLen, u8 **_X)
++{
++ unsigned X_size, x_size;
++ int X_sign;
++ u8 *X;
++
++ /* Make sure the string is the right length. The number should begin
++ * with { 0x00, 0x01, ... } so we have to account for 15 leading zero
++ * bits not being reported by MPI.
++ */
++ x_size = mpi_get_nbits(x);
++ pr_devel("size(x)=%u xLen*8=%zu\n", x_size, xLen * 8);
++ if (x_size != xLen * 8 - 15)
++ return -ERANGE;
++
++ X = mpi_get_buffer(x, &X_size, &X_sign);
++ if (!X)
++ return -ENOMEM;
++ if (X_sign < 0) {
++ kfree(X);
++ return -EBADMSG;
++ }
++ if (X_size != xLen - 1) {
++ kfree(X);
++ return -EBADMSG;
++ }
++
++ *_X = X;
++ return 0;
++}
++
++/*
++ * Perform the RSA signature verification.
++ * @H: Value of hash of data and metadata
++ * @EM: The computed signature value
++ * @k: The size of EM (EM[0] is an invalid location but should hold 0x00)
++ * @hash_size: The size of H
++ * @asn1_template: The DigestInfo ASN.1 template
++ * @asn1_size: Size of asm1_template[]
++ */
++static int RSA_verify(const u8 *H, const u8 *EM, size_t k, size_t hash_size,
++ const u8 *asn1_template, size_t asn1_size)
++{
++ unsigned PS_end, T_offset, i;
++
++ kenter(",,%zu,%zu,%zu", k, hash_size, asn1_size);
++
++ if (k < 2 + 1 + asn1_size + hash_size)
++ return -EBADMSG;
++
++ /* Decode the EMSA-PKCS1-v1_5 */
++ if (EM[1] != 0x01) {
++ kleave(" = -EBADMSG [EM[1] == %02u]", EM[1]);
++ return -EBADMSG;
++ }
++
++ T_offset = k - (asn1_size + hash_size);
++ PS_end = T_offset - 1;
++ if (EM[PS_end] != 0x00) {
++ kleave(" = -EBADMSG [EM[T-1] == %02u]", EM[PS_end]);
++ return -EBADMSG;
++ }
++
++ for (i = 2; i < PS_end; i++) {
++ if (EM[i] != 0xff) {
++ kleave(" = -EBADMSG [EM[PS%x] == %02u]", i - 2, EM[i]);
++ return -EBADMSG;
++ }
++ }
++
++ if (memcmp(asn1_template, EM + T_offset, asn1_size) != 0) {
++ kleave(" = -EBADMSG [EM[T] ASN.1 mismatch]");
++ return -EBADMSG;
++ }
++
++ if (memcmp(H, EM + T_offset + asn1_size, hash_size) != 0) {
++ kleave(" = -EKEYREJECTED [EM[T] hash mismatch]");
++ return -EKEYREJECTED;
++ }
++
++ kleave(" = 0");
++ return 0;
++}
++
++/*
++ * Perform the verification step [RFC3447 sec 8.2.2].
++ */
++static int RSA_verify_signature(const struct public_key *key,
++ const struct public_key_signature *sig)
++{
++ size_t tsize;
++ int ret;
++
++ /* Variables as per RFC3447 sec 8.2.2 */
++ const u8 *H = sig->digest;
++ u8 *EM = NULL;
++ MPI m = NULL;
++ size_t k;
++
++ kenter("");
++
++ if (!RSA_ASN1_templates[sig->pkey_hash_algo].data)
++ return -ENOTSUPP;
++
++ /* (1) Check the signature size against the public key modulus size */
++ k = (mpi_get_nbits(key->rsa.n) + 7) / 8;
++
++ tsize = (mpi_get_nbits(sig->rsa.s) + 7) / 8;
++ pr_devel("step 1: k=%zu size(S)=%zu\n", k, tsize);
++ if (tsize != k) {
++ ret = -EBADMSG;
++ goto error;
++ }
++
++ /* (2b) Apply the RSAVP1 verification primitive to the public key */
++ ret = RSAVP1(key, sig->rsa.s, &m);
++ if (ret < 0)
++ goto error;
++
++ /* (2c) Convert the message representative (m) to an encoded message
++ * (EM) of length k octets.
++ *
++ * NOTE! The leading zero byte is suppressed by MPI, so we pass a
++ * pointer to the _preceding_ byte to RSA_verify()!
++ */
++ ret = RSA_I2OSP(m, k, &EM);
++ if (ret < 0)
++ goto error;
++
++ ret = RSA_verify(H, EM - 1, k, sig->digest_size,
++ RSA_ASN1_templates[sig->pkey_hash_algo].data,
++ RSA_ASN1_templates[sig->pkey_hash_algo].size);
++
++error:
++ kfree(EM);
++ mpi_free(m);
++ kleave(" = %d", ret);
++ return ret;
++}
++
++const struct public_key_algorithm RSA_public_key_algorithm = {
++ .name = "RSA",
++ .n_pub_mpi = 2,
++ .n_sec_mpi = 3,
++ .n_sig_mpi = 1,
++ .verify_signature = RSA_verify_signature,
++};
++EXPORT_SYMBOL_GPL(RSA_public_key_algorithm);
+--
+1.7.11.4
+
+
+From e179a9b04469ea018a8fdb53f11c57222ba540a0 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:28:05 +0100
+Subject: [PATCH 11/26] RSA: Fix signature verification for shorter signatures
+
+gpg can produce a signature file where length of signature is less than the
+modulus size because the amount of space an MPI takes up is kept as low as
+possible by discarding leading zeros. This regularly happens for several
+modules during the build.
+
+Fix it by relaxing check in RSA verification code.
+
+Thanks to Tomas Mraz and Miloslav Trmac for help.
+
+Signed-off-by: Milan Broz <mbroz@redhat.com>
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ crypto/asymmetric_keys/rsa.c | 14 +++++++++++---
+ 1 file changed, 11 insertions(+), 3 deletions(-)
+
+diff --git a/crypto/asymmetric_keys/rsa.c b/crypto/asymmetric_keys/rsa.c
+index 9b31ee2..4a6a069 100644
+--- a/crypto/asymmetric_keys/rsa.c
++++ b/crypto/asymmetric_keys/rsa.c
+@@ -224,15 +224,23 @@ static int RSA_verify_signature(const struct public_key *key,
+ return -ENOTSUPP;
+
+ /* (1) Check the signature size against the public key modulus size */
+- k = (mpi_get_nbits(key->rsa.n) + 7) / 8;
++ k = mpi_get_nbits(key->rsa.n);
++ tsize = mpi_get_nbits(sig->rsa.s);
+
+- tsize = (mpi_get_nbits(sig->rsa.s) + 7) / 8;
++ /* According to RFC 4880 sec 3.2, length of MPI is computed starting
++ * from most significant bit. So the RFC 3447 sec 8.2.2 size check
++ * must be relaxed to conform with shorter signatures - so we fail here
++ * only if signature length is longer than modulus size.
++ */
+ pr_devel("step 1: k=%zu size(S)=%zu\n", k, tsize);
+- if (tsize != k) {
++ if (k < tsize) {
+ ret = -EBADMSG;
+ goto error;
+ }
+
++ /* Round up and convert to octets */
++ k = (k + 7) / 8;
++
+ /* (2b) Apply the RSAVP1 verification primitive to the public key */
+ ret = RSAVP1(key, sig->rsa.s, &m);
+ if (ret < 0)
+--
+1.7.11.4
+
+
+From d412c256ea6170b6aeceb9f1eb1737d991473634 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:30:46 +0100
+Subject: [PATCH 12/26] X.509: Implement simple static OID registry
+
+Implement a simple static OID registry that allows the mapping of an encoded
+OID to an enum value for ease of use.
+
+The OID registry index enum appears in the:
+
+ linux/oid_registry.h
+
+header file. A script generates the registry from lines in the header file
+that look like:
+
+ <sp*>OID_foo,<sp*>/*<sp*>1.2.3.4<sp*>*/
+
+The actual OID is taken to be represented by the numbers with interpolated
+dots in the comment.
+
+All other lines in the header are ignored.
+
+The registry is queries by calling:
+
+ OID look_up_oid(const void *data, size_t datasize);
+
+This returns a number from the registry enum representing the OID if found or
+OID__NR if not.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ include/linux/oid_registry.h | 90 +++++++++++++++++++
+ lib/.gitignore | 2 +-
+ lib/Kconfig | 5 ++
+ lib/Makefile | 16 ++++
+ lib/build_OID_registry | 209 +++++++++++++++++++++++++++++++++++++++++++
+ lib/oid_registry.c | 89 ++++++++++++++++++
+ 6 files changed, 410 insertions(+), 1 deletion(-)
+ create mode 100644 include/linux/oid_registry.h
+ create mode 100755 lib/build_OID_registry
+ create mode 100644 lib/oid_registry.c
+
+diff --git a/include/linux/oid_registry.h b/include/linux/oid_registry.h
+new file mode 100644
+index 0000000..5928546
+--- /dev/null
++++ b/include/linux/oid_registry.h
+@@ -0,0 +1,90 @@
++/* ASN.1 Object identifier (OID) registry
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _LINUX_OID_REGISTRY_H
++#define _LINUX_OID_REGISTRY_H
++
++#include <linux/types.h>
++
++/*
++ * OIDs are turned into these values if possible, or OID__NR if not held here.
++ *
++ * NOTE! Do not mess with the format of each line as this is read by
++ * build_OID_registry.pl to generate the data for look_up_OID().
++ */
++enum OID {
++ OID_id_dsa_with_sha1, /* 1.2.840.10030.4.3 */
++ OID_id_dsa, /* 1.2.840.10040.4.1 */
++ OID_id_ecdsa_with_sha1, /* 1.2.840.10045.4.1 */
++ OID_id_ecPublicKey, /* 1.2.840.10045.2.1 */
++
++ /* PKCS#1 {iso(1) member-body(2) us(840) rsadsi(113549) pkcs(1) pkcs-1(1)} */
++ OID_rsaEncryption, /* 1.2.840.113549.1.1.1 */
++ OID_md2WithRSAEncryption, /* 1.2.840.113549.1.1.2 */
++ OID_md3WithRSAEncryption, /* 1.2.840.113549.1.1.3 */
++ OID_md4WithRSAEncryption, /* 1.2.840.113549.1.1.4 */
++ OID_sha1WithRSAEncryption, /* 1.2.840.113549.1.1.5 */
++ OID_sha256WithRSAEncryption, /* 1.2.840.113549.1.1.11 */
++ OID_sha384WithRSAEncryption, /* 1.2.840.113549.1.1.12 */
++ OID_sha512WithRSAEncryption, /* 1.2.840.113549.1.1.13 */
++ OID_sha224WithRSAEncryption, /* 1.2.840.113549.1.1.14 */
++ /* PKCS#7 {iso(1) member-body(2) us(840) rsadsi(113549) pkcs(1) pkcs-7(7)} */
++ OID_data, /* 1.2.840.113549.1.7.1 */
++ OID_signed_data, /* 1.2.840.113549.1.7.2 */
++ /* PKCS#9 {iso(1) member-body(2) us(840) rsadsi(113549) pkcs(1) pkcs-9(9)} */
++ OID_email_address, /* 1.2.840.113549.1.9.1 */
++ OID_content_type, /* 1.2.840.113549.1.9.3 */
++ OID_messageDigest, /* 1.2.840.113549.1.9.4 */
++ OID_signingTime, /* 1.2.840.113549.1.9.5 */
++ OID_smimeCapabilites, /* 1.2.840.113549.1.9.15 */
++ OID_smimeAuthenticatedAttrs, /* 1.2.840.113549.1.9.16.2.11 */
++
++ /* {iso(1) member-body(2) us(840) rsadsi(113549) digestAlgorithm(2)} */
++ OID_md2, /* 1.2.840.113549.2.2 */
++ OID_md4, /* 1.2.840.113549.2.4 */
++ OID_md5, /* 1.2.840.113549.2.5 */
++
++ OID_certAuthInfoAccess, /* 1.3.6.1.5.5.7.1.1 */
++ OID_msOutlookExpress, /* 1.3.6.1.4.1.311.16.4 */
++ OID_sha1, /* 1.3.14.3.2.26 */
++
++ /* Distinguished Name attribute IDs [RFC 2256] */
++ OID_commonName, /* 2.5.4.3 */
++ OID_surname, /* 2.5.4.4 */
++ OID_countryName, /* 2.5.4.6 */
++ OID_locality, /* 2.5.4.7 */
++ OID_stateOrProvinceName, /* 2.5.4.8 */
++ OID_organizationName, /* 2.5.4.10 */
++ OID_organizationUnitName, /* 2.5.4.11 */
++ OID_title, /* 2.5.4.12 */
++ OID_description, /* 2.5.4.13 */
++ OID_name, /* 2.5.4.41 */
++ OID_givenName, /* 2.5.4.42 */
++ OID_initials, /* 2.5.4.43 */
++ OID_generationalQualifier, /* 2.5.4.44 */
++
++ /* Certificate extension IDs */
++ OID_subjectKeyIdentifier, /* 2.5.29.14 */
++ OID_keyUsage, /* 2.5.29.15 */
++ OID_subjectAltName, /* 2.5.29.17 */
++ OID_issuerAltName, /* 2.5.29.18 */
++ OID_basicConstraints, /* 2.5.29.19 */
++ OID_crlDistributionPoints, /* 2.5.29.31 */
++ OID_certPolicies, /* 2.5.29.32 */
++ OID_authorityKeyIdentifier, /* 2.5.29.35 */
++ OID_extKeyUsage, /* 2.5.29.37 */
++
++ OID__NR
++};
++
++extern enum OID look_up_OID(const void *data, size_t datasize);
++
++#endif /* _LINUX_OID_REGISTRY_H */
+diff --git a/lib/.gitignore b/lib/.gitignore
+index 3bef1ea..09aae85 100644
+--- a/lib/.gitignore
++++ b/lib/.gitignore
+@@ -3,4 +3,4 @@
+ #
+ gen_crc32table
+ crc32table.h
+-
++oid_registry_data.c
+diff --git a/lib/Kconfig b/lib/Kconfig
+index bb94c1b..4b31a46 100644
+--- a/lib/Kconfig
++++ b/lib/Kconfig
+@@ -396,4 +396,9 @@ config SIGNATURE
+ config LIBFDT
+ bool
+
++config OID_REGISTRY
++ tristate
++ help
++ Enable fast lookup object identifier registry.
++
+ endmenu
+diff --git a/lib/Makefile b/lib/Makefile
+index 42d283e..b042896 100644
+--- a/lib/Makefile
++++ b/lib/Makefile
+@@ -150,3 +150,19 @@ quiet_cmd_crc32 = GEN $@
+
+ $(obj)/crc32table.h: $(obj)/gen_crc32table
+ $(call cmd,crc32)
++
++#
++# Build a fast OID lookip registry from include/linux/oid_registry.h
++#
++obj-$(CONFIG_OID_REGISTRY) += oid_registry.o
++
++$(obj)/oid_registry.c: $(obj)/oid_registry_data.c
++
++$(obj)/oid_registry_data.c: $(srctree)/include/linux/oid_registry.h \
++ $(src)/build_OID_registry
++ $(call cmd,build_OID_registry)
++
++quiet_cmd_build_OID_registry = GEN $@
++ cmd_build_OID_registry = perl $(srctree)/$(src)/build_OID_registry $< $@
++
++clean-files += oid_registry_data.c
+diff --git a/lib/build_OID_registry b/lib/build_OID_registry
+new file mode 100755
+index 0000000..dfbdaab
+--- /dev/null
++++ b/lib/build_OID_registry
+@@ -0,0 +1,209 @@
++#!/usr/bin/perl -w
++#
++# Build a static ASN.1 Object Identified (OID) registry
++#
++# Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++# Written by David Howells (dhowells@redhat.com)
++#
++# This program is free software; you can redistribute it and/or
++# modify it under the terms of the GNU General Public Licence
++# as published by the Free Software Foundation; either version
++# 2 of the Licence, or (at your option) any later version.
++#
++
++use strict;
++
++my @names = ();
++my @oids = ();
++
++if ($#ARGV != 1) {
++ print STDERR "Format: ", $0, " <in-h-file> <out-c-file>\n";
++ exit(2);
++}
++
++#
++# Open the file to read from
++#
++open IN_FILE, "<$ARGV[0]" || die;
++while (<IN_FILE>) {
++ chomp;
++ if (m!\s+OID_([a-zA-z][a-zA-Z0-9_]+),\s+/[*]\s+([012][.0-9]*)\s+[*]/!) {
++ push @names, $1;
++ push @oids, $2;
++ }
++}
++close IN_FILE || die;
++
++#
++# Open the files to write into
++#
++open C_FILE, ">$ARGV[1]" or die;
++print C_FILE "/*\n";
++print C_FILE " * Automatically generated by ", $0, ". Do not edit\n";
++print C_FILE " */\n";
++
++#
++# Split the data up into separate lists and also determine the lengths of the
++# encoded data arrays.
++#
++my @indices = ();
++my @lengths = ();
++my $total_length = 0;
++
++print "Compiling ", $#names + 1, " OIDs\n";
++
++for (my $i = 0; $i <= $#names; $i++) {
++ my $name = $names[$i];
++ my $oid = $oids[$i];
++
++ my @components = split(/[.]/, $oid);
++
++ # Determine the encoded length of this OID
++ my $size = $#components;
++ for (my $loop = 2; $loop <= $#components; $loop++) {
++ my $c = $components[$loop];
++
++ # We will base128 encode the number
++ my $tmp = ($c == 0) ? 0 : int(log($c)/log(2));
++ $tmp = int($tmp / 7);
++ $size += $tmp;
++ }
++ push @lengths, $size;
++ push @indices, $total_length;
++ $total_length += $size;
++}
++
++#
++# Emit the look-up-by-OID index table
++#
++print C_FILE "\n";
++if ($total_length <= 255) {
++ print C_FILE "static const unsigned char oid_index[OID__NR + 1] = {\n";
++} else {
++ print C_FILE "static const unsigned short oid_index[OID__NR + 1] = {\n";
++}
++for (my $i = 0; $i <= $#names; $i++) {
++ print C_FILE "\t[OID_", $names[$i], "] = ", $indices[$i], ",\n"
++}
++print C_FILE "\t[OID__NR] = ", $total_length, "\n";
++print C_FILE "};\n";
++
++#
++# Encode the OIDs
++#
++my @encoded_oids = ();
++
++for (my $i = 0; $i <= $#names; $i++) {
++ my @octets = ();
++
++ my @components = split(/[.]/, $oids[$i]);
++
++ push @octets, $components[0] * 40 + $components[1];
++
++ for (my $loop = 2; $loop <= $#components; $loop++) {
++ my $c = $components[$loop];
++
++ # Base128 encode the number
++ my $tmp = ($c == 0) ? 0 : int(log($c)/log(2));
++ $tmp = int($tmp / 7);
++
++ for (; $tmp > 0; $tmp--) {
++ push @octets, (($c >> $tmp * 7) & 0x7f) | 0x80;
++ }
++ push @octets, $c & 0x7f;
++ }
++
++ push @encoded_oids, \@octets;
++}
++
++#
++# Create a hash value for each OID
++#
++my @hash_values = ();
++for (my $i = 0; $i <= $#names; $i++) {
++ my @octets = @{$encoded_oids[$i]};
++
++ my $hash = $#octets;
++ foreach (@octets) {
++ $hash += $_ * 33;
++ }
++
++ $hash = ($hash >> 24) ^ ($hash >> 16) ^ ($hash >> 8) ^ ($hash);
++
++ push @hash_values, $hash & 0xff;
++}
++
++#
++# Emit the OID data
++#
++print C_FILE "\n";
++print C_FILE "static const unsigned char oid_data[", $total_length, "] = {\n";
++for (my $i = 0; $i <= $#names; $i++) {
++ my @octets = @{$encoded_oids[$i]};
++ print C_FILE "\t";
++ print C_FILE $_, ", " foreach (@octets);
++ print C_FILE "\t// ", $names[$i];
++ print C_FILE "\n";
++}
++print C_FILE "};\n";
++
++#
++# Build the search index table (ordered by length then hash then content)
++#
++my @index_table = ( 0 .. $#names );
++
++@index_table = sort {
++ my @octets_a = @{$encoded_oids[$a]};
++ my @octets_b = @{$encoded_oids[$b]};
++
++ return $hash_values[$a] <=> $hash_values[$b]
++ if ($hash_values[$a] != $hash_values[$b]);
++ return $#octets_a <=> $#octets_b
++ if ($#octets_a != $#octets_b);
++ for (my $i = $#octets_a; $i >= 0; $i--) {
++ return $octets_a[$i] <=> $octets_b[$i]
++ if ($octets_a[$i] != $octets_b[$i]);
++ }
++ return 0;
++
++} @index_table;
++
++#
++# Emit the search index and hash value table
++#
++print C_FILE "\n";
++print C_FILE "static const struct {\n";
++print C_FILE "\tunsigned char hash;\n";
++if ($#names <= 255) {
++ print C_FILE "\tenum OID oid : 8;\n";
++} else {
++ print C_FILE "\tenum OID oid : 16;\n";
++}
++print C_FILE "} oid_search_table[OID__NR] = {\n";
++for (my $i = 0; $i <= $#names; $i++) {
++ my @octets = @{$encoded_oids[$index_table[$i]]};
++ printf(C_FILE "\t[%3u] = { %3u, OID_%-35s }, // ",
++ $i,
++ $hash_values[$index_table[$i]],
++ $names[$index_table[$i]]);
++ printf C_FILE "%02x", $_ foreach (@octets);
++ print C_FILE "\n";
++}
++print C_FILE "};\n";
++
++#
++# Emit the OID debugging name table
++#
++#print C_FILE "\n";
++#print C_FILE "const char *const oid_name_table[OID__NR + 1] = {\n";
++#
++#for (my $i = 0; $i <= $#names; $i++) {
++# print C_FILE "\t\"", $names[$i], "\",\n"
++#}
++#print C_FILE "\t\"Unknown-OID\"\n";
++#print C_FILE "};\n";
++
++#
++# Polish off
++#
++close C_FILE or die;
+diff --git a/lib/oid_registry.c b/lib/oid_registry.c
+new file mode 100644
+index 0000000..33cfd17
+--- /dev/null
++++ b/lib/oid_registry.c
+@@ -0,0 +1,89 @@
++/* ASN.1 Object identifier (OID) registry
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <linux/export.h>
++#include <linux/oid_registry.h>
++#include "oid_registry_data.c"
++
++/**
++ * look_up_OID - Find an OID registration for the specified data
++ * @data: Binary representation of the OID
++ * @datasize: Size of the binary representation
++ */
++enum OID look_up_OID(const void *data, size_t datasize)
++{
++ const unsigned char *octets = data;
++ enum OID oid;
++ unsigned char xhash;
++ unsigned i, j, k, hash;
++ size_t len;
++
++ /* Hash the OID data */
++ hash = datasize - 1;
++
++ for (i = 0; i < datasize; i++)
++ hash += octets[i] * 33;
++ hash = (hash >> 24) ^ (hash >> 16) ^ (hash >> 8) ^ hash;
++ hash &= 0xff;
++
++ /* Binary search the OID registry. OIDs are stored in ascending order
++ * of hash value then ascending order of size and then in ascending
++ * order of reverse value.
++ */
++ i = 0;
++ k = OID__NR;
++ while (i < k) {
++ j = (i + k) / 2;
++
++ xhash = oid_search_table[j].hash;
++ if (xhash > hash) {
++ k = j;
++ continue;
++ }
++ if (xhash < hash) {
++ i = j + 1;
++ continue;
++ }
++
++ oid = oid_search_table[j].oid;
++ len = oid_index[oid + 1] - oid_index[oid];
++ if (len > datasize) {
++ k = j;
++ continue;
++ }
++ if (len < datasize) {
++ i = j + 1;
++ continue;
++ }
++
++ /* Variation is most likely to be at the tail end of the
++ * OID, so do the comparison in reverse.
++ */
++ while (len > 0) {
++ unsigned char a = oid_data[oid_index[oid] + --len];
++ unsigned char b = octets[len];
++ if (a > b) {
++ k = j;
++ goto next;
++ }
++ if (a < b) {
++ i = j + 1;
++ goto next;
++ }
++ }
++ return oid;
++ next:
++ ;
++ }
++
++ return OID__NR;
++}
++EXPORT_SYMBOL_GPL(look_up_OID);
+--
+1.7.11.4
+
+
+From 098335ed1edc5ec7ae7a346416654e12bb9bcd65 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:30:51 +0100
+Subject: [PATCH 13/26] X.509: Add utility functions to render OIDs as strings
+
+Add a pair of utility functions to render OIDs as strings. The first takes an
+encoded OID and turns it into a "a.b.c.d" form string:
+
+ int sprint_oid(const void *data, size_t datasize,
+ char *buffer, size_t bufsize);
+
+The second takes an OID enum index and calls the first on the data held
+therein:
+
+ int sprint_OID(enum OID oid, char *buffer, size_t bufsize);
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ include/linux/oid_registry.h | 2 ++
+ lib/oid_registry.c | 81 ++++++++++++++++++++++++++++++++++++++++++++
+ 2 files changed, 83 insertions(+)
+
+diff --git a/include/linux/oid_registry.h b/include/linux/oid_registry.h
+index 5928546..6926db7 100644
+--- a/include/linux/oid_registry.h
++++ b/include/linux/oid_registry.h
+@@ -86,5 +86,7 @@ enum OID {
+ };
+
+ extern enum OID look_up_OID(const void *data, size_t datasize);
++extern int sprint_oid(const void *, size_t, char *, size_t);
++extern int sprint_OID(enum OID, char *, size_t);
+
+ #endif /* _LINUX_OID_REGISTRY_H */
+diff --git a/lib/oid_registry.c b/lib/oid_registry.c
+index 33cfd17..d8de11f 100644
+--- a/lib/oid_registry.c
++++ b/lib/oid_registry.c
+@@ -11,6 +11,9 @@
+
+ #include <linux/export.h>
+ #include <linux/oid_registry.h>
++#include <linux/kernel.h>
++#include <linux/errno.h>
++#include <linux/bug.h>
+ #include "oid_registry_data.c"
+
+ /**
+@@ -87,3 +90,81 @@ enum OID look_up_OID(const void *data, size_t datasize)
+ return OID__NR;
+ }
+ EXPORT_SYMBOL_GPL(look_up_OID);
++
++/*
++ * sprint_OID - Print an Object Identifier into a buffer
++ * @data: The encoded OID to print
++ * @datasize: The size of the encoded OID
++ * @buffer: The buffer to render into
++ * @bufsize: The size of the buffer
++ *
++ * The OID is rendered into the buffer in "a.b.c.d" format and the number of
++ * bytes is returned. -EBADMSG is returned if the data could not be intepreted
++ * and -ENOBUFS if the buffer was too small.
++ */
++int sprint_oid(const void *data, size_t datasize, char *buffer, size_t bufsize)
++{
++ const unsigned char *v = data, *end = v + datasize;
++ unsigned long num;
++ unsigned char n;
++ size_t ret;
++ int count;
++
++ if (v >= end)
++ return -EBADMSG;
++
++ n = *v++;
++ ret = count = snprintf(buffer, bufsize, "%u.%u", n / 40, n % 40);
++ buffer += count;
++ bufsize -= count;
++ if (bufsize == 0)
++ return -ENOBUFS;
++
++ while (v < end) {
++ num = 0;
++ n = *v++;
++ if (!(n & 0x80)) {
++ num = n;
++ } else {
++ num = n & 0x7f;
++ do {
++ if (v >= end)
++ return -EBADMSG;
++ n = *v++;
++ num <<= 7;
++ num |= n & 0x7f;
++ } while (n & 0x80);
++ }
++ ret += count = snprintf(buffer, bufsize, ".%lu", num);
++ buffer += count;
++ bufsize -= count;
++ if (bufsize == 0)
++ return -ENOBUFS;
++ }
++
++ return ret;
++}
++EXPORT_SYMBOL_GPL(sprint_oid);
++
++/**
++ * sprint_OID - Print an Object Identifier into a buffer
++ * @oid: The OID to print
++ * @buffer: The buffer to render into
++ * @bufsize: The size of the buffer
++ *
++ * The OID is rendered into the buffer in "a.b.c.d" format and the number of
++ * bytes is returned.
++ */
++int sprint_OID(enum OID oid, char *buffer, size_t bufsize)
++{
++ int ret;
++
++ BUG_ON(oid >= OID__NR);
++
++ ret = sprint_oid(oid_data + oid_index[oid],
++ oid_index[oid + 1] - oid_index[oid],
++ buffer, bufsize);
++ BUG_ON(ret == -EBADMSG);
++ return ret;
++}
++EXPORT_SYMBOL_GPL(sprint_OID);
+--
+1.7.11.4
+
+
+From 3ddb8a2f1fe420777491641d39328741d9a28565 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Fri, 21 Sep 2012 23:31:13 +0100
+Subject: [PATCH 14/26] X.509: Add simple ASN.1 grammar compiler
+
+Add a simple ASN.1 grammar compiler. This produces a bytecode output that can
+be fed to a decoder to inform the decoder how to interpret the ASN.1 stream it
+is trying to parse.
+
+Action functions can be specified in the grammar by interpolating:
+
+ ({ foo })
+
+after a type, for example:
+
+ SubjectPublicKeyInfo ::= SEQUENCE {
+ algorithm AlgorithmIdentifier,
+ subjectPublicKey BIT STRING ({ do_key_data })
+ }
+
+The decoder is expected to call these after matching this type and parsing the
+contents if it is a constructed type.
+
+The grammar compiler does not currently support the SET type (though it does
+support SET OF) as I can't see a good way of tracking which members have been
+encountered yet without using up extra stack space.
+
+Currently, the grammar compiler will fail if more than 256 bytes of bytecode
+would be produced or more than 256 actions have been specified as it uses
+8-bit jump values and action indices to keep space usage down.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ include/linux/asn1.h | 67 ++
+ include/linux/asn1_ber_bytecode.h | 87 +++
+ init/Kconfig | 8 +
+ scripts/.gitignore | 1 +
+ scripts/Makefile | 2 +
+ scripts/Makefile.build | 11 +
+ scripts/asn1_compiler.c | 1545 +++++++++++++++++++++++++++++++++++++
+ 7 files changed, 1721 insertions(+)
+ create mode 100644 include/linux/asn1.h
+ create mode 100644 include/linux/asn1_ber_bytecode.h
+ create mode 100644 scripts/asn1_compiler.c
+
+diff --git a/include/linux/asn1.h b/include/linux/asn1.h
+new file mode 100644
+index 0000000..5c3f4e4
+--- /dev/null
++++ b/include/linux/asn1.h
+@@ -0,0 +1,67 @@
++/* ASN.1 BER/DER/CER encoding definitions
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _LINUX_ASN1_H
++#define _LINUX_ASN1_H
++
++/* Class */
++enum asn1_class {
++ ASN1_UNIV = 0, /* Universal */
++ ASN1_APPL = 1, /* Application */
++ ASN1_CONT = 2, /* Context */
++ ASN1_PRIV = 3 /* Private */
++};
++#define ASN1_CLASS_BITS 0xc0
++
++
++enum asn1_method {
++ ASN1_PRIM = 0, /* Primitive */
++ ASN1_CONS = 1 /* Constructed */
++};
++#define ASN1_CONS_BIT 0x20
++
++/* Tag */
++enum asn1_tag {
++ ASN1_EOC = 0, /* End Of Contents or N/A */
++ ASN1_BOOL = 1, /* Boolean */
++ ASN1_INT = 2, /* Integer */
++ ASN1_BTS = 3, /* Bit String */
++ ASN1_OTS = 4, /* Octet String */
++ ASN1_NULL = 5, /* Null */
++ ASN1_OID = 6, /* Object Identifier */
++ ASN1_ODE = 7, /* Object Description */
++ ASN1_EXT = 8, /* External */
++ ASN1_REAL = 9, /* Real float */
++ ASN1_ENUM = 10, /* Enumerated */
++ ASN1_EPDV = 11, /* Embedded PDV */
++ ASN1_UTF8STR = 12, /* UTF8 String */
++ ASN1_RELOID = 13, /* Relative OID */
++ /* 14 - Reserved */
++ /* 15 - Reserved */
++ ASN1_SEQ = 16, /* Sequence and Sequence of */
++ ASN1_SET = 17, /* Set and Set of */
++ ASN1_NUMSTR = 18, /* Numerical String */
++ ASN1_PRNSTR = 19, /* Printable String */
++ ASN1_TEXSTR = 20, /* T61 String / Teletext String */
++ ASN1_VIDSTR = 21, /* Videotex String */
++ ASN1_IA5STR = 22, /* IA5 String */
++ ASN1_UNITIM = 23, /* Universal Time */
++ ASN1_GENTIM = 24, /* General Time */
++ ASN1_GRASTR = 25, /* Graphic String */
++ ASN1_VISSTR = 26, /* Visible String */
++ ASN1_GENSTR = 27, /* General String */
++ ASN1_UNISTR = 28, /* Universal String */
++ ASN1_CHRSTR = 29, /* Character String */
++ ASN1_BMPSTR = 30, /* BMP String */
++ ASN1_LONG_TAG = 31 /* Long form tag */
++};
++
++#endif /* _LINUX_ASN1_H */
+diff --git a/include/linux/asn1_ber_bytecode.h b/include/linux/asn1_ber_bytecode.h
+new file mode 100644
+index 0000000..945d44a
+--- /dev/null
++++ b/include/linux/asn1_ber_bytecode.h
+@@ -0,0 +1,87 @@
++/* ASN.1 BER/DER/CER parsing state machine internal definitions
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _LINUX_ASN1_BER_BYTECODE_H
++#define _LINUX_ASN1_BER_BYTECODE_H
++
++#ifdef __KERNEL__
++#include <linux/types.h>
++#endif
++#include <linux/asn1.h>
++
++typedef int (*asn1_action_t)(void *context,
++ size_t hdrlen, /* In case of ANY type */
++ unsigned char tag, /* In case of ANY type */
++ const void *value, size_t vlen);
++
++struct asn1_decoder {
++ const unsigned char *machine;
++ size_t machlen;
++ const asn1_action_t *actions;
++};
++
++enum asn1_opcode {
++ /* The tag-matching ops come first and the odd-numbered slots
++ * are for OR_SKIP ops.
++ */
++#define ASN1_OP_MATCH__SKIP 0x01
++#define ASN1_OP_MATCH__ACT 0x02
++#define ASN1_OP_MATCH__JUMP 0x04
++#define ASN1_OP_MATCH__ANY 0x08
++#define ASN1_OP_MATCH__COND 0x10
++
++ ASN1_OP_MATCH = 0x00,
++ ASN1_OP_MATCH_OR_SKIP = 0x01,
++ ASN1_OP_MATCH_ACT = 0x02,
++ ASN1_OP_MATCH_ACT_OR_SKIP = 0x03,
++ ASN1_OP_MATCH_JUMP = 0x04,
++ ASN1_OP_MATCH_JUMP_OR_SKIP = 0x05,
++ ASN1_OP_MATCH_ANY = 0x08,
++ ASN1_OP_MATCH_ANY_ACT = 0x0a,
++ /* Everything before here matches unconditionally */
++
++ ASN1_OP_COND_MATCH_OR_SKIP = 0x11,
++ ASN1_OP_COND_MATCH_ACT_OR_SKIP = 0x13,
++ ASN1_OP_COND_MATCH_JUMP_OR_SKIP = 0x15,
++ ASN1_OP_COND_MATCH_ANY = 0x18,
++ ASN1_OP_COND_MATCH_ANY_ACT = 0x1a,
++
++ /* Everything before here will want a tag from the data */
++#define ASN1_OP__MATCHES_TAG ASN1_OP_COND_MATCH_ANY_ACT
++
++ /* These are here to help fill up space */
++ ASN1_OP_COND_FAIL = 0x1b,
++ ASN1_OP_COMPLETE = 0x1c,
++ ASN1_OP_ACT = 0x1d,
++ ASN1_OP_RETURN = 0x1e,
++
++ /* The following eight have bit 0 -> SET, 1 -> OF, 2 -> ACT */
++ ASN1_OP_END_SEQ = 0x20,
++ ASN1_OP_END_SET = 0x21,
++ ASN1_OP_END_SEQ_OF = 0x22,
++ ASN1_OP_END_SET_OF = 0x23,
++ ASN1_OP_END_SEQ_ACT = 0x24,
++ ASN1_OP_END_SET_ACT = 0x25,
++ ASN1_OP_END_SEQ_OF_ACT = 0x26,
++ ASN1_OP_END_SET_OF_ACT = 0x27,
++#define ASN1_OP_END__SET 0x01
++#define ASN1_OP_END__OF 0x02
++#define ASN1_OP_END__ACT 0x04
++
++ ASN1_OP__NR
++};
++
++#define _tag(CLASS, CP, TAG) ((ASN1_##CLASS << 6) | (ASN1_##CP << 5) | ASN1_##TAG)
++#define _tagn(CLASS, CP, TAG) ((ASN1_##CLASS << 6) | (ASN1_##CP << 5) | TAG)
++#define _jump_target(N) (N)
++#define _action(N) (N)
++
++#endif /* _LINUX_ASN1_BER_BYTECODE_H */
+diff --git a/init/Kconfig b/init/Kconfig
+index af6c7f8..66cc885 100644
+--- a/init/Kconfig
++++ b/init/Kconfig
+@@ -1612,4 +1612,12 @@ config PADATA
+ depends on SMP
+ bool
+
++config ASN1
++ tristate
++ help
++ Build a simple ASN.1 grammar compiler that produces a bytecode output
++ that can be interpreted by the ASN.1 stream decoder and used to
++ inform it as to what tags are to be expected in a stream and what
++ functions to call on what tags.
++
+ source "kernel/Kconfig.locks"
+diff --git a/scripts/.gitignore b/scripts/.gitignore
+index 65f362d..fb070fa 100644
+--- a/scripts/.gitignore
++++ b/scripts/.gitignore
+@@ -10,3 +10,4 @@ ihex2fw
+ recordmcount
+ docproc
+ sortextable
++asn1_compiler
+diff --git a/scripts/Makefile b/scripts/Makefile
+index a55b006..01e7adb 100644
+--- a/scripts/Makefile
++++ b/scripts/Makefile
+@@ -16,8 +16,10 @@ hostprogs-$(CONFIG_VT) += conmakehash
+ hostprogs-$(CONFIG_IKCONFIG) += bin2c
+ hostprogs-$(BUILD_C_RECORDMCOUNT) += recordmcount
+ hostprogs-$(CONFIG_BUILDTIME_EXTABLE_SORT) += sortextable
++hostprogs-$(CONFIG_ASN1) += asn1_compiler
+
+ HOSTCFLAGS_sortextable.o = -I$(srctree)/tools/include
++HOSTCFLAGS_asn1_compiler.o = -I$(srctree)/include
+
+ always := $(hostprogs-y) $(hostprogs-m)
+
+diff --git a/scripts/Makefile.build b/scripts/Makefile.build
+index ff1720d..0e801c3 100644
+--- a/scripts/Makefile.build
++++ b/scripts/Makefile.build
+@@ -354,6 +354,17 @@ quiet_cmd_cpp_lds_S = LDS $@
+ $(obj)/%.lds: $(src)/%.lds.S FORCE
+ $(call if_changed_dep,cpp_lds_S)
+
++# ASN.1 grammar
++# ---------------------------------------------------------------------------
++quiet_cmd_asn1_compiler = ASN.1 $@
++ cmd_asn1_compiler = $(objtree)/scripts/asn1_compiler $< \
++ $(subst .h,.c,$@) $(subst .c,.h,$@)
++
++.PRECIOUS: $(objtree)/$(obj)/%-asn1.c $(objtree)/$(obj)/%-asn1.h
++
++$(obj)/%-asn1.c $(obj)/%-asn1.h: $(src)/%.asn1 $(objtree)/scripts/asn1_compiler
++ $(call cmd,asn1_compiler)
++
+ # Build the compiled-in targets
+ # ---------------------------------------------------------------------------
+
+diff --git a/scripts/asn1_compiler.c b/scripts/asn1_compiler.c
+new file mode 100644
+index 0000000..db0e5cd
+--- /dev/null
++++ b/scripts/asn1_compiler.c
+@@ -0,0 +1,1545 @@
++/* Simplified ASN.1 notation parser
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <stdarg.h>
++#include <stdio.h>
++#include <stdlib.h>
++#include <stdint.h>
++#include <string.h>
++#include <ctype.h>
++#include <unistd.h>
++#include <fcntl.h>
++#include <sys/stat.h>
++#include <linux/asn1_ber_bytecode.h>
++
++enum token_type {
++ DIRECTIVE_ABSENT,
++ DIRECTIVE_ALL,
++ DIRECTIVE_ANY,
++ DIRECTIVE_APPLICATION,
++ DIRECTIVE_AUTOMATIC,
++ DIRECTIVE_BEGIN,
++ DIRECTIVE_BIT,
++ DIRECTIVE_BMPString,
++ DIRECTIVE_BOOLEAN,
++ DIRECTIVE_BY,
++ DIRECTIVE_CHARACTER,
++ DIRECTIVE_CHOICE,
++ DIRECTIVE_CLASS,
++ DIRECTIVE_COMPONENT,
++ DIRECTIVE_COMPONENTS,
++ DIRECTIVE_CONSTRAINED,
++ DIRECTIVE_CONTAINING,
++ DIRECTIVE_DEFAULT,
++ DIRECTIVE_DEFINED,
++ DIRECTIVE_DEFINITIONS,
++ DIRECTIVE_EMBEDDED,
++ DIRECTIVE_ENCODED,
++ DIRECTIVE_ENCODING_CONTROL,
++ DIRECTIVE_END,
++ DIRECTIVE_ENUMERATED,
++ DIRECTIVE_EXCEPT,
++ DIRECTIVE_EXPLICIT,
++ DIRECTIVE_EXPORTS,
++ DIRECTIVE_EXTENSIBILITY,
++ DIRECTIVE_EXTERNAL,
++ DIRECTIVE_FALSE,
++ DIRECTIVE_FROM,
++ DIRECTIVE_GeneralString,
++ DIRECTIVE_GeneralizedTime,
++ DIRECTIVE_GraphicString,
++ DIRECTIVE_IA5String,
++ DIRECTIVE_IDENTIFIER,
++ DIRECTIVE_IMPLICIT,
++ DIRECTIVE_IMPLIED,
++ DIRECTIVE_IMPORTS,
++ DIRECTIVE_INCLUDES,
++ DIRECTIVE_INSTANCE,
++ DIRECTIVE_INSTRUCTIONS,
++ DIRECTIVE_INTEGER,
++ DIRECTIVE_INTERSECTION,
++ DIRECTIVE_ISO646String,
++ DIRECTIVE_MAX,
++ DIRECTIVE_MIN,
++ DIRECTIVE_MINUS_INFINITY,
++ DIRECTIVE_NULL,
++ DIRECTIVE_NumericString,
++ DIRECTIVE_OBJECT,
++ DIRECTIVE_OCTET,
++ DIRECTIVE_OF,
++ DIRECTIVE_OPTIONAL,
++ DIRECTIVE_ObjectDescriptor,
++ DIRECTIVE_PATTERN,
++ DIRECTIVE_PDV,
++ DIRECTIVE_PLUS_INFINITY,
++ DIRECTIVE_PRESENT,
++ DIRECTIVE_PRIVATE,
++ DIRECTIVE_PrintableString,
++ DIRECTIVE_REAL,
++ DIRECTIVE_RELATIVE_OID,
++ DIRECTIVE_SEQUENCE,
++ DIRECTIVE_SET,
++ DIRECTIVE_SIZE,
++ DIRECTIVE_STRING,
++ DIRECTIVE_SYNTAX,
++ DIRECTIVE_T61String,
++ DIRECTIVE_TAGS,
++ DIRECTIVE_TRUE,
++ DIRECTIVE_TeletexString,
++ DIRECTIVE_UNION,
++ DIRECTIVE_UNIQUE,
++ DIRECTIVE_UNIVERSAL,
++ DIRECTIVE_UTCTime,
++ DIRECTIVE_UTF8String,
++ DIRECTIVE_UniversalString,
++ DIRECTIVE_VideotexString,
++ DIRECTIVE_VisibleString,
++ DIRECTIVE_WITH,
++ NR__DIRECTIVES,
++ TOKEN_ASSIGNMENT = NR__DIRECTIVES,
++ TOKEN_OPEN_CURLY,
++ TOKEN_CLOSE_CURLY,
++ TOKEN_OPEN_SQUARE,
++ TOKEN_CLOSE_SQUARE,
++ TOKEN_OPEN_ACTION,
++ TOKEN_CLOSE_ACTION,
++ TOKEN_COMMA,
++ TOKEN_NUMBER,
++ TOKEN_TYPE_NAME,
++ TOKEN_ELEMENT_NAME,
++ NR__TOKENS
++};
++
++static const unsigned char token_to_tag[NR__TOKENS] = {
++ /* EOC goes first */
++ [DIRECTIVE_BOOLEAN] = ASN1_BOOL,
++ [DIRECTIVE_INTEGER] = ASN1_INT,
++ [DIRECTIVE_BIT] = ASN1_BTS,
++ [DIRECTIVE_OCTET] = ASN1_OTS,
++ [DIRECTIVE_NULL] = ASN1_NULL,
++ [DIRECTIVE_OBJECT] = ASN1_OID,
++ [DIRECTIVE_ObjectDescriptor] = ASN1_ODE,
++ [DIRECTIVE_EXTERNAL] = ASN1_EXT,
++ [DIRECTIVE_REAL] = ASN1_REAL,
++ [DIRECTIVE_ENUMERATED] = ASN1_ENUM,
++ [DIRECTIVE_EMBEDDED] = 0,
++ [DIRECTIVE_UTF8String] = ASN1_UTF8STR,
++ [DIRECTIVE_RELATIVE_OID] = ASN1_RELOID,
++ /* 14 */
++ /* 15 */
++ [DIRECTIVE_SEQUENCE] = ASN1_SEQ,
++ [DIRECTIVE_SET] = ASN1_SET,
++ [DIRECTIVE_NumericString] = ASN1_NUMSTR,
++ [DIRECTIVE_PrintableString] = ASN1_PRNSTR,
++ [DIRECTIVE_T61String] = ASN1_TEXSTR,
++ [DIRECTIVE_TeletexString] = ASN1_TEXSTR,
++ [DIRECTIVE_VideotexString] = ASN1_VIDSTR,
++ [DIRECTIVE_IA5String] = ASN1_IA5STR,
++ [DIRECTIVE_UTCTime] = ASN1_UNITIM,
++ [DIRECTIVE_GeneralizedTime] = ASN1_GENTIM,
++ [DIRECTIVE_GraphicString] = ASN1_GRASTR,
++ [DIRECTIVE_VisibleString] = ASN1_VISSTR,
++ [DIRECTIVE_GeneralString] = ASN1_GENSTR,
++ [DIRECTIVE_UniversalString] = ASN1_UNITIM,
++ [DIRECTIVE_CHARACTER] = ASN1_CHRSTR,
++ [DIRECTIVE_BMPString] = ASN1_BMPSTR,
++};
++
++static const char asn1_classes[4][5] = {
++ [ASN1_UNIV] = "UNIV",
++ [ASN1_APPL] = "APPL",
++ [ASN1_CONT] = "CONT",
++ [ASN1_PRIV] = "PRIV"
++};
++
++static const char asn1_methods[2][5] = {
++ [ASN1_UNIV] = "PRIM",
++ [ASN1_APPL] = "CONS"
++};
++
++static const char *const asn1_universal_tags[32] = {
++ "EOC",
++ "BOOL",
++ "INT",
++ "BTS",
++ "OTS",
++ "NULL",
++ "OID",
++ "ODE",
++ "EXT",
++ "REAL",
++ "ENUM",
++ "EPDV",
++ "UTF8STR",
++ "RELOID",
++ NULL, /* 14 */
++ NULL, /* 15 */
++ "SEQ",
++ "SET",
++ "NUMSTR",
++ "PRNSTR",
++ "TEXSTR",
++ "VIDSTR",
++ "IA5STR",
++ "UNITIM",
++ "GENTIM",
++ "GRASTR",
++ "VISSTR",
++ "GENSTR",
++ "UNISTR",
++ "CHRSTR",
++ "BMPSTR",
++ NULL /* 31 */
++};
++
++static const char *filename;
++static const char *grammar_name;
++static const char *outputname;
++static const char *headername;
++
++static const char *const directives[NR__DIRECTIVES] = {
++#define _(X) [DIRECTIVE_##X] = #X
++ _(ABSENT),
++ _(ALL),
++ _(ANY),
++ _(APPLICATION),
++ _(AUTOMATIC),
++ _(BEGIN),
++ _(BIT),
++ _(BMPString),
++ _(BOOLEAN),
++ _(BY),
++ _(CHARACTER),
++ _(CHOICE),
++ _(CLASS),
++ _(COMPONENT),
++ _(COMPONENTS),
++ _(CONSTRAINED),
++ _(CONTAINING),
++ _(DEFAULT),
++ _(DEFINED),
++ _(DEFINITIONS),
++ _(EMBEDDED),
++ _(ENCODED),
++ [DIRECTIVE_ENCODING_CONTROL] = "ENCODING-CONTROL",
++ _(END),
++ _(ENUMERATED),
++ _(EXCEPT),
++ _(EXPLICIT),
++ _(EXPORTS),
++ _(EXTENSIBILITY),
++ _(EXTERNAL),
++ _(FALSE),
++ _(FROM),
++ _(GeneralString),
++ _(GeneralizedTime),
++ _(GraphicString),
++ _(IA5String),
++ _(IDENTIFIER),
++ _(IMPLICIT),
++ _(IMPLIED),
++ _(IMPORTS),
++ _(INCLUDES),
++ _(INSTANCE),
++ _(INSTRUCTIONS),
++ _(INTEGER),
++ _(INTERSECTION),
++ _(ISO646String),
++ _(MAX),
++ _(MIN),
++ [DIRECTIVE_MINUS_INFINITY] = "MINUS-INFINITY",
++ [DIRECTIVE_NULL] = "NULL",
++ _(NumericString),
++ _(OBJECT),
++ _(OCTET),
++ _(OF),
++ _(OPTIONAL),
++ _(ObjectDescriptor),
++ _(PATTERN),
++ _(PDV),
++ [DIRECTIVE_PLUS_INFINITY] = "PLUS-INFINITY",
++ _(PRESENT),
++ _(PRIVATE),
++ _(PrintableString),
++ _(REAL),
++ [DIRECTIVE_RELATIVE_OID] = "RELATIVE-OID",
++ _(SEQUENCE),
++ _(SET),
++ _(SIZE),
++ _(STRING),
++ _(SYNTAX),
++ _(T61String),
++ _(TAGS),
++ _(TRUE),
++ _(TeletexString),
++ _(UNION),
++ _(UNIQUE),
++ _(UNIVERSAL),
++ _(UTCTime),
++ _(UTF8String),
++ _(UniversalString),
++ _(VideotexString),
++ _(VisibleString),
++ _(WITH)
++};
++
++struct action {
++ struct action *next;
++ unsigned char index;
++ char name[];
++};
++
++static struct action *action_list;
++static unsigned nr_actions;
++
++struct token {
++ unsigned short line;
++ enum token_type token_type : 8;
++ unsigned char size;
++ struct action *action;
++ const char *value;
++ struct type *type;
++};
++
++static struct token *token_list;
++static unsigned nr_tokens;
++
++static int directive_compare(const void *_key, const void *_pdir)
++{
++ const struct token *token = _key;
++ const char *const *pdir = _pdir, *dir = *pdir;
++ size_t dlen, clen;
++ int val;
++
++ dlen = strlen(dir);
++ clen = (dlen < token->size) ? dlen : token->size;
++
++ //printf("cmp(%*.*s,%s) = ",
++ // (int)token->size, (int)token->size, token->value,
++ // dir);
++
++ val = memcmp(token->value, dir, clen);
++ if (val != 0) {
++ //printf("%d [cmp]\n", val);
++ return val;
++ }
++
++ if (dlen == token->size) {
++ //printf("0\n");
++ return 0;
++ }
++ //printf("%d\n", (int)dlen - (int)token->size);
++ return dlen - token->size; /* shorter -> negative */
++}
++
++/*
++ * Tokenise an ASN.1 grammar
++ */
++static void tokenise(char *buffer, char *end)
++{
++ struct token *tokens;
++ char *line, *nl, *p, *q;
++ unsigned tix, lineno;
++
++ /* Assume we're going to have half as many tokens as we have
++ * characters
++ */
++ token_list = tokens = calloc((end - buffer) / 2, sizeof(struct token));
++ if (!tokens) {
++ perror(NULL);
++ exit(1);
++ }
++ tix = 0;
++
++ lineno = 0;
++ while (buffer < end) {
++ /* First of all, break out a line */
++ lineno++;
++ line = buffer;
++ nl = memchr(line, '\n', end - buffer);
++ if (!nl) {
++ buffer = nl = end;
++ } else {
++ buffer = nl + 1;
++ *nl = '\0';
++ }
++
++ /* Remove "--" comments */
++ p = line;
++ next_comment:
++ while ((p = memchr(p, '-', nl - p))) {
++ if (p[1] == '-') {
++ /* Found a comment; see if there's a terminator */
++ q = p + 2;
++ while ((q = memchr(q, '-', nl - q))) {
++ if (q[1] == '-') {
++ /* There is - excise the comment */
++ q += 2;
++ memmove(p, q, nl - q);
++ goto next_comment;
++ }
++ q++;
++ }
++ *p = '\0';
++ nl = p;
++ break;
++ } else {
++ p++;
++ }
++ }
++
++ p = line;
++ while (p < nl) {
++ /* Skip white space */
++ while (p < nl && isspace(*p))
++ *(p++) = 0;
++ if (p >= nl)
++ break;
++
++ tokens[tix].line = lineno;
++ tokens[tix].value = p;
++
++ /* Handle string tokens */
++ if (isalpha(*p)) {
++ const char **dir;
++
++ /* Can be a directive, type name or element
++ * name. Find the end of the name.
++ */
++ q = p + 1;
++ while (q < nl && (isalnum(*q) || *q == '-' || *q == '_'))
++ q++;
++ tokens[tix].size = q - p;
++ p = q;
++
++ /* If it begins with a lowercase letter then
++ * it's an element name
++ */
++ if (islower(tokens[tix].value[0])) {
++ tokens[tix++].token_type = TOKEN_ELEMENT_NAME;
++ continue;
++ }
++
++ /* Otherwise we need to search the directive
++ * table
++ */
++ dir = bsearch(&tokens[tix], directives,
++ sizeof(directives) / sizeof(directives[1]),
++ sizeof(directives[1]),
++ directive_compare);
++ if (dir) {
++ tokens[tix++].token_type = dir - directives;
++ continue;
++ }
++
++ tokens[tix++].token_type = TOKEN_TYPE_NAME;
++ continue;
++ }
++
++ /* Handle numbers */
++ if (isdigit(*p)) {
++ /* Find the end of the number */
++ q = p + 1;
++ while (q < nl && (isdigit(*q)))
++ q++;
++ tokens[tix].size = q - p;
++ p = q;
++ tokens[tix++].token_type = TOKEN_NUMBER;
++ continue;
++ }
++
++ if (nl - p >= 3) {
++ if (memcmp(p, "::=", 3) == 0) {
++ p += 3;
++ tokens[tix].size = 3;
++ tokens[tix++].token_type = TOKEN_ASSIGNMENT;
++ continue;
++ }
++ }
++
++ if (nl - p >= 2) {
++ if (memcmp(p, "({", 2) == 0) {
++ p += 2;
++ tokens[tix].size = 2;
++ tokens[tix++].token_type = TOKEN_OPEN_ACTION;
++ continue;
++ }
++ if (memcmp(p, "})", 2) == 0) {
++ p += 2;
++ tokens[tix].size = 2;
++ tokens[tix++].token_type = TOKEN_CLOSE_ACTION;
++ continue;
++ }
++ }
++
++ if (nl - p >= 1) {
++ tokens[tix].size = 1;
++ switch (*p) {
++ case '{':
++ p += 1;
++ tokens[tix++].token_type = TOKEN_OPEN_CURLY;
++ continue;
++ case '}':
++ p += 1;
++ tokens[tix++].token_type = TOKEN_CLOSE_CURLY;
++ continue;
++ case '[':
++ p += 1;
++ tokens[tix++].token_type = TOKEN_OPEN_SQUARE;
++ continue;
++ case ']':
++ p += 1;
++ tokens[tix++].token_type = TOKEN_CLOSE_SQUARE;
++ continue;
++ case ',':
++ p += 1;
++ tokens[tix++].token_type = TOKEN_COMMA;
++ continue;
++ default:
++ break;
++ }
++ }
++
++ fprintf(stderr, "%s:%u: Unknown character in grammar: '%c'\n",
++ filename, lineno, *p);
++ exit(1);
++ }
++ }
++
++ nr_tokens = tix;
++ printf("Extracted %u tokens\n", nr_tokens);
++
++#if 0
++ {
++ int n;
++ for (n = 0; n < nr_tokens; n++)
++ printf("Token %3u: '%*.*s'\n",
++ n,
++ (int)token_list[n].size, (int)token_list[n].size,
++ token_list[n].value);
++ }
++#endif
++}
++
++static void build_type_list(void);
++static void parse(void);
++static void render(FILE *out, FILE *hdr);
++
++/*
++ *
++ */
++int main(int argc, char **argv)
++{
++ struct stat st;
++ ssize_t readlen;
++ FILE *out, *hdr;
++ char *buffer, *p;
++ int fd;
++
++ if (argc != 4) {
++ fprintf(stderr, "Format: %s <grammar-file> <c-file> <hdr-file>\n",
++ argv[0]);
++ exit(2);
++ }
++
++ filename = argv[1];
++ outputname = argv[2];
++ headername = argv[3];
++
++ fd = open(filename, O_RDONLY);
++ if (fd < 0) {
++ perror(filename);
++ exit(1);
++ }
++
++ if (fstat(fd, &st) < 0) {
++ perror(filename);
++ exit(1);
++ }
++
++ if (!(buffer = malloc(st.st_size + 1))) {
++ perror(NULL);
++ exit(1);
++ }
++
++ if ((readlen = read(fd, buffer, st.st_size)) < 0) {
++ perror(filename);
++ exit(1);
++ }
++
++ if (close(fd) < 0) {
++ perror(filename);
++ exit(1);
++ }
++
++ if (readlen != st.st_size) {
++ fprintf(stderr, "%s: Short read\n", filename);
++ exit(1);
++ }
++
++ p = strrchr(argv[1], '/');
++ p = p ? p + 1 : argv[1];
++ grammar_name = strdup(p);
++ if (!p) {
++ perror(NULL);
++ exit(1);
++ }
++ p = strchr(grammar_name, '.');
++ if (p)
++ *p = '\0';
++
++ buffer[readlen] = 0;
++ tokenise(buffer, buffer + readlen);
++ build_type_list();
++ parse();
++
++ out = fopen(outputname, "w");
++ if (!out) {
++ perror(outputname);
++ exit(1);
++ }
++
++ hdr = fopen(headername, "w");
++ if (!out) {
++ perror(headername);
++ exit(1);
++ }
++
++ render(out, hdr);
++
++ if (fclose(out) < 0) {
++ perror(outputname);
++ exit(1);
++ }
++
++ if (fclose(hdr) < 0) {
++ perror(headername);
++ exit(1);
++ }
++
++ return 0;
++}
++
++enum compound {
++ NOT_COMPOUND,
++ SET,
++ SET_OF,
++ SEQUENCE,
++ SEQUENCE_OF,
++ CHOICE,
++ ANY,
++ TYPE_REF,
++ TAG_OVERRIDE
++};
++
++struct element {
++ struct type *type_def;
++ struct token *name;
++ struct token *type;
++ struct action *action;
++ struct element *children;
++ struct element *next;
++ struct element *render_next;
++ struct element *list_next;
++ uint8_t n_elements;
++ enum compound compound : 8;
++ enum asn1_class class : 8;
++ enum asn1_method method : 8;
++ uint8_t tag;
++ unsigned entry_index;
++ unsigned flags;
++#define ELEMENT_IMPLICIT 0x0001
++#define ELEMENT_EXPLICIT 0x0002
++#define ELEMENT_MARKED 0x0004
++#define ELEMENT_RENDERED 0x0008
++#define ELEMENT_SKIPPABLE 0x0010
++#define ELEMENT_CONDITIONAL 0x0020
++};
++
++struct type {
++ struct token *name;
++ struct token *def;
++ struct element *element;
++ unsigned ref_count;
++ unsigned flags;
++#define TYPE_STOP_MARKER 0x0001
++#define TYPE_BEGIN 0x0002
++};
++
++static struct type *type_list;
++static struct type **type_index;
++static unsigned nr_types;
++
++static int type_index_compare(const void *_a, const void *_b)
++{
++ const struct type *const *a = _a, *const *b = _b;
++
++ if ((*a)->name->size != (*b)->name->size)
++ return (*a)->name->size - (*b)->name->size;
++ else
++ return memcmp((*a)->name->value, (*b)->name->value,
++ (*a)->name->size);
++}
++
++static int type_finder(const void *_key, const void *_ti)
++{
++ const struct token *token = _key;
++ const struct type *const *ti = _ti;
++ const struct type *type = *ti;
++
++ if (token->size != type->name->size)
++ return token->size - type->name->size;
++ else
++ return memcmp(token->value, type->name->value,
++ token->size);
++}
++
++/*
++ * Build up a list of types and a sorted index to that list.
++ */
++static void build_type_list(void)
++{
++ struct type *types;
++ unsigned nr, t, n;
++
++ nr = 0;
++ for (n = 0; n < nr_tokens - 1; n++)
++ if (token_list[n + 0].token_type == TOKEN_TYPE_NAME &&
++ token_list[n + 1].token_type == TOKEN_ASSIGNMENT)
++ nr++;
++
++ if (nr == 0) {
++ fprintf(stderr, "%s: No defined types\n", filename);
++ exit(1);
++ }
++
++ nr_types = nr;
++ types = type_list = calloc(nr + 1, sizeof(type_list[0]));
++ if (!type_list) {
++ perror(NULL);
++ exit(1);
++ }
++ type_index = calloc(nr, sizeof(type_index[0]));
++ if (!type_index) {
++ perror(NULL);
++ exit(1);
++ }
++
++ t = 0;
++ types[t].flags |= TYPE_BEGIN;
++ for (n = 0; n < nr_tokens - 1; n++) {
++ if (token_list[n + 0].token_type == TOKEN_TYPE_NAME &&
++ token_list[n + 1].token_type == TOKEN_ASSIGNMENT) {
++ types[t].name = &token_list[n];
++ type_index[t] = &types[t];
++ t++;
++ }
++ }
++ types[t].name = &token_list[n + 1];
++ types[t].flags |= TYPE_STOP_MARKER;
++
++ qsort(type_index, nr, sizeof(type_index[0]), type_index_compare);
++
++ printf("Extracted %u types\n", nr_types);
++#if 0
++ for (n = 0; n < nr_types; n++) {
++ struct type *type = type_index[n];
++ printf("- %*.*s\n",
++ (int)type->name->size,
++ (int)type->name->size,
++ type->name->value);
++ }
++#endif
++}
++
++static struct element *parse_type(struct token **_cursor, struct token *stop,
++ struct token *name);
++
++/*
++ * Parse the token stream
++ */
++static void parse(void)
++{
++ struct token *cursor;
++ struct type *type;
++
++ /* Parse one type definition statement at a time */
++ type = type_list;
++ do {
++ cursor = type->name;
++
++ if (cursor[0].token_type != TOKEN_TYPE_NAME ||
++ cursor[1].token_type != TOKEN_ASSIGNMENT)
++ abort();
++ cursor += 2;
++
++ type->element = parse_type(&cursor, type[1].name, NULL);
++ type->element->type_def = type;
++
++ if (cursor != type[1].name) {
++ fprintf(stderr, "%s:%d: Parse error at token '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++
++ } while (type++, !(type->flags & TYPE_STOP_MARKER));
++
++ printf("Extracted %u actions\n", nr_actions);
++}
++
++static struct element *element_list;
++
++static struct element *alloc_elem(struct token *type)
++{
++ struct element *e = calloc(1, sizeof(*e));
++ if (!e) {
++ perror(NULL);
++ exit(1);
++ }
++ e->list_next = element_list;
++ element_list = e;
++ return e;
++}
++
++static struct element *parse_compound(struct token **_cursor, struct token *end,
++ int alternates);
++
++/*
++ * Parse one type definition statement
++ */
++static struct element *parse_type(struct token **_cursor, struct token *end,
++ struct token *name)
++{
++ struct element *top, *element;
++ struct action *action, **ppaction;
++ struct token *cursor = *_cursor;
++ struct type **ref;
++ char *p;
++ int labelled = 0, implicit = 0;
++
++ top = element = alloc_elem(cursor);
++ element->class = ASN1_UNIV;
++ element->method = ASN1_PRIM;
++ element->tag = token_to_tag[cursor->token_type];
++ element->name = name;
++
++ /* Extract the tag value if one given */
++ if (cursor->token_type == TOKEN_OPEN_SQUARE) {
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ switch (cursor->token_type) {
++ case DIRECTIVE_UNIVERSAL:
++ element->class = ASN1_UNIV;
++ cursor++;
++ break;
++ case DIRECTIVE_APPLICATION:
++ element->class = ASN1_APPL;
++ cursor++;
++ break;
++ case TOKEN_NUMBER:
++ element->class = ASN1_CONT;
++ break;
++ case DIRECTIVE_PRIVATE:
++ element->class = ASN1_PRIV;
++ cursor++;
++ break;
++ default:
++ fprintf(stderr, "%s:%d: Unrecognised tag class token '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type != TOKEN_NUMBER) {
++ fprintf(stderr, "%s:%d: Missing tag number '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++
++ element->tag &= ~0x1f;
++ element->tag |= strtoul(cursor->value, &p, 10);
++ if (p - cursor->value != cursor->size)
++ abort();
++ cursor++;
++
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type != TOKEN_CLOSE_SQUARE) {
++ fprintf(stderr, "%s:%d: Missing closing square bracket '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ labelled = 1;
++ }
++
++ /* Handle implicit and explicit markers */
++ if (cursor->token_type == DIRECTIVE_IMPLICIT) {
++ element->flags |= ELEMENT_IMPLICIT;
++ implicit = 1;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ } else if (cursor->token_type == DIRECTIVE_EXPLICIT) {
++ element->flags |= ELEMENT_EXPLICIT;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ }
++
++ if (labelled) {
++ if (!implicit)
++ element->method |= ASN1_CONS;
++ element->compound = implicit ? TAG_OVERRIDE : SEQUENCE;
++ element->children = alloc_elem(cursor);
++ element = element->children;
++ element->class = ASN1_UNIV;
++ element->method = ASN1_PRIM;
++ element->tag = token_to_tag[cursor->token_type];
++ element->name = name;
++ }
++
++ /* Extract the type we're expecting here */
++ element->type = cursor;
++ switch (cursor->token_type) {
++ case DIRECTIVE_ANY:
++ element->compound = ANY;
++ cursor++;
++ break;
++
++ case DIRECTIVE_NULL:
++ case DIRECTIVE_BOOLEAN:
++ case DIRECTIVE_ENUMERATED:
++ case DIRECTIVE_INTEGER:
++ element->compound = NOT_COMPOUND;
++ cursor++;
++ break;
++
++ case DIRECTIVE_EXTERNAL:
++ element->method = ASN1_CONS;
++
++ case DIRECTIVE_BMPString:
++ case DIRECTIVE_GeneralString:
++ case DIRECTIVE_GraphicString:
++ case DIRECTIVE_IA5String:
++ case DIRECTIVE_ISO646String:
++ case DIRECTIVE_NumericString:
++ case DIRECTIVE_PrintableString:
++ case DIRECTIVE_T61String:
++ case DIRECTIVE_TeletexString:
++ case DIRECTIVE_UniversalString:
++ case DIRECTIVE_UTF8String:
++ case DIRECTIVE_VideotexString:
++ case DIRECTIVE_VisibleString:
++ case DIRECTIVE_ObjectDescriptor:
++ case DIRECTIVE_GeneralizedTime:
++ case DIRECTIVE_UTCTime:
++ element->compound = NOT_COMPOUND;
++ cursor++;
++ break;
++
++ case DIRECTIVE_BIT:
++ case DIRECTIVE_OCTET:
++ element->compound = NOT_COMPOUND;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type != DIRECTIVE_STRING)
++ goto parse_error;
++ cursor++;
++ break;
++
++ case DIRECTIVE_OBJECT:
++ element->compound = NOT_COMPOUND;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type != DIRECTIVE_IDENTIFIER)
++ goto parse_error;
++ cursor++;
++ break;
++
++ case TOKEN_TYPE_NAME:
++ element->compound = TYPE_REF;
++ ref = bsearch(cursor, type_index, nr_types, sizeof(type_index[0]),
++ type_finder);
++ if (!ref) {
++ fprintf(stderr, "%s:%d: Type '%*.*s' undefined\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++ cursor->type = *ref;
++ (*ref)->ref_count++;
++ cursor++;
++ break;
++
++ case DIRECTIVE_CHOICE:
++ element->compound = CHOICE;
++ cursor++;
++ element->children = parse_compound(&cursor, end, 1);
++ break;
++
++ case DIRECTIVE_SEQUENCE:
++ element->compound = SEQUENCE;
++ element->method = ASN1_CONS;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type == DIRECTIVE_OF) {
++ element->compound = SEQUENCE_OF;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ element->children = parse_type(&cursor, end, NULL);
++ } else {
++ element->children = parse_compound(&cursor, end, 0);
++ }
++ break;
++
++ case DIRECTIVE_SET:
++ element->compound = SET;
++ element->method = ASN1_CONS;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type == DIRECTIVE_OF) {
++ element->compound = SET_OF;
++ cursor++;
++ if (cursor >= end)
++ goto parse_error;
++ element->children = parse_type(&cursor, end, NULL);
++ } else {
++ element->children = parse_compound(&cursor, end, 1);
++ }
++ break;
++
++ default:
++ fprintf(stderr, "%s:%d: Token '%*.*s' does not introduce a type\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++
++ /* Handle elements that are optional */
++ if (cursor < end && (cursor->token_type == DIRECTIVE_OPTIONAL ||
++ cursor->token_type == DIRECTIVE_DEFAULT)
++ ) {
++ cursor++;
++ top->flags |= ELEMENT_SKIPPABLE;
++ }
++
++ if (cursor < end && cursor->token_type == TOKEN_OPEN_ACTION) {
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type != TOKEN_ELEMENT_NAME) {
++ fprintf(stderr, "%s:%d: Token '%*.*s' is not an action function name\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++
++ action = malloc(sizeof(struct action) + cursor->size + 1);
++ if (!action) {
++ perror(NULL);
++ exit(1);
++ }
++ action->index = 0;
++ memcpy(action->name, cursor->value, cursor->size);
++ action->name[cursor->size] = 0;
++
++ for (ppaction = &action_list;
++ *ppaction;
++ ppaction = &(*ppaction)->next
++ ) {
++ int cmp = strcmp(action->name, (*ppaction)->name);
++ if (cmp == 0) {
++ free(action);
++ action = *ppaction;
++ goto found;
++ }
++ if (cmp < 0) {
++ action->next = *ppaction;
++ *ppaction = action;
++ nr_actions++;
++ goto found;
++ }
++ }
++ action->next = NULL;
++ *ppaction = action;
++ nr_actions++;
++ found:
++
++ element->action = action;
++ cursor->action = action;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type != TOKEN_CLOSE_ACTION) {
++ fprintf(stderr, "%s:%d: Missing close action, got '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++ cursor++;
++ }
++
++ *_cursor = cursor;
++ return top;
++
++parse_error:
++ fprintf(stderr, "%s:%d: Unexpected token '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++
++overrun_error:
++ fprintf(stderr, "%s: Unexpectedly hit EOF\n", filename);
++ exit(1);
++}
++
++/*
++ * Parse a compound type list
++ */
++static struct element *parse_compound(struct token **_cursor, struct token *end,
++ int alternates)
++{
++ struct element *children, **child_p = &children, *element;
++ struct token *cursor = *_cursor, *name;
++
++ if (cursor->token_type != TOKEN_OPEN_CURLY) {
++ fprintf(stderr, "%s:%d: Expected compound to start with brace not '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++
++ if (cursor->token_type == TOKEN_OPEN_CURLY) {
++ fprintf(stderr, "%s:%d: Empty compound\n",
++ filename, cursor->line);
++ exit(1);
++ }
++
++ for (;;) {
++ name = NULL;
++ if (cursor->token_type == TOKEN_ELEMENT_NAME) {
++ name = cursor;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ }
++
++ element = parse_type(&cursor, end, name);
++ if (alternates)
++ element->flags |= ELEMENT_SKIPPABLE | ELEMENT_CONDITIONAL;
++
++ *child_p = element;
++ child_p = &element->next;
++
++ if (cursor >= end)
++ goto overrun_error;
++ if (cursor->token_type != TOKEN_COMMA)
++ break;
++ cursor++;
++ if (cursor >= end)
++ goto overrun_error;
++ }
++
++ children->flags &= ~ELEMENT_CONDITIONAL;
++
++ if (cursor->token_type != TOKEN_CLOSE_CURLY) {
++ fprintf(stderr, "%s:%d: Expected compound closure, got '%*.*s'\n",
++ filename, cursor->line,
++ (int)cursor->size, (int)cursor->size, cursor->value);
++ exit(1);
++ }
++ cursor++;
++
++ *_cursor = cursor;
++ return children;
++
++overrun_error:
++ fprintf(stderr, "%s: Unexpectedly hit EOF\n", filename);
++ exit(1);
++}
++
++static void render_element(FILE *out, struct element *e, struct element *tag);
++static void render_out_of_line_list(FILE *out);
++
++static int nr_entries;
++static int render_depth = 1;
++static struct element *render_list, **render_list_p = &render_list;
++
++__attribute__((format(printf, 2, 3)))
++static void render_opcode(FILE *out, const char *fmt, ...)
++{
++ va_list va;
++
++ if (out) {
++ fprintf(out, "\t[%4d] =%*s", nr_entries, render_depth, "");
++ va_start(va, fmt);
++ vfprintf(out, fmt, va);
++ va_end(va);
++ }
++ nr_entries++;
++}
++
++__attribute__((format(printf, 2, 3)))
++static void render_more(FILE *out, const char *fmt, ...)
++{
++ va_list va;
++
++ if (out) {
++ va_start(va, fmt);
++ vfprintf(out, fmt, va);
++ va_end(va);
++ }
++}
++
++/*
++ * Render the grammar into a state machine definition.
++ */
++static void render(FILE *out, FILE *hdr)
++{
++ struct element *e;
++ struct action *action;
++ struct type *root;
++ int index;
++
++ fprintf(hdr, "/*\n");
++ fprintf(hdr, " * Automatically generated by asn1_compiler. Do not edit\n");
++ fprintf(hdr, " *\n");
++ fprintf(hdr, " * ASN.1 parser for %s\n", grammar_name);
++ fprintf(hdr, " */\n");
++ fprintf(hdr, "#include <linux/asn1_decoder.h>\n");
++ fprintf(hdr, "\n");
++ fprintf(hdr, "extern const struct asn1_decoder %s_decoder;\n", grammar_name);
++ if (ferror(hdr)) {
++ perror(headername);
++ exit(1);
++ }
++
++ fprintf(out, "/*\n");
++ fprintf(out, " * Automatically generated by asn1_compiler. Do not edit\n");
++ fprintf(out, " *\n");
++ fprintf(out, " * ASN.1 parser for %s\n", grammar_name);
++ fprintf(out, " */\n");
++ fprintf(out, "#include <linux/asn1_ber_bytecode.h>\n");
++ fprintf(out, "#include \"%s-asn1.h\"\n", grammar_name);
++ fprintf(out, "\n");
++ if (ferror(out)) {
++ perror(outputname);
++ exit(1);
++ }
++
++ /* Tabulate the action functions we might have to call */
++ fprintf(hdr, "\n");
++ index = 0;
++ for (action = action_list; action; action = action->next) {
++ action->index = index++;
++ fprintf(hdr,
++ "extern int %s(void *, size_t, unsigned char,"
++ " const void *, size_t);\n",
++ action->name);
++ }
++ fprintf(hdr, "\n");
++
++ fprintf(out, "enum %s_actions {\n", grammar_name);
++ for (action = action_list; action; action = action->next)
++ fprintf(out, "\tACT_%s = %u,\n",
++ action->name, action->index);
++ fprintf(out, "\tNR__%s_actions = %u\n", grammar_name, nr_actions);
++ fprintf(out, "};\n");
++
++ fprintf(out, "\n");
++ fprintf(out, "static const asn1_action_t %s_action_table[NR__%s_actions] = {\n",
++ grammar_name, grammar_name);
++ for (action = action_list; action; action = action->next)
++ fprintf(out, "\t[%4u] = %s,\n", action->index, action->name);
++ fprintf(out, "};\n");
++
++ if (ferror(out)) {
++ perror(outputname);
++ exit(1);
++ }
++
++ /* We do two passes - the first one calculates all the offsets */
++ printf("Pass 1\n");
++ nr_entries = 0;
++ root = &type_list[0];
++ render_element(NULL, root->element, NULL);
++ render_opcode(NULL, "ASN1_OP_COMPLETE,\n");
++ render_out_of_line_list(NULL);
++
++ for (e = element_list; e; e = e->list_next)
++ e->flags &= ~ELEMENT_RENDERED;
++
++ /* And then we actually render */
++ printf("Pass 2\n");
++ fprintf(out, "\n");
++ fprintf(out, "static const unsigned char %s_machine[] = {\n",
++ grammar_name);
++
++ nr_entries = 0;
++ root = &type_list[0];
++ render_element(out, root->element, NULL);
++ render_opcode(out, "ASN1_OP_COMPLETE,\n");
++ render_out_of_line_list(out);
++
++ fprintf(out, "};\n");
++
++ fprintf(out, "\n");
++ fprintf(out, "const struct asn1_decoder %s_decoder = {\n", grammar_name);
++ fprintf(out, "\t.machine = %s_machine,\n", grammar_name);
++ fprintf(out, "\t.machlen = sizeof(%s_machine),\n", grammar_name);
++ fprintf(out, "\t.actions = %s_action_table,\n", grammar_name);
++ fprintf(out, "};\n");
++}
++
++/*
++ * Render the out-of-line elements
++ */
++static void render_out_of_line_list(FILE *out)
++{
++ struct element *e, *ce;
++ const char *act;
++ int entry;
++
++ while ((e = render_list)) {
++ render_list = e->render_next;
++ if (!render_list)
++ render_list_p = &render_list;
++
++ render_more(out, "\n");
++ e->entry_index = entry = nr_entries;
++ render_depth++;
++ for (ce = e->children; ce; ce = ce->next)
++ render_element(out, ce, NULL);
++ render_depth--;
++
++ act = e->action ? "_ACT" : "";
++ switch (e->compound) {
++ case SEQUENCE:
++ render_opcode(out, "ASN1_OP_END_SEQ%s,\n", act);
++ break;
++ case SEQUENCE_OF:
++ render_opcode(out, "ASN1_OP_END_SEQ_OF%s,\n", act);
++ render_opcode(out, "_jump_target(%u),\n", entry);
++ break;
++ case SET:
++ render_opcode(out, "ASN1_OP_END_SET%s,\n", act);
++ break;
++ case SET_OF:
++ render_opcode(out, "ASN1_OP_END_SET_OF%s,\n", act);
++ render_opcode(out, "_jump_target(%u),\n", entry);
++ break;
++ }
++ if (e->action)
++ render_opcode(out, "_action(ACT_%s),\n",
++ e->action->name);
++ render_opcode(out, "ASN1_OP_RETURN,\n");
++ }
++}
++
++/*
++ * Render an element.
++ */
++static void render_element(FILE *out, struct element *e, struct element *tag)
++{
++ struct element *ec;
++ const char *cond, *act;
++ int entry, skippable = 0, outofline = 0;
++
++ if (e->flags & ELEMENT_SKIPPABLE ||
++ (tag && tag->flags & ELEMENT_SKIPPABLE))
++ skippable = 1;
++
++ if ((e->type_def && e->type_def->ref_count > 1) ||
++ skippable)
++ outofline = 1;
++
++ if (e->type_def && out) {
++ render_more(out, "\t// %*.*s\n",
++ (int)e->type_def->name->size, (int)e->type_def->name->size,
++ e->type_def->name->value);
++ }
++
++ /* Render the operation */
++ cond = (e->flags & ELEMENT_CONDITIONAL ||
++ (tag && tag->flags & ELEMENT_CONDITIONAL)) ? "COND_" : "";
++ act = e->action ? "_ACT" : "";
++ switch (e->compound) {
++ case ANY:
++ render_opcode(out, "ASN1_OP_%sMATCH_ANY%s,", cond, act);
++ if (e->name)
++ render_more(out, "\t\t// %*.*s",
++ (int)e->name->size, (int)e->name->size,
++ e->name->value);
++ render_more(out, "\n");
++ goto dont_render_tag;
++
++ case TAG_OVERRIDE:
++ render_element(out, e->children, e);
++ return;
++
++ case SEQUENCE:
++ case SEQUENCE_OF:
++ case SET:
++ case SET_OF:
++ render_opcode(out, "ASN1_OP_%sMATCH%s%s,",
++ cond,
++ outofline ? "_JUMP" : "",
++ skippable ? "_OR_SKIP" : "");
++ break;
++
++ case CHOICE:
++ goto dont_render_tag;
++
++ case TYPE_REF:
++ if (e->class == ASN1_UNIV && e->method == ASN1_PRIM && e->tag == 0)
++ goto dont_render_tag;
++ default:
++ render_opcode(out, "ASN1_OP_%sMATCH%s%s,",
++ cond, act,
++ skippable ? "_OR_SKIP" : "");
++ break;
++ }
++
++ if (e->name)
++ render_more(out, "\t\t// %*.*s",
++ (int)e->name->size, (int)e->name->size,
++ e->name->value);
++ render_more(out, "\n");
++
++ /* Render the tag */
++ if (!tag)
++ tag = e;
++ if (tag->class == ASN1_UNIV &&
++ tag->tag != 14 &&
++ tag->tag != 15 &&
++ tag->tag != 31)
++ render_opcode(out, "_tag(%s, %s, %s),\n",
++ asn1_classes[tag->class],
++ asn1_methods[tag->method | e->method],
++ asn1_universal_tags[tag->tag]);
++ else
++ render_opcode(out, "_tagn(%s, %s, %2u),\n",
++ asn1_classes[tag->class],
++ asn1_methods[tag->method | e->method],
++ tag->tag);
++ tag = NULL;
++dont_render_tag:
++
++ /* Deal with compound types */
++ switch (e->compound) {
++ case TYPE_REF:
++ render_element(out, e->type->type->element, tag);
++ if (e->action)
++ render_opcode(out, "ASN1_OP_ACT,\n");
++ break;
++
++ case SEQUENCE:
++ if (outofline) {
++ /* Render out-of-line for multiple use or
++ * skipability */
++ render_opcode(out, "_jump_target(%u),", e->entry_index);
++ if (e->type_def && e->type_def->name)
++ render_more(out, "\t\t// --> %*.*s",
++ (int)e->type_def->name->size,
++ (int)e->type_def->name->size,
++ e->type_def->name->value);
++ render_more(out, "\n");
++ if (!(e->flags & ELEMENT_RENDERED)) {
++ e->flags |= ELEMENT_RENDERED;
++ *render_list_p = e;
++ render_list_p = &e->render_next;
++ }
++ return;
++ } else {
++ /* Render inline for single use */
++ render_depth++;
++ for (ec = e->children; ec; ec = ec->next)
++ render_element(out, ec, NULL);
++ render_depth--;
++ render_opcode(out, "ASN1_OP_END_SEQ%s,\n", act);
++ }
++ break;
++
++ case SEQUENCE_OF:
++ case SET_OF:
++ if (outofline) {
++ /* Render out-of-line for multiple use or
++ * skipability */
++ render_opcode(out, "_jump_target(%u),", e->entry_index);
++ if (e->type_def && e->type_def->name)
++ render_more(out, "\t\t// --> %*.*s",
++ (int)e->type_def->name->size,
++ (int)e->type_def->name->size,
++ e->type_def->name->value);
++ render_more(out, "\n");
++ if (!(e->flags & ELEMENT_RENDERED)) {
++ e->flags |= ELEMENT_RENDERED;
++ *render_list_p = e;
++ render_list_p = &e->render_next;
++ }
++ return;
++ } else {
++ /* Render inline for single use */
++ entry = nr_entries;
++ render_depth++;
++ render_element(out, e->children, NULL);
++ render_depth--;
++ if (e->compound == SEQUENCE_OF)
++ render_opcode(out, "ASN1_OP_END_SEQ_OF%s,\n", act);
++ else
++ render_opcode(out, "ASN1_OP_END_SET_OF%s,\n", act);
++ render_opcode(out, "_jump_target(%u),\n", entry);
++ }
++ break;
++
++ case SET:
++ /* I can't think of a nice way to do SET support without having
++ * a stack of bitmasks to make sure no element is repeated.
++ * The bitmask has also to be checked that no non-optional
++ * elements are left out whilst not preventing optional
++ * elements from being left out.
++ */
++ fprintf(stderr, "The ASN.1 SET type is not currently supported.\n");
++ exit(1);
++
++ case CHOICE:
++ for (ec = e->children; ec; ec = ec->next)
++ render_element(out, ec, NULL);
++ if (!skippable)
++ render_opcode(out, "ASN1_OP_COND_FAIL,\n");
++ if (e->action)
++ render_opcode(out, "ASN1_OP_ACT,\n");
++ break;
++
++ default:
++ break;
++ }
++
++ if (e->action)
++ render_opcode(out, "_action(ACT_%s),\n", e->action->name);
++}
+--
+1.7.11.4
+
+
+From 8ea3f94cc16a23e3edbebf12f4223e654eb8219d Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:11:16 +0100
+Subject: [PATCH 15/26] X.509: Add an ASN.1 decoder
+
+Add an ASN.1 BER/DER/CER decoder. This uses the bytecode from the ASN.1
+compiler in the previous patch to inform it as to what to expect to find in the
+encoded byte stream. The output from the compiler also tells it what functions
+to call on what tags, thus allowing the caller to retrieve information.
+
+The decoder is called as follows:
+
+ int asn1_decoder(const struct asn1_decoder *decoder,
+ void *context,
+ const unsigned char *data,
+ size_t datalen);
+
+The decoder argument points to the bytecode from the ASN.1 compiler. context
+is the caller's context and is passed to the action functions. data and
+datalen define the byte stream to be decoded.
+
+
+Note that the decoder is currently limited to datalen being less than 64K.
+This reduces the amount of stack space used by the decoder because ASN.1 is a
+nested construct. Similarly, the decoder is limited to a maximum of 10 levels
+of constructed data outside of a leaf node also in an effort to keep stack
+usage down.
+
+These restrictions can be raised if necessary.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ include/linux/asn1_decoder.h | 24 +++
+ lib/Makefile | 2 +
+ lib/asn1_decoder.c | 477 +++++++++++++++++++++++++++++++++++++++++++
+ 3 files changed, 503 insertions(+)
+ create mode 100644 include/linux/asn1_decoder.h
+ create mode 100644 lib/asn1_decoder.c
+
+diff --git a/include/linux/asn1_decoder.h b/include/linux/asn1_decoder.h
+new file mode 100644
+index 0000000..fa2ff5b
+--- /dev/null
++++ b/include/linux/asn1_decoder.h
+@@ -0,0 +1,24 @@
++/* ASN.1 decoder
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#ifndef _LINUX_ASN1_DECODER_H
++#define _LINUX_ASN1_DECODER_H
++
++#include <linux/asn1.h>
++
++struct asn1_decoder;
++
++extern int asn1_ber_decoder(const struct asn1_decoder *decoder,
++ void *context,
++ const unsigned char *data,
++ size_t datalen);
++
++#endif /* _LINUX_ASN1_DECODER_H */
+diff --git a/lib/Makefile b/lib/Makefile
+index b042896..ca856b6 100644
+--- a/lib/Makefile
++++ b/lib/Makefile
+@@ -140,6 +140,8 @@ $(foreach file, $(libfdt_files), \
+ $(eval CFLAGS_$(file) = -I$(src)/../scripts/dtc/libfdt))
+ lib-$(CONFIG_LIBFDT) += $(libfdt_files)
+
++obj-$(CONFIG_ASN1) += asn1_decoder.o
++
+ hostprogs-y := gen_crc32table
+ clean-files := crc32table.h
+
+diff --git a/lib/asn1_decoder.c b/lib/asn1_decoder.c
+new file mode 100644
+index 0000000..2e4196d
+--- /dev/null
++++ b/lib/asn1_decoder.c
+@@ -0,0 +1,477 @@
++/* Decoder for ASN.1 BER/DER/CER encoded bytestream
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <linux/export.h>
++#include <linux/kernel.h>
++#include <linux/errno.h>
++#include <linux/asn1_decoder.h>
++#include <linux/asn1_ber_bytecode.h>
++
++static const unsigned char asn1_op_lengths[ASN1_OP__NR] = {
++ /* OPC TAG JMP ACT */
++ [ASN1_OP_MATCH] = 1 + 1,
++ [ASN1_OP_MATCH_OR_SKIP] = 1 + 1,
++ [ASN1_OP_MATCH_ACT] = 1 + 1 + 1,
++ [ASN1_OP_MATCH_ACT_OR_SKIP] = 1 + 1 + 1,
++ [ASN1_OP_MATCH_JUMP] = 1 + 1 + 1,
++ [ASN1_OP_MATCH_JUMP_OR_SKIP] = 1 + 1 + 1,
++ [ASN1_OP_MATCH_ANY] = 1,
++ [ASN1_OP_MATCH_ANY_ACT] = 1 + 1,
++ [ASN1_OP_COND_MATCH_OR_SKIP] = 1 + 1,
++ [ASN1_OP_COND_MATCH_ACT_OR_SKIP] = 1 + 1 + 1,
++ [ASN1_OP_COND_MATCH_JUMP_OR_SKIP] = 1 + 1 + 1,
++ [ASN1_OP_COND_MATCH_ANY] = 1,
++ [ASN1_OP_COND_MATCH_ANY_ACT] = 1 + 1,
++ [ASN1_OP_COND_FAIL] = 1,
++ [ASN1_OP_COMPLETE] = 1,
++ [ASN1_OP_ACT] = 1 + 1,
++ [ASN1_OP_RETURN] = 1,
++ [ASN1_OP_END_SEQ] = 1,
++ [ASN1_OP_END_SEQ_OF] = 1 + 1,
++ [ASN1_OP_END_SET] = 1,
++ [ASN1_OP_END_SET_OF] = 1 + 1,
++ [ASN1_OP_END_SEQ_ACT] = 1 + 1,
++ [ASN1_OP_END_SEQ_OF_ACT] = 1 + 1 + 1,
++ [ASN1_OP_END_SET_ACT] = 1 + 1,
++ [ASN1_OP_END_SET_OF_ACT] = 1 + 1 + 1,
++};
++
++/*
++ * Find the length of an indefinite length object
++ */
++static int asn1_find_indefinite_length(const unsigned char *data, size_t datalen,
++ const char **_errmsg, size_t *_err_dp)
++{
++ unsigned char tag, tmp;
++ size_t dp = 0, len, n;
++ int indef_level = 1;
++
++next_tag:
++ if (unlikely(datalen - dp < 2)) {
++ if (datalen == dp)
++ goto missing_eoc;
++ goto data_overrun_error;
++ }
++
++ /* Extract a tag from the data */
++ tag = data[dp++];
++ if (tag == 0) {
++ /* It appears to be an EOC. */
++ if (data[dp++] != 0)
++ goto invalid_eoc;
++ if (--indef_level <= 0)
++ return dp;
++ goto next_tag;
++ }
++
++ if (unlikely((tag & 0x1f) == 0x1f)) {
++ do {
++ if (unlikely(datalen - dp < 2))
++ goto data_overrun_error;
++ tmp = data[dp++];
++ } while (tmp & 0x80);
++ }
++
++ /* Extract the length */
++ len = data[dp++];
++ if (len < 0x7f) {
++ dp += len;
++ goto next_tag;
++ }
++
++ if (unlikely(len == 0x80)) {
++ /* Indefinite length */
++ if (unlikely((tag & ASN1_CONS_BIT) == ASN1_PRIM << 5))
++ goto indefinite_len_primitive;
++ indef_level++;
++ goto next_tag;
++ }
++
++ n = len - 0x80;
++ if (unlikely(n > sizeof(size_t) - 1))
++ goto length_too_long;
++ if (unlikely(n > datalen - dp))
++ goto data_overrun_error;
++ for (len = 0; n > 0; n--) {
++ len <<= 8;
++ len |= data[dp++];
++ }
++ dp += len;
++ goto next_tag;
++
++length_too_long:
++ *_errmsg = "Unsupported length";
++ goto error;
++indefinite_len_primitive:
++ *_errmsg = "Indefinite len primitive not permitted";
++ goto error;
++invalid_eoc:
++ *_errmsg = "Invalid length EOC";
++ goto error;
++data_overrun_error:
++ *_errmsg = "Data overrun error";
++ goto error;
++missing_eoc:
++ *_errmsg = "Missing EOC in indefinite len cons";
++error:
++ *_err_dp = dp;
++ return -1;
++}
++
++/**
++ * asn1_ber_decoder - Decoder BER/DER/CER ASN.1 according to pattern
++ * @decoder: The decoder definition (produced by asn1_compiler)
++ * @context: The caller's context (to be passed to the action functions)
++ * @data: The encoded data
++ * @datasize: The size of the encoded data
++ *
++ * Decode BER/DER/CER encoded ASN.1 data according to a bytecode pattern
++ * produced by asn1_compiler. Action functions are called on marked tags to
++ * allow the caller to retrieve significant data.
++ *
++ * LIMITATIONS:
++ *
++ * To keep down the amount of stack used by this function, the following limits
++ * have been imposed:
++ *
++ * (1) This won't handle datalen > 65535 without increasing the size of the
++ * cons stack elements and length_too_long checking.
++ *
++ * (2) The stack of constructed types is 10 deep. If the depth of non-leaf
++ * constructed types exceeds this, the decode will fail.
++ *
++ * (3) The SET type (not the SET OF type) isn't really supported as tracking
++ * what members of the set have been seen is a pain.
++ */
++int asn1_ber_decoder(const struct asn1_decoder *decoder,
++ void *context,
++ const unsigned char *data,
++ size_t datalen)
++{
++ const unsigned char *machine = decoder->machine;
++ const asn1_action_t *actions = decoder->actions;
++ size_t machlen = decoder->machlen;
++ enum asn1_opcode op;
++ unsigned char tag = 0, csp = 0, jsp = 0, optag = 0, hdr = 0;
++ const char *errmsg;
++ size_t pc = 0, dp = 0, tdp = 0, len = 0;
++ int ret;
++
++ unsigned char flags = 0;
++#define FLAG_INDEFINITE_LENGTH 0x01
++#define FLAG_MATCHED 0x02
++#define FLAG_CONS 0x20 /* Corresponds to CONS bit in the opcode tag
++ * - ie. whether or not we are going to parse
++ * a compound type.
++ */
++
++#define NR_CONS_STACK 10
++ unsigned short cons_dp_stack[NR_CONS_STACK];
++ unsigned short cons_datalen_stack[NR_CONS_STACK];
++ unsigned char cons_hdrlen_stack[NR_CONS_STACK];
++#define NR_JUMP_STACK 10
++ unsigned char jump_stack[NR_JUMP_STACK];
++
++ if (datalen > 65535)
++ return -EMSGSIZE;
++
++next_op:
++ pr_debug("next_op: pc=\e[32m%zu\e[m/%zu dp=\e[33m%zu\e[m/%zu C=%d J=%d\n",
++ pc, machlen, dp, datalen, csp, jsp);
++ if (unlikely(pc >= machlen))
++ goto machine_overrun_error;
++ op = machine[pc];
++ if (unlikely(pc + asn1_op_lengths[op] > machlen))
++ goto machine_overrun_error;
++
++ /* If this command is meant to match a tag, then do that before
++ * evaluating the command.
++ */
++ if (op <= ASN1_OP__MATCHES_TAG) {
++ unsigned char tmp;
++
++ /* Skip conditional matches if possible */
++ if ((op & ASN1_OP_MATCH__COND &&
++ flags & FLAG_MATCHED) ||
++ dp == datalen) {
++ pc += asn1_op_lengths[op];
++ goto next_op;
++ }
++
++ flags = 0;
++ hdr = 2;
++
++ /* Extract a tag from the data */
++ if (unlikely(dp >= datalen - 1))
++ goto data_overrun_error;
++ tag = data[dp++];
++ if (unlikely((tag & 0x1f) == 0x1f))
++ goto long_tag_not_supported;
++
++ if (op & ASN1_OP_MATCH__ANY) {
++ pr_debug("- any %02x\n", tag);
++ } else {
++ /* Extract the tag from the machine
++ * - Either CONS or PRIM are permitted in the data if
++ * CONS is not set in the op stream, otherwise CONS
++ * is mandatory.
++ */
++ optag = machine[pc + 1];
++ flags |= optag & FLAG_CONS;
++
++ /* Determine whether the tag matched */
++ tmp = optag ^ tag;
++ tmp &= ~(optag & ASN1_CONS_BIT);
++ pr_debug("- match? %02x %02x %02x\n", tag, optag, tmp);
++ if (tmp != 0) {
++ /* All odd-numbered tags are MATCH_OR_SKIP. */
++ if (op & ASN1_OP_MATCH__SKIP) {
++ pc += asn1_op_lengths[op];
++ dp--;
++ goto next_op;
++ }
++ goto tag_mismatch;
++ }
++ }
++ flags |= FLAG_MATCHED;
++
++ len = data[dp++];
++ if (len > 0x7f) {
++ if (unlikely(len == 0x80)) {
++ /* Indefinite length */
++ if (unlikely(!(tag & ASN1_CONS_BIT)))
++ goto indefinite_len_primitive;
++ flags |= FLAG_INDEFINITE_LENGTH;
++ if (unlikely(2 > datalen - dp))
++ goto data_overrun_error;
++ } else {
++ int n = len - 0x80;
++ if (unlikely(n > 2))
++ goto length_too_long;
++ if (unlikely(dp >= datalen - n))
++ goto data_overrun_error;
++ hdr += n;
++ for (len = 0; n > 0; n--) {
++ len <<= 8;
++ len |= data[dp++];
++ }
++ if (unlikely(len > datalen - dp))
++ goto data_overrun_error;
++ }
++ }
++
++ if (flags & FLAG_CONS) {
++ /* For expected compound forms, we stack the positions
++ * of the start and end of the data.
++ */
++ if (unlikely(csp >= NR_CONS_STACK))
++ goto cons_stack_overflow;
++ cons_dp_stack[csp] = dp;
++ cons_hdrlen_stack[csp] = hdr;
++ if (!(flags & FLAG_INDEFINITE_LENGTH)) {
++ cons_datalen_stack[csp] = datalen;
++ datalen = dp + len;
++ } else {
++ cons_datalen_stack[csp] = 0;
++ }
++ csp++;
++ }
++
++ pr_debug("- TAG: %02x %zu%s\n",
++ tag, len, flags & FLAG_CONS ? " CONS" : "");
++ tdp = dp;
++ }
++
++ /* Decide how to handle the operation */
++ switch (op) {
++ case ASN1_OP_MATCH_ANY_ACT:
++ case ASN1_OP_COND_MATCH_ANY_ACT:
++ ret = actions[machine[pc + 1]](context, hdr, tag, data + dp, len);
++ if (ret < 0)
++ return ret;
++ goto skip_data;
++
++ case ASN1_OP_MATCH_ACT:
++ case ASN1_OP_MATCH_ACT_OR_SKIP:
++ case ASN1_OP_COND_MATCH_ACT_OR_SKIP:
++ ret = actions[machine[pc + 2]](context, hdr, tag, data + dp, len);
++ if (ret < 0)
++ return ret;
++ goto skip_data;
++
++ case ASN1_OP_MATCH:
++ case ASN1_OP_MATCH_OR_SKIP:
++ case ASN1_OP_MATCH_ANY:
++ case ASN1_OP_COND_MATCH_OR_SKIP:
++ case ASN1_OP_COND_MATCH_ANY:
++ skip_data:
++ if (!(flags & FLAG_CONS)) {
++ if (flags & FLAG_INDEFINITE_LENGTH) {
++ len = asn1_find_indefinite_length(
++ data + dp, datalen - dp, &errmsg, &dp);
++ if (len < 0)
++ goto error;
++ }
++ pr_debug("- LEAF: %zu\n", len);
++ dp += len;
++ }
++ pc += asn1_op_lengths[op];
++ goto next_op;
++
++ case ASN1_OP_MATCH_JUMP:
++ case ASN1_OP_MATCH_JUMP_OR_SKIP:
++ case ASN1_OP_COND_MATCH_JUMP_OR_SKIP:
++ pr_debug("- MATCH_JUMP\n");
++ if (unlikely(jsp == NR_JUMP_STACK))
++ goto jump_stack_overflow;
++ jump_stack[jsp++] = pc + asn1_op_lengths[op];
++ pc = machine[pc + 2];
++ goto next_op;
++
++ case ASN1_OP_COND_FAIL:
++ if (unlikely(!(flags & FLAG_MATCHED)))
++ goto tag_mismatch;
++ pc += asn1_op_lengths[op];
++ goto next_op;
++
++ case ASN1_OP_COMPLETE:
++ if (unlikely(jsp != 0 || csp != 0)) {
++ pr_err("ASN.1 decoder error: Stacks not empty at completion (%u, %u)\n",
++ jsp, csp);
++ return -EBADMSG;
++ }
++ return 0;
++
++ case ASN1_OP_END_SET:
++ case ASN1_OP_END_SET_ACT:
++ if (unlikely(!(flags & FLAG_MATCHED)))
++ goto tag_mismatch;
++ case ASN1_OP_END_SEQ:
++ case ASN1_OP_END_SET_OF:
++ case ASN1_OP_END_SEQ_OF:
++ case ASN1_OP_END_SEQ_ACT:
++ case ASN1_OP_END_SET_OF_ACT:
++ case ASN1_OP_END_SEQ_OF_ACT:
++ if (unlikely(csp <= 0))
++ goto cons_stack_underflow;
++ csp--;
++ tdp = cons_dp_stack[csp];
++ hdr = cons_hdrlen_stack[csp];
++ len = datalen;
++ datalen = cons_datalen_stack[csp];
++ pr_debug("- end cons t=%zu dp=%zu l=%zu/%zu\n",
++ tdp, dp, len, datalen);
++ if (datalen == 0) {
++ /* Indefinite length - check for the EOC. */
++ datalen = len;
++ if (unlikely(datalen - dp < 2))
++ goto data_overrun_error;
++ if (data[dp++] != 0) {
++ if (op & ASN1_OP_END__OF) {
++ dp--;
++ csp++;
++ pc = machine[pc + 1];
++ pr_debug("- continue\n");
++ goto next_op;
++ }
++ goto missing_eoc;
++ }
++ if (data[dp++] != 0)
++ goto invalid_eoc;
++ len = dp - tdp - 2;
++ } else {
++ if (dp < len && (op & ASN1_OP_END__OF)) {
++ datalen = len;
++ csp++;
++ pc = machine[pc + 1];
++ pr_debug("- continue\n");
++ goto next_op;
++ }
++ if (dp != len)
++ goto cons_length_error;
++ len -= tdp;
++ pr_debug("- cons len l=%zu d=%zu\n", len, dp - tdp);
++ }
++
++ if (op & ASN1_OP_END__ACT) {
++ unsigned char act;
++ if (op & ASN1_OP_END__OF)
++ act = machine[pc + 2];
++ else
++ act = machine[pc + 1];
++ ret = actions[act](context, hdr, 0, data + tdp, len);
++ }
++ pc += asn1_op_lengths[op];
++ goto next_op;
++
++ case ASN1_OP_ACT:
++ ret = actions[machine[pc + 1]](context, hdr, tag, data + tdp, len);
++ pc += asn1_op_lengths[op];
++ goto next_op;
++
++ case ASN1_OP_RETURN:
++ if (unlikely(jsp <= 0))
++ goto jump_stack_underflow;
++ pc = jump_stack[--jsp];
++ goto next_op;
++
++ default:
++ break;
++ }
++
++ /* Shouldn't reach here */
++ pr_err("ASN.1 decoder error: Found reserved opcode (%u)\n", op);
++ return -EBADMSG;
++
++data_overrun_error:
++ errmsg = "Data overrun error";
++ goto error;
++machine_overrun_error:
++ errmsg = "Machine overrun error";
++ goto error;
++jump_stack_underflow:
++ errmsg = "Jump stack underflow";
++ goto error;
++jump_stack_overflow:
++ errmsg = "Jump stack overflow";
++ goto error;
++cons_stack_underflow:
++ errmsg = "Cons stack underflow";
++ goto error;
++cons_stack_overflow:
++ errmsg = "Cons stack overflow";
++ goto error;
++cons_length_error:
++ errmsg = "Cons length error";
++ goto error;
++missing_eoc:
++ errmsg = "Missing EOC in indefinite len cons";
++ goto error;
++invalid_eoc:
++ errmsg = "Invalid length EOC";
++ goto error;
++length_too_long:
++ errmsg = "Unsupported length";
++ goto error;
++indefinite_len_primitive:
++ errmsg = "Indefinite len primitive not permitted";
++ goto error;
++tag_mismatch:
++ errmsg = "Unexpected tag";
++ goto error;
++long_tag_not_supported:
++ errmsg = "Long tag not supported";
++error:
++ pr_debug("\nASN1: %s [m=%zu d=%zu ot=%02x t=%02x l=%zu]\n",
++ errmsg, pc, dp, optag, tag, len);
++ return -EBADMSG;
++}
++EXPORT_SYMBOL_GPL(asn1_ber_decoder);
+--
+1.7.11.4
+
+
+From f055a9091c35be0171d39ca8e76bb4677d89eef1 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:11:27 +0100
+Subject: [PATCH 16/26] MPILIB: Provide a function to read raw data into an
+ MPI
+
+Provide a function to read raw data of a predetermined size into an MPI rather
+than expecting the size to be encoded within the data. The data is assumed to
+represent an unsigned integer, and the resulting MPI will be positive.
+
+The function looks like this:
+
+ MPI mpi_read_raw_data(const void *, size_t);
+
+This is useful for reading ASN.1 integer primitives where the length is encoded
+in the ASN.1 metadata.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ include/linux/mpi.h | 1 +
+ lib/mpi/mpicoder.c | 55 +++++++++++++++++++++++++++++++++++++++++++++++++++++
+ 2 files changed, 56 insertions(+)
+
+diff --git a/include/linux/mpi.h b/include/linux/mpi.h
+index d02cca6..5af1b81 100644
+--- a/include/linux/mpi.h
++++ b/include/linux/mpi.h
+@@ -76,6 +76,7 @@ void mpi_swap(MPI a, MPI b);
+
+ /*-- mpicoder.c --*/
+ MPI do_encode_md(const void *sha_buffer, unsigned nbits);
++MPI mpi_read_raw_data(const void *xbuffer, size_t nbytes);
+ MPI mpi_read_from_buffer(const void *buffer, unsigned *ret_nread);
+ int mpi_fromstr(MPI val, const char *str);
+ u32 mpi_get_keyid(MPI a, u32 *keyid);
+diff --git a/lib/mpi/mpicoder.c b/lib/mpi/mpicoder.c
+index f0fa659..3962b7f 100644
+--- a/lib/mpi/mpicoder.c
++++ b/lib/mpi/mpicoder.c
+@@ -18,10 +18,65 @@
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
+ */
+
++#include <linux/bitops.h>
++#include <asm-generic/bitops/count_zeros.h>
+ #include "mpi-internal.h"
+
+ #define MAX_EXTERN_MPI_BITS 16384
+
++/**
++ * mpi_read_raw_data - Read a raw byte stream as a positive integer
++ * @xbuffer: The data to read
++ * @nbytes: The amount of data to read
++ */
++MPI mpi_read_raw_data(const void *xbuffer, size_t nbytes)
++{
++ const uint8_t *buffer = xbuffer;
++ int i, j;
++ unsigned nbits, nlimbs;
++ mpi_limb_t a;
++ MPI val = NULL;
++
++ while (nbytes >= 0 && buffer[0] == 0) {
++ buffer++;
++ nbytes--;
++ }
++
++ nbits = nbytes * 8;
++ if (nbits > MAX_EXTERN_MPI_BITS) {
++ pr_info("MPI: mpi too large (%u bits)\n", nbits);
++ return NULL;
++ }
++ if (nbytes > 0)
++ nbits -= count_leading_zeros(buffer[0]);
++ else
++ nbits = 0;
++
++ nlimbs = (nbytes + BYTES_PER_MPI_LIMB - 1) / BYTES_PER_MPI_LIMB;
++ val = mpi_alloc(nlimbs);
++ if (!val)
++ return NULL;
++ val->nbits = nbits;
++ val->sign = 0;
++ val->nlimbs = nlimbs;
++
++ if (nbytes > 0) {
++ i = BYTES_PER_MPI_LIMB - nbytes % BYTES_PER_MPI_LIMB;
++ i %= BYTES_PER_MPI_LIMB;
++ for (j = nlimbs; j > 0; j--) {
++ a = 0;
++ for (; i < BYTES_PER_MPI_LIMB; i++) {
++ a <<= 8;
++ a |= *buffer++;
++ }
++ i = 0;
++ val->d[j - 1] = a;
++ }
++ }
++ return val;
++}
++EXPORT_SYMBOL_GPL(mpi_read_raw_data);
++
+ MPI mpi_read_from_buffer(const void *xbuffer, unsigned *ret_nread)
+ {
+ const uint8_t *buffer = xbuffer;
+--
+1.7.11.4
+
+
+From 3d816cdad8cdd5412ecc8f539bb09daef52ba361 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:11:48 +0100
+Subject: [PATCH 17/26] X.509: Add a crypto key parser for binary (DER) X.509
+ certificates
+
+Add a crypto key parser for binary (DER) encoded X.509 certificates. The
+certificate is parsed and, if possible, the signature is verified.
+
+An X.509 key can be added like this:
+
+ # keyctl padd crypto bar @s </tmp/x509.cert
+ 15768135
+
+and displayed like this:
+
+ # cat /proc/keys
+ 00f09a47 I--Q--- 1 perm 39390000 0 0 asymmetri bar: X509.RSA e9fd6d08 []
+
+Note that this only works with binary certificates. PEM encoded certificates
+are ignored by the parser.
+
+Note also that the X.509 key ID is not congruent with the PGP key ID, but for
+the moment, they will match.
+
+If a NULL or "" name is given to add_key(), then the parser will generate a key
+description from the CertificateSerialNumber and Name fields of the
+TBSCertificate:
+
+ 00aefc4e I--Q--- 1 perm 39390000 0 0 asymmetri bfbc0cd76d050ea4:/C=GB/L=Cambridge/O=Red Hat/CN=kernel key: X509.RSA 0c688c7b []
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ crypto/asymmetric_keys/.gitignore | 1 +
+ crypto/asymmetric_keys/Kconfig | 10 +
+ crypto/asymmetric_keys/Makefile | 17 +
+ crypto/asymmetric_keys/x509.asn1 | 60 ++++
+ crypto/asymmetric_keys/x509_cert_parser.c | 497 ++++++++++++++++++++++++++++++
+ crypto/asymmetric_keys/x509_parser.h | 36 +++
+ crypto/asymmetric_keys/x509_public_key.c | 207 +++++++++++++
+ crypto/asymmetric_keys/x509_rsakey.asn1 | 4 +
+ 8 files changed, 832 insertions(+)
+ create mode 100644 crypto/asymmetric_keys/.gitignore
+ create mode 100644 crypto/asymmetric_keys/x509.asn1
+ create mode 100644 crypto/asymmetric_keys/x509_cert_parser.c
+ create mode 100644 crypto/asymmetric_keys/x509_parser.h
+ create mode 100644 crypto/asymmetric_keys/x509_public_key.c
+ create mode 100644 crypto/asymmetric_keys/x509_rsakey.asn1
+
+diff --git a/crypto/asymmetric_keys/.gitignore b/crypto/asymmetric_keys/.gitignore
+new file mode 100644
+index 0000000..ee32837
+--- /dev/null
++++ b/crypto/asymmetric_keys/.gitignore
+@@ -0,0 +1 @@
++*-asn1.[ch]
+diff --git a/crypto/asymmetric_keys/Kconfig b/crypto/asymmetric_keys/Kconfig
+index 561759d..6d2c2ea 100644
+--- a/crypto/asymmetric_keys/Kconfig
++++ b/crypto/asymmetric_keys/Kconfig
+@@ -25,4 +25,14 @@ config PUBLIC_KEY_ALGO_RSA
+ help
+ This option enables support for the RSA algorithm (PKCS#1, RFC3447).
+
++config X509_CERTIFICATE_PARSER
++ tristate "X.509 certificate parser"
++ depends on ASYMMETRIC_PUBLIC_KEY_SUBTYPE
++ select ASN1
++ select OID_REGISTRY
++ help
++ This option procides support for parsing X.509 format blobs for key
++ data and provides the ability to instantiate a crypto key from a
++ public key packet found inside the certificate.
++
+ endif # ASYMMETRIC_KEY_TYPE
+diff --git a/crypto/asymmetric_keys/Makefile b/crypto/asymmetric_keys/Makefile
+index 7c92691..0727204 100644
+--- a/crypto/asymmetric_keys/Makefile
++++ b/crypto/asymmetric_keys/Makefile
+@@ -8,3 +8,20 @@ asymmetric_keys-y := asymmetric_type.o signature.o
+
+ obj-$(CONFIG_ASYMMETRIC_PUBLIC_KEY_SUBTYPE) += public_key.o
+ obj-$(CONFIG_PUBLIC_KEY_ALGO_RSA) += rsa.o
++
++#
++# X.509 Certificate handling
++#
++obj-$(CONFIG_X509_CERTIFICATE_PARSER) += x509_key_parser.o
++x509_key_parser-y := \
++ x509-asn1.o \
++ x509_rsakey-asn1.o \
++ x509_cert_parser.o \
++ x509_public_key.o
++
++$(obj)/x509_cert_parser.o: $(obj)/x509-asn1.h $(obj)/x509_rsakey-asn1.h
++$(obj)/x509-asn1.o: $(obj)/x509-asn1.c $(obj)/x509-asn1.h
++$(obj)/x509_rsakey-asn1.o: $(obj)/x509_rsakey-asn1.c $(obj)/x509_rsakey-asn1.h
++
++clean-files += x509-asn1.c x509-asn1.h
++clean-files += x509_rsakey-asn1.c x509_rsakey-asn1.h
+diff --git a/crypto/asymmetric_keys/x509.asn1 b/crypto/asymmetric_keys/x509.asn1
+new file mode 100644
+index 0000000..bf32b3d
+--- /dev/null
++++ b/crypto/asymmetric_keys/x509.asn1
+@@ -0,0 +1,60 @@
++Certificate ::= SEQUENCE {
++ tbsCertificate TBSCertificate ({ x509_note_tbs_certificate }),
++ signatureAlgorithm AlgorithmIdentifier,
++ signature BIT STRING ({ x509_note_signature })
++ }
++
++TBSCertificate ::= SEQUENCE {
++ version [ 0 ] Version DEFAULT,
++ serialNumber CertificateSerialNumber,
++ signature AlgorithmIdentifier ({ x509_note_pkey_algo }),
++ issuer Name ({ x509_note_issuer }),
++ validity Validity,
++ subject Name ({ x509_note_subject }),
++ subjectPublicKeyInfo SubjectPublicKeyInfo,
++ issuerUniqueID [ 1 ] IMPLICIT UniqueIdentifier OPTIONAL,
++ subjectUniqueID [ 2 ] IMPLICIT UniqueIdentifier OPTIONAL,
++ extensions [ 3 ] Extensions OPTIONAL
++ }
++
++Version ::= INTEGER
++CertificateSerialNumber ::= INTEGER
++
++AlgorithmIdentifier ::= SEQUENCE {
++ algorithm OBJECT IDENTIFIER ({ x509_note_OID }),
++ parameters ANY OPTIONAL
++}
++
++Name ::= SEQUENCE OF RelativeDistinguishedName
++
++RelativeDistinguishedName ::= SET OF AttributeValueAssertion
++
++AttributeValueAssertion ::= SEQUENCE {
++ attributeType OBJECT IDENTIFIER ({ x509_note_OID }),
++ attributeValue ANY ({ x509_extract_name_segment })
++ }
++
++Validity ::= SEQUENCE {
++ notBefore Time ({ x509_note_not_before }),
++ notAfter Time ({ x509_note_not_after })
++ }
++
++Time ::= CHOICE {
++ utcTime UTCTime,
++ generalTime GeneralizedTime
++ }
++
++SubjectPublicKeyInfo ::= SEQUENCE {
++ algorithm AlgorithmIdentifier,
++ subjectPublicKey BIT STRING ({ x509_extract_key_data })
++ }
++
++UniqueIdentifier ::= BIT STRING
++
++Extensions ::= SEQUENCE OF Extension
++
++Extension ::= SEQUENCE {
++ extnid OBJECT IDENTIFIER ({ x509_note_OID }),
++ critical BOOLEAN DEFAULT,
++ extnValue OCTET STRING ({ x509_process_extension })
++ }
+diff --git a/crypto/asymmetric_keys/x509_cert_parser.c b/crypto/asymmetric_keys/x509_cert_parser.c
+new file mode 100644
+index 0000000..8fcac94
+--- /dev/null
++++ b/crypto/asymmetric_keys/x509_cert_parser.c
+@@ -0,0 +1,497 @@
++/* X.509 certificate parser
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#define pr_fmt(fmt) "X.509: "fmt
++#include <linux/kernel.h>
++#include <linux/slab.h>
++#include <linux/err.h>
++#include <linux/oid_registry.h>
++#include "public_key.h"
++#include "x509_parser.h"
++#include "x509-asn1.h"
++#include "x509_rsakey-asn1.h"
++
++struct x509_parse_context {
++ struct x509_certificate *cert; /* Certificate being constructed */
++ unsigned long data; /* Start of data */
++ const void *cert_start; /* Start of cert content */
++ const void *key; /* Key data */
++ size_t key_size; /* Size of key data */
++ enum OID last_oid; /* Last OID encountered */
++ enum OID algo_oid; /* Algorithm OID */
++ unsigned char nr_mpi; /* Number of MPIs stored */
++ u8 o_size; /* Size of organizationName (O) */
++ u8 cn_size; /* Size of commonName (CN) */
++ u8 email_size; /* Size of emailAddress */
++ u16 o_offset; /* Offset of organizationName (O) */
++ u16 cn_offset; /* Offset of commonName (CN) */
++ u16 email_offset; /* Offset of emailAddress */
++};
++
++/*
++ * Free an X.509 certificate
++ */
++void x509_free_certificate(struct x509_certificate *cert)
++{
++ if (cert) {
++ public_key_destroy(cert->pub);
++ kfree(cert->issuer);
++ kfree(cert->subject);
++ kfree(cert->fingerprint);
++ kfree(cert->authority);
++ kfree(cert);
++ }
++}
++
++/*
++ * Parse an X.509 certificate
++ */
++struct x509_certificate *x509_cert_parse(const void *data, size_t datalen)
++{
++ struct x509_certificate *cert;
++ struct x509_parse_context *ctx;
++ long ret;
++
++ ret = -ENOMEM;
++ cert = kzalloc(sizeof(struct x509_certificate), GFP_KERNEL);
++ if (!cert)
++ goto error_no_cert;
++ cert->pub = kzalloc(sizeof(struct public_key), GFP_KERNEL);
++ if (!cert->pub)
++ goto error_no_ctx;
++ ctx = kzalloc(sizeof(struct x509_parse_context), GFP_KERNEL);
++ if (!ctx)
++ goto error_no_ctx;
++
++ ctx->cert = cert;
++ ctx->data = (unsigned long)data;
++
++ /* Attempt to decode the certificate */
++ ret = asn1_ber_decoder(&x509_decoder, ctx, data, datalen);
++ if (ret < 0)
++ goto error_decode;
++
++ /* Decode the public key */
++ ret = asn1_ber_decoder(&x509_rsakey_decoder, ctx,
++ ctx->key, ctx->key_size);
++ if (ret < 0)
++ goto error_decode;
++
++ kfree(ctx);
++ return cert;
++
++error_decode:
++ kfree(ctx);
++error_no_ctx:
++ x509_free_certificate(cert);
++error_no_cert:
++ return ERR_PTR(ret);
++}
++
++/*
++ * Note an OID when we find one for later processing when we know how
++ * to interpret it.
++ */
++int x509_note_OID(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++
++ ctx->last_oid = look_up_OID(value, vlen);
++ if (ctx->last_oid == OID__NR) {
++ char buffer[50];
++ sprint_oid(value, vlen, buffer, sizeof(buffer));
++ pr_debug("Unknown OID: [%zu] %s\n",
++ (unsigned long)value - ctx->data, buffer);
++ }
++ return 0;
++}
++
++/*
++ * Save the position of the TBS data so that we can check the signature over it
++ * later.
++ */
++int x509_note_tbs_certificate(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++
++ pr_debug("x509_note_tbs_certificate(,%zu,%02x,%ld,%zu)!\n",
++ hdrlen, tag, (unsigned long)value - ctx->data, vlen);
++
++ ctx->cert->tbs = value - hdrlen;
++ ctx->cert->tbs_size = vlen + hdrlen;
++ return 0;
++}
++
++/*
++ * Record the public key algorithm
++ */
++int x509_note_pkey_algo(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++
++ pr_debug("PubKey Algo: %u\n", ctx->last_oid);
++
++ switch (ctx->last_oid) {
++ case OID_md2WithRSAEncryption:
++ case OID_md3WithRSAEncryption:
++ default:
++ return -ENOPKG; /* Unsupported combination */
++
++ case OID_md4WithRSAEncryption:
++ ctx->cert->sig_hash_algo = PKEY_HASH_MD5;
++ ctx->cert->sig_pkey_algo = PKEY_ALGO_RSA;
++ break;
++
++ case OID_sha1WithRSAEncryption:
++ ctx->cert->sig_hash_algo = PKEY_HASH_SHA1;
++ ctx->cert->sig_pkey_algo = PKEY_ALGO_RSA;
++ break;
++
++ case OID_sha256WithRSAEncryption:
++ ctx->cert->sig_hash_algo = PKEY_HASH_SHA256;
++ ctx->cert->sig_pkey_algo = PKEY_ALGO_RSA;
++ break;
++
++ case OID_sha384WithRSAEncryption:
++ ctx->cert->sig_hash_algo = PKEY_HASH_SHA384;
++ ctx->cert->sig_pkey_algo = PKEY_ALGO_RSA;
++ break;
++
++ case OID_sha512WithRSAEncryption:
++ ctx->cert->sig_hash_algo = PKEY_HASH_SHA512;
++ ctx->cert->sig_pkey_algo = PKEY_ALGO_RSA;
++ break;
++
++ case OID_sha224WithRSAEncryption:
++ ctx->cert->sig_hash_algo = PKEY_HASH_SHA224;
++ ctx->cert->sig_pkey_algo = PKEY_ALGO_RSA;
++ break;
++ }
++
++ ctx->algo_oid = ctx->last_oid;
++ return 0;
++}
++
++/*
++ * Note the whereabouts and type of the signature.
++ */
++int x509_note_signature(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++
++ pr_debug("Signature type: %u size %zu\n", ctx->last_oid, vlen);
++
++ if (ctx->last_oid != ctx->algo_oid) {
++ pr_warn("Got cert with pkey (%u) and sig (%u) algorithm OIDs\n",
++ ctx->algo_oid, ctx->last_oid);
++ return -EINVAL;
++ }
++
++ ctx->cert->sig = value;
++ ctx->cert->sig_size = vlen;
++ return 0;
++}
++
++/*
++ * Note some of the name segments from which we'll fabricate a name.
++ */
++int x509_extract_name_segment(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++
++ switch (ctx->last_oid) {
++ case OID_commonName:
++ ctx->cn_size = vlen;
++ ctx->cn_offset = (unsigned long)value - ctx->data;
++ break;
++ case OID_organizationName:
++ ctx->o_size = vlen;
++ ctx->o_offset = (unsigned long)value - ctx->data;
++ break;
++ case OID_email_address:
++ ctx->email_size = vlen;
++ ctx->email_offset = (unsigned long)value - ctx->data;
++ break;
++ default:
++ break;
++ }
++
++ return 0;
++}
++
++/*
++ * Fabricate and save the issuer and subject names
++ */
++static int x509_fabricate_name(struct x509_parse_context *ctx, size_t hdrlen,
++ unsigned char tag,
++ char **_name, size_t vlen)
++{
++ const void *name, *data = (const void *)ctx->data;
++ size_t namesize;
++ char *buffer;
++
++ if (*_name)
++ return -EINVAL;
++
++ /* Empty name string if no material */
++ if (!ctx->cn_size && !ctx->o_size && !ctx->email_size) {
++ buffer = kmalloc(1, GFP_KERNEL);
++ if (!buffer)
++ return -ENOMEM;
++ buffer[0] = 0;
++ goto done;
++ }
++
++ if (ctx->cn_size && ctx->o_size) {
++ /* Consider combining O and CN, but use only the CN if it is
++ * prefixed by the O, or a significant portion thereof.
++ */
++ namesize = ctx->cn_size;
++ name = data + ctx->cn_offset;
++ if (ctx->cn_size >= ctx->o_size &&
++ memcmp(data + ctx->cn_offset, data + ctx->o_offset,
++ ctx->o_size) == 0)
++ goto single_component;
++ if (ctx->cn_size >= 7 &&
++ ctx->o_size >= 7 &&
++ memcmp(data + ctx->cn_offset, data + ctx->o_offset, 7) == 0)
++ goto single_component;
++
++ buffer = kmalloc(ctx->o_size + 2 + ctx->cn_size + 1,
++ GFP_KERNEL);
++ if (!buffer)
++ return -ENOMEM;
++
++ memcpy(buffer,
++ data + ctx->o_offset, ctx->o_size);
++ buffer[ctx->o_size + 0] = ':';
++ buffer[ctx->o_size + 1] = ' ';
++ memcpy(buffer + ctx->o_size + 2,
++ data + ctx->cn_offset, ctx->cn_size);
++ buffer[ctx->o_size + 2 + ctx->cn_size] = 0;
++ goto done;
++
++ } else if (ctx->cn_size) {
++ namesize = ctx->cn_size;
++ name = data + ctx->cn_offset;
++ } else if (ctx->o_size) {
++ namesize = ctx->o_size;
++ name = data + ctx->o_offset;
++ } else {
++ namesize = ctx->email_size;
++ name = data + ctx->email_offset;
++ }
++
++single_component:
++ buffer = kmalloc(namesize + 1, GFP_KERNEL);
++ if (!buffer)
++ return -ENOMEM;
++ memcpy(buffer, name, namesize);
++ buffer[namesize] = 0;
++
++done:
++ *_name = buffer;
++ ctx->cn_size = 0;
++ ctx->o_size = 0;
++ ctx->email_size = 0;
++ return 0;
++}
++
++int x509_note_issuer(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++ return x509_fabricate_name(ctx, hdrlen, tag, &ctx->cert->issuer, vlen);
++}
++
++int x509_note_subject(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++ return x509_fabricate_name(ctx, hdrlen, tag, &ctx->cert->subject, vlen);
++}
++
++/*
++ * Extract the data for the public key algorithm
++ */
++int x509_extract_key_data(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++
++ if (ctx->last_oid != OID_rsaEncryption)
++ return -ENOPKG;
++
++ /* There seems to be an extraneous 0 byte on the front of the data */
++ ctx->cert->pkey_algo = PKEY_ALGO_RSA;
++ ctx->key = value + 1;
++ ctx->key_size = vlen - 1;
++ return 0;
++}
++
++/*
++ * Extract a RSA public key value
++ */
++int rsa_extract_mpi(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++ MPI mpi;
++
++ if (ctx->nr_mpi >= ARRAY_SIZE(ctx->cert->pub->mpi)) {
++ pr_err("Too many public key MPIs in certificate\n");
++ return -EBADMSG;
++ }
++
++ mpi = mpi_read_raw_data(value, vlen);
++ if (!mpi)
++ return -ENOMEM;
++
++ ctx->cert->pub->mpi[ctx->nr_mpi++] = mpi;
++ return 0;
++}
++
++/*
++ * Process certificate extensions that are used to qualify the certificate.
++ */
++int x509_process_extension(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++ const unsigned char *v = value;
++ char *f;
++ int i;
++
++ pr_debug("Extension: %u\n", ctx->last_oid);
++
++ if (ctx->last_oid == OID_subjectKeyIdentifier) {
++ /* Get hold of the key fingerprint */
++ if (vlen < 3)
++ return -EBADMSG;
++ if (v[0] != ASN1_OTS || v[1] != vlen - 2)
++ return -EBADMSG;
++ v += 2;
++ vlen -= 2;
++
++ f = kmalloc(vlen * 2 + 1, GFP_KERNEL);
++ if (!f)
++ return -ENOMEM;
++ for (i = 0; i < vlen; i++)
++ sprintf(f + i * 2, "%02x", v[i]);
++ pr_debug("fingerprint %s\n", f);
++ ctx->cert->fingerprint = f;
++ return 0;
++ }
++
++ if (ctx->last_oid == OID_authorityKeyIdentifier) {
++ /* Get hold of the CA key fingerprint */
++ if (vlen < 5)
++ return -EBADMSG;
++ if (v[0] != (ASN1_SEQ | (ASN1_CONS << 5)) ||
++ v[1] != vlen - 2 ||
++ v[2] != (ASN1_CONT << 6) ||
++ v[3] != vlen - 4)
++ return -EBADMSG;
++ v += 4;
++ vlen -= 4;
++
++ f = kmalloc(vlen * 2 + 1, GFP_KERNEL);
++ if (!f)
++ return -ENOMEM;
++ for (i = 0; i < vlen; i++)
++ sprintf(f + i * 2, "%02x", v[i]);
++ pr_debug("authority %s\n", f);
++ ctx->cert->authority = f;
++ return 0;
++ }
++
++ return 0;
++}
++
++/*
++ * Record a certificate time.
++ */
++static int x509_note_time(time_t *_time, size_t hdrlen,
++ unsigned char tag,
++ const unsigned char *value, size_t vlen)
++{
++ unsigned YY, MM, DD, hh, mm, ss;
++ const unsigned char *p = value;
++
++#define dec2bin(X) ((X) - '0')
++#define DD2bin(P) ({ unsigned x = dec2bin(P[0]) * 10 + dec2bin(P[1]); P += 2; x; })
++
++ if (tag == ASN1_UNITIM) {
++ /* UTCTime: YYMMDDHHMMSSZ */
++ if (vlen != 13)
++ goto unsupported_time;
++ YY = DD2bin(p);
++ if (YY > 50)
++ YY += 1900;
++ else
++ YY += 2000;
++ } else if (tag == ASN1_GENTIM) {
++ /* GenTime: YYYYMMDDHHMMSSZ */
++ if (vlen != 15)
++ goto unsupported_time;
++ YY = DD2bin(p) * 100 + DD2bin(p);
++ } else {
++ goto unsupported_time;
++ }
++
++ MM = DD2bin(p);
++ DD = DD2bin(p);
++ hh = DD2bin(p);
++ mm = DD2bin(p);
++ ss = DD2bin(p);
++
++ if (*p != 'Z')
++ goto unsupported_time;
++
++ *_time = mktime(YY, MM, DD, hh, mm, ss);
++ return 0;
++
++unsupported_time:
++ pr_debug("Got unsupported time [tag %02x]: '%*.*s'\n",
++ tag, (int)vlen, (int)vlen, value);
++ return -EBADMSG;
++}
++
++int x509_note_not_before(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++ return x509_note_time(&ctx->cert->valid_from, hdrlen, tag, value, vlen);
++}
++
++int x509_note_not_after(void *context, size_t hdrlen,
++ unsigned char tag,
++ const void *value, size_t vlen)
++{
++ struct x509_parse_context *ctx = context;
++ return x509_note_time(&ctx->cert->valid_to, hdrlen, tag, value, vlen);
++}
+diff --git a/crypto/asymmetric_keys/x509_parser.h b/crypto/asymmetric_keys/x509_parser.h
+new file mode 100644
+index 0000000..635053f
+--- /dev/null
++++ b/crypto/asymmetric_keys/x509_parser.h
+@@ -0,0 +1,36 @@
++/* X.509 certificate parser internal definitions
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <crypto/public_key.h>
++
++struct x509_certificate {
++ struct x509_certificate *next;
++ struct public_key *pub; /* Public key details */
++ char *issuer; /* Name of certificate issuer */
++ char *subject; /* Name of certificate subject */
++ char *fingerprint; /* Key fingerprint as hex */
++ char *authority; /* Authority key fingerprint as hex */
++ time_t valid_from;
++ time_t valid_to;
++ enum pkey_algo pkey_algo : 8; /* Public key algorithm */
++ enum pkey_algo sig_pkey_algo : 8; /* Signature public key algorithm */
++ enum pkey_hash_algo sig_hash_algo : 8; /* Signature hash algorithm */
++ const void *tbs; /* Signed data */
++ size_t tbs_size; /* Size of signed data */
++ const void *sig; /* Signature data */
++ size_t sig_size; /* Size of sigature */
++};
++
++/*
++ * x509_cert_parser.c
++ */
++extern void x509_free_certificate(struct x509_certificate *cert);
++extern struct x509_certificate *x509_cert_parse(const void *data, size_t datalen);
+diff --git a/crypto/asymmetric_keys/x509_public_key.c b/crypto/asymmetric_keys/x509_public_key.c
+new file mode 100644
+index 0000000..716917c
+--- /dev/null
++++ b/crypto/asymmetric_keys/x509_public_key.c
+@@ -0,0 +1,207 @@
++/* Instantiate a public key crypto key from an X.509 Certificate
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#define pr_fmt(fmt) "X.509: "fmt
++#include <linux/module.h>
++#include <linux/kernel.h>
++#include <linux/slab.h>
++#include <linux/err.h>
++#include <linux/mpi.h>
++#include <linux/asn1_decoder.h>
++#include <keys/asymmetric-subtype.h>
++#include <keys/asymmetric-parser.h>
++#include <crypto/hash.h>
++#include "asymmetric_keys.h"
++#include "public_key.h"
++#include "x509_parser.h"
++
++static const
++struct public_key_algorithm *x509_public_key_algorithms[PKEY_ALGO__LAST] = {
++ [PKEY_ALGO_DSA] = NULL,
++#if defined(CONFIG_PUBLIC_KEY_ALGO_RSA) || \
++ defined(CONFIG_PUBLIC_KEY_ALGO_RSA_MODULE)
++ [PKEY_ALGO_RSA] = &RSA_public_key_algorithm,
++#endif
++};
++
++/*
++ * Check the signature on a certificate using the provided public key
++ */
++static int x509_check_signature(const struct public_key *pub,
++ const struct x509_certificate *cert)
++{
++ struct public_key_signature *sig;
++ struct crypto_shash *tfm;
++ struct shash_desc *desc;
++ size_t digest_size, desc_size;
++ int ret;
++
++ pr_devel("==>%s()\n", __func__);
++
++ /* Allocate the hashing algorithm we're going to need and find out how
++ * big the hash operational data will be.
++ */
++ tfm = crypto_alloc_shash(pkey_hash_algo[cert->sig_hash_algo], 0, 0);
++ if (IS_ERR(tfm))
++ return (PTR_ERR(tfm) == -ENOENT) ? -ENOPKG : PTR_ERR(tfm);
++
++ desc_size = crypto_shash_descsize(tfm) + sizeof(*desc);
++ digest_size = crypto_shash_digestsize(tfm);
++
++ /* We allocate the hash operational data storage on the end of our
++ * context data.
++ */
++ ret = -ENOMEM;
++ sig = kzalloc(sizeof(*sig) + desc_size + digest_size, GFP_KERNEL);
++ if (!sig)
++ goto error_no_sig;
++
++ sig->pkey_hash_algo = cert->sig_hash_algo;
++ sig->digest = (u8 *)sig + sizeof(*sig) + desc_size;
++ sig->digest_size = digest_size;
++
++ desc = (void *)sig + sizeof(*sig);
++ desc->tfm = tfm;
++ desc->flags = CRYPTO_TFM_REQ_MAY_SLEEP;
++
++ ret = crypto_shash_init(desc);
++ if (ret < 0)
++ goto error;
++
++ ret = -ENOMEM;
++ sig->rsa.s = mpi_read_raw_data(cert->sig, cert->sig_size);
++ if (!sig->rsa.s)
++ goto error;
++
++ ret = crypto_shash_finup(desc, cert->tbs, cert->tbs_size, sig->digest);
++ if (ret < 0)
++ goto error_mpi;
++
++ ret = pub->algo->verify_signature(pub, sig);
++
++ pr_debug("Cert Verification: %d\n", ret);
++
++error_mpi:
++ mpi_free(sig->rsa.s);
++error:
++ kfree(sig);
++error_no_sig:
++ crypto_free_shash(tfm);
++
++ pr_devel("<==%s() = %d\n", __func__, ret);
++ return ret;
++}
++
++/*
++ * Attempt to parse a data blob for a key as an X509 certificate.
++ */
++static int x509_key_preparse(struct key_preparsed_payload *prep)
++{
++ struct x509_certificate *cert;
++ time_t now;
++ size_t srlen, sulen;
++ char *desc = NULL;
++ int ret;
++
++ cert = x509_cert_parse(prep->data, prep->datalen);
++ if (IS_ERR(cert))
++ return PTR_ERR(cert);
++
++ pr_devel("Cert Issuer: %s\n", cert->issuer);
++ pr_devel("Cert Subject: %s\n", cert->subject);
++ pr_devel("Cert Key Algo: %s\n", pkey_algo[cert->pkey_algo]);
++ pr_devel("Cert Valid: %lu - %lu\n", cert->valid_from, cert->valid_to);
++ pr_devel("Cert Signature: %s + %s\n",
++ pkey_algo[cert->sig_pkey_algo],
++ pkey_hash_algo[cert->sig_hash_algo]);
++
++ if (!cert->fingerprint || !cert->authority) {
++ pr_warn("Cert for '%s' must have SubjKeyId and AuthKeyId extensions\n",
++ cert->subject);
++ ret = -EKEYREJECTED;
++ goto error_free_cert;
++ }
++
++ now = CURRENT_TIME.tv_sec;
++ if (now < cert->valid_from) {
++ pr_warn("Cert %s is not yet valid\n", cert->fingerprint);
++ ret = -EKEYREJECTED;
++ goto error_free_cert;
++ }
++ if (now >= cert->valid_to) {
++ pr_warn("Cert %s has expired\n", cert->fingerprint);
++ ret = -EKEYEXPIRED;
++ goto error_free_cert;
++ }
++
++ cert->pub->algo = x509_public_key_algorithms[cert->pkey_algo];
++ cert->pub->id_type = PKEY_ID_X509;
++
++ /* Check the signature on the key */
++ if (strcmp(cert->fingerprint, cert->authority) == 0) {
++ ret = x509_check_signature(cert->pub, cert);
++ if (ret < 0)
++ goto error_free_cert;
++ }
++
++ /* Propose a description */
++ sulen = strlen(cert->subject);
++ srlen = strlen(cert->fingerprint);
++ ret = -ENOMEM;
++ desc = kmalloc(sulen + 2 + srlen + 1, GFP_KERNEL);
++ if (!desc)
++ goto error_free_cert;
++ memcpy(desc, cert->subject, sulen);
++ desc[sulen] = ':';
++ desc[sulen + 1] = ' ';
++ memcpy(desc + sulen + 2, cert->fingerprint, srlen);
++ desc[sulen + 2 + srlen] = 0;
++
++ /* We're pinning the module by being linked against it */
++ __module_get(public_key_subtype.owner);
++ prep->type_data[0] = &public_key_subtype;
++ prep->type_data[1] = cert->fingerprint;
++ prep->payload = cert->pub;
++ prep->description = desc;
++ prep->quotalen = 100;
++
++ /* We've finished with the certificate */
++ cert->pub = NULL;
++ cert->fingerprint = NULL;
++ desc = NULL;
++ ret = 0;
++
++error_free_cert:
++ x509_free_certificate(cert);
++ return ret;
++}
++
++static struct asymmetric_key_parser x509_key_parser = {
++ .owner = THIS_MODULE,
++ .name = "x509",
++ .parse = x509_key_preparse,
++};
++
++/*
++ * Module stuff
++ */
++static int __init x509_key_init(void)
++{
++ return register_asymmetric_key_parser(&x509_key_parser);
++}
++
++static void __exit x509_key_exit(void)
++{
++ unregister_asymmetric_key_parser(&x509_key_parser);
++}
++
++module_init(x509_key_init);
++module_exit(x509_key_exit);
+diff --git a/crypto/asymmetric_keys/x509_rsakey.asn1 b/crypto/asymmetric_keys/x509_rsakey.asn1
+new file mode 100644
+index 0000000..4ec7cc6
+--- /dev/null
++++ b/crypto/asymmetric_keys/x509_rsakey.asn1
+@@ -0,0 +1,4 @@
++RSAPublicKey ::= SEQUENCE {
++ modulus INTEGER ({ rsa_extract_mpi }), -- n
++ publicExponent INTEGER ({ rsa_extract_mpi }) -- e
++ }
+--
+1.7.11.4
+
+
+From 955fc6ec995f6bec6c487eb46027e108e240ebe3 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:13:25 +0100
+Subject: [PATCH 18/26] MOD: Fix Rusty's module_sig_check()
+
+Make the following fixes to Rusty's module_sig_check() function:
+
+ (1) mod_verify_sig() is not defined, resulting in a compilation error and
+ thereby breaking git bisect, so provide a dummy that returns an error.
+
+ (2) Using strlen() on a static string is a waste of resources. Further, you
+ may end up with two copies of the string emitted.
+
+ (3) Doing a memchr() of the bytes beyond the last position that the marker
+ can be in is a waste of resources.
+
+While we're at it, push responsibility for the return value entirely off to
+mod_verify_sig() if we find a signature.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ kernel/Makefile | 1 +
+ kernel/module-internal.h | 14 ++++++++++++++
+ kernel/module.c | 41 +++++++++++++++++++++++------------------
+ kernel/module_signing.c | 23 +++++++++++++++++++++++
+ 4 files changed, 61 insertions(+), 18 deletions(-)
+ create mode 100644 kernel/module-internal.h
+ create mode 100644 kernel/module_signing.c
+
+diff --git a/kernel/Makefile b/kernel/Makefile
+index c0cc67a..08ba8a6 100644
+--- a/kernel/Makefile
++++ b/kernel/Makefile
+@@ -55,6 +55,7 @@ obj-$(CONFIG_DEBUG_SPINLOCK) += spinlock.o
+ obj-$(CONFIG_PROVE_LOCKING) += spinlock.o
+ obj-$(CONFIG_UID16) += uid16.o
+ obj-$(CONFIG_MODULES) += module.o
++obj-$(CONFIG_MODULE_SIG) += module_signing.o
+ obj-$(CONFIG_KALLSYMS) += kallsyms.o
+ obj-$(CONFIG_BSD_PROCESS_ACCT) += acct.o
+ obj-$(CONFIG_KEXEC) += kexec.o
+diff --git a/kernel/module-internal.h b/kernel/module-internal.h
+new file mode 100644
+index 0000000..14da0ea
+--- /dev/null
++++ b/kernel/module-internal.h
+@@ -0,0 +1,14 @@
++/* Module internals
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++extern int mod_verify_sig(const void *mod, unsigned long modlen,
++ const void *sig, unsigned long siglen,
++ bool *_sig_ok);
+diff --git a/kernel/module.c b/kernel/module.c
+index 5c6f65c..ab69599 100644
+--- a/kernel/module.c
++++ b/kernel/module.c
+@@ -58,6 +58,7 @@
+ #include <linux/jump_label.h>
+ #include <linux/pfn.h>
+ #include <linux/bsearch.h>
++#include "module-internal.h"
+
+ #define CREATE_TRACE_POINTS
+ #include <trace/events/module.h>
+@@ -2438,34 +2439,38 @@ static inline void kmemleak_load_module(const struct module *mod,
+ #endif
+
+ #ifdef CONFIG_MODULE_SIG
++
+ static int module_sig_check(struct load_info *info,
+ const void *mod, unsigned long *len)
+ {
++ static const char module_sig_string[] = MODULE_SIG_STRING;
+ int err = 0;
+- const unsigned long markerlen = strlen(MODULE_SIG_STRING);
+- const void *p = mod, *end = mod + *len;
+-
+- /* Poor man's memmem. */
+- while ((p = memchr(p, MODULE_SIG_STRING[0], end - p))) {
+- if (p + markerlen > end)
+- break;
+-
+- if (memcmp(p, MODULE_SIG_STRING, markerlen) == 0) {
+- const void *sig = p + markerlen;
+- /* Truncate module up to signature. */
+- *len = p - mod;
+- err = mod_verify_sig(mod, *len,
+- sig, end - sig,
+- &info->sig_ok);
+- break;
+- }
+- p++;
++ const unsigned long markerlen = sizeof(module_sig_string) - 1;
++ const void *p = mod, *end = mod + *len, *sig;
++ const void *limit = end - markerlen - 1;
++
++ if (markerlen < *len) {
++ /* Poor man's memmem. */
++ do {
++ p = memchr(p, MODULE_SIG_STRING[0], limit - p);
++ if (!p)
++ break;
++ if (memcmp(p, module_sig_string, markerlen) != 0)
++ continue;
++ goto found_marker;
++ } while (++p < limit);
+ }
+
+ /* Not having a signature is only an error if we're strict. */
+ if (!err && !info->sig_ok && sig_enforce)
+ err = -EKEYREJECTED;
+ return err;
++
++found_marker:
++ sig = p + markerlen;
++ /* Truncate module up to signature. */
++ *len = p - mod;
++ return mod_verify_sig(mod, *len, sig, end - sig, &info->sig_ok);
+ }
+ #else /* !CONFIG_MODULE_SIG */
+ static int module_sig_check(struct load_info *info,
+diff --git a/kernel/module_signing.c b/kernel/module_signing.c
+new file mode 100644
+index 0000000..0af10a5
+--- /dev/null
++++ b/kernel/module_signing.c
+@@ -0,0 +1,23 @@
++/* Module signature checker
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <linux/kernel.h>
++#include "module-internal.h"
++
++/*
++ * Verify the signature on a module.
++ */
++int mod_verify_sig(const void *mod, unsigned long modlen,
++ const void *sig, unsigned long siglen,
++ bool *_sig_ok)
++{
++ return -EKEYREJECTED;
++}
+--
+1.7.11.4
+
+
+From 4c31831859550149cdba65a37d72416c87dbbef6 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:13:25 +0100
+Subject: [PATCH 19/26] MODSIGN: Provide gitignore and make clean rules for
+ extra files
+
+Provide gitignore and make clean rules for extra files to hide and clean up the
+extra files produced by module signing stuff once it is added. Also add a
+clean up rule for the module content extractor program used to extract the data
+to be signed.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ .gitignore | 13 +++++++++++++
+ Makefile | 1 +
+ 2 files changed, 14 insertions(+)
+
+diff --git a/.gitignore b/.gitignore
+index 57af07c..9736304 100644
+--- a/.gitignore
++++ b/.gitignore
+@@ -14,6 +14,10 @@
+ *.o.*
+ *.a
+ *.s
++*.ko.unsigned
++*.ko.stripped
++*.ko.stripped.dig
++*.ko.stripped.sig
+ *.ko
+ *.so
+ *.so.dbg
+@@ -84,3 +88,12 @@ GTAGS
+ *.orig
+ *~
+ \#*#
++
++#
++# Leavings from module signing
++#
++extra_certificates
++signing_key.priv
++signing_key.x509
++signing_key.x509.keyid
++signing_key.x509.signer
+diff --git a/Makefile b/Makefile
+index 371ce88..644048d 100644
+--- a/Makefile
++++ b/Makefile
+@@ -1239,6 +1239,7 @@ clean: $(clean-dirs)
+ $(call cmd,rmfiles)
+ @find $(if $(KBUILD_EXTMOD), $(KBUILD_EXTMOD), .) $(RCS_FIND_IGNORE) \
+ \( -name '*.[oas]' -o -name '*.ko' -o -name '.*.cmd' \
++ -o -name '*.ko.*' \
+ -o -name '.*.d' -o -name '.*.tmp' -o -name '*.mod.c' \
+ -o -name '*.symtypes' -o -name 'modules.order' \
+ -o -name modules.builtin -o -name '.tmp_*.o.*' \
+--
+1.7.11.4
+
+
+From 6977e69eef4379f34a2ad264856d74ac292284df Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:13:25 +0100
+Subject: [PATCH 20/26] MODSIGN: Provide Kconfig options
+
+Provide kernel configuration options for module signing.
+
+The following configuration options are added:
+
+ CONFIG_MODULE_SIG_SHA1
+ CONFIG_MODULE_SIG_SHA224
+ CONFIG_MODULE_SIG_SHA256
+ CONFIG_MODULE_SIG_SHA384
+ CONFIG_MODULE_SIG_SHA512
+
+These select the cryptographic hash used to digest the data prior to signing.
+Additionally, the crypto module selected will be built into the kernel as it
+won't be possible to load it as a module without incurring a circular
+dependency when the kernel tries to check its signature.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ init/Kconfig | 38 ++++++++++++++++++++++++++++++++++++++
+ 1 file changed, 38 insertions(+)
+
+diff --git a/init/Kconfig b/init/Kconfig
+index fa8ccad..00d4579 100644
+--- a/init/Kconfig
++++ b/init/Kconfig
+@@ -1593,12 +1593,50 @@ config MODULE_SIG
+ is simply appended to the module. For more information see
+ Documentation/module-signing.txt.
+
++ !!!WARNING!!! If you enable this option, you MUST make sure that the
++ module DOES NOT get stripped after being signed. This includes the
++ debuginfo strip done by some packagers (such as rpmbuild) and
++ inclusion into an initramfs that wants the module size reduced.
++
+ config MODULE_SIG_FORCE
+ bool "Require modules to be validly signed"
+ depends on MODULE_SIG
+ help
+ Reject unsigned modules or signed modules for which we don't have a
+ key. Without this, such modules will simply taint the kernel.
++
++choice
++ prompt "Which hash algorithm should modules be signed with?"
++ depends on MODULE_SIG
++ help
++ This determines which sort of hashing algorithm will be used during
++ signature generation. This algorithm _must_ be built into the kernel
++ directly so that signature verification can take place. It is not
++ possible to load a signed module containing the algorithm to check
++ the signature on that module.
++
++config MODULE_SIG_SHA1
++ bool "Sign modules with SHA-1"
++ select CRYPTO_SHA1
++
++config MODULE_SIG_SHA224
++ bool "Sign modules with SHA-224"
++ select CRYPTO_SHA256
++
++config MODULE_SIG_SHA256
++ bool "Sign modules with SHA-256"
++ select CRYPTO_SHA256
++
++config MODULE_SIG_SHA384
++ bool "Sign modules with SHA-384"
++ select CRYPTO_SHA512
++
++config MODULE_SIG_SHA512
++ bool "Sign modules with SHA-512"
++ select CRYPTO_SHA512
++
++endchoice
++
+ endif # MODULES
+
+ config INIT_ALL_POSSIBLE
+--
+1.7.11.4
+
+
+From 60378703cf88ed221ee602727c37ae241565d44a Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:13:25 +0100
+Subject: [PATCH 21/26] MODSIGN: Automatically generate module signing keys if
+ missing
+
+Automatically generate keys for module signing if they're absent so that
+allyesconfig doesn't break. The builder should consider generating their own
+key and certificate, however, so that the keys are appropriately named.
+
+The private key for the module signer should be placed in signing_key.priv
+(unencrypted!) and the public key in an X.509 certificate as signing_key.x509.
+
+If a transient key is desired for signing the modules, a config file for
+'openssl req' can be placed in x509.genkey, looking something like the
+following:
+
+ [ req ]
+ default_bits = 4096
+ distinguished_name = req_distinguished_name
+ prompt = no
+ x509_extensions = myexts
+
+ [ req_distinguished_name ]
+ O = Magarathea
+ CN = Glacier signing key
+ emailAddress = slartibartfast@magrathea.h2g2
+
+ [ myexts ]
+ basicConstraints=critical,CA:FALSE
+ keyUsage=digitalSignature
+ subjectKeyIdentifier=hash
+ authorityKeyIdentifier=hash
+
+The build process will use this to configure:
+
+ openssl req -new -nodes -utf8 -sha1 -days 36500 -batch \
+ -x509 -config x509.genkey \
+ -outform DER -out signing_key.x509 \
+ -keyout signing_key.priv
+
+to generate the key.
+
+Note that it is required that the X.509 certificate have a subjectKeyIdentifier
+and an authorityKeyIdentifier. Without those, the certificate will be
+rejected. These can be used to check the validity of a certificate.
+
+Note that 'make distclean' will remove signing_key.{priv,x509} and x509.genkey,
+whether or not they were generated automatically.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ kernel/Makefile | 49 +++++++++++++++++++++++++++++++++++++++++++++++++
+ 1 file changed, 49 insertions(+)
+
+diff --git a/kernel/Makefile b/kernel/Makefile
+index 08ba8a6..83f1565 100644
+--- a/kernel/Makefile
++++ b/kernel/Makefile
+@@ -132,3 +132,52 @@ quiet_cmd_timeconst = TIMEC $@
+ targets += timeconst.h
+ $(obj)/timeconst.h: $(src)/timeconst.pl FORCE
+ $(call if_changed,timeconst)
++
++ifeq ($(CONFIG_MODULE_SIG),y)
++
++###############################################################################
++#
++# If module signing is requested, say by allyesconfig, but a key has not been
++# supplied, then one will need to be generated to make sure the build does not
++# fail and that the kernel may be used afterwards.
++#
++###############################################################################
++signing_key.priv signing_key.x509: x509.genkey
++ @echo "###"
++ @echo "### Now generating an X.509 key pair to be used for signing modules."
++ @echo "###"
++ @echo "### If this takes a long time, you might wish to run rngd in the"
++ @echo "### background to keep the supply of entropy topped up. It"
++ @echo "### needs to be run as root, and should use a hardware random"
++ @echo "### number generator if one is available, eg:"
++ @echo "###"
++ @echo "### rngd -r /dev/hwrandom"
++ @echo "###"
++ openssl req -new -nodes -utf8 -sha1 -days 36500 -batch \
++ -x509 -config x509.genkey \
++ -outform DER -out signing_key.x509 \
++ -keyout signing_key.priv
++ @echo "###"
++ @echo "### Key pair generated."
++ @echo "###"
++
++x509.genkey:
++ @echo Generating X.509 key generation config
++ @echo >x509.genkey "[ req ]"
++ @echo >>x509.genkey "default_bits = 4096"
++ @echo >>x509.genkey "distinguished_name = req_distinguished_name"
++ @echo >>x509.genkey "prompt = no"
++ @echo >>x509.genkey "x509_extensions = myexts"
++ @echo >>x509.genkey
++ @echo >>x509.genkey "[ req_distinguished_name ]"
++ @echo >>x509.genkey "O = Magarathea"
++ @echo >>x509.genkey "CN = Glacier signing key"
++ @echo >>x509.genkey "emailAddress = slartibartfast@magrathea.h2g2"
++ @echo >>x509.genkey
++ @echo >>x509.genkey "[ myexts ]"
++ @echo >>x509.genkey "basicConstraints=critical,CA:FALSE"
++ @echo >>x509.genkey "keyUsage=digitalSignature"
++ @echo >>x509.genkey "subjectKeyIdentifier=hash"
++ @echo >>x509.genkey "authorityKeyIdentifier=keyid"
++endif
++CLEAN_FILES += signing_key.priv signing_key.x509 x509.genkey
+--
+1.7.11.4
+
+
+From 798049d9df83c8fd87fd5ddb97a77054558f4361 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:13:25 +0100
+Subject: [PATCH 22/26] MODSIGN: Provide module signing public keys to the
+ kernel
+
+Include a PGP keyring containing the public keys required to perform module
+verification in the kernel image during build and create a special keyring
+during boot which is then populated with keys of crypto type holding the public
+keys found in the PGP keyring.
+
+These can be seen by root:
+
+[root@andromeda ~]# cat /proc/keys
+07ad4ee0 I----- 1 perm 3f010000 0 0 crypto modsign.0: RSA 87b9b3bd []
+15c7f8c3 I----- 1 perm 1f030000 0 0 keyring .module_sign: 1/4
+...
+
+It is probably worth permitting root to invalidate these keys, resulting in
+their removal and preventing further modules from being loaded with that key.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ kernel/Makefile | 11 ++++-
+ kernel/modsign_pubkey.c | 112 +++++++++++++++++++++++++++++++++++++++++++++++
+ kernel/module-internal.h | 2 +
+ 3 files changed, 123 insertions(+), 2 deletions(-)
+ create mode 100644 kernel/modsign_pubkey.c
+
+diff --git a/kernel/Makefile b/kernel/Makefile
+index 83f1565..63f8386 100644
+--- a/kernel/Makefile
++++ b/kernel/Makefile
+@@ -55,7 +55,7 @@ obj-$(CONFIG_DEBUG_SPINLOCK) += spinlock.o
+ obj-$(CONFIG_PROVE_LOCKING) += spinlock.o
+ obj-$(CONFIG_UID16) += uid16.o
+ obj-$(CONFIG_MODULES) += module.o
+-obj-$(CONFIG_MODULE_SIG) += module_signing.o
++obj-$(CONFIG_MODULE_SIG) += module_signing.o modsign_pubkey.o
+ obj-$(CONFIG_KALLSYMS) += kallsyms.o
+ obj-$(CONFIG_BSD_PROCESS_ACCT) += acct.o
+ obj-$(CONFIG_KEXEC) += kexec.o
+@@ -134,6 +134,13 @@ $(obj)/timeconst.h: $(src)/timeconst.pl FORCE
+ $(call if_changed,timeconst)
+
+ ifeq ($(CONFIG_MODULE_SIG),y)
++#
++# Pull the signing certificate and any extra certificates into the kernel
++#
++extra_certificates:
++ touch $@
++
++kernel/modsign_pubkey.o: signing_key.x509 extra_certificates
+
+ ###############################################################################
+ #
+@@ -180,4 +187,4 @@ x509.genkey:
+ @echo >>x509.genkey "subjectKeyIdentifier=hash"
+ @echo >>x509.genkey "authorityKeyIdentifier=keyid"
+ endif
+-CLEAN_FILES += signing_key.priv signing_key.x509 x509.genkey
++CLEAN_FILES += signing_key.priv signing_key.x509 x509.genkey extra_certificates
+diff --git a/kernel/modsign_pubkey.c b/kernel/modsign_pubkey.c
+new file mode 100644
+index 0000000..f504d9f
+--- /dev/null
++++ b/kernel/modsign_pubkey.c
+@@ -0,0 +1,112 @@
++/* Public keys for module signature verification
++ *
++ * Copyright (C) 2012 Red Hat, Inc. All Rights Reserved.
++ * Written by David Howells (dhowells@redhat.com)
++ *
++ * This program is free software; you can redistribute it and/or
++ * modify it under the terms of the GNU General Public Licence
++ * as published by the Free Software Foundation; either version
++ * 2 of the Licence, or (at your option) any later version.
++ */
++
++#include <linux/kernel.h>
++#include <linux/sched.h>
++#include <linux/cred.h>
++#include <linux/err.h>
++#include <keys/asymmetric-type.h>
++#include "module-internal.h"
++
++struct key *modsign_keyring;
++
++extern __initdata const u8 modsign_certificate_list[];
++extern __initdata const u8 modsign_certificate_list_end[];
++asm(".section .init.data,\"aw\"\n"
++ "modsign_certificate_list:\n"
++ ".incbin \"signing_key.x509\"\n"
++ ".incbin \"extra_certificates\"\n"
++ "modsign_certificate_list_end:"
++ );
++
++/*
++ * We need to make sure ccache doesn't cache the .o file as it doesn't notice
++ * if modsign.pub changes.
++ */
++static __initdata const char annoy_ccache[] = __TIME__ "foo";
++
++/*
++ * Load the compiled-in keys
++ */
++static __init int module_verify_init(void)
++{
++ pr_notice("Initialise module verification\n");
++
++ modsign_keyring = key_alloc(&key_type_keyring, ".module_sign",
++ 0, 0, current_cred(),
++ (KEY_POS_ALL & ~KEY_POS_SETATTR) |
++ KEY_USR_VIEW | KEY_USR_READ,
++ KEY_ALLOC_NOT_IN_QUOTA);
++ if (IS_ERR(modsign_keyring))
++ panic("Can't allocate module signing keyring\n");
++
++ if (key_instantiate_and_link(modsign_keyring, NULL, 0, NULL, NULL) < 0)
++ panic("Can't instantiate module signing keyring\n");
++
++ return 0;
++}
++
++/*
++ * Must be initialised before we try and load the keys into the keyring.
++ */
++device_initcall(module_verify_init);
++
++/*
++ * Load the compiled-in keys
++ */
++static __init int load_module_signing_keys(void)
++{
++ key_ref_t key;
++ const u8 *p, *end;
++ size_t plen;
++
++ pr_notice("Loading module verification certificates\n");
++
++ end = modsign_certificate_list_end;
++ p = modsign_certificate_list;
++ while (p < end) {
++ /* Each cert begins with an ASN.1 SEQUENCE tag and must be more
++ * than 256 bytes in size.
++ */
++ if (end - p < 4)
++ goto dodgy_cert;
++ if (p[0] != 0x30 &&
++ p[1] != 0x82)
++ goto dodgy_cert;
++ plen = (p[2] << 8) | p[3];
++ plen += 4;
++ if (plen > end - p)
++ goto dodgy_cert;
++
++ key = key_create_or_update(make_key_ref(modsign_keyring, 1),
++ "asymmetric",
++ NULL,
++ p,
++ plen,
++ (KEY_POS_ALL & ~KEY_POS_SETATTR) |
++ KEY_USR_VIEW,
++ KEY_ALLOC_NOT_IN_QUOTA);
++ if (IS_ERR(key))
++ pr_err("MODSIGN: Problem loading in-kernel X.509 certificate (%ld)\n",
++ PTR_ERR(key));
++ else
++ pr_notice("MODSIGN: Loaded cert '%s'\n",
++ key_ref_to_ptr(key)->description);
++ p += plen;
++ }
++
++ return 0;
++
++dodgy_cert:
++ pr_err("MODSIGN: Problem parsing in-kernel X.509 certificate list\n");
++ return 0;
++}
++late_initcall(load_module_signing_keys);
+diff --git a/kernel/module-internal.h b/kernel/module-internal.h
+index 14da0ea..648f481 100644
+--- a/kernel/module-internal.h
++++ b/kernel/module-internal.h
+@@ -9,6 +9,8 @@
+ * 2 of the Licence, or (at your option) any later version.
+ */
+
++extern struct key *modsign_keyring;
++
+ extern int mod_verify_sig(const void *mod, unsigned long modlen,
+ const void *sig, unsigned long siglen,
+ bool *_sig_ok);
+--
+1.7.11.4
+
+
+From dcac300c703bc9a237deab7c7f0301f3803a3a9a Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:13:26 +0100
+Subject: [PATCH 23/26] MODSIGN: Implement module signature checking
+
+Check the signature on the module against the keys compiled into the kernel or
+available in a hardware key store.
+
+Currently, only RSA keys are supported - though that's easy enough to change,
+and the signature is expected to contain raw components (so not a PGP or
+PKCS#7 formatted blob).
+
+The signature blob is expected to consist of the following pieces in order:
+
+ (1) The binary identifier for the key. This is expected to match the
+ SubjectKeyIdentifier from an X.509 certificate. Only X.509 type
+ identifiers are currently supported.
+
+ (2) The signature data, consisting of a series of MPIs in which each is in
+ the format of a 2-byte BE word sizes followed by the content data.
+
+ (3) A 12 byte information block of the form:
+
+ struct module_signature {
+ enum pkey_algo algo : 8;
+ enum pkey_hash_algo hash : 8;
+ enum pkey_id_type id_type : 8;
+ u8 __pad;
+ __be32 id_length;
+ __be32 sig_length;
+ };
+
+ The three enums are defined in crypto/public_key.h.
+
+ 'algo' contains the public-key algorithm identifier (0->DSA, 1->RSA).
+
+ 'hash' contains the digest algorithm identifier (0->MD4, 1->MD5, 2->SHA1,
+ etc.).
+
+ 'id_type' contains the public-key identifier type (0->PGP, 1->X.509).
+
+ '__pad' should be 0.
+
+ 'id_length' should contain in the binary identifier length in BE form.
+
+ 'sig_length' should contain in the signature data length in BE form.
+
+ The lengths are in BE order rather than CPU order to make dealing with
+ cross-compilation easier.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ init/Kconfig | 8 ++
+ kernel/module_signing.c | 223 +++++++++++++++++++++++++++++++++++++++++++++++-
+ 2 files changed, 230 insertions(+), 1 deletion(-)
+
+diff --git a/init/Kconfig b/init/Kconfig
+index 00d4579..63fcbeb 100644
+--- a/init/Kconfig
++++ b/init/Kconfig
+@@ -1588,6 +1588,14 @@ config MODULE_SRCVERSION_ALL
+ config MODULE_SIG
+ bool "Module signature verification"
+ depends on MODULES
++ select CONFIG_KEYS
++ select CONFIG_CRYPTO
++ select ASYMMETRIC_KEY_TYPE
++ select ASYMMETRIC_PUBLIC_KEY_SUBTYPE
++ select PUBLIC_KEY_ALGO_RSA
++ select ASN1
++ select OID_REGISTRY
++ select X509_CERTIFICATE_PARSER
+ help
+ Check modules for valid signatures upon load: the signature
+ is simply appended to the module. For more information see
+diff --git a/kernel/module_signing.c b/kernel/module_signing.c
+index 0af10a5..83eb505 100644
+--- a/kernel/module_signing.c
++++ b/kernel/module_signing.c
+@@ -10,14 +10,235 @@
+ */
+
+ #include <linux/kernel.h>
++#include <linux/err.h>
++#include <crypto/public_key.h>
++#include <crypto/hash.h>
++#include <keys/asymmetric-type.h>
+ #include "module-internal.h"
+
+ /*
++ * Module signature information block.
++ *
++ * The constituents of the signature section are, in order:
++ *
++ * - Signer's name
++ * - Key identifier
++ * - Signature data
++ * - Information block
++ */
++struct module_signature {
++ enum pkey_algo algo : 8; /* Public-key crypto algorithm */
++ enum pkey_hash_algo hash : 8; /* Digest algorithm */
++ enum pkey_id_type id_type : 8; /* Key identifier type */
++ u8 signer_len; /* Length of signer's name */
++ u8 key_id_len; /* Length of key identifier */
++ u8 __pad[3];
++ __be32 sig_len; /* Length of signature data */
++};
++
++/*
++ * Digest the module contents.
++ */
++static struct public_key_signature *mod_make_digest(enum pkey_hash_algo hash,
++ const void *mod,
++ unsigned long modlen)
++{
++ struct public_key_signature *pks;
++ struct crypto_shash *tfm;
++ struct shash_desc *desc;
++ size_t digest_size, desc_size;
++ int ret;
++
++ pr_devel("==>%s()\n", __func__);
++
++ /* Allocate the hashing algorithm we're going to need and find out how
++ * big the hash operational data will be.
++ */
++ tfm = crypto_alloc_shash(pkey_hash_algo[hash], 0, 0);
++ if (IS_ERR(tfm))
++ return (PTR_ERR(tfm) == -ENOENT) ? ERR_PTR(-ENOPKG) : ERR_CAST(tfm);
++
++ desc_size = crypto_shash_descsize(tfm) + sizeof(*desc);
++ digest_size = crypto_shash_digestsize(tfm);
++
++ /* We allocate the hash operational data storage on the end of our
++ * context data and the digest output buffer on the end of that.
++ */
++ ret = -ENOMEM;
++ pks = kzalloc(digest_size + sizeof(*pks) + desc_size, GFP_KERNEL);
++ if (!pks)
++ goto error_no_pks;
++
++ pks->pkey_hash_algo = hash;
++ pks->digest = (u8 *)pks + sizeof(*pks) + desc_size;
++ pks->digest_size = digest_size;
++
++ desc = (void *)pks + sizeof(*pks);
++ desc->tfm = tfm;
++ desc->flags = CRYPTO_TFM_REQ_MAY_SLEEP;
++
++ ret = crypto_shash_init(desc);
++ if (ret < 0)
++ goto error;
++
++ ret = crypto_shash_finup(desc, mod, modlen, pks->digest);
++ if (ret < 0)
++ goto error;
++
++ crypto_free_shash(tfm);
++ pr_devel("<==%s() = ok\n", __func__);
++ return pks;
++
++error:
++ kfree(pks);
++error_no_pks:
++ crypto_free_shash(tfm);
++ pr_devel("<==%s() = %d\n", __func__, ret);
++ return ERR_PTR(ret);
++}
++
++/*
++ * Extract an MPI array from the signature data. This represents the actual
++ * signature. Each raw MPI is prefaced by a BE 2-byte value indicating the
++ * size of the MPI in bytes.
++ *
++ * RSA signatures only have one MPI, so currently we only read one.
++ */
++static int mod_extract_mpi_array(struct public_key_signature *pks,
++ const void *data, size_t len)
++{
++ size_t nbytes;
++ MPI mpi;
++
++ if (len < 3)
++ return -EBADMSG;
++ nbytes = ((const u8 *)data)[0] << 8 | ((const u8 *)data)[1];
++ data += 2;
++ len -= 2;
++ if (len != nbytes)
++ return -EBADMSG;
++
++ mpi = mpi_read_raw_data(data, nbytes);
++ if (!mpi)
++ return -ENOMEM;
++ pks->mpi[0] = mpi;
++ pks->nr_mpi = 1;
++ return 0;
++}
++
++/*
++ * Request an asymmetric key.
++ */
++static struct key *request_asymmetric_key(const char *signer, size_t signer_len,
++ const u8 *key_id, size_t key_id_len)
++{
++ key_ref_t key;
++ size_t i;
++ char *id, *q;
++
++ pr_devel("==>%s(,%zu,,%zu)\n", __func__, signer_len, key_id_len);
++
++ /* Construct an identifier. */
++ id = kmalloc(signer_len + 2 + key_id_len * 2 + 1, GFP_KERNEL);
++ if (!id)
++ return ERR_PTR(-ENOKEY);
++
++ memcpy(id, signer, signer_len);
++
++ q = id + signer_len;
++ *q++ = ':';
++ *q++ = ' ';
++ for (i = 0; i < key_id_len; i++) {
++ *q++ = hex_asc[*key_id >> 4];
++ *q++ = hex_asc[*key_id++ & 0x0f];
++ }
++
++ *q = 0;
++
++ pr_debug("Look up: \"%s\"\n", id);
++
++ key = keyring_search(make_key_ref(modsign_keyring, 1),
++ &key_type_asymmetric, id);
++ kfree(id);
++
++ if (IS_ERR(key)) {
++ switch (PTR_ERR(key)) {
++ /* Hide some search errors */
++ case -EACCES:
++ case -ENOTDIR:
++ case -EAGAIN:
++ return ERR_PTR(-ENOKEY);
++ default:
++ return ERR_CAST(key);
++ }
++ }
++
++ pr_devel("<==%s() = 0 [%x]\n", __func__, key_serial(key_ref_to_ptr(key)));
++ return key_ref_to_ptr(key);
++}
++
++/*
+ * Verify the signature on a module.
+ */
+ int mod_verify_sig(const void *mod, unsigned long modlen,
+ const void *sig, unsigned long siglen,
+ bool *_sig_ok)
+ {
+- return -EKEYREJECTED;
++ struct public_key_signature *pks;
++ struct module_signature ms;
++ struct key *key;
++ size_t sig_len;
++ int ret;
++
++ pr_devel("==>%s(,%lu,,%lu,)\n", __func__, modlen, siglen);
++
++ if (siglen <= sizeof(ms))
++ return -EBADMSG;
++
++ memcpy(&ms, sig + (siglen - sizeof(ms)), sizeof(ms));
++ siglen -= sizeof(ms);
++
++ sig_len = be32_to_cpu(ms.sig_len);
++ if (sig_len >= siglen ||
++ siglen - sig_len != (size_t)ms.signer_len + ms.key_id_len)
++ return -EBADMSG;
++
++ /* For the moment, only support RSA and X.509 identifiers */
++ if (ms.algo != PKEY_ALGO_RSA ||
++ ms.id_type != PKEY_ID_X509)
++ return -ENOPKG;
++
++ if (ms.hash >= PKEY_HASH__LAST ||
++ !pkey_hash_algo[ms.hash])
++ return -ENOPKG;
++
++ key = request_asymmetric_key(sig, ms.signer_len,
++ sig + ms.signer_len, ms.key_id_len);
++ if (IS_ERR(key))
++ return PTR_ERR(key);
++
++ pks = mod_make_digest(ms.hash, mod, modlen);
++ if (IS_ERR(pks)) {
++ ret = PTR_ERR(pks);
++ goto error_put_key;
++ }
++
++ ret = mod_extract_mpi_array(pks, sig + ms.signer_len + ms.key_id_len,
++ sig_len);
++ if (ret < 0)
++ goto error_free_pks;
++
++ ret = verify_signature(key, pks);
++ pr_devel("verify_signature() = %d\n", ret);
++
++ if (ret == 0)
++ *_sig_ok = true;
++
++error_free_pks:
++ mpi_free(pks->rsa.s);
++ kfree(pks);
++error_put_key:
++ key_put(key);
++ pr_devel("<==%s() = %d\n", __func__, ret);
++ return ret;
+ }
+--
+1.7.11.4
+
+
+From 89fbf1de73ed1ef29a3943d9f3bcf69a433191da Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 17:13:26 +0100
+Subject: [PATCH 24/26] MODSIGN: Provide a script for generating a key ID from
+ an X.509 cert
+
+Provide a script to parse an X.509 certificate and certain pieces of
+information from it in order to generate a key identifier to be included within
+a module signature.
+
+The script takes the Subject Name and extracts (if present) the
+organizationName (O), the commonName (CN) and the emailAddress and fabricates
+the signer's name from them:
+
+ (1) If both O and CN exist, then the name will be "O: CN", unless:
+
+ (a) CN is prefixed by O, in which case only CN is used.
+
+ (b) CN and O share at least the first 7 characters, in which case only CN
+ is used.
+
+ (2) Otherwise, CN is used if present.
+
+ (3) Otherwise, O is used if present.
+
+ (4) Otherwise the emailAddress is used, if present.
+
+ (5) Otherwise a blank name is used.
+
+The script emits a binary encoded identifier in the following form:
+
+ - 2 BE bytes indicating the length of the signer's name.
+
+ - 2 BE bytes indicating the length of the subject key identifier.
+
+ - The characters of the signer's name.
+
+ - The bytes of the subject key identifier.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ scripts/x509keyid | 268 ++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ 1 file changed, 268 insertions(+)
+ create mode 100755 scripts/x509keyid
+
+diff --git a/scripts/x509keyid b/scripts/x509keyid
+new file mode 100755
+index 0000000..c8e91a4
+--- /dev/null
++++ b/scripts/x509keyid
+@@ -0,0 +1,268 @@
++#!/usr/bin/perl -w
++#
++# Generate an identifier from an X.509 certificate that can be placed in a
++# module signature to indentify the key to use.
++#
++# Format:
++#
++# ./scripts/x509keyid <x509-cert> <signer's-name> <key-id>
++#
++# We read the DER-encoded X509 certificate and parse it to extract the Subject
++# name and Subject Key Identifier. The provide the data we need to build the
++# certificate identifier.
++#
++# The signer's name part of the identifier is fabricated from the commonName,
++# the organizationName or the emailAddress components of the X.509 subject
++# name and written to the second named file.
++#
++# The subject key ID to select which of that signer's certificates we're
++# intending to use to sign the module is written to the third named file.
++#
++use strict;
++
++my $raw_data;
++
++die "Need three filenames\n" if ($#ARGV != 2);
++
++my $src = $ARGV[0];
++
++open(FD, "<$src") || die $src;
++binmode FD;
++my @st = stat(FD);
++die $src if (!@st);
++read(FD, $raw_data, $st[7]) || die $src;
++close(FD);
++
++my $UNIV = 0 << 6;
++my $APPL = 1 << 6;
++my $CONT = 2 << 6;
++my $PRIV = 3 << 6;
++
++my $CONS = 0x20;
++
++my $BOOLEAN = 0x01;
++my $INTEGER = 0x02;
++my $BIT_STRING = 0x03;
++my $OCTET_STRING = 0x04;
++my $NULL = 0x05;
++my $OBJ_ID = 0x06;
++my $UTF8String = 0x0c;
++my $SEQUENCE = 0x10;
++my $SET = 0x11;
++my $UTCTime = 0x17;
++my $GeneralizedTime = 0x18;
++
++my %OIDs = (
++ pack("CCC", 85, 4, 3) => "commonName",
++ pack("CCC", 85, 4, 6) => "countryName",
++ pack("CCC", 85, 4, 10) => "organizationName",
++ pack("CCC", 85, 4, 11) => "organizationUnitName",
++ pack("CCCCCCCCC", 42, 134, 72, 134, 247, 13, 1, 1, 1) => "rsaEncryption",
++ pack("CCCCCCCCC", 42, 134, 72, 134, 247, 13, 1, 1, 5) => "sha1WithRSAEncryption",
++ pack("CCCCCCCCC", 42, 134, 72, 134, 247, 13, 1, 9, 1) => "emailAddress",
++ pack("CCC", 85, 29, 35) => "authorityKeyIdentifier",
++ pack("CCC", 85, 29, 14) => "subjectKeyIdentifier",
++ pack("CCC", 85, 29, 19) => "basicConstraints"
++);
++
++###############################################################################
++#
++# Extract an ASN.1 element from a string and return information about it.
++#
++###############################################################################
++sub asn1_extract($$@)
++{
++ my ($cursor, $expected_tag, $optional) = @_;
++
++ return [ -1 ]
++ if ($cursor->[1] == 0 && $optional);
++
++ die $src, ": ", $cursor->[0], ": ASN.1 data underrun (elem ", $cursor->[1], ")\n"
++ if ($cursor->[1] < 2);
++
++ my ($tag, $len) = unpack("CC", substr(${$cursor->[2]}, $cursor->[0], 2));
++
++ if ($expected_tag != -1 && $tag != $expected_tag) {
++ return [ -1 ]
++ if ($optional);
++ die $src, ": ", $cursor->[0], ": ASN.1 unexpected tag (", $tag,
++ " not ", $expected_tag, ")\n";
++ }
++
++ $cursor->[0] += 2;
++ $cursor->[1] -= 2;
++
++ die $src, ": ", $cursor->[0], ": ASN.1 long tag\n"
++ if (($tag & 0x1f) == 0x1f);
++ die $src, ": ", $cursor->[0], ": ASN.1 indefinite length\n"
++ if ($len == 0x80);
++
++ if ($len > 0x80) {
++ my $l = $len - 0x80;
++ die $src, ": ", $cursor->[0], ": ASN.1 data underrun (len len $l)\n"
++ if ($cursor->[1] < $l);
++
++ if ($l == 0x1) {
++ $len = unpack("C", substr(${$cursor->[2]}, $cursor->[0], 1));
++ } elsif ($l = 0x2) {
++ $len = unpack("n", substr(${$cursor->[2]}, $cursor->[0], 2));
++ } elsif ($l = 0x3) {
++ $len = unpack("C", substr(${$cursor->[2]}, $cursor->[0], 1)) << 16;
++ $len = unpack("n", substr(${$cursor->[2]}, $cursor->[0] + 1, 2));
++ } elsif ($l = 0x4) {
++ $len = unpack("N", substr(${$cursor->[2]}, $cursor->[0], 4));
++ } else {
++ die $src, ": ", $cursor->[0], ": ASN.1 element too long (", $l, ")\n";
++ }
++
++ $cursor->[0] += $l;
++ $cursor->[1] -= $l;
++ }
++
++ die $src, ": ", $cursor->[0], ": ASN.1 data underrun (", $len, ")\n"
++ if ($cursor->[1] < $len);
++
++ my $ret = [ $tag, [ $cursor->[0], $len, $cursor->[2] ] ];
++ $cursor->[0] += $len;
++ $cursor->[1] -= $len;
++
++ return $ret;
++}
++
++###############################################################################
++#
++# Retrieve the data referred to by a cursor
++#
++###############################################################################
++sub asn1_retrieve($)
++{
++ my ($cursor) = @_;
++ my ($offset, $len, $data) = @$cursor;
++ return substr($$data, $offset, $len);
++}
++
++###############################################################################
++#
++# Roughly parse the X.509 certificate
++#
++###############################################################################
++my $cursor = [ 0, length($raw_data), \$raw_data ];
++
++my $cert = asn1_extract($cursor, $UNIV | $CONS | $SEQUENCE);
++my $tbs = asn1_extract($cert->[1], $UNIV | $CONS | $SEQUENCE);
++my $version = asn1_extract($tbs->[1], $CONT | $CONS | 0, 1);
++my $serial_number = asn1_extract($tbs->[1], $UNIV | $INTEGER);
++my $sig_type = asn1_extract($tbs->[1], $UNIV | $CONS | $SEQUENCE);
++my $issuer = asn1_extract($tbs->[1], $UNIV | $CONS | $SEQUENCE);
++my $validity = asn1_extract($tbs->[1], $UNIV | $CONS | $SEQUENCE);
++my $subject = asn1_extract($tbs->[1], $UNIV | $CONS | $SEQUENCE);
++my $key = asn1_extract($tbs->[1], $UNIV | $CONS | $SEQUENCE);
++my $issuer_uid = asn1_extract($tbs->[1], $CONT | $CONS | 1, 1);
++my $subject_uid = asn1_extract($tbs->[1], $CONT | $CONS | 2, 1);
++my $extension_list = asn1_extract($tbs->[1], $CONT | $CONS | 3, 1);
++
++my $subject_key_id = ();
++my $authority_key_id = ();
++
++#
++# Parse the extension list
++#
++if ($extension_list->[0] != -1) {
++ my $extensions = asn1_extract($extension_list->[1], $UNIV | $CONS | $SEQUENCE);
++
++ while ($extensions->[1]->[1] > 0) {
++ my $ext = asn1_extract($extensions->[1], $UNIV | $CONS | $SEQUENCE);
++ my $x_oid = asn1_extract($ext->[1], $UNIV | $OBJ_ID);
++ my $x_crit = asn1_extract($ext->[1], $UNIV | $BOOLEAN, 1);
++ my $x_val = asn1_extract($ext->[1], $UNIV | $OCTET_STRING);
++
++ my $raw_oid = asn1_retrieve($x_oid->[1]);
++ next if (!exists($OIDs{$raw_oid}));
++ my $x_type = $OIDs{$raw_oid};
++
++ my $raw_value = asn1_retrieve($x_val->[1]);
++
++ if ($x_type eq "subjectKeyIdentifier") {
++ my $vcursor = [ 0, length($raw_value), \$raw_value ];
++
++ $subject_key_id = asn1_extract($vcursor, $UNIV | $OCTET_STRING);
++ }
++ }
++}
++
++###############################################################################
++#
++# Determine what we're going to use as the signer's name. In order of
++# preference, take one of: commonName, organizationName or emailAddress.
++#
++###############################################################################
++my $org = "";
++my $cn = "";
++my $email = "";
++
++while ($subject->[1]->[1] > 0) {
++ my $rdn = asn1_extract($subject->[1], $UNIV | $CONS | $SET);
++ my $attr = asn1_extract($rdn->[1], $UNIV | $CONS | $SEQUENCE);
++ my $n_oid = asn1_extract($attr->[1], $UNIV | $OBJ_ID);
++ my $n_val = asn1_extract($attr->[1], -1);
++
++ my $raw_oid = asn1_retrieve($n_oid->[1]);
++ next if (!exists($OIDs{$raw_oid}));
++ my $n_type = $OIDs{$raw_oid};
++
++ my $raw_value = asn1_retrieve($n_val->[1]);
++
++ if ($n_type eq "organizationName") {
++ $org = $raw_value;
++ } elsif ($n_type eq "commonName") {
++ $cn = $raw_value;
++ } elsif ($n_type eq "emailAddress") {
++ $email = $raw_value;
++ }
++}
++
++my $id_name = $email;
++
++if ($org && $cn) {
++ # Don't use the organizationName if the commonName repeats it
++ if (length($org) <= length($cn) &&
++ substr($cn, 0, length($org)) eq $org) {
++ $id_name = $cn;
++ goto got_id_name;
++ }
++
++ # Or a signifcant chunk of it
++ if (length($org) >= 7 &&
++ length($cn) >= 7 &&
++ substr($cn, 0, 7) eq substr($org, 0, 7)) {
++ $id_name = $cn;
++ goto got_id_name;
++ }
++
++ $id_name = $org . ": " . $cn;
++} elsif ($org) {
++ $id_name = $org;
++} elsif ($cn) {
++ $id_name = $cn;
++}
++
++got_id_name:
++
++###############################################################################
++#
++# Output the signer's name and the key identifier that we're going to include
++# in module signatures.
++#
++###############################################################################
++die $src, ": ", "X.509: Couldn't find the Subject Key Identifier extension\n"
++ if (!$subject_key_id);
++
++my $id_key_id = asn1_retrieve($subject_key_id->[1]);
++
++open(OUTFD, ">$ARGV[1]") || die $ARGV[1];
++print OUTFD $id_name;
++close OUTFD || die $ARGV[1];
++
++open(OUTFD, ">$ARGV[2]") || die $ARGV[2];
++print OUTFD $id_key_id;
++close OUTFD || die $ARGV[2];
+--
+1.7.11.4
+
+
+From 007c8fc3412f0a55cd3a32c5e42236703a17d1c1 Mon Sep 17 00:00:00 2001
+From: Josh Boyer <jwboyer@redhat.com>
+Date: Mon, 24 Sep 2012 10:46:36 -0400
+Subject: [PATCH 25/26] MODSIGN: Add modules_sign make target
+
+If CONFIG_MODULE_SIG is set, and 'make modules_sign' is called then this
+patch will cause the modules to get a signature installed. The make target
+is intended to be run after 'make modules_install', and will modify the
+modules in-place in the installed location.
+
+The signature will be appended to the module, along with some information
+about the signature size and a magic string that indicates the presence of
+the signature. This requires private and public keys to be available. By
+default these are expected to be found in files:
+
+ signing_key.priv
+ signing_key.x509
+
+in the base directory of the build. The first is the private key in PEM
+form and the second is the X.509 certificate in DER form as can be generated
+from openssl:
+
+ openssl req \
+ -new -x509 -outform PEM -out signing_key.x509 \
+ -keyout signing_key.priv -nodes \
+ -subj "/CN=H2G2/O=Magrathea/CN=Slartibartfast"
+
+If the secret key is not found then signing will be skipped and the unsigned
+module from (1) will just be copied to foo.ko.
+
+If signing occurs, lines like the following will be seen:
+
+ SIGN [M] <install path>/fs/foo/foo.ko
+
+will appear in the build log. If the signature step will be skipped and the
+following will be seen:
+
+ NO SIGN [M] <install path>/fs/foo/foo.ko
+
+NOTE! After the signature step, the signed module must not be passed through
+strip. If you wish to strip or otherwise modify the kernel modules, use the
+built-in stripping capabilities with 'make modules_install' or perform said
+modifications before calling this make target. This restriction may affect
+packaging tools (such as rpmbuild) and initramfs composition tools.
+
+Based heavily on work by: David Howells <dhowells@redhat.com>
+Signed-off-by: Josh Boyer <jwboyer@redhat.com>
+---
+ Makefile | 6 +++
+ scripts/Makefile.modsign | 72 +++++++++++++++++++++++++++++
+ scripts/sign-file | 115 +++++++++++++++++++++++++++++++++++++++++++++++
+ 3 files changed, 193 insertions(+)
+ create mode 100644 scripts/Makefile.modsign
+ create mode 100644 scripts/sign-file
+
+diff --git a/Makefile b/Makefile
+index 644048d..4479147 100644
+--- a/Makefile
++++ b/Makefile
+@@ -965,6 +965,12 @@ _modinst_post: _modinst_
+ $(Q)$(MAKE) -f $(srctree)/scripts/Makefile.fwinst obj=firmware __fw_modinst
+ $(call cmd,depmod)
+
++ifeq ($(CONFIG_MODULE_SIG), y)
++PHONY += modules_sign
++modules_sign:
++ $(Q)$(MAKE) -f $(srctree)/scripts/Makefile.modsign
++endif
++
+ else # CONFIG_MODULES
+
+ # Modules not configured
+diff --git a/scripts/Makefile.modsign b/scripts/Makefile.modsign
+new file mode 100644
+index 0000000..17326bc
+--- /dev/null
++++ b/scripts/Makefile.modsign
+@@ -0,0 +1,72 @@
++# ==========================================================================
++# Signing modules
++# ==========================================================================
++
++PHONY := __modsign
++__modsign:
++
++include scripts/Kbuild.include
++
++__modules := $(sort $(shell grep -h '\.ko' /dev/null $(wildcard $(MODVERDIR)/*.mod)))
++modules := $(patsubst %.o,%.ko,$(wildcard $(__modules:.ko=.o)))
++
++PHONY += $(modules)
++__modsign: $(modules)
++ @:
++
++MODSECKEY = ./signing_key.priv
++MODPUBKEY = ./signing_key.x509
++
++ifeq ($(wildcard $(MODSECKEY))+$(wildcard $(MODPUBKEY)),$(MODSECKEY)+$(MODPUBKEY))
++ifeq ($(KBUILD_SRC),)
++ # no O= is being used
++ SCRIPTS_DIR := scripts
++else
++ SCRIPTS_DIR := $(KBUILD_SRC)/scripts
++endif
++SIGN_MODULES := 1
++else
++SIGN_MODULES := 0
++endif
++
++# only sign if it's an in-tree module
++ifneq ($(KBUILD_EXTMOD),)
++SIGN_MODULES := 0
++endif
++
++ifeq ($(SIGN_MODULES),1)
++
++quiet_cmd_genkeyid = GENKEYID $@
++ cmd_genkeyid = \
++ perl $(SCRIPTS_DIR)/x509keyid $< $<.signer $<.keyid
++
++%.signer %.keyid: %
++ $(call if_changed,genkeyid)
++
++KEYRING_DEP := $(MODSECKEY) $(MODPUBKEY) $(MODPUBKEY).signer $(MODPUBKEY).keyid
++quiet_cmd_sign_ko = SIGN [M] $(2)/$(notdir $@)
++ cmd_sign_ko = \
++ sh $(SCRIPTS_DIR)/sign-file $(MODSECKEY) $(MODPUBKEY) \
++ $(2)/$(notdir $@) $(2)/$(notdir $@).signed && \
++ mv $(2)/$(notdir $@).signed $(2)/$(notdir $@) && \
++ rm -rf $(2)/$(notdir $@).{dig,sig}
++else
++KEYRING_DEP :=
++quiet_cmd_sign_ko = NO SIGN [M] $@
++ cmd_sign_ko = \
++ true
++endif
++
++# Modules built outside the kernel source tree go into extra by default
++INSTALL_MOD_DIR ?= extra
++ext-mod-dir = $(INSTALL_MOD_DIR)$(subst $(patsubst %/,%,$(KBUILD_EXTMOD)),,$(@D))
++
++modinst_dir = $(if $(KBUILD_EXTMOD),$(ext-mod-dir),kernel/$(@D))
++
++$(modules): $(KEYRING_DEP)
++ $(call cmd,sign_ko,$(MODLIB)/$(modinst_dir))
++
++# Declare the contents of the .PHONY variable as phony. We keep that
++# # information in a variable se we can use it in if_changed and friends.
++
++.PHONY: $(PHONY)
+diff --git a/scripts/sign-file b/scripts/sign-file
+new file mode 100644
+index 0000000..1a472bb
+--- /dev/null
++++ b/scripts/sign-file
+@@ -0,0 +1,115 @@
++#!/bin/sh
++#
++# Sign a module file using the given key.
++#
++# Format: sign-file <key> <x509> <src-file> <dst-file>
++#
++
++scripts=`dirname $0`
++
++CONFIG_MODULE_SIG_SHA512=y
++if [ -r .config ]
++then
++ source ./.config
++fi
++
++key="$1"
++x509="$2"
++src="$3"
++dst="$4"
++
++if [ ! -r "$key" ]
++then
++ echo "Can't read private key" >&2
++ exit 2
++fi
++
++if [ ! -r "$x509" ]
++then
++ echo "Can't read X.509 certificate" >&2
++ exit 2
++fi
++if [ ! -r "$x509.signer" ]
++then
++ echo "Can't read Signer name" >&2
++ exit 2;
++fi
++if [ ! -r "$x509.keyid" ]
++then
++ echo "Can't read Key identifier" >&2
++ exit 2;
++fi
++
++#
++# Signature parameters
++#
++algo=1 # Public-key crypto algorithm: RSA
++hash= # Digest algorithm
++id_type=1 # Identifier type: X.509
++
++#
++# Digest the data
++#
++dgst=
++if [ "$CONFIG_MODULE_SIG_SHA1" = "y" ]
++then
++ prologue="0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2B, 0x0E, 0x03, 0x02, 0x1A, 0x05, 0x00, 0x04, 0x14"
++ dgst=-sha1
++ hash=2
++elif [ "$CONFIG_MODULE_SIG_SHA224" = "y" ]
++then
++ prologue="0x30, 0x2d, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x04, 0x05, 0x00, 0x04, 0x1C"
++ dgst=-sha224
++ hash=7
++elif [ "$CONFIG_MODULE_SIG_SHA256" = "y" ]
++then
++ prologue="0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01, 0x05, 0x00, 0x04, 0x20"
++ dgst=-sha256
++ hash=4
++elif [ "$CONFIG_MODULE_SIG_SHA384" = "y" ]
++then
++ prologue="0x30, 0x41, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x02, 0x05, 0x00, 0x04, 0x30"
++ dgst=-sha384
++ hash=5
++elif [ "$CONFIG_MODULE_SIG_SHA512" = "y" ]
++then
++ prologue="0x30, 0x51, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x03, 0x05, 0x00, 0x04, 0x40"
++ dgst=-sha512
++ hash=6
++else
++ echo "$0: Can't determine hash algorithm" >&2
++ exit 2
++fi
++
++(
++perl -e "binmode STDOUT; print pack(\"C*\", $prologue)" || exit $?
++openssl dgst $dgst -binary $src || exit $?
++) >$src.dig || exit $?
++
++#
++# Generate the binary signature, which will be just the integer that comprises
++# the signature with no metadata attached.
++#
++openssl rsautl -sign -inkey $key -keyform PEM -in $src.dig -out $src.sig || exit $?
++signerlen=`stat -c %s $x509.signer`
++keyidlen=`stat -c %s $x509.keyid`
++siglen=`stat -c %s $src.sig`
++
++#
++# Build the signed binary
++#
++(
++ cat $src || exit $?
++ echo '~Module signature appended~' || exit $?
++ cat $x509.signer $x509.keyid || exit $?
++
++ # Preface each signature integer with a 2-byte BE length
++ perl -e "binmode STDOUT; print pack(\"n\", $siglen)" || exit $?
++ cat $src.sig || exit $?
++
++ # Generate the information block
++ perl -e "binmode STDOUT; print pack(\"CCCCCxxxN\", $algo, $hash, $id_type, $signerlen, $keyidlen, $siglen + 2)" || exit $?
++) >$dst~ || exit $?
++
++# Permit in-place signing
++mv $dst~ $dst || exit $?
+--
+1.7.11.4
+
+
+From 33c6737b352ec10a7e0d7b053fbb084c0b7a9d36 Mon Sep 17 00:00:00 2001
+From: David Howells <dhowells@redhat.com>
+Date: Mon, 24 Sep 2012 20:51:59 +0100
+Subject: [PATCH 26/26] MODSIGN: Extend the policy on signature check failure
+
+Extend the policy on handling various sorts of signature check failure such as
+not having the requisite key available or being in FIPS mode.
+
+If the key specified is not known (ENOKEY) permit the module to be loaded in
+non-enforcing mode, otherwise reject it.
+
+If in FIPS mode, any loading failure shall cause a panic.
+
+Also print a warning if we try and fail to find a module signing key:
+
+ Request for unknown module key 'Magrathea: Glacier signing key: 5dd0839552bd6af498253f8af1e65da3472941c6' err -11
+
+This contains the identity field and the key ID from the signature as well as
+the error code. The error codes are the raw return from keyring_search() and
+may be translated to -ENOKEY.
+
+Signed-off-by: David Howells <dhowells@redhat.com>
+---
+ kernel/Makefile | 2 +-
+ kernel/module.c | 9 ++++++++-
+ kernel/module_signing.c | 3 +++
+ 3 files changed, 12 insertions(+), 2 deletions(-)
+
+diff --git a/kernel/Makefile b/kernel/Makefile
+index 63f8386..111a845 100644
+--- a/kernel/Makefile
++++ b/kernel/Makefile
+@@ -177,7 +177,7 @@ x509.genkey:
+ @echo >>x509.genkey "x509_extensions = myexts"
+ @echo >>x509.genkey
+ @echo >>x509.genkey "[ req_distinguished_name ]"
+- @echo >>x509.genkey "O = Magarathea"
++ @echo >>x509.genkey "O = Magrathea"
+ @echo >>x509.genkey "CN = Glacier signing key"
+ @echo >>x509.genkey "emailAddress = slartibartfast@magrathea.h2g2"
+ @echo >>x509.genkey
+diff --git a/kernel/module.c b/kernel/module.c
+index ab69599..de16959 100644
+--- a/kernel/module.c
++++ b/kernel/module.c
+@@ -58,6 +58,7 @@
+ #include <linux/jump_label.h>
+ #include <linux/pfn.h>
+ #include <linux/bsearch.h>
++#include <linux/fips.h>
+ #include "module-internal.h"
+
+ #define CREATE_TRACE_POINTS
+@@ -2470,7 +2471,13 @@ found_marker:
+ sig = p + markerlen;
+ /* Truncate module up to signature. */
+ *len = p - mod;
+- return mod_verify_sig(mod, *len, sig, end - sig, &info->sig_ok);
++ err = mod_verify_sig(mod, *len, sig, end - sig, &info->sig_ok);
++ if (err < 0 && fips_enabled)
++ panic("Module verification failed with error %d in FIPS mode\n",
++ err);
++ if (err == -ENOKEY && !sig_enforce)
++ err = 0;
++ return err;
+ }
+ #else /* !CONFIG_MODULE_SIG */
+ static int module_sig_check(struct load_info *info,
+diff --git a/kernel/module_signing.c b/kernel/module_signing.c
+index 83eb505..2beea56 100644
+--- a/kernel/module_signing.c
++++ b/kernel/module_signing.c
+@@ -159,6 +159,9 @@ static struct key *request_asymmetric_key(const char *signer, size_t signer_len,
+
+ key = keyring_search(make_key_ref(modsign_keyring, 1),
+ &key_type_asymmetric, id);
++ if (IS_ERR(key))
++ pr_warn("Request for unknown module key '%s' err %ld\n",
++ id, PTR_ERR(key));
+ kfree(id);
+
+ if (IS_ERR(key)) {
+--
+1.7.11.4
+