Yangtao Li | 055f5df | 2019-04-13 11:33:04 +0100 | [diff] [blame] | 1 | // SPDX-License-Identifier: GPL-2.0+ |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 2 | /* |
| 3 | * Allwinner sunXi SoCs Security ID support. |
| 4 | * |
| 5 | * Copyright (c) 2013 Oliver Schinagl <oliver@schinagl.nl> |
| 6 | * Copyright (C) 2014 Maxime Ripard <maxime.ripard@free-electrons.com> |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 7 | */ |
| 8 | |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 9 | #include <linux/device.h> |
| 10 | #include <linux/io.h> |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 11 | #include <linux/iopoll.h> |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 12 | #include <linux/module.h> |
| 13 | #include <linux/nvmem-provider.h> |
| 14 | #include <linux/of.h> |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 15 | #include <linux/of_device.h> |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 16 | #include <linux/platform_device.h> |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 17 | #include <linux/slab.h> |
| 18 | #include <linux/random.h> |
| 19 | |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 20 | /* Registers and special values for doing register-based SID readout on H3 */ |
| 21 | #define SUN8I_SID_PRCTL 0x40 |
| 22 | #define SUN8I_SID_RDKEY 0x60 |
| 23 | |
| 24 | #define SUN8I_SID_OFFSET_MASK 0x1FF |
| 25 | #define SUN8I_SID_OFFSET_SHIFT 16 |
| 26 | #define SUN8I_SID_OP_LOCK (0xAC << 8) |
| 27 | #define SUN8I_SID_READ BIT(1) |
| 28 | |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 29 | struct sunxi_sid_cfg { |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 30 | u32 value_offset; |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 31 | u32 size; |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 32 | bool need_register_readout; |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 33 | }; |
| 34 | |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 35 | struct sunxi_sid { |
| 36 | void __iomem *base; |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 37 | u32 value_offset; |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 38 | }; |
| 39 | |
Srinivas Kandagatla | 9c7b16e | 2016-04-24 20:28:10 +0100 | [diff] [blame] | 40 | static int sunxi_sid_read(void *context, unsigned int offset, |
| 41 | void *val, size_t bytes) |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 42 | { |
| 43 | struct sunxi_sid *sid = context; |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 44 | |
Chen-Yu Tsai | 273a474 | 2019-04-13 11:32:52 +0100 | [diff] [blame] | 45 | memcpy_fromio(val, sid->base + sid->value_offset + offset, bytes); |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 46 | |
| 47 | return 0; |
| 48 | } |
| 49 | |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 50 | static int sun8i_sid_register_readout(const struct sunxi_sid *sid, |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 51 | const unsigned int offset, |
| 52 | u32 *out) |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 53 | { |
| 54 | u32 reg_val; |
| 55 | int ret; |
| 56 | |
| 57 | /* Set word, lock access, and set read command */ |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 58 | reg_val = (offset & SUN8I_SID_OFFSET_MASK) |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 59 | << SUN8I_SID_OFFSET_SHIFT; |
| 60 | reg_val |= SUN8I_SID_OP_LOCK | SUN8I_SID_READ; |
| 61 | writel(reg_val, sid->base + SUN8I_SID_PRCTL); |
| 62 | |
| 63 | ret = readl_poll_timeout(sid->base + SUN8I_SID_PRCTL, reg_val, |
| 64 | !(reg_val & SUN8I_SID_READ), 100, 250000); |
| 65 | if (ret) |
| 66 | return ret; |
| 67 | |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 68 | if (out) |
| 69 | *out = readl(sid->base + SUN8I_SID_RDKEY); |
| 70 | |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 71 | writel(0, sid->base + SUN8I_SID_PRCTL); |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 72 | |
| 73 | return 0; |
| 74 | } |
| 75 | |
| 76 | /* |
| 77 | * On Allwinner H3, the value on the 0x200 offset of the SID controller seems |
| 78 | * to be not reliable at all. |
| 79 | * Read by the registers instead. |
| 80 | */ |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 81 | static int sun8i_sid_read_by_reg(void *context, unsigned int offset, |
| 82 | void *val, size_t bytes) |
| 83 | { |
| 84 | struct sunxi_sid *sid = context; |
Chen-Yu Tsai | de2a3ea | 2019-04-13 11:32:50 +0100 | [diff] [blame] | 85 | u32 word; |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 86 | int ret; |
| 87 | |
Chen-Yu Tsai | de2a3ea | 2019-04-13 11:32:50 +0100 | [diff] [blame] | 88 | /* .stride = 4 so offset is guaranteed to be aligned */ |
| 89 | while (bytes >= 4) { |
| 90 | ret = sun8i_sid_register_readout(sid, offset, val); |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 91 | if (ret) |
| 92 | return ret; |
Chen-Yu Tsai | de2a3ea | 2019-04-13 11:32:50 +0100 | [diff] [blame] | 93 | |
| 94 | val += 4; |
| 95 | offset += 4; |
| 96 | bytes -= 4; |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 97 | } |
| 98 | |
Chen-Yu Tsai | de2a3ea | 2019-04-13 11:32:50 +0100 | [diff] [blame] | 99 | if (!bytes) |
| 100 | return 0; |
| 101 | |
| 102 | /* Handle any trailing bytes */ |
| 103 | ret = sun8i_sid_register_readout(sid, offset, &word); |
| 104 | if (ret) |
| 105 | return ret; |
| 106 | |
| 107 | memcpy(val, &word, bytes); |
| 108 | |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 109 | return 0; |
| 110 | } |
| 111 | |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 112 | static int sunxi_sid_probe(struct platform_device *pdev) |
| 113 | { |
| 114 | struct device *dev = &pdev->dev; |
| 115 | struct resource *res; |
Chen-Yu Tsai | 7fa5ad2 | 2019-04-13 11:32:51 +0100 | [diff] [blame] | 116 | struct nvmem_config *nvmem_cfg; |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 117 | struct nvmem_device *nvmem; |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 118 | struct sunxi_sid *sid; |
Chen-Yu Tsai | 9c4adfb | 2019-04-13 11:32:49 +0100 | [diff] [blame] | 119 | int size; |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 120 | char *randomness; |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 121 | const struct sunxi_sid_cfg *cfg; |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 122 | |
| 123 | sid = devm_kzalloc(dev, sizeof(*sid), GFP_KERNEL); |
| 124 | if (!sid) |
| 125 | return -ENOMEM; |
| 126 | |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 127 | cfg = of_device_get_match_data(dev); |
| 128 | if (!cfg) |
| 129 | return -EINVAL; |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 130 | sid->value_offset = cfg->value_offset; |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 131 | |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 132 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
| 133 | sid->base = devm_ioremap_resource(dev, res); |
| 134 | if (IS_ERR(sid->base)) |
| 135 | return PTR_ERR(sid->base); |
| 136 | |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 137 | size = cfg->size; |
| 138 | |
Chen-Yu Tsai | 7fa5ad2 | 2019-04-13 11:32:51 +0100 | [diff] [blame] | 139 | nvmem_cfg = devm_kzalloc(dev, sizeof(*nvmem_cfg), GFP_KERNEL); |
| 140 | if (!nvmem_cfg) |
| 141 | return -ENOMEM; |
| 142 | |
| 143 | nvmem_cfg->dev = dev; |
| 144 | nvmem_cfg->name = "sunxi-sid"; |
| 145 | nvmem_cfg->read_only = true; |
| 146 | nvmem_cfg->size = cfg->size; |
| 147 | nvmem_cfg->word_size = 1; |
| 148 | nvmem_cfg->stride = 4; |
| 149 | nvmem_cfg->priv = sid; |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 150 | if (cfg->need_register_readout) |
Chen-Yu Tsai | 7fa5ad2 | 2019-04-13 11:32:51 +0100 | [diff] [blame] | 151 | nvmem_cfg->reg_read = sun8i_sid_read_by_reg; |
Icenowy Zheng | 0ab09d6 | 2018-03-09 14:47:17 +0000 | [diff] [blame] | 152 | else |
Chen-Yu Tsai | 7fa5ad2 | 2019-04-13 11:32:51 +0100 | [diff] [blame] | 153 | nvmem_cfg->reg_read = sunxi_sid_read; |
| 154 | |
| 155 | nvmem = devm_nvmem_register(dev, nvmem_cfg); |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 156 | if (IS_ERR(nvmem)) |
| 157 | return PTR_ERR(nvmem); |
| 158 | |
Kees Cook | 6396bb2 | 2018-06-12 14:03:40 -0700 | [diff] [blame] | 159 | randomness = kzalloc(size, GFP_KERNEL); |
Bartosz Golaszewski | 6eed8dd | 2018-09-21 06:40:10 -0700 | [diff] [blame] | 160 | if (!randomness) |
| 161 | return -ENOMEM; |
Maxime Ripard | fb72707 | 2015-09-30 13:36:31 +0100 | [diff] [blame] | 162 | |
Chen-Yu Tsai | 7fa5ad2 | 2019-04-13 11:32:51 +0100 | [diff] [blame] | 163 | nvmem_cfg->reg_read(sid, 0, randomness, size); |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 164 | add_device_randomness(randomness, size); |
| 165 | kfree(randomness); |
| 166 | |
| 167 | platform_set_drvdata(pdev, nvmem); |
| 168 | |
| 169 | return 0; |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 170 | } |
| 171 | |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 172 | static const struct sunxi_sid_cfg sun4i_a10_cfg = { |
| 173 | .size = 0x10, |
| 174 | }; |
| 175 | |
| 176 | static const struct sunxi_sid_cfg sun7i_a20_cfg = { |
| 177 | .size = 0x200, |
| 178 | }; |
| 179 | |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 180 | static const struct sunxi_sid_cfg sun8i_h3_cfg = { |
| 181 | .value_offset = 0x200, |
| 182 | .size = 0x100, |
| 183 | .need_register_readout = true, |
| 184 | }; |
| 185 | |
Icenowy Zheng | b7fe57b | 2017-10-24 10:54:34 +0100 | [diff] [blame] | 186 | static const struct sunxi_sid_cfg sun50i_a64_cfg = { |
| 187 | .value_offset = 0x200, |
| 188 | .size = 0x100, |
Stefan Mavrodiev | 2ac00e3 | 2019-08-18 10:33:41 +0100 | [diff] [blame] | 189 | .need_register_readout = true, |
Icenowy Zheng | b7fe57b | 2017-10-24 10:54:34 +0100 | [diff] [blame] | 190 | }; |
| 191 | |
Yangtao Li | fc1eb6e | 2019-04-13 11:33:05 +0100 | [diff] [blame] | 192 | static const struct sunxi_sid_cfg sun50i_h6_cfg = { |
| 193 | .value_offset = 0x200, |
| 194 | .size = 0x200, |
| 195 | }; |
| 196 | |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 197 | static const struct of_device_id sunxi_sid_of_match[] = { |
Icenowy Zheng | 4a72cda | 2017-03-31 13:44:47 +0100 | [diff] [blame] | 198 | { .compatible = "allwinner,sun4i-a10-sid", .data = &sun4i_a10_cfg }, |
| 199 | { .compatible = "allwinner,sun7i-a20-sid", .data = &sun7i_a20_cfg }, |
Chen-Yu Tsai | da75b89 | 2019-04-13 11:32:53 +0100 | [diff] [blame] | 200 | { .compatible = "allwinner,sun8i-a83t-sid", .data = &sun50i_a64_cfg }, |
Icenowy Zheng | 1a96364 | 2017-03-31 13:44:48 +0100 | [diff] [blame] | 201 | { .compatible = "allwinner,sun8i-h3-sid", .data = &sun8i_h3_cfg }, |
Icenowy Zheng | b7fe57b | 2017-10-24 10:54:34 +0100 | [diff] [blame] | 202 | { .compatible = "allwinner,sun50i-a64-sid", .data = &sun50i_a64_cfg }, |
Chen-Yu Tsai | da75b89 | 2019-04-13 11:32:53 +0100 | [diff] [blame] | 203 | { .compatible = "allwinner,sun50i-h5-sid", .data = &sun50i_a64_cfg }, |
Yangtao Li | fc1eb6e | 2019-04-13 11:33:05 +0100 | [diff] [blame] | 204 | { .compatible = "allwinner,sun50i-h6-sid", .data = &sun50i_h6_cfg }, |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 205 | {/* sentinel */}, |
| 206 | }; |
| 207 | MODULE_DEVICE_TABLE(of, sunxi_sid_of_match); |
| 208 | |
| 209 | static struct platform_driver sunxi_sid_driver = { |
| 210 | .probe = sunxi_sid_probe, |
Maxime Ripard | 3d0b16a | 2015-07-27 12:17:09 +0100 | [diff] [blame] | 211 | .driver = { |
| 212 | .name = "eeprom-sunxi-sid", |
| 213 | .of_match_table = sunxi_sid_of_match, |
| 214 | }, |
| 215 | }; |
| 216 | module_platform_driver(sunxi_sid_driver); |
| 217 | |
| 218 | MODULE_AUTHOR("Oliver Schinagl <oliver@schinagl.nl>"); |
| 219 | MODULE_DESCRIPTION("Allwinner sunxi security id driver"); |
| 220 | MODULE_LICENSE("GPL"); |