162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Character device interface driver for Remoteproc framework. 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (c) 2020, The Linux Foundation. All rights reserved. 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include <linux/cdev.h> 962306a36Sopenharmony_ci#include <linux/compat.h> 1062306a36Sopenharmony_ci#include <linux/fs.h> 1162306a36Sopenharmony_ci#include <linux/module.h> 1262306a36Sopenharmony_ci#include <linux/remoteproc.h> 1362306a36Sopenharmony_ci#include <linux/uaccess.h> 1462306a36Sopenharmony_ci#include <uapi/linux/remoteproc_cdev.h> 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci#include "remoteproc_internal.h" 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#define NUM_RPROC_DEVICES 64 1962306a36Sopenharmony_cistatic dev_t rproc_major; 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_cistatic ssize_t rproc_cdev_write(struct file *filp, const char __user *buf, size_t len, loff_t *pos) 2262306a36Sopenharmony_ci{ 2362306a36Sopenharmony_ci struct rproc *rproc = container_of(filp->f_inode->i_cdev, struct rproc, cdev); 2462306a36Sopenharmony_ci int ret = 0; 2562306a36Sopenharmony_ci char cmd[10]; 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci if (!len || len > sizeof(cmd)) 2862306a36Sopenharmony_ci return -EINVAL; 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci ret = copy_from_user(cmd, buf, len); 3162306a36Sopenharmony_ci if (ret) 3262306a36Sopenharmony_ci return -EFAULT; 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci if (!strncmp(cmd, "start", len)) { 3562306a36Sopenharmony_ci ret = rproc_boot(rproc); 3662306a36Sopenharmony_ci } else if (!strncmp(cmd, "stop", len)) { 3762306a36Sopenharmony_ci ret = rproc_shutdown(rproc); 3862306a36Sopenharmony_ci } else if (!strncmp(cmd, "detach", len)) { 3962306a36Sopenharmony_ci ret = rproc_detach(rproc); 4062306a36Sopenharmony_ci } else { 4162306a36Sopenharmony_ci dev_err(&rproc->dev, "Unrecognized option\n"); 4262306a36Sopenharmony_ci ret = -EINVAL; 4362306a36Sopenharmony_ci } 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_ci return ret ? ret : len; 4662306a36Sopenharmony_ci} 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_cistatic long rproc_device_ioctl(struct file *filp, unsigned int ioctl, unsigned long arg) 4962306a36Sopenharmony_ci{ 5062306a36Sopenharmony_ci struct rproc *rproc = container_of(filp->f_inode->i_cdev, struct rproc, cdev); 5162306a36Sopenharmony_ci void __user *argp = (void __user *)arg; 5262306a36Sopenharmony_ci s32 param; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci switch (ioctl) { 5562306a36Sopenharmony_ci case RPROC_SET_SHUTDOWN_ON_RELEASE: 5662306a36Sopenharmony_ci if (copy_from_user(¶m, argp, sizeof(s32))) 5762306a36Sopenharmony_ci return -EFAULT; 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci rproc->cdev_put_on_release = !!param; 6062306a36Sopenharmony_ci break; 6162306a36Sopenharmony_ci case RPROC_GET_SHUTDOWN_ON_RELEASE: 6262306a36Sopenharmony_ci param = (s32)rproc->cdev_put_on_release; 6362306a36Sopenharmony_ci if (copy_to_user(argp, ¶m, sizeof(s32))) 6462306a36Sopenharmony_ci return -EFAULT; 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci break; 6762306a36Sopenharmony_ci default: 6862306a36Sopenharmony_ci dev_err(&rproc->dev, "Unsupported ioctl\n"); 6962306a36Sopenharmony_ci return -EINVAL; 7062306a36Sopenharmony_ci } 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci return 0; 7362306a36Sopenharmony_ci} 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_cistatic int rproc_cdev_release(struct inode *inode, struct file *filp) 7662306a36Sopenharmony_ci{ 7762306a36Sopenharmony_ci struct rproc *rproc = container_of(inode->i_cdev, struct rproc, cdev); 7862306a36Sopenharmony_ci int ret = 0; 7962306a36Sopenharmony_ci 8062306a36Sopenharmony_ci if (!rproc->cdev_put_on_release) 8162306a36Sopenharmony_ci return 0; 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci if (rproc->state == RPROC_RUNNING) 8462306a36Sopenharmony_ci rproc_shutdown(rproc); 8562306a36Sopenharmony_ci else if (rproc->state == RPROC_ATTACHED) 8662306a36Sopenharmony_ci ret = rproc_detach(rproc); 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci return ret; 8962306a36Sopenharmony_ci} 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cistatic const struct file_operations rproc_fops = { 9262306a36Sopenharmony_ci .write = rproc_cdev_write, 9362306a36Sopenharmony_ci .unlocked_ioctl = rproc_device_ioctl, 9462306a36Sopenharmony_ci .compat_ioctl = compat_ptr_ioctl, 9562306a36Sopenharmony_ci .release = rproc_cdev_release, 9662306a36Sopenharmony_ci}; 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_ciint rproc_char_device_add(struct rproc *rproc) 9962306a36Sopenharmony_ci{ 10062306a36Sopenharmony_ci int ret; 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci cdev_init(&rproc->cdev, &rproc_fops); 10362306a36Sopenharmony_ci rproc->cdev.owner = THIS_MODULE; 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci rproc->dev.devt = MKDEV(MAJOR(rproc_major), rproc->index); 10662306a36Sopenharmony_ci cdev_set_parent(&rproc->cdev, &rproc->dev.kobj); 10762306a36Sopenharmony_ci ret = cdev_add(&rproc->cdev, rproc->dev.devt, 1); 10862306a36Sopenharmony_ci if (ret < 0) 10962306a36Sopenharmony_ci dev_err(&rproc->dev, "Failed to add char dev for %s\n", rproc->name); 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci return ret; 11262306a36Sopenharmony_ci} 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_civoid rproc_char_device_remove(struct rproc *rproc) 11562306a36Sopenharmony_ci{ 11662306a36Sopenharmony_ci cdev_del(&rproc->cdev); 11762306a36Sopenharmony_ci} 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_civoid __init rproc_init_cdev(void) 12062306a36Sopenharmony_ci{ 12162306a36Sopenharmony_ci int ret; 12262306a36Sopenharmony_ci 12362306a36Sopenharmony_ci ret = alloc_chrdev_region(&rproc_major, 0, NUM_RPROC_DEVICES, "remoteproc"); 12462306a36Sopenharmony_ci if (ret < 0) 12562306a36Sopenharmony_ci pr_err("Failed to alloc rproc_cdev region, err %d\n", ret); 12662306a36Sopenharmony_ci} 127