Skip to content

Commit

Permalink
PKCS#7: Check codeSigning EKU for kernel module and kexec pe verifica…
Browse files Browse the repository at this point in the history
…tion

This patch adds the logic for checking the CodeSigning extended
key usage when verifying signature of kernel module or
kexec PE binary in PKCS#7.

Signed-off-by: "Lee, Chun-Yi" <jlee@suse.com>
  • Loading branch information
Lee, Chun-Yi authored and intel-lab-lkp committed May 22, 2021
1 parent a5934a4 commit 9285555
Show file tree
Hide file tree
Showing 6 changed files with 63 additions and 10 deletions.
6 changes: 4 additions & 2 deletions certs/blacklist.c
Expand Up @@ -16,6 +16,7 @@
#include <linux/seq_file.h>
#include <linux/uidgid.h>
#include <keys/system_keyring.h>
#include <keys/asymmetric-type.h>
#include "blacklist.h"
#include "common.h"

Expand Down Expand Up @@ -181,11 +182,12 @@ int add_key_to_revocation_list(const char *data, size_t size)
* is_key_on_revocation_list - Determine if the key for a PKCS#7 message is revoked
* @pkcs7: The PKCS#7 message to check
*/
int is_key_on_revocation_list(struct pkcs7_message *pkcs7)
int is_key_on_revocation_list(struct pkcs7_message *pkcs7,
enum key_being_used_for usage)
{
int ret;

ret = pkcs7_validate_trust(pkcs7, blacklist_keyring);
ret = pkcs7_validate_trust(pkcs7, blacklist_keyring, usage, false);

if (ret == 0)
return -EKEYREJECTED;
Expand Down
4 changes: 2 additions & 2 deletions certs/system_keyring.c
Expand Up @@ -221,13 +221,13 @@ int verify_pkcs7_message_sig(const void *data, size_t len,
goto error;
}

ret = is_key_on_revocation_list(pkcs7);
ret = is_key_on_revocation_list(pkcs7, usage);
if (ret != -ENOKEY) {
pr_devel("PKCS#7 platform key is on revocation list\n");
goto error;
}
}
ret = pkcs7_validate_trust(pkcs7, trusted_keys);
ret = pkcs7_validate_trust(pkcs7, trusted_keys, usage, true);
if (ret < 0) {
if (ret == -ENOKEY)
pr_devel("PKCS#7 signature not signed with a trusted key\n");
Expand Down
9 changes: 9 additions & 0 deletions crypto/asymmetric_keys/Kconfig
Expand Up @@ -96,4 +96,13 @@ config SIGNED_PE_FILE_VERIFICATION
This option provides support for verifying the signature(s) on a
signed PE binary.

config CHECK_CODESIGN_EKU
bool "Check codeSigning extended key usage"
depends on PKCS7_MESSAGE_PARSER=y
depends on SYSTEM_DATA_VERIFICATION
help
This option provides support for checking the codeSigning extended
key usage when verifying the signature in PKCS#7. It affects kernel
module verification and kexec PE binary verification.

endif # ASYMMETRIC_KEY_TYPE
43 changes: 40 additions & 3 deletions crypto/asymmetric_keys/pkcs7_trust.c
Expand Up @@ -16,12 +16,40 @@
#include <crypto/public_key.h>
#include "pkcs7_parser.h"

#ifdef CONFIG_CHECK_CODESIGN_EKU
static bool check_eku_by_usage(struct key *key, enum key_being_used_for usage)
{
struct public_key *public_key = key->payload.data[asym_crypto];
bool ret = true;

switch (usage) {
case VERIFYING_MODULE_SIGNATURE:
case VERIFYING_KEXEC_PE_SIGNATURE:
ret = !!(public_key->eku & EKU_codeSigning);
if (!ret)
pr_warn("The signer '%s' key is not CodeSigning\n",
key->description);
break;
default:
break;
}
return ret;
}
#else
static bool check_eku_by_usage(struct key *key, enum key_being_used_for usage)
{
return true;
}
#endif

/*
* Check the trust on one PKCS#7 SignedInfo block.
*/
static int pkcs7_validate_trust_one(struct pkcs7_message *pkcs7,
struct pkcs7_signed_info *sinfo,
struct key *trust_keyring)
struct key *trust_keyring,
enum key_being_used_for usage,
bool check_eku)
{
struct public_key_signature *sig = sinfo->sig;
struct x509_certificate *x509, *last = NULL, *p;
Expand Down Expand Up @@ -112,6 +140,10 @@ static int pkcs7_validate_trust_one(struct pkcs7_message *pkcs7,
return -ENOKEY;

matched:
if (check_eku && !check_eku_by_usage(key, usage)) {
key_put(key);
return -ENOKEY;
}
ret = verify_signature(key, sig);
key_put(key);
if (ret < 0) {
Expand All @@ -135,6 +167,8 @@ static int pkcs7_validate_trust_one(struct pkcs7_message *pkcs7,
* pkcs7_validate_trust - Validate PKCS#7 trust chain
* @pkcs7: The PKCS#7 certificate to validate
* @trust_keyring: Signing certificates to use as starting points
* @usage: The use to which the key is being put.
* @check_eku: Check EKU (Extended Key Usage)
*
* Validate that the certificate chain inside the PKCS#7 message intersects
* keys we already know and trust.
Expand All @@ -156,7 +190,9 @@ static int pkcs7_validate_trust_one(struct pkcs7_message *pkcs7,
* May also return -ENOMEM.
*/
int pkcs7_validate_trust(struct pkcs7_message *pkcs7,
struct key *trust_keyring)
struct key *trust_keyring,
enum key_being_used_for usage,
bool check_eku)
{
struct pkcs7_signed_info *sinfo;
struct x509_certificate *p;
Expand All @@ -167,7 +203,8 @@ int pkcs7_validate_trust(struct pkcs7_message *pkcs7,
p->seen = false;

for (sinfo = pkcs7->signed_infos; sinfo; sinfo = sinfo->next) {
ret = pkcs7_validate_trust_one(pkcs7, sinfo, trust_keyring);
ret = pkcs7_validate_trust_one(pkcs7, sinfo, trust_keyring,
usage, check_eku);
switch (ret) {
case -ENOKEY:
continue;
Expand Down
4 changes: 3 additions & 1 deletion include/crypto/pkcs7.h
Expand Up @@ -30,7 +30,9 @@ extern int pkcs7_get_content_data(const struct pkcs7_message *pkcs7,
* pkcs7_trust.c
*/
extern int pkcs7_validate_trust(struct pkcs7_message *pkcs7,
struct key *trust_keyring);
struct key *trust_keyring,
enum key_being_used_for usage,
bool check_eku);

/*
* pkcs7_verify.c
Expand Down
7 changes: 5 additions & 2 deletions include/keys/system_keyring.h
Expand Up @@ -9,6 +9,7 @@
#define _KEYS_SYSTEM_KEYRING_H

#include <linux/key.h>
#include <keys/asymmetric-type.h>

#ifdef CONFIG_SYSTEM_TRUSTED_KEYRING

Expand Down Expand Up @@ -59,13 +60,15 @@ static inline int is_binary_blacklisted(const u8 *hash, size_t hash_len)

#ifdef CONFIG_SYSTEM_REVOCATION_LIST
extern int add_key_to_revocation_list(const char *data, size_t size);
extern int is_key_on_revocation_list(struct pkcs7_message *pkcs7);
extern int is_key_on_revocation_list(struct pkcs7_message *pkcs7,
enum key_being_used_for usage);
#else
static inline int add_key_to_revocation_list(const char *data, size_t size)
{
return 0;
}
static inline int is_key_on_revocation_list(struct pkcs7_message *pkcs7)
static inline int is_key_on_revocation_list(struct pkcs7_message *pkcs7,
enum key_being_used_for usage);
{
return -ENOKEY;
}
Expand Down

0 comments on commit 9285555

Please sign in to comment.