2011-10-05 02:54:46 -06:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2011 Intel Corporation
|
|
|
|
*
|
|
|
|
* Author:
|
|
|
|
* Dmitry Kasatkin <dmitry.kasatkin@intel.com>
|
|
|
|
*
|
|
|
|
* This program 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, version 2 of the License.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/rbtree.h>
|
|
|
|
#include <linux/key-type.h>
|
|
|
|
#include <linux/digsig.h>
|
|
|
|
|
|
|
|
#include "integrity.h"
|
|
|
|
|
|
|
|
static struct key *keyring[INTEGRITY_KEYRING_MAX];
|
|
|
|
|
|
|
|
static const char *keyring_name[INTEGRITY_KEYRING_MAX] = {
|
|
|
|
"_evm",
|
|
|
|
"_module",
|
|
|
|
"_ima",
|
|
|
|
};
|
|
|
|
|
|
|
|
int integrity_digsig_verify(const unsigned int id, const char *sig, int siglen,
|
2013-10-10 00:56:13 -06:00
|
|
|
const char *digest, int digestlen)
|
2011-10-05 02:54:46 -06:00
|
|
|
{
|
|
|
|
if (id >= INTEGRITY_KEYRING_MAX)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (!keyring[id]) {
|
|
|
|
keyring[id] =
|
2013-11-23 17:36:35 -07:00
|
|
|
request_key(&key_type_keyring, keyring_name[id], NULL);
|
2011-10-05 02:54:46 -06:00
|
|
|
if (IS_ERR(keyring[id])) {
|
|
|
|
int err = PTR_ERR(keyring[id]);
|
|
|
|
pr_err("no %s keyring: %d\n", keyring_name[id], err);
|
|
|
|
keyring[id] = NULL;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-10 01:12:03 -06:00
|
|
|
switch (sig[1]) {
|
2013-02-06 15:12:08 -07:00
|
|
|
case 1:
|
2013-10-10 01:12:03 -06:00
|
|
|
/* v1 API expect signature without xattr type */
|
|
|
|
return digsig_verify(keyring[id], sig + 1, siglen - 1,
|
2013-02-06 15:12:08 -07:00
|
|
|
digest, digestlen);
|
|
|
|
case 2:
|
|
|
|
return asymmetric_verify(keyring[id], sig, siglen,
|
|
|
|
digest, digestlen);
|
|
|
|
}
|
|
|
|
|
|
|
|
return -EOPNOTSUPP;
|
2011-10-05 02:54:46 -06:00
|
|
|
}
|