aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/nvdimm/dimm_devs.c11
-rw-r--r--drivers/nvdimm/nd-core.h7
-rw-r--r--drivers/nvdimm/security.c58
3 files changed, 69 insertions, 7 deletions
diff --git a/drivers/nvdimm/dimm_devs.c b/drivers/nvdimm/dimm_devs.c
index 7f42cc4e119b..1cc3a6af3d0e 100644
--- a/drivers/nvdimm/dimm_devs.c
+++ b/drivers/nvdimm/dimm_devs.c
@@ -392,8 +392,9 @@ static ssize_t security_show(struct device *dev,
392} 392}
393 393
394#define OPS \ 394#define OPS \
395 C( OP_FREEZE, "freeze", 1), \ 395 C( OP_FREEZE, "freeze", 1), \
396 C( OP_DISABLE, "disable", 2) 396 C( OP_DISABLE, "disable", 2), \
397 C( OP_UPDATE, "update", 3)
397#undef C 398#undef C
398#define C(a, b, c) a 399#define C(a, b, c) a
399enum nvdimmsec_op_ids { OPS }; 400enum nvdimmsec_op_ids { OPS };
@@ -444,6 +445,9 @@ static ssize_t __security_store(struct device *dev, const char *buf, size_t len)
444 } else if (i == OP_DISABLE) { 445 } else if (i == OP_DISABLE) {
445 dev_dbg(dev, "disable %u\n", key); 446 dev_dbg(dev, "disable %u\n", key);
446 rc = nvdimm_security_disable(nvdimm, key); 447 rc = nvdimm_security_disable(nvdimm, key);
448 } else if (i == OP_UPDATE) {
449 dev_dbg(dev, "update %u %u\n", key, newkey);
450 rc = nvdimm_security_update(nvdimm, key, newkey);
447 } else 451 } else
448 return -EINVAL; 452 return -EINVAL;
449 453
@@ -493,7 +497,8 @@ static umode_t nvdimm_visible(struct kobject *kobj, struct attribute *a, int n)
493 if (nvdimm->sec.state < 0) 497 if (nvdimm->sec.state < 0)
494 return 0; 498 return 0;
495 /* Are there any state mutation ops? */ 499 /* Are there any state mutation ops? */
496 if (nvdimm->sec.ops->freeze || nvdimm->sec.ops->disable) 500 if (nvdimm->sec.ops->freeze || nvdimm->sec.ops->disable
501 || nvdimm->sec.ops->change_key)
497 return a->mode; 502 return a->mode;
498 return 0444; 503 return 0444;
499} 504}
diff --git a/drivers/nvdimm/nd-core.h b/drivers/nvdimm/nd-core.h
index d1351c0b1119..c2567f9ae07b 100644
--- a/drivers/nvdimm/nd-core.h
+++ b/drivers/nvdimm/nd-core.h
@@ -59,12 +59,19 @@ static inline enum nvdimm_security_state nvdimm_security_state(
59int nvdimm_security_freeze(struct nvdimm *nvdimm); 59int nvdimm_security_freeze(struct nvdimm *nvdimm);
60#if IS_ENABLED(CONFIG_NVDIMM_KEYS) 60#if IS_ENABLED(CONFIG_NVDIMM_KEYS)
61int nvdimm_security_disable(struct nvdimm *nvdimm, unsigned int keyid); 61int nvdimm_security_disable(struct nvdimm *nvdimm, unsigned int keyid);
62int nvdimm_security_update(struct nvdimm *nvdimm, unsigned int keyid,
63 unsigned int new_keyid);
62#else 64#else
63static inline int nvdimm_security_disable(struct nvdimm *nvdimm, 65static inline int nvdimm_security_disable(struct nvdimm *nvdimm,
64 unsigned int keyid) 66 unsigned int keyid)
65{ 67{
66 return -EOPNOTSUPP; 68 return -EOPNOTSUPP;
67} 69}
70static inline int nvdimm_security_update(struct nvdimm *nvdimm, unsigned int keyid,
71 unsigned int new_keyid)
72{
73 return -EOPNOTSUPP;
74}
68#endif 75#endif
69 76
70/** 77/**
diff --git a/drivers/nvdimm/security.c b/drivers/nvdimm/security.c
index 647a99dd3182..df7f070e96fb 100644
--- a/drivers/nvdimm/security.c
+++ b/drivers/nvdimm/security.c
@@ -15,6 +15,9 @@
15#include "nd-core.h" 15#include "nd-core.h"
16#include "nd.h" 16#include "nd.h"
17 17
18#define NVDIMM_BASE_KEY 0
19#define NVDIMM_NEW_KEY 1
20
18static bool key_revalidate = true; 21static bool key_revalidate = true;
19module_param(key_revalidate, bool, 0444); 22module_param(key_revalidate, bool, 0444);
20MODULE_PARM_DESC(key_revalidate, "Require key validation at init."); 23MODULE_PARM_DESC(key_revalidate, "Require key validation at init.");
@@ -70,7 +73,7 @@ static struct key *nvdimm_request_key(struct nvdimm *nvdimm)
70} 73}
71 74
72static struct key *nvdimm_lookup_user_key(struct nvdimm *nvdimm, 75static struct key *nvdimm_lookup_user_key(struct nvdimm *nvdimm,
73 key_serial_t id) 76 key_serial_t id, int subclass)
74{ 77{
75 key_ref_t keyref; 78 key_ref_t keyref;
76 struct key *key; 79 struct key *key;
@@ -86,10 +89,10 @@ static struct key *nvdimm_lookup_user_key(struct nvdimm *nvdimm,
86 key_put(key); 89 key_put(key);
87 return NULL; 90 return NULL;
88 } 91 }
89 dev_dbg(dev, "%s: key found: %#x\n", __func__, key_serial(key));
90 92
93 dev_dbg(dev, "%s: key found: %#x\n", __func__, key_serial(key));
91 94
92 down_read(&key->sem); 95 down_read_nested(&key->sem, subclass);
93 epayload = dereference_key_locked(key); 96 epayload = dereference_key_locked(key);
94 if (epayload->decrypted_datalen != NVDIMM_PASSPHRASE_LEN) { 97 if (epayload->decrypted_datalen != NVDIMM_PASSPHRASE_LEN) {
95 up_read(&key->sem); 98 up_read(&key->sem);
@@ -197,7 +200,7 @@ int nvdimm_security_disable(struct nvdimm *nvdimm, unsigned int keyid)
197 return -EIO; 200 return -EIO;
198 } 201 }
199 202
200 key = nvdimm_lookup_user_key(nvdimm, keyid); 203 key = nvdimm_lookup_user_key(nvdimm, keyid, NVDIMM_BASE_KEY);
201 if (!key) 204 if (!key)
202 return -ENOKEY; 205 return -ENOKEY;
203 206
@@ -209,3 +212,50 @@ int nvdimm_security_disable(struct nvdimm *nvdimm, unsigned int keyid)
209 nvdimm->sec.state = nvdimm_security_state(nvdimm); 212 nvdimm->sec.state = nvdimm_security_state(nvdimm);
210 return rc; 213 return rc;
211} 214}
215
216int nvdimm_security_update(struct nvdimm *nvdimm, unsigned int keyid,
217 unsigned int new_keyid)
218{
219 struct device *dev = &nvdimm->dev;
220 struct nvdimm_bus *nvdimm_bus = walk_to_nvdimm_bus(dev);
221 struct key *key, *newkey;
222 int rc;
223
224 /* The bus lock should be held at the top level of the call stack */
225 lockdep_assert_held(&nvdimm_bus->reconfig_mutex);
226
227 if (!nvdimm->sec.ops || !nvdimm->sec.ops->change_key
228 || nvdimm->sec.state < 0)
229 return -EOPNOTSUPP;
230
231 if (nvdimm->sec.state >= NVDIMM_SECURITY_FROZEN) {
232 dev_warn(dev, "Incorrect security state: %d\n",
233 nvdimm->sec.state);
234 return -EIO;
235 }
236
237 if (keyid == 0)
238 key = NULL;
239 else {
240 key = nvdimm_lookup_user_key(nvdimm, keyid, NVDIMM_BASE_KEY);
241 if (!key)
242 return -ENOKEY;
243 }
244
245 newkey = nvdimm_lookup_user_key(nvdimm, new_keyid, NVDIMM_NEW_KEY);
246 if (!newkey) {
247 nvdimm_put_key(key);
248 return -ENOKEY;
249 }
250
251 rc = nvdimm->sec.ops->change_key(nvdimm, key ? key_data(key) : NULL,
252 key_data(newkey));
253 dev_dbg(dev, "key: %d %d update: %s\n",
254 key_serial(key), key_serial(newkey),
255 rc == 0 ? "success" : "fail");
256
257 nvdimm_put_key(newkey);
258 nvdimm_put_key(key);
259 nvdimm->sec.state = nvdimm_security_state(nvdimm);
260 return rc;
261}