/* SPDX-License-Identifier: GPL-2.0 */ /* * Copyright (C) 2015-2019 Intel Corp. All rights reserved * Copyright (C) 2021-2022 Linaro Ltd */ #ifndef __RPMB_H__ #define __RPMB_H__ #include #include /** * enum rpmb_type - type of underlying storage technology * * @RPMB_TYPE_EMMC : emmc (JESD84-B50.1) * @RPMB_TYPE_UFS : UFS (JESD220) * @RPMB_TYPE_NVME : NVM Express */ enum rpmb_type { RPMB_TYPE_EMMC, RPMB_TYPE_UFS, RPMB_TYPE_NVME, }; /** * struct rpmb_descr - RPMB description provided by the underlying block device * * @type : block device type * @route_frames : routes frames to and from the RPMB device * @dev_id : unique device identifier read from the hardware * @dev_id_len : length of unique device identifier * @reliable_wr_count: number of sectors that can be written in one access * @capacity : capacity of the device in units of 128K * * @dev_id is intended to be used as input when deriving the authenticaion key. */ struct rpmb_descr { enum rpmb_type type; int (*route_frames)(struct device *dev, u8 *req, unsigned int req_len, u8 *resp, unsigned int resp_len); u8 *dev_id; size_t dev_id_len; u16 reliable_wr_count; u16 capacity; }; /** * struct rpmb_dev - device which can support RPMB partition * * @dev : device * @id : device_id * @list_node : linked list node * @descr : RPMB description */ struct rpmb_dev { struct device dev; int id; struct list_head list_node; struct rpmb_descr descr; }; #define to_rpmb_dev(x) container_of((x), struct rpmb_dev, dev) /** * struct rpmb_frame - RPMB frame structure for authenticated access * * @stuff : stuff bytes, a padding/reserved area of 196 bytes at the * beginning of the RPMB frame. They don’t carry meaningful * data but are required to make the frame exactly 512 bytes. * @key_mac : The authentication key or the message authentication * code (MAC) depending on the request/response type. * The MAC will be delivered in the last (or the only) * block of data. * @data : Data to be written or read by signed access. * @nonce : Random number generated by the host for the requests * and copied to the response by the RPMB engine. * @write_counter: Counter value for the total amount of the successful * authenticated data write requests made by the host. * @addr : Address of the data to be programmed to or read * from the RPMB. Address is the serial number of * the accessed block (half sector 256B). * @block_count : Number of blocks (half sectors, 256B) requested to be * read/programmed. * @result : Includes information about the status of the write counter * (valid, expired) and result of the access made to the RPMB. * @req_resp : Defines the type of request and response to/from the memory. * * The stuff bytes and big-endian properties are modeled to fit to the spec. */ struct rpmb_frame { u8 stuff[196]; u8 key_mac[32]; u8 data[256]; u8 nonce[16]; __be32 write_counter; __be16 addr; __be16 block_count; __be16 result; __be16 req_resp; }; #define RPMB_PROGRAM_KEY 0x1 /* Program RPMB Authentication Key */ #define RPMB_GET_WRITE_COUNTER 0x2 /* Read RPMB write counter */ #define RPMB_WRITE_DATA 0x3 /* Write data to RPMB partition */ #define RPMB_READ_DATA 0x4 /* Read data from RPMB partition */ #define RPMB_RESULT_READ 0x5 /* Read result request (Internal) */ #if IS_ENABLED(CONFIG_RPMB) struct rpmb_dev *rpmb_dev_get(struct rpmb_dev *rdev); void rpmb_dev_put(struct rpmb_dev *rdev); struct rpmb_dev *rpmb_dev_find_device(const void *data, const struct rpmb_dev *start, int (*match)(struct device *dev, const void *data)); int rpmb_interface_register(struct class_interface *intf); void rpmb_interface_unregister(struct class_interface *intf); struct rpmb_dev *rpmb_dev_register(struct device *dev, struct rpmb_descr *descr); int rpmb_dev_unregister(struct rpmb_dev *rdev); int rpmb_route_frames(struct rpmb_dev *rdev, u8 *req, unsigned int req_len, u8 *resp, unsigned int resp_len); #else static inline struct rpmb_dev *rpmb_dev_get(struct rpmb_dev *rdev) { return NULL; } static inline void rpmb_dev_put(struct rpmb_dev *rdev) { } static inline struct rpmb_dev * rpmb_dev_find_device(const void *data, const struct rpmb_dev *start, int (*match)(struct device *dev, const void *data)) { return NULL; } static inline int rpmb_interface_register(struct class_interface *intf) { return -EOPNOTSUPP; } static inline void rpmb_interface_unregister(struct class_interface *intf) { } static inline struct rpmb_dev * rpmb_dev_register(struct device *dev, struct rpmb_descr *descr) { return NULL; } static inline int rpmb_dev_unregister(struct rpmb_dev *dev) { return 0; } static inline int rpmb_route_frames(struct rpmb_dev *rdev, u8 *req, unsigned int req_len, u8 *resp, unsigned int resp_len) { return -EOPNOTSUPP; } #endif /* CONFIG_RPMB */ #endif /* __RPMB_H__ */