xattr.c 3.8 KB
Newer Older
R
Roberto Sassu 已提交
1
/*
R
Roberto Sassu 已提交
2
 * Copyright (C) 2017-2020 Huawei Technologies Duesseldorf GmbH
R
Roberto Sassu 已提交
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
 *
 * Author: Roberto Sassu <roberto.sassu@huawei.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.
 *
 * File: xattr.c
 *      Write IMA xattrs.
 */

#include <errno.h>
#include <unistd.h>
#include <sys/xattr.h>

#include "xattr.h"


int write_ima_xattr(int dirfd, char *path, u8 *keyid, size_t keyid_len,
R
Roberto Sassu 已提交
23
		    u8 *sig, size_t sig_len, enum hash_algo algo)
R
Roberto Sassu 已提交
24
{
R
Roberto Sassu 已提交
25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
	struct signature_v2_hdr *hdr;
	u8 *xattr_buf;
	size_t xattr_buf_len;
	int ret, fd;

	xattr_buf_len = sizeof(*hdr) + sig_len;
	xattr_buf = calloc(sizeof(u8), xattr_buf_len);
	if (!xattr_buf) {
		printf("Out of memory\n");
		return -ENOMEM;
	}

	hdr = (struct signature_v2_hdr *)xattr_buf;
	hdr->type = EVM_IMA_XATTR_DIGSIG;
	hdr->version = 2;
	hdr->hash_algo = algo;
	if (keyid_len)
		memcpy(&hdr->keyid, keyid, keyid_len);
	hdr->sig_size = cpu_to_be16(sig_len);
	memcpy(&hdr->sig, sig, sig_len);

	if (dirfd != -1) {
		fd = openat(dirfd, path, O_RDONLY);
		if (fd < 0) {
			printf("Cannot open %s\n", path);
			ret = fd;
			goto out;
		}

		ret = fsetxattr(fd, XATTR_NAME_IMA,
				xattr_buf, xattr_buf_len, 0);
		close(fd);
	} else {
		ret = lsetxattr(path, XATTR_NAME_IMA,
				xattr_buf, xattr_buf_len, 0);
	}
R
Roberto Sassu 已提交
61
out:
62
	if (ret < 0 && errno != ENOTSUP && errno != EOPNOTSUPP)
R
Roberto Sassu 已提交
63 64
		printf("Cannot add %s xattr to %s: %s\n", XATTR_NAME_IMA, path,
		       strerror(errno));
R
Roberto Sassu 已提交
65

R
Roberto Sassu 已提交
66 67
	free(xattr_buf);
	return 0;
R
Roberto Sassu 已提交
68 69
}

70 71
int write_evm_xattr(char *path, enum hash_algo algo)
{
R
Roberto Sassu 已提交
72 73
	struct signature_v2_hdr hdr = { 0 };
	int ret;
74

R
Roberto Sassu 已提交
75 76 77
	hdr.type = EVM_IMA_XATTR_DIGEST_LIST;
	hdr.version = 2;
	hdr.hash_algo = algo;
78

R
Roberto Sassu 已提交
79 80
	ret = lsetxattr(path, XATTR_NAME_EVM, &hdr,
			offsetof(struct signature_v2_hdr, keyid), 0);
81
	if (ret < 0 && errno != ENOTSUP && errno != EOPNOTSUPP)
R
Roberto Sassu 已提交
82 83
		printf("Cannot add %s xattr to %s: %s\n", XATTR_NAME_EVM, path,
		       strerror(errno));
84

R
Roberto Sassu 已提交
85
	return 0;
86 87
}

R
Roberto Sassu 已提交
88
int parse_ima_xattr(u8 *buf, size_t buf_len, u8 **keyid, size_t *keyid_len,
R
Roberto Sassu 已提交
89
		    u8 **sig, size_t *sig_len, enum hash_algo *algo)
R
Roberto Sassu 已提交
90
{
R
Roberto Sassu 已提交
91
	struct signature_v2_hdr *hdr;
R
Roberto Sassu 已提交
92

R
Roberto Sassu 已提交
93 94
	if (buf_len < sizeof(*hdr))
		return -EINVAL;
R
Roberto Sassu 已提交
95

R
Roberto Sassu 已提交
96
	hdr = (struct signature_v2_hdr *)buf;
R
Roberto Sassu 已提交
97

R
Roberto Sassu 已提交
98 99 100
	if (hdr->type != EVM_IMA_XATTR_DIGSIG &&
	    hdr->type != EVM_XATTR_PORTABLE_DIGSIG)
		return -EINVAL;
R
Roberto Sassu 已提交
101

R
Roberto Sassu 已提交
102 103
	if (hdr->version != 2)
		return -EINVAL;
R
Roberto Sassu 已提交
104

R
Roberto Sassu 已提交
105 106 107 108 109
	*algo = hdr->hash_algo;
	*keyid = (u8 *)&hdr->keyid;
	*keyid_len = sizeof(hdr->keyid);
	*sig = hdr->sig;
	*sig_len = be16_to_cpu(hdr->sig_size);
R
Roberto Sassu 已提交
110

R
Roberto Sassu 已提交
111 112
	if (buf_len != sizeof(*hdr) + *sig_len)
		return -EINVAL;
R
Roberto Sassu 已提交
113

R
Roberto Sassu 已提交
114
	return 0;
R
Roberto Sassu 已提交
115 116 117
}

int read_ima_xattr(int dirfd, char *path, u8 **buf, size_t *buf_len,
R
Roberto Sassu 已提交
118 119
		   u8 **keyid, size_t *keyid_len, u8 **sig, size_t *sig_len,
		   enum hash_algo *algo)
R
Roberto Sassu 已提交
120
{
R
Roberto Sassu 已提交
121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146
	ssize_t ret;
	int fd;

	fd = openat(dirfd, path, O_RDONLY);
	if (fd < 0) {
		printf("Cannot open %s\n", path);
		return fd;
	}

	ret = fgetxattr(fd, XATTR_NAME_IMA, NULL, 0);
	if (ret < 0)
		return -ENODATA;

	*buf_len = ret;
	*buf = malloc(*buf_len);
	if (!*buf)
		return -ENOMEM;

	ret = fgetxattr(fd, XATTR_NAME_IMA, *buf, ret);
	if (ret < 0)
		return -ENODATA;

	ret = parse_ima_xattr(*buf, *buf_len, keyid, keyid_len, sig, sig_len,
			      algo);
	close(fd);
	return ret;
R
Roberto Sassu 已提交
147
}
R
Roberto Sassu 已提交
148

R
Roberto Sassu 已提交
149
int gen_write_ima_xattr(u8 *buf, int *buf_len, char *path, u8 algo, u8 *digest,
R
Roberto Sassu 已提交
150
			bool immutable, bool write)
R
Roberto Sassu 已提交
151
{
R
Roberto Sassu 已提交
152 153
	struct evm_ima_xattr_data *ima_xattr = (struct evm_ima_xattr_data *)buf;
	int ret;
R
Roberto Sassu 已提交
154

R
Roberto Sassu 已提交
155 156 157 158
	*buf_len = 1 + 1 + hash_digest_size[algo];
	ima_xattr->type = IMA_XATTR_DIGEST_NG;
	ima_xattr->digest[0] = algo;
	memcpy(&ima_xattr->digest[1], digest, hash_digest_size[algo]);
R
Roberto Sassu 已提交
159

R
Roberto Sassu 已提交
160 161
	if (!write)
		return 0;
R
Roberto Sassu 已提交
162

R
Roberto Sassu 已提交
163
	ret = lsetxattr(path, XATTR_NAME_IMA, buf, *buf_len, 0);
164
	if (ret < 0 && errno != ENOTSUP && errno != EOPNOTSUPP)
R
Roberto Sassu 已提交
165 166
		printf("Cannot add %s xattr to %s: %s\n", XATTR_NAME_IMA, path,
		       strerror(errno));
167

R
Roberto Sassu 已提交
168
	return 0;
R
Roberto Sassu 已提交
169
}