/* Copyright (c) 2010-2014,2017-2018 The Linux Foundation. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 and * only version 2 as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * */ /* QTI Over the Air (OTA) Crypto driver */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "qce.h" #include "qce_ota.h" enum qce_ota_oper_enum { QCE_OTA_F8_OPER = 0, QCE_OTA_MPKT_F8_OPER = 1, QCE_OTA_F9_OPER = 2, QCE_OTA_VAR_MPKT_F8_OPER = 3, QCE_OTA_OPER_LAST }; struct ota_dev_control; struct ota_async_req { struct list_head rlist; struct completion complete; int err; enum qce_ota_oper_enum op; union { struct qce_f9_req f9_req; struct qce_f8_req f8_req; struct qce_f8_multi_pkt_req f8_mp_req; struct qce_f8_variable_multi_pkt_req f8_v_mp_req; } req; unsigned int steps; struct ota_qce_dev *pqce; }; /* * Register ourselves as a char device /dev/qcota0 to be able to access the ota * from userspace. */ #define QCOTA_DEV "qcota0" struct ota_dev_control { /* char device */ struct cdev cdev; int minor; struct list_head ready_commands; unsigned int magic; struct list_head qce_dev; spinlock_t lock; struct mutex register_lock; bool registered; uint32_t total_units; }; struct ota_qce_dev { struct list_head qlist; /* qce handle */ void *qce; /* platform device */ struct platform_device *pdev; struct ota_async_req *active_command; struct tasklet_struct done_tasklet; struct ota_dev_control *podev; uint32_t unit; u64 total_req; u64 err_req; }; #define OTA_MAGIC 0x4f544143 static long qcota_ioctl(struct file *file, unsigned int cmd, unsigned long arg); static int qcota_open(struct inode *inode, struct file *file); static int qcota_release(struct inode *inode, struct file *file); static int start_req(struct ota_qce_dev *pqce, struct ota_async_req *areq); static void f8_cb(void *cookie, unsigned char *icv, unsigned char *iv, int ret); static const struct file_operations qcota_fops = { .owner = THIS_MODULE, .unlocked_ioctl = qcota_ioctl, .open = qcota_open, .release = qcota_release, }; static struct ota_dev_control qcota_dev = { .magic = OTA_MAGIC, }; static dev_t qcota_device_no; static struct class *driver_class; static struct device *class_dev; #define DEBUG_MAX_FNAME 16 #define DEBUG_MAX_RW_BUF 1024 struct qcota_stat { u64 f8_req; u64 f8_mp_req; u64 f8_v_mp_req; u64 f9_req; u64 f8_op_success; u64 f8_op_fail; u64 f8_mp_op_success; u64 f8_mp_op_fail; u64 f8_v_mp_op_success; u64 f8_v_mp_op_fail; u64 f9_op_success; u64 f9_op_fail; }; static struct qcota_stat _qcota_stat; static struct dentry *_debug_dent; static char _debug_read_buf[DEBUG_MAX_RW_BUF]; static int _debug_qcota; static struct ota_dev_control *qcota_control(void) { return &qcota_dev; } static int qcota_open(struct inode *inode, struct file *file) { struct ota_dev_control *podev; podev = qcota_control(); if (podev == NULL) { pr_err("%s: no such device %d\n", __func__, MINOR(inode->i_rdev)); return -ENOENT; } file->private_data = podev; return 0; } static int qcota_release(struct inode *inode, struct file *file) { struct ota_dev_control *podev; podev = file->private_data; if (podev != NULL && podev->magic != OTA_MAGIC) { pr_err("%s: invalid handle %pK\n", __func__, podev); } file->private_data = NULL; return 0; } static bool _next_v_mp_req(struct ota_async_req *areq) { unsigned char *p; if (areq->err) return false; if (++areq->steps >= areq->req.f8_v_mp_req.num_pkt) return false; p = areq->req.f8_v_mp_req.qce_f8_req.data_in; p += areq->req.f8_v_mp_req.qce_f8_req.data_len; p = (uint8_t *) ALIGN(((uintptr_t)p), L1_CACHE_BYTES); areq->req.f8_v_mp_req.qce_f8_req.data_out = p; areq->req.f8_v_mp_req.qce_f8_req.data_in = p; areq->req.f8_v_mp_req.qce_f8_req.data_len = areq->req.f8_v_mp_req.cipher_iov[areq->steps].size; areq->req.f8_v_mp_req.qce_f8_req.count_c++; return true; } static void req_done(unsigned long data) { struct ota_qce_dev *pqce = (struct ota_qce_dev *)data; struct ota_dev_control *podev = pqce->podev; struct ota_async_req *areq; unsigned long flags; struct ota_async_req *new_req = NULL; int ret = 0; bool schedule = true; spin_lock_irqsave(&podev->lock, flags); areq = pqce->active_command; if (unlikely(areq == NULL)) pr_err("ota_crypto: %s, no active request\n", __func__); else if (areq->op == QCE_OTA_VAR_MPKT_F8_OPER) { if (_next_v_mp_req(areq)) { /* execute next subcommand */ spin_unlock_irqrestore(&podev->lock, flags); ret = start_req(pqce, areq); if (unlikely(ret)) { areq->err = ret; schedule = true; spin_lock_irqsave(&podev->lock, flags); } else { areq = NULL; schedule = false; } } else { /* done with this variable mp req */ schedule = true; } } while (schedule) { if (!list_empty(&podev->ready_commands)) { new_req = container_of(podev->ready_commands.next, struct ota_async_req, rlist); list_del(&new_req->rlist); pqce->active_command = new_req; spin_unlock_irqrestore(&podev->lock, flags); if (new_req) { new_req->err = 0; /* start a new request */ ret = start_req(pqce, new_req); } if (unlikely(new_req && ret)) { new_req->err = ret; complete(&new_req->complete); ret = 0; new_req = NULL; spin_lock_irqsave(&podev->lock, flags); } else { schedule = false; } } else { pqce->active_command = NULL; spin_unlock_irqrestore(&podev->lock, flags); schedule = false; }; } if (areq) complete(&areq->complete); } static void f9_cb(void *cookie, unsigned char *icv, unsigned char *iv, int ret) { struct ota_async_req *areq = (struct ota_async_req *) cookie; struct ota_qce_dev *pqce; pqce = areq->pqce; areq->req.f9_req.mac_i = *((uint32_t *)icv); if (ret) { pqce->err_req++; areq->err = -ENXIO; } else areq->err = 0; tasklet_schedule(&pqce->done_tasklet); } static void f8_cb(void *cookie, unsigned char *icv, unsigned char *iv, int ret) { struct ota_async_req *areq = (struct ota_async_req *) cookie; struct ota_qce_dev *pqce; pqce = areq->pqce; if (ret) { pqce->err_req++; areq->err = -ENXIO; } else { areq->err = 0; } tasklet_schedule(&pqce->done_tasklet); } static int start_req(struct ota_qce_dev *pqce, struct ota_async_req *areq) { struct qce_f9_req *pf9; struct qce_f8_multi_pkt_req *p_mp_f8; struct qce_f8_req *pf8; int ret = 0; /* command should be on the podev->active_command */ areq->pqce = pqce; switch (areq->op) { case QCE_OTA_F8_OPER: pf8 = &areq->req.f8_req; ret = qce_f8_req(pqce->qce, pf8, areq, f8_cb); break; case QCE_OTA_MPKT_F8_OPER: p_mp_f8 = &areq->req.f8_mp_req; ret = qce_f8_multi_pkt_req(pqce->qce, p_mp_f8, areq, f8_cb); break; case QCE_OTA_F9_OPER: pf9 = &areq->req.f9_req; ret = qce_f9_req(pqce->qce, pf9, areq, f9_cb); break; case QCE_OTA_VAR_MPKT_F8_OPER: pf8 = &areq->req.f8_v_mp_req.qce_f8_req; ret = qce_f8_req(pqce->qce, pf8, areq, f8_cb); break; default: ret = -ENOTSUPP; break; }; areq->err = ret; pqce->total_req++; if (ret) pqce->err_req++; return ret; } static struct ota_qce_dev *schedule_qce(struct ota_dev_control *podev) { /* do this function with spinlock set */ struct ota_qce_dev *p; if (unlikely(list_empty(&podev->qce_dev))) { pr_err("%s: no valid qce to schedule\n", __func__); return NULL; } list_for_each_entry(p, &podev->qce_dev, qlist) { if (p->active_command == NULL) return p; } return NULL; } static int submit_req(struct ota_async_req *areq, struct ota_dev_control *podev) { unsigned long flags; int ret = 0; struct qcota_stat *pstat; struct ota_qce_dev *pqce; areq->err = 0; spin_lock_irqsave(&podev->lock, flags); pqce = schedule_qce(podev); if (pqce) { pqce->active_command = areq; spin_unlock_irqrestore(&podev->lock, flags); ret = start_req(pqce, areq); if (ret != 0) { spin_lock_irqsave(&podev->lock, flags); pqce->active_command = NULL; spin_unlock_irqrestore(&podev->lock, flags); } } else { list_add_tail(&areq->rlist, &podev->ready_commands); spin_unlock_irqrestore(&podev->lock, flags); } if (ret == 0) wait_for_completion(&areq->complete); pstat = &_qcota_stat; switch (areq->op) { case QCE_OTA_F8_OPER: if (areq->err) pstat->f8_op_fail++; else pstat->f8_op_success++; break; case QCE_OTA_MPKT_F8_OPER: if (areq->err) pstat->f8_mp_op_fail++; else pstat->f8_mp_op_success++; break; case QCE_OTA_F9_OPER: if (areq->err) pstat->f9_op_fail++; else pstat->f9_op_success++; break; case QCE_OTA_VAR_MPKT_F8_OPER: default: if (areq->err) pstat->f8_v_mp_op_fail++; else pstat->f8_v_mp_op_success++; break; }; return areq->err; } static long qcota_ioctl(struct file *file, unsigned int cmd, unsigned long arg) { int err = 0; struct ota_dev_control *podev; uint8_t *user_src; uint8_t *user_dst; uint8_t *k_buf = NULL; struct ota_async_req areq; uint32_t total, temp; struct qcota_stat *pstat; int i; uint8_t *p = NULL; podev = file->private_data; if (podev == NULL || podev->magic != OTA_MAGIC) { pr_err("%s: invalid handle %pK\n", __func__, podev); return -ENOENT; } /* Verify user arguments. */ if (_IOC_TYPE(cmd) != QCOTA_IOC_MAGIC) return -ENOTTY; init_completion(&areq.complete); pstat = &_qcota_stat; switch (cmd) { case QCOTA_F9_REQ: if (!access_ok(VERIFY_WRITE, (void __user *)arg, sizeof(struct qce_f9_req))) return -EFAULT; if (copy_from_user(&areq.req.f9_req, (void __user *)arg, sizeof(struct qce_f9_req))) return -EFAULT; user_src = areq.req.f9_req.message; if (!access_ok(VERIFY_READ, (void __user *)user_src, areq.req.f9_req.msize)) return -EFAULT; if (areq.req.f9_req.msize == 0) return 0; k_buf = kmalloc(areq.req.f9_req.msize, GFP_KERNEL); if (k_buf == NULL) return -ENOMEM; if (copy_from_user(k_buf, (void __user *)user_src, areq.req.f9_req.msize)) { kfree(k_buf); return -EFAULT; } areq.req.f9_req.message = k_buf; areq.op = QCE_OTA_F9_OPER; pstat->f9_req++; err = submit_req(&areq, podev); areq.req.f9_req.message = user_src; if (err == 0 && copy_to_user((void __user *)arg, &areq.req.f9_req, sizeof(struct qce_f9_req))) { err = -EFAULT; } kfree(k_buf); break; case QCOTA_F8_REQ: if (!access_ok(VERIFY_WRITE, (void __user *)arg, sizeof(struct qce_f8_req))) return -EFAULT; if (copy_from_user(&areq.req.f8_req, (void __user *)arg, sizeof(struct qce_f8_req))) return -EFAULT; total = areq.req.f8_req.data_len; user_src = areq.req.f8_req.data_in; if (user_src != NULL) { if (!access_ok(VERIFY_READ, (void __user *) user_src, total)) return -EFAULT; }; user_dst = areq.req.f8_req.data_out; if (!access_ok(VERIFY_WRITE, (void __user *) user_dst, total)) return -EFAULT; if (!total) return 0; k_buf = kmalloc(total, GFP_KERNEL); if (k_buf == NULL) return -ENOMEM; /* k_buf returned from kmalloc should be cache line aligned */ if (user_src && copy_from_user(k_buf, (void __user *)user_src, total)) { kfree(k_buf); return -EFAULT; } if (user_src) areq.req.f8_req.data_in = k_buf; else areq.req.f8_req.data_in = NULL; areq.req.f8_req.data_out = k_buf; areq.op = QCE_OTA_F8_OPER; pstat->f8_req++; err = submit_req(&areq, podev); if (err == 0 && copy_to_user(user_dst, k_buf, total)) err = -EFAULT; kfree(k_buf); break; case QCOTA_F8_MPKT_REQ: if (!access_ok(VERIFY_WRITE, (void __user *)arg, sizeof(struct qce_f8_multi_pkt_req))) return -EFAULT; if (copy_from_user(&areq.req.f8_mp_req, (void __user *)arg, sizeof(struct qce_f8_multi_pkt_req))) return -EFAULT; temp = areq.req.f8_mp_req.qce_f8_req.data_len; if (temp < (uint32_t) areq.req.f8_mp_req.cipher_start + areq.req.f8_mp_req.cipher_size) return -EINVAL; total = (uint32_t) areq.req.f8_mp_req.num_pkt * areq.req.f8_mp_req.qce_f8_req.data_len; user_src = areq.req.f8_mp_req.qce_f8_req.data_in; if (!access_ok(VERIFY_READ, (void __user *) user_src, total)) return -EFAULT; user_dst = areq.req.f8_mp_req.qce_f8_req.data_out; if (!access_ok(VERIFY_WRITE, (void __user *) user_dst, total)) return -EFAULT; if (!total) return 0; k_buf = kmalloc(total, GFP_KERNEL); if (k_buf == NULL) return -ENOMEM; /* k_buf returned from kmalloc should be cache line aligned */ if (copy_from_user(k_buf, (void __user *)user_src, total)) { kfree(k_buf); return -EFAULT; } areq.req.f8_mp_req.qce_f8_req.data_out = k_buf; areq.req.f8_mp_req.qce_f8_req.data_in = k_buf; areq.op = QCE_OTA_MPKT_F8_OPER; pstat->f8_mp_req++; err = submit_req(&areq, podev); if (err == 0 && copy_to_user(user_dst, k_buf, total)) err = -EFAULT; kfree(k_buf); break; case QCOTA_F8_V_MPKT_REQ: if (!access_ok(VERIFY_WRITE, (void __user *)arg, sizeof(struct qce_f8_variable_multi_pkt_req))) return -EFAULT; if (copy_from_user(&areq.req.f8_v_mp_req, (void __user *)arg, sizeof(struct qce_f8_variable_multi_pkt_req))) return -EFAULT; if (areq.req.f8_v_mp_req.num_pkt > MAX_NUM_V_MULTI_PKT) return -EINVAL; for (i = 0, total = 0; i < areq.req.f8_v_mp_req.num_pkt; i++) { if (!access_ok(VERIFY_WRITE, (void __user *) areq.req.f8_v_mp_req.cipher_iov[i].addr, areq.req.f8_v_mp_req.cipher_iov[i].size)) return -EFAULT; total += areq.req.f8_v_mp_req.cipher_iov[i].size; total = ALIGN(total, L1_CACHE_BYTES); } if (!total) return 0; k_buf = kmalloc(total, GFP_KERNEL); if (k_buf == NULL) return -ENOMEM; for (i = 0, p = k_buf; i < areq.req.f8_v_mp_req.num_pkt; i++) { user_src = areq.req.f8_v_mp_req.cipher_iov[i].addr; if (copy_from_user(p, (void __user *)user_src, areq.req.f8_v_mp_req.cipher_iov[i].size)) { kfree(k_buf); return -EFAULT; } p += areq.req.f8_v_mp_req.cipher_iov[i].size; p = (uint8_t *) ALIGN(((uintptr_t)p), L1_CACHE_BYTES); } areq.req.f8_v_mp_req.qce_f8_req.data_out = k_buf; areq.req.f8_v_mp_req.qce_f8_req.data_in = k_buf; areq.req.f8_v_mp_req.qce_f8_req.data_len = areq.req.f8_v_mp_req.cipher_iov[0].size; areq.steps = 0; areq.op = QCE_OTA_VAR_MPKT_F8_OPER; pstat->f8_v_mp_req++; err = submit_req(&areq, podev); if (err != 0) { kfree(k_buf); return err; } for (i = 0, p = k_buf; i < areq.req.f8_v_mp_req.num_pkt; i++) { user_dst = areq.req.f8_v_mp_req.cipher_iov[i].addr; if (copy_to_user(user_dst, p, areq.req.f8_v_mp_req.cipher_iov[i].size)) { kfree(k_buf); return -EFAULT; } p += areq.req.f8_v_mp_req.cipher_iov[i].size; p = (uint8_t *) ALIGN(((uintptr_t)p), L1_CACHE_BYTES); } kfree(k_buf); break; default: return -ENOTTY; } return err; } static int qcota_probe(struct platform_device *pdev) { void *handle = NULL; int rc = 0; struct ota_dev_control *podev; struct ce_hw_support ce_support; struct ota_qce_dev *pqce; unsigned long flags; podev = &qcota_dev; pqce = kzalloc(sizeof(*pqce), GFP_KERNEL); if (!pqce) return -ENOMEM; rc = alloc_chrdev_region(&qcota_device_no, 0, 1, QCOTA_DEV); if (rc < 0) { pr_err("alloc_chrdev_region failed %d\n", rc); return rc; } driver_class = class_create(THIS_MODULE, QCOTA_DEV); if (IS_ERR(driver_class)) { rc = -ENOMEM; pr_err("class_create failed %d\n", rc); goto exit_unreg_chrdev_region; } class_dev = device_create(driver_class, NULL, qcota_device_no, NULL, QCOTA_DEV); if (IS_ERR(class_dev)) { pr_err("class_device_create failed %d\n", rc); rc = -ENOMEM; goto exit_destroy_class; } cdev_init(&podev->cdev, &qcota_fops); podev->cdev.owner = THIS_MODULE; rc = cdev_add(&podev->cdev, MKDEV(MAJOR(qcota_device_no), 0), 1); if (rc < 0) { pr_err("cdev_add failed %d\n", rc); goto exit_destroy_device; } podev->minor = 0; pqce->podev = podev; pqce->active_command = NULL; tasklet_init(&pqce->done_tasklet, req_done, (unsigned long)pqce); /* open qce */ handle = qce_open(pdev, &rc); if (handle == NULL) { pr_err("%s: device %s, can not open qce\n", __func__, pdev->name); goto exit_del_cdev; } if (qce_hw_support(handle, &ce_support) < 0 || ce_support.ota == false) { pr_err("%s: device %s, qce does not support ota capability\n", __func__, pdev->name); rc = -ENODEV; goto err; } pqce->qce = handle; pqce->pdev = pdev; pqce->total_req = 0; pqce->err_req = 0; platform_set_drvdata(pdev, pqce); mutex_lock(&podev->register_lock); rc = 0; if (podev->registered == false) { if (rc == 0) { pqce->unit = podev->total_units; podev->total_units++; podev->registered = true; }; } else { pqce->unit = podev->total_units; podev->total_units++; } mutex_unlock(&podev->register_lock); if (rc) { pr_err("ion: failed to register misc device.\n"); goto err; } spin_lock_irqsave(&podev->lock, flags); list_add_tail(&pqce->qlist, &podev->qce_dev); spin_unlock_irqrestore(&podev->lock, flags); return 0; err: if (handle) qce_close(handle); platform_set_drvdata(pdev, NULL); tasklet_kill(&pqce->done_tasklet); exit_del_cdev: cdev_del(&podev->cdev); exit_destroy_device: device_destroy(driver_class, qcota_device_no); exit_destroy_class: class_destroy(driver_class); exit_unreg_chrdev_region: unregister_chrdev_region(qcota_device_no, 1); kfree(pqce); return rc; } static int qcota_remove(struct platform_device *pdev) { struct ota_dev_control *podev; struct ota_qce_dev *pqce; unsigned long flags; pqce = platform_get_drvdata(pdev); if (!pqce) return 0; if (pqce->qce) qce_close(pqce->qce); podev = pqce->podev; if (!podev) goto ret; spin_lock_irqsave(&podev->lock, flags); list_del(&pqce->qlist); spin_unlock_irqrestore(&podev->lock, flags); mutex_lock(&podev->register_lock); if (--podev->total_units == 0) { cdev_del(&podev->cdev); device_destroy(driver_class, qcota_device_no); class_destroy(driver_class); unregister_chrdev_region(qcota_device_no, 1); podev->registered = false; } mutex_unlock(&podev->register_lock); ret: tasklet_kill(&pqce->done_tasklet); kfree(pqce); return 0; } static const struct of_device_id qcota_match[] = { { .compatible = "qcom,qcota", }, {} }; static struct platform_driver qcota_plat_driver = { .probe = qcota_probe, .remove = qcota_remove, .driver = { .name = "qcota", .owner = THIS_MODULE, .of_match_table = qcota_match, }, }; static int _disp_stats(void) { struct qcota_stat *pstat; int len = 0; struct ota_dev_control *podev = &qcota_dev; unsigned long flags; struct ota_qce_dev *p; pstat = &_qcota_stat; len = scnprintf(_debug_read_buf, DEBUG_MAX_RW_BUF - 1, "\nQTI OTA crypto accelerator Statistics:\n"); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 request : %llu\n", pstat->f8_req); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 operation success : %llu\n", pstat->f8_op_success); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 operation fail : %llu\n", pstat->f8_op_fail); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 MP request : %llu\n", pstat->f8_mp_req); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 MP operation success : %llu\n", pstat->f8_mp_op_success); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 MP operation fail : %llu\n", pstat->f8_mp_op_fail); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 Variable MP request : %llu\n", pstat->f8_v_mp_req); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 Variable MP operation success: %llu\n", pstat->f8_v_mp_op_success); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F8 Variable MP operation fail : %llu\n", pstat->f8_v_mp_op_fail); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F9 request : %llu\n", pstat->f9_req); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F9 operation success : %llu\n", pstat->f9_op_success); len += scnprintf(_debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " F9 operation fail : %llu\n", pstat->f9_op_fail); spin_lock_irqsave(&podev->lock, flags); list_for_each_entry(p, &podev->qce_dev, qlist) { len += scnprintf( _debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " Engine %4d Req : %llu\n", p->unit, p->total_req ); len += scnprintf( _debug_read_buf + len, DEBUG_MAX_RW_BUF - len - 1, " Engine %4d Req Error : %llu\n", p->unit, p->err_req ); } spin_unlock_irqrestore(&podev->lock, flags); return len; } static int _debug_stats_open(struct inode *inode, struct file *file) { file->private_data = inode->i_private; return 0; } static ssize_t _debug_stats_read(struct file *file, char __user *buf, size_t count, loff_t *ppos) { int rc = -EINVAL; int len; len = _disp_stats(); if (len <= count) rc = simple_read_from_buffer((void __user *) buf, len, ppos, (void *) _debug_read_buf, len); return rc; } static ssize_t _debug_stats_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos) { struct ota_dev_control *podev = &qcota_dev; unsigned long flags; struct ota_qce_dev *p; memset((char *)&_qcota_stat, 0, sizeof(struct qcota_stat)); spin_lock_irqsave(&podev->lock, flags); list_for_each_entry(p, &podev->qce_dev, qlist) { p->total_req = 0; p->err_req = 0; } spin_unlock_irqrestore(&podev->lock, flags); return count; } static const struct file_operations _debug_stats_ops = { .open = _debug_stats_open, .read = _debug_stats_read, .write = _debug_stats_write, }; static int _qcota_debug_init(void) { int rc; char name[DEBUG_MAX_FNAME]; struct dentry *dent; _debug_dent = debugfs_create_dir("qcota", NULL); if (IS_ERR(_debug_dent)) { pr_err("qcota debugfs_create_dir fail, error %ld\n", PTR_ERR(_debug_dent)); return PTR_ERR(_debug_dent); } snprintf(name, DEBUG_MAX_FNAME-1, "stats-0"); _debug_qcota = 0; dent = debugfs_create_file(name, 0644, _debug_dent, &_debug_qcota, &_debug_stats_ops); if (dent == NULL) { pr_err("qcota debugfs_create_file fail, error %ld\n", PTR_ERR(dent)); rc = PTR_ERR(dent); goto err; } return 0; err: debugfs_remove_recursive(_debug_dent); return rc; } static int __init qcota_init(void) { int rc; struct ota_dev_control *podev; rc = _qcota_debug_init(); if (rc) return rc; podev = &qcota_dev; INIT_LIST_HEAD(&podev->ready_commands); INIT_LIST_HEAD(&podev->qce_dev); spin_lock_init(&podev->lock); mutex_init(&podev->register_lock); podev->registered = false; podev->total_units = 0; return platform_driver_register(&qcota_plat_driver); } static void __exit qcota_exit(void) { debugfs_remove_recursive(_debug_dent); platform_driver_unregister(&qcota_plat_driver); } MODULE_LICENSE("GPL v2"); MODULE_DESCRIPTION("QTI Ota Crypto driver"); module_init(qcota_init); module_exit(qcota_exit);