pstore: Replace arguments for write_buf_user() API
authorKees Cook <keescook@chromium.org>
Sun, 5 Mar 2017 08:56:38 +0000 (00:56 -0800)
committerKees Cook <keescook@chromium.org>
Tue, 7 Mar 2017 22:01:01 +0000 (14:01 -0800)
Removes argument list in favor of pstore record, though the user buffer
remains passed separately since it must carry the __user annotation.

Signed-off-by: Kees Cook <keescook@chromium.org>
fs/pstore/platform.c
fs/pstore/pmsg.c
fs/pstore/ram.c
include/linux/pstore.h

index 5eecf9012459dd18169ae04036766f00073b1063..1e6642a2063ee7b4e635e479a81c94553b8a8763 100644 (file)
@@ -639,47 +639,36 @@ static int pstore_write_compat(struct pstore_record *record)
        return record->psi->write_buf(record);
 }
 
-static int pstore_write_buf_user_compat(enum pstore_type_id type,
-                              enum kmsg_dump_reason reason,
-                              u64 *id, unsigned int part,
-                              const char __user *buf,
-                              bool compressed, size_t size,
-                              struct pstore_info *psi)
+static int pstore_write_buf_user_compat(struct pstore_record *record,
+                                       const char __user *buf)
 {
        unsigned long flags = 0;
-       size_t i, bufsize = size;
+       size_t i, bufsize, total_size = record->size;
        long ret = 0;
 
-       if (unlikely(!access_ok(VERIFY_READ, buf, size)))
+       if (unlikely(!access_ok(VERIFY_READ, buf, total_size)))
                return -EFAULT;
+       bufsize = total_size;
        if (bufsize > psinfo->bufsize)
                bufsize = psinfo->bufsize;
+       record->buf = psinfo->buf;
        spin_lock_irqsave(&psinfo->buf_lock, flags);
-       for (i = 0; i < size; ) {
-               struct pstore_record record = {
-                       .type = type,
-                       .reason = reason,
-                       .id = id,
-                       .part = part,
-                       .buf = psinfo->buf,
-                       .compressed = compressed,
-                       .psi = psi,
-               };
-               size_t c = min(size - i, bufsize);
+       for (i = 0; i < total_size; ) {
+               size_t c = min(total_size - i, bufsize);
 
-               ret = __copy_from_user(psinfo->buf, buf + i, c);
+               ret = __copy_from_user(record->buf, buf + i, c);
                if (unlikely(ret != 0)) {
                        ret = -EFAULT;
                        break;
                }
-               record.size = c;
-               ret = psi->write_buf(&record);
+               record->size = c;
+               ret = record->psi->write_buf(record);
                if (unlikely(ret < 0))
                        break;
                i += c;
        }
        spin_unlock_irqrestore(&psinfo->buf_lock, flags);
-       return unlikely(ret < 0) ? ret : size;
+       return unlikely(ret < 0) ? ret : total_size;
 }
 
 /*
index 78f6176c020f8256dad089221af85d9560e5ac2f..ce35907602de642d91f1ffc177dea24bb186be63 100644 (file)
@@ -23,7 +23,11 @@ static DEFINE_MUTEX(pmsg_lock);
 static ssize_t write_pmsg(struct file *file, const char __user *buf,
                          size_t count, loff_t *ppos)
 {
-       u64 id;
+       struct pstore_record record = {
+               .type = PSTORE_TYPE_PMSG,
+               .size = count,
+               .psi = psinfo,
+       };
        int ret;
 
        if (!count)
@@ -34,8 +38,7 @@ static ssize_t write_pmsg(struct file *file, const char __user *buf,
                return -EFAULT;
 
        mutex_lock(&pmsg_lock);
-       ret = psinfo->write_buf_user(PSTORE_TYPE_PMSG, 0, &id, 0, buf, 0, count,
-                                    psinfo);
+       ret = psinfo->write_buf_user(&record, buf);
        mutex_unlock(&pmsg_lock);
        return ret ? ret : count;
 }
index a7cdde60b1f93d62353cbc896ed717bc91fbdb95..d85e1adae1b6939038f084226beb3abf988cae68 100644 (file)
@@ -451,19 +451,15 @@ static int notrace ramoops_pstore_write_buf(struct pstore_record *record)
        return 0;
 }
 
-static int notrace ramoops_pstore_write_buf_user(enum pstore_type_id type,
-                                                enum kmsg_dump_reason reason,
-                                                u64 *id, unsigned int part,
-                                                const char __user *buf,
-                                                bool compressed, size_t size,
-                                                struct pstore_info *psi)
+static int notrace ramoops_pstore_write_buf_user(struct pstore_record *record,
+                                                const char __user *buf)
 {
-       if (type == PSTORE_TYPE_PMSG) {
-               struct ramoops_context *cxt = psi->data;
+       if (record->type == PSTORE_TYPE_PMSG) {
+               struct ramoops_context *cxt = record->psi->data;
 
                if (!cxt->mprz)
                        return -ENOMEM;
-               return persistent_ram_write_user(cxt->mprz, buf, size);
+               return persistent_ram_write_user(cxt->mprz, buf, record->size);
        }
 
        return -EINVAL;
index cbf5e561778d8736133eee551197603d409f8c76..9b85d3eeca83ece85d0282b219ead2b880a5d7bb 100644 (file)
@@ -152,18 +152,11 @@ struct pstore_record {
  *
  * @write_buf_user:
  *     Perform a frontend write to a backend record, using a specified
- *     buffer that is coming directly from userspace.
- *
- *     @type:  in: pstore record type to write
- *     @reason:
- *             in: pstore write reason
- *     @id:    out: unique identifier for the record
- *     @part:  in: position in a multipart write
- *     @buf:   in: pointer to userspace contents to write to backend record
- *     @compressed:
- *             in: if the record is compressed
- *     @size:  in: size of the write
- *     @psi:   in: pointer to the struct pstore_info for the backend
+ *     buffer that is coming directly from userspace, instead of the
+ *     @record @buf.
+ *
+ *     @record:        pointer to record metadata.
+ *     @buf:           pointer to userspace contents to write to backend
  *
  *     Returns 0 on success, and non-zero on error.
  *
@@ -196,10 +189,8 @@ struct pstore_info {
        ssize_t         (*read)(struct pstore_record *record);
        int             (*write)(struct pstore_record *record);
        int             (*write_buf)(struct pstore_record *record);
-       int             (*write_buf_user)(enum pstore_type_id type,
-                       enum kmsg_dump_reason reason, u64 *id,
-                       unsigned int part, const char __user *buf,
-                       bool compressed, size_t size, struct pstore_info *psi);
+       int             (*write_buf_user)(struct pstore_record *record,
+                                         const char __user *buf);
        int             (*erase)(struct pstore_record *record);
 };