Commit 50b97748 authored by Matthew Garrett's avatar Matthew Garrett Committed by Mimi Zohar

EVM: Add support for portable signature format

The EVM signature includes the inode number and (optionally) the
filesystem UUID, making it impractical to ship EVM signatures in
packages. This patch adds a new portable format intended to allow
distributions to include EVM signatures. It is identical to the existing
format but hardcodes the inode and generation numbers to 0 and does not
include the filesystem UUID even if the kernel is configured to do so.

Removing the inode means that the metadata and signature from one file
could be copied to another file without invalidating it. This is avoided
by ensuring that an IMA xattr is present during EVM validation.

Portable signatures are intended to be immutable - ie, they will never
be transformed into HMACs.

Based on earlier work by Dmitry Kasatkin and Mikhail Kurinnoi.
Signed-off-by: default avatarMatthew Garrett <mjg59@google.com>
Cc: Dmitry Kasatkin <dmitry.kasatkin@huawei.com>
Cc: Mikhail Kurinnoi <viewizard@viewizard.com>
Signed-off-by: default avatarMimi Zohar <zohar@linux.vnet.ibm.com>
parent ae1ba167
...@@ -14,6 +14,7 @@ ...@@ -14,6 +14,7 @@
enum integrity_status { enum integrity_status {
INTEGRITY_PASS = 0, INTEGRITY_PASS = 0,
INTEGRITY_PASS_IMMUTABLE,
INTEGRITY_FAIL, INTEGRITY_FAIL,
INTEGRITY_NOLABEL, INTEGRITY_NOLABEL,
INTEGRITY_NOXATTRS, INTEGRITY_NOXATTRS,
......
...@@ -54,7 +54,7 @@ int evm_calc_hmac(struct dentry *dentry, const char *req_xattr_name, ...@@ -54,7 +54,7 @@ int evm_calc_hmac(struct dentry *dentry, const char *req_xattr_name,
size_t req_xattr_value_len, char *digest); size_t req_xattr_value_len, char *digest);
int evm_calc_hash(struct dentry *dentry, const char *req_xattr_name, int evm_calc_hash(struct dentry *dentry, const char *req_xattr_name,
const char *req_xattr_value, const char *req_xattr_value,
size_t req_xattr_value_len, char *digest); size_t req_xattr_value_len, char type, char *digest);
int evm_init_hmac(struct inode *inode, const struct xattr *xattr, int evm_init_hmac(struct inode *inode, const struct xattr *xattr,
char *hmac_val); char *hmac_val);
int evm_init_secfs(void); int evm_init_secfs(void);
......
...@@ -138,7 +138,7 @@ static struct shash_desc *init_desc(char type) ...@@ -138,7 +138,7 @@ static struct shash_desc *init_desc(char type)
* protection.) * protection.)
*/ */
static void hmac_add_misc(struct shash_desc *desc, struct inode *inode, static void hmac_add_misc(struct shash_desc *desc, struct inode *inode,
char *digest) char type, char *digest)
{ {
struct h_misc { struct h_misc {
unsigned long ino; unsigned long ino;
...@@ -149,8 +149,13 @@ static void hmac_add_misc(struct shash_desc *desc, struct inode *inode, ...@@ -149,8 +149,13 @@ static void hmac_add_misc(struct shash_desc *desc, struct inode *inode,
} hmac_misc; } hmac_misc;
memset(&hmac_misc, 0, sizeof(hmac_misc)); memset(&hmac_misc, 0, sizeof(hmac_misc));
hmac_misc.ino = inode->i_ino; /* Don't include the inode or generation number in portable
hmac_misc.generation = inode->i_generation; * signatures
*/
if (type != EVM_XATTR_PORTABLE_DIGSIG) {
hmac_misc.ino = inode->i_ino;
hmac_misc.generation = inode->i_generation;
}
/* The hmac uid and gid must be encoded in the initial user /* The hmac uid and gid must be encoded in the initial user
* namespace (not the filesystems user namespace) as encoding * namespace (not the filesystems user namespace) as encoding
* them in the filesystems user namespace allows an attack * them in the filesystems user namespace allows an attack
...@@ -163,7 +168,8 @@ static void hmac_add_misc(struct shash_desc *desc, struct inode *inode, ...@@ -163,7 +168,8 @@ static void hmac_add_misc(struct shash_desc *desc, struct inode *inode,
hmac_misc.gid = from_kgid(&init_user_ns, inode->i_gid); hmac_misc.gid = from_kgid(&init_user_ns, inode->i_gid);
hmac_misc.mode = inode->i_mode; hmac_misc.mode = inode->i_mode;
crypto_shash_update(desc, (const u8 *)&hmac_misc, sizeof(hmac_misc)); crypto_shash_update(desc, (const u8 *)&hmac_misc, sizeof(hmac_misc));
if (evm_hmac_attrs & EVM_ATTR_FSUUID) if ((evm_hmac_attrs & EVM_ATTR_FSUUID) &&
type != EVM_XATTR_PORTABLE_DIGSIG)
crypto_shash_update(desc, &inode->i_sb->s_uuid.b[0], crypto_shash_update(desc, &inode->i_sb->s_uuid.b[0],
sizeof(inode->i_sb->s_uuid)); sizeof(inode->i_sb->s_uuid));
crypto_shash_final(desc, digest); crypto_shash_final(desc, digest);
...@@ -189,6 +195,7 @@ static int evm_calc_hmac_or_hash(struct dentry *dentry, ...@@ -189,6 +195,7 @@ static int evm_calc_hmac_or_hash(struct dentry *dentry,
char *xattr_value = NULL; char *xattr_value = NULL;
int error; int error;
int size; int size;
bool ima_present = false;
if (!(inode->i_opflags & IOP_XATTR)) if (!(inode->i_opflags & IOP_XATTR))
return -EOPNOTSUPP; return -EOPNOTSUPP;
...@@ -199,11 +206,18 @@ static int evm_calc_hmac_or_hash(struct dentry *dentry, ...@@ -199,11 +206,18 @@ static int evm_calc_hmac_or_hash(struct dentry *dentry,
error = -ENODATA; error = -ENODATA;
for (xattrname = evm_config_xattrnames; *xattrname != NULL; xattrname++) { for (xattrname = evm_config_xattrnames; *xattrname != NULL; xattrname++) {
bool is_ima = false;
if (strcmp(*xattrname, XATTR_NAME_IMA) == 0)
is_ima = true;
if ((req_xattr_name && req_xattr_value) if ((req_xattr_name && req_xattr_value)
&& !strcmp(*xattrname, req_xattr_name)) { && !strcmp(*xattrname, req_xattr_name)) {
error = 0; error = 0;
crypto_shash_update(desc, (const u8 *)req_xattr_value, crypto_shash_update(desc, (const u8 *)req_xattr_value,
req_xattr_value_len); req_xattr_value_len);
if (is_ima)
ima_present = true;
continue; continue;
} }
size = vfs_getxattr_alloc(dentry, *xattrname, size = vfs_getxattr_alloc(dentry, *xattrname,
...@@ -218,9 +232,14 @@ static int evm_calc_hmac_or_hash(struct dentry *dentry, ...@@ -218,9 +232,14 @@ static int evm_calc_hmac_or_hash(struct dentry *dentry,
error = 0; error = 0;
xattr_size = size; xattr_size = size;
crypto_shash_update(desc, (const u8 *)xattr_value, xattr_size); crypto_shash_update(desc, (const u8 *)xattr_value, xattr_size);
if (is_ima)
ima_present = true;
} }
hmac_add_misc(desc, inode, digest); hmac_add_misc(desc, inode, type, digest);
/* Portable EVM signatures must include an IMA hash */
if (type == EVM_XATTR_PORTABLE_DIGSIG && !ima_present)
return -EPERM;
out: out:
kfree(xattr_value); kfree(xattr_value);
kfree(desc); kfree(desc);
...@@ -232,17 +251,45 @@ int evm_calc_hmac(struct dentry *dentry, const char *req_xattr_name, ...@@ -232,17 +251,45 @@ int evm_calc_hmac(struct dentry *dentry, const char *req_xattr_name,
char *digest) char *digest)
{ {
return evm_calc_hmac_or_hash(dentry, req_xattr_name, req_xattr_value, return evm_calc_hmac_or_hash(dentry, req_xattr_name, req_xattr_value,
req_xattr_value_len, EVM_XATTR_HMAC, digest); req_xattr_value_len, EVM_XATTR_HMAC, digest);
} }
int evm_calc_hash(struct dentry *dentry, const char *req_xattr_name, int evm_calc_hash(struct dentry *dentry, const char *req_xattr_name,
const char *req_xattr_value, size_t req_xattr_value_len, const char *req_xattr_value, size_t req_xattr_value_len,
char *digest) char type, char *digest)
{ {
return evm_calc_hmac_or_hash(dentry, req_xattr_name, req_xattr_value, return evm_calc_hmac_or_hash(dentry, req_xattr_name, req_xattr_value,
req_xattr_value_len, IMA_XATTR_DIGEST, digest); req_xattr_value_len, type, digest);
}
static int evm_is_immutable(struct dentry *dentry, struct inode *inode)
{
const struct evm_ima_xattr_data *xattr_data = NULL;
struct integrity_iint_cache *iint;
int rc = 0;
iint = integrity_iint_find(inode);
if (iint && (iint->flags & EVM_IMMUTABLE_DIGSIG))
return 1;
/* Do this the hard way */
rc = vfs_getxattr_alloc(dentry, XATTR_NAME_EVM, (char **)&xattr_data, 0,
GFP_NOFS);
if (rc <= 0) {
if (rc == -ENODATA)
return 0;
return rc;
}
if (xattr_data->type == EVM_XATTR_PORTABLE_DIGSIG)
rc = 1;
else
rc = 0;
kfree(xattr_data);
return rc;
} }
/* /*
* Calculate the hmac and update security.evm xattr * Calculate the hmac and update security.evm xattr
* *
...@@ -255,6 +302,16 @@ int evm_update_evmxattr(struct dentry *dentry, const char *xattr_name, ...@@ -255,6 +302,16 @@ int evm_update_evmxattr(struct dentry *dentry, const char *xattr_name,
struct evm_ima_xattr_data xattr_data; struct evm_ima_xattr_data xattr_data;
int rc = 0; int rc = 0;
/*
* Don't permit any transformation of the EVM xattr if the signature
* is of an immutable type
*/
rc = evm_is_immutable(dentry, inode);
if (rc < 0)
return rc;
if (rc)
return -EPERM;
rc = evm_calc_hmac(dentry, xattr_name, xattr_value, rc = evm_calc_hmac(dentry, xattr_name, xattr_value,
xattr_value_len, xattr_data.digest); xattr_value_len, xattr_data.digest);
if (rc == 0) { if (rc == 0) {
...@@ -280,7 +337,7 @@ int evm_init_hmac(struct inode *inode, const struct xattr *lsm_xattr, ...@@ -280,7 +337,7 @@ int evm_init_hmac(struct inode *inode, const struct xattr *lsm_xattr,
} }
crypto_shash_update(desc, lsm_xattr->value, lsm_xattr->value_len); crypto_shash_update(desc, lsm_xattr->value, lsm_xattr->value_len);
hmac_add_misc(desc, inode, hmac_val); hmac_add_misc(desc, inode, EVM_XATTR_HMAC, hmac_val);
kfree(desc); kfree(desc);
return 0; return 0;
} }
......
...@@ -31,7 +31,7 @@ ...@@ -31,7 +31,7 @@
int evm_initialized; int evm_initialized;
static char *integrity_status_msg[] = { static char *integrity_status_msg[] = {
"pass", "fail", "no_label", "no_xattrs", "unknown" "pass", "pass_immutable", "fail", "no_label", "no_xattrs", "unknown"
}; };
char *evm_hmac = "hmac(sha1)"; char *evm_hmac = "hmac(sha1)";
char *evm_hash = "sha1"; char *evm_hash = "sha1";
...@@ -128,7 +128,8 @@ static enum integrity_status evm_verify_hmac(struct dentry *dentry, ...@@ -128,7 +128,8 @@ static enum integrity_status evm_verify_hmac(struct dentry *dentry,
enum integrity_status evm_status = INTEGRITY_PASS; enum integrity_status evm_status = INTEGRITY_PASS;
int rc, xattr_len; int rc, xattr_len;
if (iint && iint->evm_status == INTEGRITY_PASS) if (iint && (iint->evm_status == INTEGRITY_PASS ||
iint->evm_status == INTEGRITY_PASS_IMMUTABLE))
return iint->evm_status; return iint->evm_status;
/* if status is not PASS, try to check again - against -ENOMEM */ /* if status is not PASS, try to check again - against -ENOMEM */
...@@ -169,22 +170,26 @@ static enum integrity_status evm_verify_hmac(struct dentry *dentry, ...@@ -169,22 +170,26 @@ static enum integrity_status evm_verify_hmac(struct dentry *dentry,
rc = -EINVAL; rc = -EINVAL;
break; break;
case EVM_IMA_XATTR_DIGSIG: case EVM_IMA_XATTR_DIGSIG:
case EVM_XATTR_PORTABLE_DIGSIG:
rc = evm_calc_hash(dentry, xattr_name, xattr_value, rc = evm_calc_hash(dentry, xattr_name, xattr_value,
xattr_value_len, calc.digest); xattr_value_len, xattr_data->type,
calc.digest);
if (rc) if (rc)
break; break;
rc = integrity_digsig_verify(INTEGRITY_KEYRING_EVM, rc = integrity_digsig_verify(INTEGRITY_KEYRING_EVM,
(const char *)xattr_data, xattr_len, (const char *)xattr_data, xattr_len,
calc.digest, sizeof(calc.digest)); calc.digest, sizeof(calc.digest));
if (!rc) { if (!rc) {
/* Replace RSA with HMAC if not mounted readonly and if (xattr_data->type == EVM_XATTR_PORTABLE_DIGSIG) {
* not immutable if (iint)
*/ iint->flags |= EVM_IMMUTABLE_DIGSIG;
if (!IS_RDONLY(d_backing_inode(dentry)) && evm_status = INTEGRITY_PASS_IMMUTABLE;
!IS_IMMUTABLE(d_backing_inode(dentry))) } else if (!IS_RDONLY(d_backing_inode(dentry)) &&
!IS_IMMUTABLE(d_backing_inode(dentry))) {
evm_update_evmxattr(dentry, xattr_name, evm_update_evmxattr(dentry, xattr_name,
xattr_value, xattr_value,
xattr_value_len); xattr_value_len);
}
} }
break; break;
default: default:
...@@ -285,7 +290,7 @@ static enum integrity_status evm_verify_current_integrity(struct dentry *dentry) ...@@ -285,7 +290,7 @@ static enum integrity_status evm_verify_current_integrity(struct dentry *dentry)
* affect security.evm. An interesting side affect of writing posix xattr * affect security.evm. An interesting side affect of writing posix xattr
* acls is their modifying of the i_mode, which is included in security.evm. * acls is their modifying of the i_mode, which is included in security.evm.
* For posix xattr acls only, permit security.evm, even if it currently * For posix xattr acls only, permit security.evm, even if it currently
* doesn't exist, to be updated. * doesn't exist, to be updated unless the EVM signature is immutable.
*/ */
static int evm_protect_xattr(struct dentry *dentry, const char *xattr_name, static int evm_protect_xattr(struct dentry *dentry, const char *xattr_name,
const void *xattr_value, size_t xattr_value_len) const void *xattr_value, size_t xattr_value_len)
...@@ -360,7 +365,8 @@ int evm_inode_setxattr(struct dentry *dentry, const char *xattr_name, ...@@ -360,7 +365,8 @@ int evm_inode_setxattr(struct dentry *dentry, const char *xattr_name,
if (strcmp(xattr_name, XATTR_NAME_EVM) == 0) { if (strcmp(xattr_name, XATTR_NAME_EVM) == 0) {
if (!xattr_value_len) if (!xattr_value_len)
return -EINVAL; return -EINVAL;
if (xattr_data->type != EVM_IMA_XATTR_DIGSIG) if (xattr_data->type != EVM_IMA_XATTR_DIGSIG &&
xattr_data->type != EVM_XATTR_PORTABLE_DIGSIG)
return -EPERM; return -EPERM;
} }
return evm_protect_xattr(dentry, xattr_name, xattr_value, return evm_protect_xattr(dentry, xattr_name, xattr_value,
...@@ -443,6 +449,9 @@ void evm_inode_post_removexattr(struct dentry *dentry, const char *xattr_name) ...@@ -443,6 +449,9 @@ void evm_inode_post_removexattr(struct dentry *dentry, const char *xattr_name)
/** /**
* evm_inode_setattr - prevent updating an invalid EVM extended attribute * evm_inode_setattr - prevent updating an invalid EVM extended attribute
* @dentry: pointer to the affected dentry * @dentry: pointer to the affected dentry
*
* Permit update of file attributes when files have a valid EVM signature,
* except in the case of them having an immutable portable signature.
*/ */
int evm_inode_setattr(struct dentry *dentry, struct iattr *attr) int evm_inode_setattr(struct dentry *dentry, struct iattr *attr)
{ {
......
...@@ -230,7 +230,9 @@ int ima_appraise_measurement(enum ima_hooks func, ...@@ -230,7 +230,9 @@ int ima_appraise_measurement(enum ima_hooks func,
} }
status = evm_verifyxattr(dentry, XATTR_NAME_IMA, xattr_value, rc, iint); status = evm_verifyxattr(dentry, XATTR_NAME_IMA, xattr_value, rc, iint);
if ((status != INTEGRITY_PASS) && (status != INTEGRITY_UNKNOWN)) { if ((status != INTEGRITY_PASS) &&
(status != INTEGRITY_PASS_IMMUTABLE) &&
(status != INTEGRITY_UNKNOWN)) {
if ((status == INTEGRITY_NOLABEL) if ((status == INTEGRITY_NOLABEL)
|| (status == INTEGRITY_NOXATTRS)) || (status == INTEGRITY_NOXATTRS))
cause = "missing-HMAC"; cause = "missing-HMAC";
......
...@@ -33,6 +33,7 @@ ...@@ -33,6 +33,7 @@
#define IMA_DIGSIG_REQUIRED 0x02000000 #define IMA_DIGSIG_REQUIRED 0x02000000
#define IMA_PERMIT_DIRECTIO 0x04000000 #define IMA_PERMIT_DIRECTIO 0x04000000
#define IMA_NEW_FILE 0x08000000 #define IMA_NEW_FILE 0x08000000
#define EVM_IMMUTABLE_DIGSIG 0x10000000
#define IMA_DO_MASK (IMA_MEASURE | IMA_APPRAISE | IMA_AUDIT | \ #define IMA_DO_MASK (IMA_MEASURE | IMA_APPRAISE | IMA_AUDIT | \
IMA_APPRAISE_SUBMASK) IMA_APPRAISE_SUBMASK)
...@@ -58,6 +59,7 @@ enum evm_ima_xattr_type { ...@@ -58,6 +59,7 @@ enum evm_ima_xattr_type {
EVM_XATTR_HMAC, EVM_XATTR_HMAC,
EVM_IMA_XATTR_DIGSIG, EVM_IMA_XATTR_DIGSIG,
IMA_XATTR_DIGEST_NG, IMA_XATTR_DIGEST_NG,
EVM_XATTR_PORTABLE_DIGSIG,
IMA_XATTR_LAST IMA_XATTR_LAST
}; };
......
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