693 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			693 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Copyright (c) 2021 MediaTek Inc.
 | |
|  */
 | |
| 
 | |
| #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
 | |
| 
 | |
| #include <linux/module.h>
 | |
| #include <linux/slab.h>
 | |
| #include <linux/device.h>
 | |
| #include <crypto/hash.h>
 | |
| #include <linux/scatterlist.h>
 | |
| #include <linux/sizes.h>
 | |
| 
 | |
| #include <linux/rpmb.h>
 | |
| 
 | |
| static const char id[] = "RPMB:SIM";
 | |
| #define CAPACITY_UNIT SZ_128K
 | |
| #define CAPACITY_MIN  SZ_128K
 | |
| #define CAPACITY_MAX  SZ_16M
 | |
| #define BLK_UNIT      SZ_256
 | |
| 
 | |
| static unsigned int max_wr_blks = 2;
 | |
| module_param(max_wr_blks, uint, 0644);
 | |
| MODULE_PARM_DESC(max_wr_blks, "max blocks that can be written in a single command (default: 2)");
 | |
| 
 | |
| static unsigned int daunits = 1;
 | |
| module_param(daunits, uint, 0644);
 | |
| MODULE_PARM_DESC(daunits, "number of data area units of 128K (default: 1)");
 | |
| 
 | |
| struct blk {
 | |
| 	u8 data[BLK_UNIT];
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * struct rpmb_sim_dev
 | |
|  *
 | |
|  * @dev:  back pointer device
 | |
|  * @rdev: rpmb device
 | |
|  * @auth_key: Authentication key register which is used to authenticate
 | |
|  *            accesses when MAC is calculated;
 | |
|  * @auth_key_set: true if auth key was set
 | |
|  * @write_counter: Counter value for the total amount of successful
 | |
|  *             authenticated data write requests made by the host.
 | |
|  *             The initial value of this register after production is 00000000h.
 | |
|  *             The value will be incremented by one along with each successful
 | |
|  *             programming access. The value cannot be reset. After the counter
 | |
|  *             has reached the maximum value of FFFFFFFFh,
 | |
|  *             it will not be incremented anymore (overflow prevention)
 | |
|  * @hash_tfm:  hmac(sha256) tfm
 | |
|  *
 | |
|  * @res_frames: frame that holds the result of the last write operation
 | |
|  * @out_frames: next read operation result frames
 | |
|  * @out_frames_cnt: number of the output frames
 | |
|  *
 | |
|  * @capacity: size of the partition in bytes multiple of 128K
 | |
|  * @blkcnt:   block count
 | |
|  * @da:       data area in blocks
 | |
|  */
 | |
| struct rpmb_sim_dev {
 | |
| 	struct device *dev;
 | |
| 	struct rpmb_dev *rdev;
 | |
| 	u8 auth_key[32];
 | |
| 	bool auth_key_set;
 | |
| 	u32 write_counter;
 | |
| 	struct crypto_shash *hash_tfm;
 | |
| 
 | |
| 	struct rpmb_frame res_frames[1];
 | |
| 	struct rpmb_frame *out_frames;
 | |
| 	unsigned int out_frames_cnt;
 | |
| 
 | |
| 	size_t capacity;
 | |
| 	size_t blkcnt;
 | |
| 	struct blk *da;
 | |
| };
 | |
| 
 | |
| static __be16 op_result(struct rpmb_sim_dev *rsdev, u16 result)
 | |
| {
 | |
| 	if (!rsdev->auth_key_set)
 | |
| 		return cpu_to_be16(RPMB_ERR_NO_KEY);
 | |
| 
 | |
| 	if (rsdev->write_counter == 0xFFFFFFFF)
 | |
| 		result |=  RPMB_ERR_COUNTER_EXPIRED;
 | |
| 
 | |
| 	return cpu_to_be16(result);
 | |
| }
 | |
| 
 | |
| static __be16 req_to_resp(u16 req)
 | |
| {
 | |
| 	return cpu_to_be16(RPMB_REQ2RESP(req));
 | |
| }
 | |
| 
 | |
| static int rpmb_sim_calc_hmac(struct rpmb_sim_dev *rsdev,
 | |
| 			      struct rpmb_frame *frames,
 | |
| 			      unsigned int blks, u8 *mac)
 | |
| {
 | |
| 	SHASH_DESC_ON_STACK(desc, rsdev->hash_tfm);
 | |
| 	int i;
 | |
| 	int ret;
 | |
| 
 | |
| 	desc->tfm = rsdev->hash_tfm;
 | |
| 	desc->flags = 0;
 | |
| 
 | |
| 	ret = crypto_shash_init(desc);
 | |
| 	if (ret)
 | |
| 		goto out;
 | |
| 
 | |
| 	for (i = 0; i < blks; i++) {
 | |
| 		ret = crypto_shash_update(desc, frames[i].data, hmac_data_len);
 | |
| 		if (ret)
 | |
| 			goto out;
 | |
| 	}
 | |
| 	ret = crypto_shash_final(desc, mac);
 | |
| out:
 | |
| 	if (ret)
 | |
| 		dev_notice(rsdev->dev, "digest error = %d", ret);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int rpmb_op_not_programmed(struct rpmb_sim_dev *rsdev, u16 req)
 | |
| {
 | |
| 	struct rpmb_frame *res_frame = rsdev->res_frames;
 | |
| 
 | |
| 	res_frame->req_resp = req_to_resp(req);
 | |
| 	res_frame->result = op_result(rsdev, RPMB_ERR_NO_KEY);
 | |
| 
 | |
| 	dev_notice(rsdev->dev, "not programmed\n");
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rpmb_op_program_key(struct rpmb_sim_dev *rsdev,
 | |
| 			       struct rpmb_frame *in_frame, u32 cnt)
 | |
| {
 | |
| 	struct rpmb_frame *res_frame = rsdev->res_frames;
 | |
| 	u16 req;
 | |
| 	int ret;
 | |
| 	u16 err = RPMB_ERR_OK;
 | |
| 
 | |
| 	req = be16_to_cpu(in_frame[0].req_resp);
 | |
| 
 | |
| 	if (req != RPMB_PROGRAM_KEY)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (cnt != 1) {
 | |
| 		dev_notice(rsdev->dev, "wrong number of frames %d != 1\n", cnt);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	if (rsdev->auth_key_set) {
 | |
| 		dev_notice(rsdev->dev, "key allread set\n");
 | |
| 		err = RPMB_ERR_WRITE;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	ret = crypto_shash_setkey(rsdev->hash_tfm, in_frame[0].key_mac, 32);
 | |
| 	if (ret) {
 | |
| 		dev_notice(rsdev->dev, "set key failed = %d\n", ret);
 | |
| 		err = RPMB_ERR_GENERAL;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	dev_dbg(rsdev->dev, "digest size %u\n",
 | |
| 		crypto_shash_digestsize(rsdev->hash_tfm));
 | |
| 
 | |
| 	memcpy(rsdev->auth_key, in_frame[0].key_mac, 32);
 | |
| 	rsdev->auth_key_set = true;
 | |
| out:
 | |
| 
 | |
| 	memset(res_frame, 0, sizeof(struct rpmb_frame));
 | |
| 	res_frame->req_resp = req_to_resp(req);
 | |
| 	res_frame->result = op_result(rsdev, err);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rpmb_op_get_wr_counter(struct rpmb_sim_dev *rsdev,
 | |
| 				  struct rpmb_frame *in_frame, u32 cnt)
 | |
| {
 | |
| 	struct rpmb_frame *frame;
 | |
| 	int ret = 0;
 | |
| 	u16 req;
 | |
| 	u16 err;
 | |
| 
 | |
| 	req = be16_to_cpu(in_frame[0].req_resp);
 | |
| 	if (req != RPMB_GET_WRITE_COUNTER)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (cnt != 1) {
 | |
| 		dev_notice(rsdev->dev, "wrong number of frames %d != 1\n", cnt);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	frame = kcalloc(1, sizeof(struct rpmb_frame), GFP_KERNEL);
 | |
| 	if (!frame) {
 | |
| 		err = RPMB_ERR_READ;
 | |
| 		ret = -ENOMEM;
 | |
| 		rsdev->out_frames = rsdev->res_frames;
 | |
| 		rsdev->out_frames_cnt = cnt;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	rsdev->out_frames = frame;
 | |
| 	rsdev->out_frames_cnt = cnt;
 | |
| 
 | |
| 	frame->req_resp = req_to_resp(req);
 | |
| 	frame->write_counter = cpu_to_be32(rsdev->write_counter);
 | |
| 	memcpy(frame->nonce, in_frame[0].nonce, 16);
 | |
| 
 | |
| 	err = RPMB_ERR_OK;
 | |
| 	if (rpmb_sim_calc_hmac(rsdev, frame, cnt, frame->key_mac))
 | |
| 		err = RPMB_ERR_READ;
 | |
| 
 | |
| out:
 | |
| 	rsdev->out_frames[0].req_resp = req_to_resp(req);
 | |
| 	rsdev->out_frames[0].result = op_result(rsdev, err);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int rpmb_op_write_data(struct rpmb_sim_dev *rsdev,
 | |
| 			      struct rpmb_frame *in_frame, u32 cnt)
 | |
| {
 | |
| 	struct rpmb_frame *res_frame = rsdev->res_frames;
 | |
| 	u8 mac[32];
 | |
| 	u16 req, err, addr, blks;
 | |
| 	unsigned int i;
 | |
| 	int ret = 0;
 | |
| 
 | |
| 	req = be16_to_cpu(in_frame[0].req_resp);
 | |
| 	if (req != RPMB_WRITE_DATA)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (rsdev->write_counter == 0xFFFFFFFF) {
 | |
| 		err = RPMB_ERR_WRITE;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	blks = be16_to_cpu(in_frame[0].block_count);
 | |
| 	if (blks == 0 || blks > cnt) {
 | |
| 		dev_notice(rsdev->dev, "wrong number of frames %u > %u\n",
 | |
| 			blks, cnt);
 | |
| 		ret = -EINVAL;
 | |
| 		err = RPMB_ERR_GENERAL;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (blks > max_wr_blks) {
 | |
| 		err = RPMB_ERR_WRITE;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	addr = be16_to_cpu(in_frame[0].addr);
 | |
| 	if (addr >= rsdev->blkcnt) {
 | |
| 		err = RPMB_ERR_ADDRESS;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (rpmb_sim_calc_hmac(rsdev, in_frame, blks, mac)) {
 | |
| 		err = RPMB_ERR_AUTH;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	/* mac is in the last frame */
 | |
| 	if (memcmp(mac, in_frame[blks - 1].key_mac, sizeof(mac)) != 0) {
 | |
| 		err = RPMB_ERR_AUTH;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (be32_to_cpu(in_frame[0].write_counter) != rsdev->write_counter) {
 | |
| 		err = RPMB_ERR_COUNTER;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (addr + blks > rsdev->blkcnt) {
 | |
| 		err = RPMB_ERR_WRITE;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	dev_dbg(rsdev->dev, "Writing = %u blokcs at addr = 0x%X\n", blks, addr);
 | |
| 	err = RPMB_ERR_OK;
 | |
| 	for (i = 0; i < blks; i++)
 | |
| 		memcpy(rsdev->da[addr + i].data, in_frame[i].data, BLK_UNIT);
 | |
| 
 | |
| 	rsdev->write_counter++;
 | |
| 
 | |
| 	memset(res_frame, 0, sizeof(struct rpmb_frame));
 | |
| 	res_frame->req_resp = req_to_resp(req);
 | |
| 	res_frame->write_counter = cpu_to_be32(rsdev->write_counter);
 | |
| 	res_frame->addr = cpu_to_be16(addr);
 | |
| 	if (rpmb_sim_calc_hmac(rsdev, res_frame, 1, res_frame->key_mac))
 | |
| 		err = RPMB_ERR_READ;
 | |
| 
 | |
| out:
 | |
| 	if (err != RPMB_ERR_OK) {
 | |
| 		memset(res_frame, 0, sizeof(struct rpmb_frame));
 | |
| 		res_frame->req_resp = req_to_resp(req);
 | |
| 	}
 | |
| 	res_frame->result = op_result(rsdev, err);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int rpmb_do_read_data(struct rpmb_sim_dev *rsdev,
 | |
| 			     struct rpmb_frame *in_frame, u32 cnt)
 | |
| {
 | |
| 	struct rpmb_frame *res_frame = rsdev->res_frames;
 | |
| 	struct rpmb_frame *out_frames = NULL;
 | |
| 	u8 mac[32];
 | |
| 	u16 req, err, addr, blks;
 | |
| 	unsigned int i;
 | |
| 	int ret;
 | |
| 
 | |
| 	req = be16_to_cpu(in_frame->req_resp);
 | |
| 	if (req != RPMB_READ_DATA)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	/* eMMc intentially set 0 here */
 | |
| 	blks = be16_to_cpu(in_frame->block_count);
 | |
| 	blks = blks ?: cnt;
 | |
| 	if (blks > cnt) {
 | |
| 		dev_notice(rsdev->dev, "wrong number of frames cnt %u\n", blks);
 | |
| 		ret = -EINVAL;
 | |
| 		err = RPMB_ERR_GENERAL;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	out_frames = kcalloc(blks, sizeof(struct rpmb_frame), GFP_KERNEL);
 | |
| 	if (!out_frames) {
 | |
| 		ret = -ENOMEM;
 | |
| 		err = RPMB_ERR_READ;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	ret = 0;
 | |
| 	addr = be16_to_cpu(in_frame[0].addr);
 | |
| 	if (addr >= rsdev->blkcnt) {
 | |
| 		err = RPMB_ERR_ADDRESS;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (addr + blks > rsdev->blkcnt) {
 | |
| 		err = RPMB_ERR_READ;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	dev_dbg(rsdev->dev, "reading = %u blokcs at addr = 0x%X\n", blks, addr);
 | |
| 	for (i = 0; i < blks; i++) {
 | |
| 		memcpy(out_frames[i].data, rsdev->da[addr + i].data, BLK_UNIT);
 | |
| 		memcpy(out_frames[i].nonce, in_frame[0].nonce, 16);
 | |
| 		out_frames[i].req_resp = req_to_resp(req);
 | |
| 		out_frames[i].addr = in_frame[0].addr;
 | |
| 		out_frames[i].block_count = cpu_to_be16(blks);
 | |
| 	}
 | |
| 
 | |
| 	if (rpmb_sim_calc_hmac(rsdev, out_frames, blks, mac)) {
 | |
| 		err = RPMB_ERR_AUTH;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	memcpy(out_frames[blks - 1].key_mac, mac, sizeof(mac));
 | |
| 
 | |
| 	err = RPMB_ERR_OK;
 | |
| 	for (i = 0; i < blks; i++)
 | |
| 		out_frames[i].result = op_result(rsdev, err);
 | |
| 
 | |
| 	rsdev->out_frames = out_frames;
 | |
| 	rsdev->out_frames_cnt = cnt;
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| out:
 | |
| 	memset(res_frame, 0, sizeof(struct rpmb_frame));
 | |
| 	res_frame->req_resp = req_to_resp(req);
 | |
| 	res_frame->result = op_result(rsdev, err);
 | |
| 	kfree(out_frames);
 | |
| 	rsdev->out_frames = res_frame;
 | |
| 	rsdev->out_frames_cnt = 1;
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int rpmb_op_read_data(struct rpmb_sim_dev *rsdev,
 | |
| 			     struct rpmb_frame *in_frame, u32 cnt)
 | |
| {
 | |
| 	struct rpmb_frame *res_frame = rsdev->res_frames;
 | |
| 	u16 req;
 | |
| 
 | |
| 	req = be16_to_cpu(in_frame->req_resp);
 | |
| 	if (req != RPMB_READ_DATA)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	memcpy(res_frame, in_frame, sizeof(*res_frame));
 | |
| 
 | |
| 	rsdev->out_frames = res_frame;
 | |
| 	rsdev->out_frames_cnt = 1;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rpmb_op_result_read(struct rpmb_sim_dev *rsdev,
 | |
| 			       struct rpmb_frame *frames, u32 cnt)
 | |
| {
 | |
| 	u16 req = be16_to_cpu(frames[0].req_resp);
 | |
| 	u16 blks = be16_to_cpu(frames[0].block_count);
 | |
| 
 | |
| 	if (req != RPMB_RESULT_READ)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (blks != 0) {
 | |
| 		dev_notice(rsdev->dev, "wrong number of frames %u != 0\n",
 | |
| 				blks);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	rsdev->out_frames = rsdev->res_frames;
 | |
| 	rsdev->out_frames_cnt = 1;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rpmb_sim_write(struct rpmb_sim_dev *rsdev,
 | |
| 			  struct rpmb_frame *frames, u32 cnt)
 | |
| {
 | |
| 	u16 req;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (!frames || !cnt)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	req = be16_to_cpu(frames[0].req_resp);
 | |
| 	if (!rsdev->auth_key_set && req != RPMB_PROGRAM_KEY)
 | |
| 		return rpmb_op_not_programmed(rsdev, req);
 | |
| 
 | |
| 	switch (req) {
 | |
| 	case RPMB_PROGRAM_KEY:
 | |
| 		dev_dbg(rsdev->dev, "rpmb: program key\n");
 | |
| 		ret = rpmb_op_program_key(rsdev, frames, cnt);
 | |
| 		break;
 | |
| 	case RPMB_WRITE_DATA:
 | |
| 		dev_dbg(rsdev->dev, "rpmb: write data\n");
 | |
| 		ret = rpmb_op_write_data(rsdev, frames, cnt);
 | |
| 		break;
 | |
| 	case RPMB_GET_WRITE_COUNTER:
 | |
| 		dev_dbg(rsdev->dev, "rpmb: get write counter\n");
 | |
| 		ret = rpmb_op_get_wr_counter(rsdev, frames, cnt);
 | |
| 		break;
 | |
| 	case RPMB_READ_DATA:
 | |
| 		dev_dbg(rsdev->dev, "rpmb: read data\n");
 | |
| 		ret = rpmb_op_read_data(rsdev, frames, cnt);
 | |
| 		break;
 | |
| 	case RPMB_RESULT_READ:
 | |
| 		dev_dbg(rsdev->dev, "rpmb: result read\n");
 | |
| 		ret = rpmb_op_result_read(rsdev, frames, cnt);
 | |
| 		break;
 | |
| 	default:
 | |
| 		dev_notice(rsdev->dev, "unsupported command %u\n", req);
 | |
| 		ret = -EINVAL;
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	dev_dbg(rsdev->dev, "rpmb: ret=%d\n", ret);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int rpmb_sim_read(struct rpmb_sim_dev *rsdev,
 | |
| 			 struct rpmb_frame *frames, u32 cnt)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	if (!frames || !cnt)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (!rsdev->out_frames || rsdev->out_frames_cnt == 0) {
 | |
| 		dev_notice(rsdev->dev, "out_frames are not set\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	if (rsdev->out_frames->req_resp == cpu_to_be16(RPMB_READ_DATA))
 | |
| 		rpmb_do_read_data(rsdev, rsdev->out_frames, cnt);
 | |
| 
 | |
| 	for (i = 0; i < min_t(u32, rsdev->out_frames_cnt, cnt); i++)
 | |
| 		memcpy(&frames[i], &rsdev->out_frames[i], sizeof(frames[i]));
 | |
| 
 | |
| 	if (rsdev->out_frames != rsdev->res_frames)
 | |
| 		kfree(rsdev->out_frames);
 | |
| 
 | |
| 	rsdev->out_frames = NULL;
 | |
| 	rsdev->out_frames_cnt = 0;
 | |
| 	dev_dbg(rsdev->dev, "rpmb: cnt=%d\n", cnt);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rpmb_sim_cmd_seq(struct device *dev,
 | |
| 			    struct rpmb_cmd *cmds, u32 ncmds, u8 region)
 | |
| {
 | |
| 	struct rpmb_sim_dev *rsdev;
 | |
| 	int i;
 | |
| 	int ret;
 | |
| 	struct rpmb_cmd *cmd;
 | |
| 
 | |
| 	if (!dev)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	dev_notice(dev, "rpmb_cmd_seq\n");
 | |
| 
 | |
| 	rsdev = dev_get_drvdata(dev);
 | |
| 
 | |
| 	if (!rsdev)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	for (ret = 0, i = 0; i < ncmds && !ret; i++) {
 | |
| 		cmd = &cmds[i];
 | |
| 		if (cmd->flags & RPMB_F_WRITE)
 | |
| 			ret = rpmb_sim_write(rsdev, cmd->frames, cmd->nframes);
 | |
| 		else
 | |
| 			ret = rpmb_sim_read(rsdev, cmd->frames, cmd->nframes);
 | |
| 	}
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static struct rpmb_ops rpmb_sim_ops = {
 | |
| 	.cmd_seq = rpmb_sim_cmd_seq,
 | |
| 	.type = RPMB_TYPE_EMMC,
 | |
| };
 | |
| 
 | |
| static int rpmb_sim_hmac_256_alloc(struct rpmb_sim_dev *rsdev)
 | |
| {
 | |
| 	struct crypto_shash *hash_tfm;
 | |
| 
 | |
| 	hash_tfm = crypto_alloc_shash("hmac(sha256)", 0, 0);
 | |
| 	if (IS_ERR(hash_tfm))
 | |
| 		return PTR_ERR(hash_tfm);
 | |
| 
 | |
| 	rsdev->hash_tfm = hash_tfm;
 | |
| 
 | |
| 	dev_dbg(rsdev->dev, "hamac(sha256) registered\n");
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void rpmb_sim_hmac_256_free(struct rpmb_sim_dev *rsdev)
 | |
| {
 | |
| 	if (rsdev->hash_tfm)
 | |
| 		crypto_free_shash(rsdev->hash_tfm);
 | |
| 
 | |
| 	rsdev->hash_tfm = NULL;
 | |
| }
 | |
| 
 | |
| static int rpmb_sim_probe(struct device *dev)
 | |
| {
 | |
| 	struct rpmb_sim_dev *rsdev;
 | |
| 	int ret;
 | |
| 
 | |
| 	rsdev = kzalloc(sizeof(*rsdev), GFP_KERNEL);
 | |
| 	if (!rsdev)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	rsdev->dev = dev;
 | |
| 
 | |
| 	ret = rpmb_sim_hmac_256_alloc(rsdev);
 | |
| 	if (ret)
 | |
| 		goto err;
 | |
| 
 | |
| 	rsdev->capacity = CAPACITY_UNIT * daunits;
 | |
| 	rsdev->blkcnt  = rsdev->capacity / BLK_UNIT;
 | |
| 	rsdev->da = kzalloc(rsdev->capacity, GFP_KERNEL);
 | |
| 	if (!rsdev->da) {
 | |
| 		ret = -ENOMEM;
 | |
| 		goto err;
 | |
| 	}
 | |
| 
 | |
| 	rpmb_sim_ops.dev_id_len = strlen(id);
 | |
| 	rpmb_sim_ops.dev_id = id;
 | |
| 	rpmb_sim_ops.reliable_wr_cnt = max_wr_blks;
 | |
| 
 | |
| 	rsdev->rdev = rpmb_dev_register(rsdev->dev, &rpmb_sim_ops);
 | |
| 	if (IS_ERR(rsdev->rdev)) {
 | |
| 		ret = PTR_ERR(rsdev->rdev);
 | |
| 		goto err;
 | |
| 	}
 | |
| 
 | |
| 	dev_info(dev, "registered RPMB capacity = %zu of %zu blocks\n",
 | |
| 		 rsdev->capacity, rsdev->blkcnt);
 | |
| 
 | |
| 	dev_set_drvdata(dev, rsdev);
 | |
| 
 | |
| 	return 0;
 | |
| err:
 | |
| 	rpmb_sim_hmac_256_free(rsdev);
 | |
| 	if (rsdev)
 | |
| 		kfree(rsdev->da);
 | |
| 	kfree(rsdev);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int rpmb_sim_remove(struct device *dev)
 | |
| {
 | |
| 	struct rpmb_sim_dev *rsdev;
 | |
| 
 | |
| 	rsdev = dev_get_drvdata(dev);
 | |
| 
 | |
| 	rpmb_dev_unregister(rsdev->dev);
 | |
| 
 | |
| 	dev_set_drvdata(dev, NULL);
 | |
| 
 | |
| 	rpmb_sim_hmac_256_free(rsdev);
 | |
| 
 | |
| 	kfree(rsdev->da);
 | |
| 	kfree(rsdev);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void rpmb_sim_shutdown(struct device *dev)
 | |
| {
 | |
| 	rpmb_sim_remove(dev);
 | |
| }
 | |
| 
 | |
| static int rpmb_sim_match(struct device *dev, struct device_driver *drv)
 | |
| {
 | |
| 	return 1;
 | |
| }
 | |
| 
 | |
| static struct bus_type rpmb_sim_bus = {
 | |
| 	.name = "rpmb_sim",
 | |
| 	.match = rpmb_sim_match,
 | |
| };
 | |
| 
 | |
| static struct device_driver rpmb_sim_drv = {
 | |
| 	.name  = "rpmb_sim",
 | |
| 	.probe = rpmb_sim_probe,
 | |
| 	.remove = rpmb_sim_remove,
 | |
| 	.shutdown = rpmb_sim_shutdown,
 | |
| };
 | |
| 
 | |
| static void rpmb_sim_dev_release(struct device *dev)
 | |
| {
 | |
| }
 | |
| 
 | |
| static struct device rpmb_sim_dev;
 | |
| 
 | |
| static int __init rpmb_sim_init(void)
 | |
| {
 | |
| 	int ret;
 | |
| 	struct device *dev = &rpmb_sim_dev;
 | |
| 	struct device_driver *drv = &rpmb_sim_drv;
 | |
| 
 | |
| 	ret = bus_register(&rpmb_sim_bus);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	dev->bus = &rpmb_sim_bus;
 | |
| 	dev->release = rpmb_sim_dev_release;
 | |
| 	dev_set_name(dev, "%s", "rpmb_sim");
 | |
| 	ret = device_register(dev);
 | |
| 	if (ret) {
 | |
| 		pr_notice("device register failed %d\n", ret);
 | |
| 		goto err_device;
 | |
| 	}
 | |
| 
 | |
| 	drv->bus = &rpmb_sim_bus;
 | |
| 	ret = driver_register(drv);
 | |
| 	if (ret) {
 | |
| 		pr_notice("driver register failed %d\n", ret);
 | |
| 		goto err_driver;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_driver:
 | |
| 	device_unregister(dev);
 | |
| err_device:
 | |
| 	bus_unregister(&rpmb_sim_bus);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static void __exit rpmb_sim_exit(void)
 | |
| {
 | |
| 	struct device *dev = &rpmb_sim_dev;
 | |
| 	struct device_driver *drv = &rpmb_sim_drv;
 | |
| 
 | |
| 	device_unregister(dev);
 | |
| 	driver_unregister(drv);
 | |
| 	bus_unregister(&rpmb_sim_bus);
 | |
| }
 | |
| 
 | |
| module_init(rpmb_sim_init);
 | |
| module_exit(rpmb_sim_exit);
 | |
| 
 | |
| MODULE_AUTHOR("Tomas Winkler <tomas.winkler@intel.com");
 | |
| MODULE_LICENSE("Dual BSD/GPL");
 | |
| MODULE_ALIAS("rpmb_sim:rpmb_sim");
 |