Commit b3e2fd85 authored by David Howells's avatar David Howells

KEYS: Add facility to check key trustworthiness upon link creation

Add a facility whereby if KEY_FLAG_TRUSTED_ONLY is set on the destination
keyring, the creation of a link to a candidate key will cause the
trustworthiness of that key to be evaluated against the already present
contents of that keyring.  This affects operations like add_key(),
KEYCTL_LINK and KEYCTL_INSTANTIATE.

To this end:

 (1) A new key type method is provided:

	int (*verify_trust)(const union key_payload *payload,
			    struct key *keyring);

     This is implemented by key types for which verification of one key by
     another is appropriate.  It is primarily intended for use with the
     asymmetric key type.

     When called, it is given the payload or prospective payload[*] of the
     candidate key to verify and a pointer to the destination keyring.  The
     method is expected to search the keying for an appropriate key with
     which to verify the candidate.

     [*] If called during add_key(), preparse is called before this method,
     	 but a key isn't actually allocated unless the verification is
     	 successful.

 (2) KEY_FLAG_TRUSTED is removed.  A key is now trusted by virtue of being
     contained in the trusted-only keyring being searched.

 (3) KEY_ALLOC_TRUSTED now acts as an override.  If this is passed to
     key_create_or_update() then the ->verify_trust() method will be
     ignored and the key will be added anyway.
Signed-off-by: default avatarDavid Howells <dhowells@redhat.com>
parent be4dc974
...@@ -1183,6 +1183,23 @@ The structure has a number of fields, some of which are mandatory: ...@@ -1183,6 +1183,23 @@ The structure has a number of fields, some of which are mandatory:
successfully, even if instantiate() or update() succeed. successfully, even if instantiate() or update() succeed.
(*) int (*verify_trust)(const union key_payload *payload, struct key *keyring);
If the keyring to which a candidate key is being added/linked is marked as
KEY_FLAG_TRUSTED_ONLY then this function will get called in the candidate
key type to verify the key or proposed key based on its payload. It is
expected to use the contents of the supplied destination keyring to
determine whether the candidate key is to be trusted and added to the
keyring.
The method should return 0 to allow the addition and an error otherwise,
typically ENOKEY if there's no key in the keyring to verify this key and
EKEYREJECTED if the selected key fails to verify the candidate.
This method is optional. If it is not supplied, keys of this type cannot
be added to trusted-only keyrings and EPERM will be returned.
(*) int (*instantiate)(struct key *key, struct key_preparsed_payload *prep); (*) int (*instantiate)(struct key *key, struct key_preparsed_payload *prep);
This method is called to attach a payload to a key during construction. This method is called to attach a payload to a key during construction.
......
...@@ -318,10 +318,10 @@ static int x509_key_preparse(struct key_preparsed_payload *prep) ...@@ -318,10 +318,10 @@ static int x509_key_preparse(struct key_preparsed_payload *prep)
ret = x509_check_signature(cert->pub, cert); /* self-signed */ ret = x509_check_signature(cert->pub, cert); /* self-signed */
if (ret < 0) if (ret < 0)
goto error_free_cert; goto error_free_cert;
} else if (!prep->trusted) { } else {
ret = x509_validate_trust(cert, get_system_trusted_keyring()); ret = x509_validate_trust(cert, get_system_trusted_keyring());
if (!ret) if (ret == -EKEYREJECTED)
prep->trusted = 1; goto error_free_cert;
} }
/* Propose a description */ /* Propose a description */
......
...@@ -45,7 +45,6 @@ struct key_preparsed_payload { ...@@ -45,7 +45,6 @@ struct key_preparsed_payload {
size_t datalen; /* Raw datalen */ size_t datalen; /* Raw datalen */
size_t quotalen; /* Quota length for proposed payload */ size_t quotalen; /* Quota length for proposed payload */
time_t expiry; /* Expiry time of key */ time_t expiry; /* Expiry time of key */
bool trusted; /* True if key is trusted */
}; };
typedef int (*request_key_actor_t)(struct key_construction *key, typedef int (*request_key_actor_t)(struct key_construction *key,
...@@ -95,6 +94,15 @@ struct key_type { ...@@ -95,6 +94,15 @@ struct key_type {
*/ */
void (*free_preparse)(struct key_preparsed_payload *prep); void (*free_preparse)(struct key_preparsed_payload *prep);
/* Verify the trust on a key when added to a trusted-only keyring.
*
* If this method isn't provided then it is assumed that the concept of
* trust is irrelevant to keys of this type and an attempt to add one
* to a trusted-only keyring will be rejected.
*/
int (*verify_trust)(const union key_payload *payload,
struct key *keyring);
/* instantiate a key of this type /* instantiate a key of this type
* - this method should call key_payload_reserve() to determine if the * - this method should call key_payload_reserve() to determine if the
* user's quota will hold the payload * user's quota will hold the payload
......
...@@ -173,10 +173,12 @@ struct key { ...@@ -173,10 +173,12 @@ struct key {
#define KEY_FLAG_NEGATIVE 5 /* set if key is negative */ #define KEY_FLAG_NEGATIVE 5 /* set if key is negative */
#define KEY_FLAG_ROOT_CAN_CLEAR 6 /* set if key can be cleared by root without permission */ #define KEY_FLAG_ROOT_CAN_CLEAR 6 /* set if key can be cleared by root without permission */
#define KEY_FLAG_INVALIDATED 7 /* set if key has been invalidated */ #define KEY_FLAG_INVALIDATED 7 /* set if key has been invalidated */
#define KEY_FLAG_TRUSTED 8 /* set if key is trusted */ #define KEY_FLAG_TRUSTED_ONLY 8 /* set to make keyring only accept links to keys that can
#define KEY_FLAG_TRUSTED_ONLY 9 /* set if keyring only accepts links to trusted keys */ * be verified by one of the keys it already contains or
#define KEY_FLAG_BUILTIN 10 /* set if key is builtin */ * if KEY_ALLOC_TRUSTED is flagged.
#define KEY_FLAG_ROOT_CAN_INVAL 11 /* set if key can be invalidated by root without permission */ */
#define KEY_FLAG_BUILTIN 9 /* set if key is builtin */
#define KEY_FLAG_ROOT_CAN_INVAL 10 /* set if key can be invalidated by root without permission */
/* the key type and key description string /* the key type and key description string
* - the desc is used to match a key against search criteria * - the desc is used to match a key against search criteria
...@@ -217,7 +219,7 @@ extern struct key *key_alloc(struct key_type *type, ...@@ -217,7 +219,7 @@ extern struct key *key_alloc(struct key_type *type,
#define KEY_ALLOC_IN_QUOTA 0x0000 /* add to quota, reject if would overrun */ #define KEY_ALLOC_IN_QUOTA 0x0000 /* add to quota, reject if would overrun */
#define KEY_ALLOC_QUOTA_OVERRUN 0x0001 /* add to quota, permit even if overrun */ #define KEY_ALLOC_QUOTA_OVERRUN 0x0001 /* add to quota, permit even if overrun */
#define KEY_ALLOC_NOT_IN_QUOTA 0x0002 /* not in quota */ #define KEY_ALLOC_NOT_IN_QUOTA 0x0002 /* not in quota */
#define KEY_ALLOC_TRUSTED 0x0004 /* Key should be flagged as trusted */ #define KEY_ALLOC_TRUSTED 0x0004 /* Override the verification check on trusted keyrings */
extern void key_revoke(struct key *key); extern void key_revoke(struct key *key);
extern void key_invalidate(struct key *key); extern void key_invalidate(struct key *key);
......
...@@ -294,8 +294,6 @@ struct key *key_alloc(struct key_type *type, const char *desc, ...@@ -294,8 +294,6 @@ struct key *key_alloc(struct key_type *type, const char *desc,
if (!(flags & KEY_ALLOC_NOT_IN_QUOTA)) if (!(flags & KEY_ALLOC_NOT_IN_QUOTA))
key->flags |= 1 << KEY_FLAG_IN_QUOTA; key->flags |= 1 << KEY_FLAG_IN_QUOTA;
if (flags & KEY_ALLOC_TRUSTED)
key->flags |= 1 << KEY_FLAG_TRUSTED;
#ifdef KEY_DEBUGGING #ifdef KEY_DEBUGGING
key->magic = KEY_DEBUG_MAGIC; key->magic = KEY_DEBUG_MAGIC;
...@@ -478,6 +476,11 @@ int key_instantiate_and_link(struct key *key, ...@@ -478,6 +476,11 @@ int key_instantiate_and_link(struct key *key,
struct assoc_array_edit *edit; struct assoc_array_edit *edit;
int ret; int ret;
if (keyring &&
unlikely(test_bit(KEY_FLAG_TRUSTED_ONLY, &keyring->flags)) &&
!key->type->verify_trust)
return -EPERM;
memset(&prep, 0, sizeof(prep)); memset(&prep, 0, sizeof(prep));
prep.data = data; prep.data = data;
prep.datalen = datalen; prep.datalen = datalen;
...@@ -490,6 +493,13 @@ int key_instantiate_and_link(struct key *key, ...@@ -490,6 +493,13 @@ int key_instantiate_and_link(struct key *key,
} }
if (keyring) { if (keyring) {
if (unlikely(test_bit(KEY_FLAG_TRUSTED_ONLY,
&keyring->flags)) &&
key->type->verify_trust) {
ret = key->type->verify_trust(&prep.payload, keyring);
if (ret < 0)
goto error;
}
ret = __key_link_begin(keyring, &key->index_key, &edit); ret = __key_link_begin(keyring, &key->index_key, &edit);
if (ret < 0) if (ret < 0)
goto error; goto error;
...@@ -545,8 +555,12 @@ int key_reject_and_link(struct key *key, ...@@ -545,8 +555,12 @@ int key_reject_and_link(struct key *key,
awaken = 0; awaken = 0;
ret = -EBUSY; ret = -EBUSY;
if (keyring) if (keyring) {
if (unlikely(test_bit(KEY_FLAG_TRUSTED_ONLY, &keyring->flags)))
return -EPERM;
link_ret = __key_link_begin(keyring, &key->index_key, &edit); link_ret = __key_link_begin(keyring, &key->index_key, &edit);
}
mutex_lock(&key_construction_mutex); mutex_lock(&key_construction_mutex);
...@@ -786,6 +800,7 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref, ...@@ -786,6 +800,7 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
const struct cred *cred = current_cred(); const struct cred *cred = current_cred();
struct key *keyring, *key = NULL; struct key *keyring, *key = NULL;
key_ref_t key_ref; key_ref_t key_ref;
bool verify_trust;
int ret; int ret;
/* look up the key type to see if it's one of the registered kernel /* look up the key type to see if it's one of the registered kernel
...@@ -802,9 +817,18 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref, ...@@ -802,9 +817,18 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
goto error_put_type; goto error_put_type;
keyring = key_ref_to_ptr(keyring_ref); keyring = key_ref_to_ptr(keyring_ref);
key_check(keyring); key_check(keyring);
key_ref = ERR_PTR(-EPERM);
if (!test_bit(KEY_FLAG_TRUSTED_ONLY, &keyring->flags))
verify_trust = false;
else if (flags & KEY_ALLOC_TRUSTED)
verify_trust = false;
else if (index_key.type->verify_trust)
verify_trust = true;
else
goto error_put_type;
key_ref = ERR_PTR(-ENOTDIR); key_ref = ERR_PTR(-ENOTDIR);
if (keyring->type != &key_type_keyring) if (keyring->type != &key_type_keyring)
goto error_put_type; goto error_put_type;
...@@ -813,7 +837,6 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref, ...@@ -813,7 +837,6 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
prep.data = payload; prep.data = payload;
prep.datalen = plen; prep.datalen = plen;
prep.quotalen = index_key.type->def_datalen; prep.quotalen = index_key.type->def_datalen;
prep.trusted = flags & KEY_ALLOC_TRUSTED;
prep.expiry = TIME_T_MAX; prep.expiry = TIME_T_MAX;
if (index_key.type->preparse) { if (index_key.type->preparse) {
ret = index_key.type->preparse(&prep); ret = index_key.type->preparse(&prep);
...@@ -829,10 +852,13 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref, ...@@ -829,10 +852,13 @@ key_ref_t key_create_or_update(key_ref_t keyring_ref,
} }
index_key.desc_len = strlen(index_key.description); index_key.desc_len = strlen(index_key.description);
key_ref = ERR_PTR(-EPERM); if (verify_trust) {
if (!prep.trusted && test_bit(KEY_FLAG_TRUSTED_ONLY, &keyring->flags)) ret = index_key.type->verify_trust(&prep.payload, keyring);
if (ret < 0) {
key_ref = ERR_PTR(ret);
goto error_free_prep; goto error_free_prep;
flags |= prep.trusted ? KEY_ALLOC_TRUSTED : 0; }
}
ret = __key_link_begin(keyring, &index_key, &edit); ret = __key_link_begin(keyring, &index_key, &edit);
if (ret < 0) { if (ret < 0) {
......
...@@ -1191,6 +1191,18 @@ void __key_link_end(struct key *keyring, ...@@ -1191,6 +1191,18 @@ void __key_link_end(struct key *keyring,
up_write(&keyring->sem); up_write(&keyring->sem);
} }
/*
* Verify a trusted-only keyring link.
*/
static int __key_link_verify(struct key *keyring, struct key *key)
{
if (!test_bit(KEY_FLAG_TRUSTED_ONLY, &keyring->flags))
return 0;
if (!key->type->verify_trust)
return -EPERM;
return key->type->verify_trust(&key->payload, keyring);
}
/** /**
* key_link - Link a key to a keyring * key_link - Link a key to a keyring
* @keyring: The keyring to make the link in. * @keyring: The keyring to make the link in.
...@@ -1222,12 +1234,14 @@ int key_link(struct key *keyring, struct key *key) ...@@ -1222,12 +1234,14 @@ int key_link(struct key *keyring, struct key *key)
key_check(key); key_check(key);
if (test_bit(KEY_FLAG_TRUSTED_ONLY, &keyring->flags) && if (test_bit(KEY_FLAG_TRUSTED_ONLY, &keyring->flags) &&
!test_bit(KEY_FLAG_TRUSTED, &key->flags)) !key->type->verify_trust)
return -EPERM; return -EPERM;
ret = __key_link_begin(keyring, &key->index_key, &edit); ret = __key_link_begin(keyring, &key->index_key, &edit);
if (ret == 0) { if (ret == 0) {
kdebug("begun {%d,%d}", keyring->serial, atomic_read(&keyring->usage)); kdebug("begun {%d,%d}", keyring->serial, atomic_read(&keyring->usage));
ret = __key_link_verify(keyring, key);
if (ret == 0)
ret = __key_link_check_live_key(keyring, key); ret = __key_link_check_live_key(keyring, key);
if (ret == 0) if (ret == 0)
__key_link(key, &edit); __key_link(key, &edit);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment