#include <linux/sed-opal.h>
 #include <linux/string.h>
 #include <linux/kdev_t.h>
+#include <linux/key.h>
+#include <linux/key-type.h>
+#include <keys/user-type.h>
 
 #include "opal_proto.h"
 
 /* Number of bytes needed by cmd_finalize. */
 #define CMD_FINALIZE_BYTES_NEEDED 7
 
+static struct key *sed_opal_keyring;
+
 struct opal_step {
        int (*fn)(struct opal_dev *dev, void *data);
        void *data;
 #endif
 }
 
+/*
+ * Allocate/update a SED Opal key and add it to the SED Opal keyring.
+ */
+static int update_sed_opal_key(const char *desc, u_char *key_data, int keylen)
+{
+       key_ref_t kr;
+
+       if (!sed_opal_keyring)
+               return -ENOKEY;
+
+       kr = key_create_or_update(make_key_ref(sed_opal_keyring, true), "user",
+                                 desc, (const void *)key_data, keylen,
+                                 KEY_USR_VIEW | KEY_USR_SEARCH | KEY_USR_WRITE,
+                                 KEY_ALLOC_NOT_IN_QUOTA | KEY_ALLOC_BUILT_IN |
+                                       KEY_ALLOC_BYPASS_RESTRICTION);
+       if (IS_ERR(kr)) {
+               pr_err("Error adding SED key (%ld)\n", PTR_ERR(kr));
+               return PTR_ERR(kr);
+       }
+
+       return 0;
+}
+
+/*
+ * Read a SED Opal key from the SED Opal keyring.
+ */
+static int read_sed_opal_key(const char *key_name, u_char *buffer, int buflen)
+{
+       int ret;
+       key_ref_t kref;
+       struct key *key;
+
+       if (!sed_opal_keyring)
+               return -ENOKEY;
+
+       kref = keyring_search(make_key_ref(sed_opal_keyring, true),
+                             &key_type_user, key_name, true);
+
+       if (IS_ERR(kref))
+               ret = PTR_ERR(kref);
+
+       key = key_ref_to_ptr(kref);
+       down_read(&key->sem);
+       ret = key_validate(key);
+       if (ret == 0) {
+               if (buflen > key->datalen)
+                       buflen = key->datalen;
+
+               ret = key->type->read(key, (char *)buffer, buflen);
+       }
+       up_read(&key->sem);
+
+       key_ref_put(kref);
+
+       return ret;
+}
+
+static int opal_get_key(struct opal_dev *dev, struct opal_key *key)
+{
+       int ret = 0;
+
+       switch (key->key_type) {
+       case OPAL_INCLUDED:
+               /* the key is ready to use */
+               break;
+       case OPAL_KEYRING:
+               /* the key is in the keyring */
+               ret = read_sed_opal_key(OPAL_AUTH_KEY, key->key, OPAL_KEY_MAX);
+               if (ret > 0) {
+                       if (ret > U8_MAX) {
+                               ret = -ENOSPC;
+                               goto error;
+                       }
+                       key->key_len = ret;
+                       key->key_type = OPAL_INCLUDED;
+               }
+               break;
+       default:
+               ret = -EINVAL;
+               break;
+       }
+       if (ret < 0)
+               goto error;
+
+       /* must have a PEK by now or it's an error */
+       if (key->key_type != OPAL_INCLUDED || key->key_len == 0) {
+               ret = -EINVAL;
+               goto error;
+       }
+       return 0;
+error:
+       pr_debug("Error getting password: %d\n", ret);
+       return ret;
+}
+
 static bool check_tper(const void *data)
 {
        const struct d0_tper_features *tper = data;
        };
        int ret;
 
+       ret = opal_get_key(dev, &opal_session->opal_key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, erase_steps, ARRAY_SIZE(erase_steps));
        };
        int ret;
 
+       ret = opal_get_key(dev, &rev->key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, steps, ARRAY_SIZE(steps));
        };
        int ret;
 
+       ret = opal_get_key(dev, &opal_session->opal_key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, erase_steps, ARRAY_SIZE(erase_steps));
            opal_mbr->enable_disable != OPAL_MBR_DISABLE)
                return -EINVAL;
 
+       ret = opal_get_key(dev, &opal_mbr->key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, mbr_steps, ARRAY_SIZE(mbr_steps));
            mbr_done->done_flag != OPAL_MBR_NOT_DONE)
                return -EINVAL;
 
+       ret = opal_get_key(dev, &mbr_done->key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, mbr_steps, ARRAY_SIZE(mbr_steps));
        if (info->size == 0)
                return 0;
 
+       ret = opal_get_key(dev, &info->key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, mbr_steps, ARRAY_SIZE(mbr_steps));
                return -EINVAL;
        }
 
+       ret = opal_get_key(dev, &lk_unlk->session.opal_key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, steps, ARRAY_SIZE(steps));
 
        int ret;
 
+       ret = opal_get_key(dev, opal);
+
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        if (psid)
        if (lk_unlk->session.who > OPAL_USER9)
                return -EINVAL;
 
+       ret = opal_get_key(dev, &lk_unlk->session.opal_key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        opal_lock_check_for_saved_key(dev, lk_unlk);
        ret = __opal_lock_unlock(dev, lk_unlk);
        if (!dev)
                return -ENODEV;
 
+       ret = opal_get_key(dev, opal);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, owner_steps, ARRAY_SIZE(owner_steps));
        if (!opal_lr_act->num_lrs || opal_lr_act->num_lrs > OPAL_MAX_LRS)
                return -EINVAL;
 
+       ret = opal_get_key(dev, &opal_lr_act->key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, active_steps, ARRAY_SIZE(active_steps));
        };
        int ret;
 
+       ret = opal_get_key(dev, &opal_lrs->session.opal_key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, lr_steps, ARRAY_SIZE(lr_steps));
        ret = execute_steps(dev, pw_steps, ARRAY_SIZE(pw_steps));
        mutex_unlock(&dev->dev_lock);
 
+       if (ret)
+               return ret;
+
+       /* update keyring with new password */
+       ret = update_sed_opal_key(OPAL_AUTH_KEY,
+                                 opal_pw->new_user_pw.opal_key.key,
+                                 opal_pw->new_user_pw.opal_key.key_len);
+
        return ret;
 }
 
                return -EINVAL;
        }
 
+       ret = opal_get_key(dev, &opal_session->opal_key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
        ret = execute_steps(dev, act_steps, ARRAY_SIZE(act_steps));
 {
        int ret, bit_set;
 
+       ret = opal_get_key(dev, &rw_tbl->key);
+       if (ret)
+               return ret;
        mutex_lock(&dev->dev_lock);
        setup_opal_dev(dev);
 
        if (!capable(CAP_SYS_ADMIN))
                return -EACCES;
        if (!dev)
-               return -ENOTSUPP;
+               return -EOPNOTSUPP;
        if (!(dev->flags & OPAL_FL_SUPPORTED))
-               return -ENOTSUPP;
+               return -EOPNOTSUPP;
 
        if (cmd & IOC_IN) {
                p = memdup_user(arg, _IOC_SIZE(cmd));
        return ret;
 }
 EXPORT_SYMBOL_GPL(sed_ioctl);
+
+static int __init sed_opal_init(void)
+{
+       struct key *kr;
+
+       kr = keyring_alloc(".sed_opal",
+                          GLOBAL_ROOT_UID, GLOBAL_ROOT_GID, current_cred(),
+                          (KEY_POS_ALL & ~KEY_POS_SETATTR) | KEY_USR_VIEW |
+                          KEY_USR_READ | KEY_USR_SEARCH | KEY_USR_WRITE,
+                          KEY_ALLOC_NOT_IN_QUOTA,
+                          NULL, NULL);
+       if (IS_ERR(kr))
+               return PTR_ERR(kr);
+
+       sed_opal_keyring = kr;
+
+       return 0;
+}
+late_initcall(sed_opal_init);