18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci *      	An implementation of a loadable kernel mode driver providing
48c2ecf20Sopenharmony_ci *		multiple kernel/user space bidirectional communications links.
58c2ecf20Sopenharmony_ci *
68c2ecf20Sopenharmony_ci * 		Author: 	Alan Cox <alan@lxorguk.ukuu.org.uk>
78c2ecf20Sopenharmony_ci *
88c2ecf20Sopenharmony_ci *              Adapted to become the Linux 2.0 Coda pseudo device
98c2ecf20Sopenharmony_ci *              Peter  Braam  <braam@maths.ox.ac.uk>
108c2ecf20Sopenharmony_ci *              Michael Callahan <mjc@emmy.smith.edu>
118c2ecf20Sopenharmony_ci *
128c2ecf20Sopenharmony_ci *              Changes for Linux 2.1
138c2ecf20Sopenharmony_ci *              Copyright (c) 1997 Carnegie-Mellon University
148c2ecf20Sopenharmony_ci */
158c2ecf20Sopenharmony_ci
168c2ecf20Sopenharmony_ci#include <linux/module.h>
178c2ecf20Sopenharmony_ci#include <linux/errno.h>
188c2ecf20Sopenharmony_ci#include <linux/kernel.h>
198c2ecf20Sopenharmony_ci#include <linux/major.h>
208c2ecf20Sopenharmony_ci#include <linux/time.h>
218c2ecf20Sopenharmony_ci#include <linux/sched/signal.h>
228c2ecf20Sopenharmony_ci#include <linux/slab.h>
238c2ecf20Sopenharmony_ci#include <linux/ioport.h>
248c2ecf20Sopenharmony_ci#include <linux/fcntl.h>
258c2ecf20Sopenharmony_ci#include <linux/delay.h>
268c2ecf20Sopenharmony_ci#include <linux/skbuff.h>
278c2ecf20Sopenharmony_ci#include <linux/proc_fs.h>
288c2ecf20Sopenharmony_ci#include <linux/vmalloc.h>
298c2ecf20Sopenharmony_ci#include <linux/fs.h>
308c2ecf20Sopenharmony_ci#include <linux/file.h>
318c2ecf20Sopenharmony_ci#include <linux/poll.h>
328c2ecf20Sopenharmony_ci#include <linux/init.h>
338c2ecf20Sopenharmony_ci#include <linux/list.h>
348c2ecf20Sopenharmony_ci#include <linux/mutex.h>
358c2ecf20Sopenharmony_ci#include <linux/device.h>
368c2ecf20Sopenharmony_ci#include <linux/pid_namespace.h>
378c2ecf20Sopenharmony_ci#include <asm/io.h>
388c2ecf20Sopenharmony_ci#include <linux/uaccess.h>
398c2ecf20Sopenharmony_ci
408c2ecf20Sopenharmony_ci#include <linux/coda.h>
418c2ecf20Sopenharmony_ci#include "coda_psdev.h"
428c2ecf20Sopenharmony_ci#include "coda_linux.h"
438c2ecf20Sopenharmony_ci
448c2ecf20Sopenharmony_ci#include "coda_int.h"
458c2ecf20Sopenharmony_ci
468c2ecf20Sopenharmony_ci/* statistics */
478c2ecf20Sopenharmony_ciint           coda_hard;         /* allows signals during upcalls */
488c2ecf20Sopenharmony_ciunsigned long coda_timeout = 30; /* .. secs, then signals will dequeue */
498c2ecf20Sopenharmony_ci
508c2ecf20Sopenharmony_ci
518c2ecf20Sopenharmony_cistruct venus_comm coda_comms[MAX_CODADEVS];
528c2ecf20Sopenharmony_cistatic struct class *coda_psdev_class;
538c2ecf20Sopenharmony_ci
548c2ecf20Sopenharmony_ci/*
558c2ecf20Sopenharmony_ci * Device operations
568c2ecf20Sopenharmony_ci */
578c2ecf20Sopenharmony_ci
588c2ecf20Sopenharmony_cistatic __poll_t coda_psdev_poll(struct file *file, poll_table * wait)
598c2ecf20Sopenharmony_ci{
608c2ecf20Sopenharmony_ci        struct venus_comm *vcp = (struct venus_comm *) file->private_data;
618c2ecf20Sopenharmony_ci	__poll_t mask = EPOLLOUT | EPOLLWRNORM;
628c2ecf20Sopenharmony_ci
638c2ecf20Sopenharmony_ci	poll_wait(file, &vcp->vc_waitq, wait);
648c2ecf20Sopenharmony_ci	mutex_lock(&vcp->vc_mutex);
658c2ecf20Sopenharmony_ci	if (!list_empty(&vcp->vc_pending))
668c2ecf20Sopenharmony_ci                mask |= EPOLLIN | EPOLLRDNORM;
678c2ecf20Sopenharmony_ci	mutex_unlock(&vcp->vc_mutex);
688c2ecf20Sopenharmony_ci
698c2ecf20Sopenharmony_ci	return mask;
708c2ecf20Sopenharmony_ci}
718c2ecf20Sopenharmony_ci
728c2ecf20Sopenharmony_cistatic long coda_psdev_ioctl(struct file * filp, unsigned int cmd, unsigned long arg)
738c2ecf20Sopenharmony_ci{
748c2ecf20Sopenharmony_ci	unsigned int data;
758c2ecf20Sopenharmony_ci
768c2ecf20Sopenharmony_ci	switch(cmd) {
778c2ecf20Sopenharmony_ci	case CIOC_KERNEL_VERSION:
788c2ecf20Sopenharmony_ci		data = CODA_KERNEL_VERSION;
798c2ecf20Sopenharmony_ci		return put_user(data, (int __user *) arg);
808c2ecf20Sopenharmony_ci	default:
818c2ecf20Sopenharmony_ci		return -ENOTTY;
828c2ecf20Sopenharmony_ci	}
838c2ecf20Sopenharmony_ci
848c2ecf20Sopenharmony_ci	return 0;
858c2ecf20Sopenharmony_ci}
868c2ecf20Sopenharmony_ci
878c2ecf20Sopenharmony_ci/*
888c2ecf20Sopenharmony_ci *	Receive a message written by Venus to the psdev
898c2ecf20Sopenharmony_ci */
908c2ecf20Sopenharmony_ci
918c2ecf20Sopenharmony_cistatic ssize_t coda_psdev_write(struct file *file, const char __user *buf,
928c2ecf20Sopenharmony_ci				size_t nbytes, loff_t *off)
938c2ecf20Sopenharmony_ci{
948c2ecf20Sopenharmony_ci        struct venus_comm *vcp = (struct venus_comm *) file->private_data;
958c2ecf20Sopenharmony_ci        struct upc_req *req = NULL;
968c2ecf20Sopenharmony_ci        struct upc_req *tmp;
978c2ecf20Sopenharmony_ci	struct list_head *lh;
988c2ecf20Sopenharmony_ci	struct coda_in_hdr hdr;
998c2ecf20Sopenharmony_ci	ssize_t retval = 0, count = 0;
1008c2ecf20Sopenharmony_ci	int error;
1018c2ecf20Sopenharmony_ci
1028c2ecf20Sopenharmony_ci	/* make sure there is enough to copy out the (opcode, unique) values */
1038c2ecf20Sopenharmony_ci	if (nbytes < (2 * sizeof(u_int32_t)))
1048c2ecf20Sopenharmony_ci		return -EINVAL;
1058c2ecf20Sopenharmony_ci
1068c2ecf20Sopenharmony_ci        /* Peek at the opcode, uniquefier */
1078c2ecf20Sopenharmony_ci	if (copy_from_user(&hdr, buf, 2 * sizeof(u_int32_t)))
1088c2ecf20Sopenharmony_ci	        return -EFAULT;
1098c2ecf20Sopenharmony_ci
1108c2ecf20Sopenharmony_ci        if (DOWNCALL(hdr.opcode)) {
1118c2ecf20Sopenharmony_ci		union outputArgs *dcbuf;
1128c2ecf20Sopenharmony_ci		int size = sizeof(*dcbuf);
1138c2ecf20Sopenharmony_ci
1148c2ecf20Sopenharmony_ci		if  ( nbytes < sizeof(struct coda_out_hdr) ) {
1158c2ecf20Sopenharmony_ci			pr_warn("coda_downcall opc %d uniq %d, not enough!\n",
1168c2ecf20Sopenharmony_ci				hdr.opcode, hdr.unique);
1178c2ecf20Sopenharmony_ci			count = nbytes;
1188c2ecf20Sopenharmony_ci			goto out;
1198c2ecf20Sopenharmony_ci		}
1208c2ecf20Sopenharmony_ci		if ( nbytes > size ) {
1218c2ecf20Sopenharmony_ci			pr_warn("downcall opc %d, uniq %d, too much!",
1228c2ecf20Sopenharmony_ci				hdr.opcode, hdr.unique);
1238c2ecf20Sopenharmony_ci		        nbytes = size;
1248c2ecf20Sopenharmony_ci		}
1258c2ecf20Sopenharmony_ci		dcbuf = kvmalloc(nbytes, GFP_KERNEL);
1268c2ecf20Sopenharmony_ci		if (!dcbuf) {
1278c2ecf20Sopenharmony_ci			retval = -ENOMEM;
1288c2ecf20Sopenharmony_ci			goto out;
1298c2ecf20Sopenharmony_ci		}
1308c2ecf20Sopenharmony_ci		if (copy_from_user(dcbuf, buf, nbytes)) {
1318c2ecf20Sopenharmony_ci			kvfree(dcbuf);
1328c2ecf20Sopenharmony_ci			retval = -EFAULT;
1338c2ecf20Sopenharmony_ci			goto out;
1348c2ecf20Sopenharmony_ci		}
1358c2ecf20Sopenharmony_ci
1368c2ecf20Sopenharmony_ci		/* what downcall errors does Venus handle ? */
1378c2ecf20Sopenharmony_ci		error = coda_downcall(vcp, hdr.opcode, dcbuf, nbytes);
1388c2ecf20Sopenharmony_ci
1398c2ecf20Sopenharmony_ci		kvfree(dcbuf);
1408c2ecf20Sopenharmony_ci		if (error) {
1418c2ecf20Sopenharmony_ci			pr_warn("%s: coda_downcall error: %d\n",
1428c2ecf20Sopenharmony_ci				__func__, error);
1438c2ecf20Sopenharmony_ci			retval = error;
1448c2ecf20Sopenharmony_ci			goto out;
1458c2ecf20Sopenharmony_ci		}
1468c2ecf20Sopenharmony_ci		count = nbytes;
1478c2ecf20Sopenharmony_ci		goto out;
1488c2ecf20Sopenharmony_ci	}
1498c2ecf20Sopenharmony_ci
1508c2ecf20Sopenharmony_ci	/* Look for the message on the processing queue. */
1518c2ecf20Sopenharmony_ci	mutex_lock(&vcp->vc_mutex);
1528c2ecf20Sopenharmony_ci	list_for_each(lh, &vcp->vc_processing) {
1538c2ecf20Sopenharmony_ci		tmp = list_entry(lh, struct upc_req , uc_chain);
1548c2ecf20Sopenharmony_ci		if (tmp->uc_unique == hdr.unique) {
1558c2ecf20Sopenharmony_ci			req = tmp;
1568c2ecf20Sopenharmony_ci			list_del(&req->uc_chain);
1578c2ecf20Sopenharmony_ci			break;
1588c2ecf20Sopenharmony_ci		}
1598c2ecf20Sopenharmony_ci	}
1608c2ecf20Sopenharmony_ci	mutex_unlock(&vcp->vc_mutex);
1618c2ecf20Sopenharmony_ci
1628c2ecf20Sopenharmony_ci	if (!req) {
1638c2ecf20Sopenharmony_ci		pr_warn("%s: msg (%d, %d) not found\n",
1648c2ecf20Sopenharmony_ci			__func__, hdr.opcode, hdr.unique);
1658c2ecf20Sopenharmony_ci		retval = -ESRCH;
1668c2ecf20Sopenharmony_ci		goto out;
1678c2ecf20Sopenharmony_ci	}
1688c2ecf20Sopenharmony_ci
1698c2ecf20Sopenharmony_ci        /* move data into response buffer. */
1708c2ecf20Sopenharmony_ci	if (req->uc_outSize < nbytes) {
1718c2ecf20Sopenharmony_ci		pr_warn("%s: too much cnt: %d, cnt: %ld, opc: %d, uniq: %d.\n",
1728c2ecf20Sopenharmony_ci			__func__, req->uc_outSize, (long)nbytes,
1738c2ecf20Sopenharmony_ci			hdr.opcode, hdr.unique);
1748c2ecf20Sopenharmony_ci		nbytes = req->uc_outSize; /* don't have more space! */
1758c2ecf20Sopenharmony_ci	}
1768c2ecf20Sopenharmony_ci        if (copy_from_user(req->uc_data, buf, nbytes)) {
1778c2ecf20Sopenharmony_ci		req->uc_flags |= CODA_REQ_ABORT;
1788c2ecf20Sopenharmony_ci		wake_up(&req->uc_sleep);
1798c2ecf20Sopenharmony_ci		retval = -EFAULT;
1808c2ecf20Sopenharmony_ci		goto out;
1818c2ecf20Sopenharmony_ci	}
1828c2ecf20Sopenharmony_ci
1838c2ecf20Sopenharmony_ci	/* adjust outsize. is this useful ?? */
1848c2ecf20Sopenharmony_ci	req->uc_outSize = nbytes;
1858c2ecf20Sopenharmony_ci	req->uc_flags |= CODA_REQ_WRITE;
1868c2ecf20Sopenharmony_ci	count = nbytes;
1878c2ecf20Sopenharmony_ci
1888c2ecf20Sopenharmony_ci	/* Convert filedescriptor into a file handle */
1898c2ecf20Sopenharmony_ci	if (req->uc_opcode == CODA_OPEN_BY_FD) {
1908c2ecf20Sopenharmony_ci		struct coda_open_by_fd_out *outp =
1918c2ecf20Sopenharmony_ci			(struct coda_open_by_fd_out *)req->uc_data;
1928c2ecf20Sopenharmony_ci		if (!outp->oh.result) {
1938c2ecf20Sopenharmony_ci			outp->fh = fget(outp->fd);
1948c2ecf20Sopenharmony_ci			if (!outp->fh)
1958c2ecf20Sopenharmony_ci				return -EBADF;
1968c2ecf20Sopenharmony_ci		}
1978c2ecf20Sopenharmony_ci	}
1988c2ecf20Sopenharmony_ci
1998c2ecf20Sopenharmony_ci        wake_up(&req->uc_sleep);
2008c2ecf20Sopenharmony_ciout:
2018c2ecf20Sopenharmony_ci        return(count ? count : retval);
2028c2ecf20Sopenharmony_ci}
2038c2ecf20Sopenharmony_ci
2048c2ecf20Sopenharmony_ci/*
2058c2ecf20Sopenharmony_ci *	Read a message from the kernel to Venus
2068c2ecf20Sopenharmony_ci */
2078c2ecf20Sopenharmony_ci
2088c2ecf20Sopenharmony_cistatic ssize_t coda_psdev_read(struct file * file, char __user * buf,
2098c2ecf20Sopenharmony_ci			       size_t nbytes, loff_t *off)
2108c2ecf20Sopenharmony_ci{
2118c2ecf20Sopenharmony_ci	DECLARE_WAITQUEUE(wait, current);
2128c2ecf20Sopenharmony_ci        struct venus_comm *vcp = (struct venus_comm *) file->private_data;
2138c2ecf20Sopenharmony_ci        struct upc_req *req;
2148c2ecf20Sopenharmony_ci	ssize_t retval = 0, count = 0;
2158c2ecf20Sopenharmony_ci
2168c2ecf20Sopenharmony_ci	if (nbytes == 0)
2178c2ecf20Sopenharmony_ci		return 0;
2188c2ecf20Sopenharmony_ci
2198c2ecf20Sopenharmony_ci	mutex_lock(&vcp->vc_mutex);
2208c2ecf20Sopenharmony_ci
2218c2ecf20Sopenharmony_ci	add_wait_queue(&vcp->vc_waitq, &wait);
2228c2ecf20Sopenharmony_ci	set_current_state(TASK_INTERRUPTIBLE);
2238c2ecf20Sopenharmony_ci
2248c2ecf20Sopenharmony_ci	while (list_empty(&vcp->vc_pending)) {
2258c2ecf20Sopenharmony_ci		if (file->f_flags & O_NONBLOCK) {
2268c2ecf20Sopenharmony_ci			retval = -EAGAIN;
2278c2ecf20Sopenharmony_ci			break;
2288c2ecf20Sopenharmony_ci		}
2298c2ecf20Sopenharmony_ci		if (signal_pending(current)) {
2308c2ecf20Sopenharmony_ci			retval = -ERESTARTSYS;
2318c2ecf20Sopenharmony_ci			break;
2328c2ecf20Sopenharmony_ci		}
2338c2ecf20Sopenharmony_ci		mutex_unlock(&vcp->vc_mutex);
2348c2ecf20Sopenharmony_ci		schedule();
2358c2ecf20Sopenharmony_ci		mutex_lock(&vcp->vc_mutex);
2368c2ecf20Sopenharmony_ci	}
2378c2ecf20Sopenharmony_ci
2388c2ecf20Sopenharmony_ci	set_current_state(TASK_RUNNING);
2398c2ecf20Sopenharmony_ci	remove_wait_queue(&vcp->vc_waitq, &wait);
2408c2ecf20Sopenharmony_ci
2418c2ecf20Sopenharmony_ci	if (retval)
2428c2ecf20Sopenharmony_ci		goto out;
2438c2ecf20Sopenharmony_ci
2448c2ecf20Sopenharmony_ci	req = list_entry(vcp->vc_pending.next, struct upc_req,uc_chain);
2458c2ecf20Sopenharmony_ci	list_del(&req->uc_chain);
2468c2ecf20Sopenharmony_ci
2478c2ecf20Sopenharmony_ci	/* Move the input args into userspace */
2488c2ecf20Sopenharmony_ci	count = req->uc_inSize;
2498c2ecf20Sopenharmony_ci	if (nbytes < req->uc_inSize) {
2508c2ecf20Sopenharmony_ci		pr_warn("%s: Venus read %ld bytes of %d in message\n",
2518c2ecf20Sopenharmony_ci			__func__, (long)nbytes, req->uc_inSize);
2528c2ecf20Sopenharmony_ci		count = nbytes;
2538c2ecf20Sopenharmony_ci        }
2548c2ecf20Sopenharmony_ci
2558c2ecf20Sopenharmony_ci	if (copy_to_user(buf, req->uc_data, count))
2568c2ecf20Sopenharmony_ci	        retval = -EFAULT;
2578c2ecf20Sopenharmony_ci
2588c2ecf20Sopenharmony_ci	/* If request was not a signal, enqueue and don't free */
2598c2ecf20Sopenharmony_ci	if (!(req->uc_flags & CODA_REQ_ASYNC)) {
2608c2ecf20Sopenharmony_ci		req->uc_flags |= CODA_REQ_READ;
2618c2ecf20Sopenharmony_ci		list_add_tail(&(req->uc_chain), &vcp->vc_processing);
2628c2ecf20Sopenharmony_ci		goto out;
2638c2ecf20Sopenharmony_ci	}
2648c2ecf20Sopenharmony_ci
2658c2ecf20Sopenharmony_ci	kvfree(req->uc_data);
2668c2ecf20Sopenharmony_ci	kfree(req);
2678c2ecf20Sopenharmony_ciout:
2688c2ecf20Sopenharmony_ci	mutex_unlock(&vcp->vc_mutex);
2698c2ecf20Sopenharmony_ci	return (count ? count : retval);
2708c2ecf20Sopenharmony_ci}
2718c2ecf20Sopenharmony_ci
2728c2ecf20Sopenharmony_cistatic int coda_psdev_open(struct inode * inode, struct file * file)
2738c2ecf20Sopenharmony_ci{
2748c2ecf20Sopenharmony_ci	struct venus_comm *vcp;
2758c2ecf20Sopenharmony_ci	int idx, err;
2768c2ecf20Sopenharmony_ci
2778c2ecf20Sopenharmony_ci	if (task_active_pid_ns(current) != &init_pid_ns)
2788c2ecf20Sopenharmony_ci		return -EINVAL;
2798c2ecf20Sopenharmony_ci
2808c2ecf20Sopenharmony_ci	if (current_user_ns() != &init_user_ns)
2818c2ecf20Sopenharmony_ci		return -EINVAL;
2828c2ecf20Sopenharmony_ci
2838c2ecf20Sopenharmony_ci	idx = iminor(inode);
2848c2ecf20Sopenharmony_ci	if (idx < 0 || idx >= MAX_CODADEVS)
2858c2ecf20Sopenharmony_ci		return -ENODEV;
2868c2ecf20Sopenharmony_ci
2878c2ecf20Sopenharmony_ci	err = -EBUSY;
2888c2ecf20Sopenharmony_ci	vcp = &coda_comms[idx];
2898c2ecf20Sopenharmony_ci	mutex_lock(&vcp->vc_mutex);
2908c2ecf20Sopenharmony_ci
2918c2ecf20Sopenharmony_ci	if (!vcp->vc_inuse) {
2928c2ecf20Sopenharmony_ci		vcp->vc_inuse++;
2938c2ecf20Sopenharmony_ci
2948c2ecf20Sopenharmony_ci		INIT_LIST_HEAD(&vcp->vc_pending);
2958c2ecf20Sopenharmony_ci		INIT_LIST_HEAD(&vcp->vc_processing);
2968c2ecf20Sopenharmony_ci		init_waitqueue_head(&vcp->vc_waitq);
2978c2ecf20Sopenharmony_ci		vcp->vc_sb = NULL;
2988c2ecf20Sopenharmony_ci		vcp->vc_seq = 0;
2998c2ecf20Sopenharmony_ci
3008c2ecf20Sopenharmony_ci		file->private_data = vcp;
3018c2ecf20Sopenharmony_ci		err = 0;
3028c2ecf20Sopenharmony_ci	}
3038c2ecf20Sopenharmony_ci
3048c2ecf20Sopenharmony_ci	mutex_unlock(&vcp->vc_mutex);
3058c2ecf20Sopenharmony_ci	return err;
3068c2ecf20Sopenharmony_ci}
3078c2ecf20Sopenharmony_ci
3088c2ecf20Sopenharmony_ci
3098c2ecf20Sopenharmony_cistatic int coda_psdev_release(struct inode * inode, struct file * file)
3108c2ecf20Sopenharmony_ci{
3118c2ecf20Sopenharmony_ci	struct venus_comm *vcp = (struct venus_comm *) file->private_data;
3128c2ecf20Sopenharmony_ci	struct upc_req *req, *tmp;
3138c2ecf20Sopenharmony_ci
3148c2ecf20Sopenharmony_ci	if (!vcp || !vcp->vc_inuse ) {
3158c2ecf20Sopenharmony_ci		pr_warn("%s: Not open.\n", __func__);
3168c2ecf20Sopenharmony_ci		return -1;
3178c2ecf20Sopenharmony_ci	}
3188c2ecf20Sopenharmony_ci
3198c2ecf20Sopenharmony_ci	mutex_lock(&vcp->vc_mutex);
3208c2ecf20Sopenharmony_ci
3218c2ecf20Sopenharmony_ci	/* Wakeup clients so they can return. */
3228c2ecf20Sopenharmony_ci	list_for_each_entry_safe(req, tmp, &vcp->vc_pending, uc_chain) {
3238c2ecf20Sopenharmony_ci		list_del(&req->uc_chain);
3248c2ecf20Sopenharmony_ci
3258c2ecf20Sopenharmony_ci		/* Async requests need to be freed here */
3268c2ecf20Sopenharmony_ci		if (req->uc_flags & CODA_REQ_ASYNC) {
3278c2ecf20Sopenharmony_ci			kvfree(req->uc_data);
3288c2ecf20Sopenharmony_ci			kfree(req);
3298c2ecf20Sopenharmony_ci			continue;
3308c2ecf20Sopenharmony_ci		}
3318c2ecf20Sopenharmony_ci		req->uc_flags |= CODA_REQ_ABORT;
3328c2ecf20Sopenharmony_ci		wake_up(&req->uc_sleep);
3338c2ecf20Sopenharmony_ci	}
3348c2ecf20Sopenharmony_ci
3358c2ecf20Sopenharmony_ci	list_for_each_entry_safe(req, tmp, &vcp->vc_processing, uc_chain) {
3368c2ecf20Sopenharmony_ci		list_del(&req->uc_chain);
3378c2ecf20Sopenharmony_ci
3388c2ecf20Sopenharmony_ci		req->uc_flags |= CODA_REQ_ABORT;
3398c2ecf20Sopenharmony_ci		wake_up(&req->uc_sleep);
3408c2ecf20Sopenharmony_ci	}
3418c2ecf20Sopenharmony_ci
3428c2ecf20Sopenharmony_ci	file->private_data = NULL;
3438c2ecf20Sopenharmony_ci	vcp->vc_inuse--;
3448c2ecf20Sopenharmony_ci	mutex_unlock(&vcp->vc_mutex);
3458c2ecf20Sopenharmony_ci	return 0;
3468c2ecf20Sopenharmony_ci}
3478c2ecf20Sopenharmony_ci
3488c2ecf20Sopenharmony_ci
3498c2ecf20Sopenharmony_cistatic const struct file_operations coda_psdev_fops = {
3508c2ecf20Sopenharmony_ci	.owner		= THIS_MODULE,
3518c2ecf20Sopenharmony_ci	.read		= coda_psdev_read,
3528c2ecf20Sopenharmony_ci	.write		= coda_psdev_write,
3538c2ecf20Sopenharmony_ci	.poll		= coda_psdev_poll,
3548c2ecf20Sopenharmony_ci	.unlocked_ioctl	= coda_psdev_ioctl,
3558c2ecf20Sopenharmony_ci	.open		= coda_psdev_open,
3568c2ecf20Sopenharmony_ci	.release	= coda_psdev_release,
3578c2ecf20Sopenharmony_ci	.llseek		= noop_llseek,
3588c2ecf20Sopenharmony_ci};
3598c2ecf20Sopenharmony_ci
3608c2ecf20Sopenharmony_cistatic int __init init_coda_psdev(void)
3618c2ecf20Sopenharmony_ci{
3628c2ecf20Sopenharmony_ci	int i, err = 0;
3638c2ecf20Sopenharmony_ci	if (register_chrdev(CODA_PSDEV_MAJOR, "coda", &coda_psdev_fops)) {
3648c2ecf20Sopenharmony_ci		pr_err("%s: unable to get major %d\n",
3658c2ecf20Sopenharmony_ci		       __func__, CODA_PSDEV_MAJOR);
3668c2ecf20Sopenharmony_ci		return -EIO;
3678c2ecf20Sopenharmony_ci	}
3688c2ecf20Sopenharmony_ci	coda_psdev_class = class_create(THIS_MODULE, "coda");
3698c2ecf20Sopenharmony_ci	if (IS_ERR(coda_psdev_class)) {
3708c2ecf20Sopenharmony_ci		err = PTR_ERR(coda_psdev_class);
3718c2ecf20Sopenharmony_ci		goto out_chrdev;
3728c2ecf20Sopenharmony_ci	}
3738c2ecf20Sopenharmony_ci	for (i = 0; i < MAX_CODADEVS; i++) {
3748c2ecf20Sopenharmony_ci		mutex_init(&(&coda_comms[i])->vc_mutex);
3758c2ecf20Sopenharmony_ci		device_create(coda_psdev_class, NULL,
3768c2ecf20Sopenharmony_ci			      MKDEV(CODA_PSDEV_MAJOR, i), NULL, "cfs%d", i);
3778c2ecf20Sopenharmony_ci	}
3788c2ecf20Sopenharmony_ci	coda_sysctl_init();
3798c2ecf20Sopenharmony_ci	goto out;
3808c2ecf20Sopenharmony_ci
3818c2ecf20Sopenharmony_ciout_chrdev:
3828c2ecf20Sopenharmony_ci	unregister_chrdev(CODA_PSDEV_MAJOR, "coda");
3838c2ecf20Sopenharmony_ciout:
3848c2ecf20Sopenharmony_ci	return err;
3858c2ecf20Sopenharmony_ci}
3868c2ecf20Sopenharmony_ci
3878c2ecf20Sopenharmony_ciMODULE_AUTHOR("Jan Harkes, Peter J. Braam");
3888c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Coda Distributed File System VFS interface");
3898c2ecf20Sopenharmony_ciMODULE_ALIAS_CHARDEV_MAJOR(CODA_PSDEV_MAJOR);
3908c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL");
3918c2ecf20Sopenharmony_ciMODULE_VERSION("7.0");
3928c2ecf20Sopenharmony_ci
3938c2ecf20Sopenharmony_cistatic int __init init_coda(void)
3948c2ecf20Sopenharmony_ci{
3958c2ecf20Sopenharmony_ci	int status;
3968c2ecf20Sopenharmony_ci	int i;
3978c2ecf20Sopenharmony_ci
3988c2ecf20Sopenharmony_ci	status = coda_init_inodecache();
3998c2ecf20Sopenharmony_ci	if (status)
4008c2ecf20Sopenharmony_ci		goto out2;
4018c2ecf20Sopenharmony_ci	status = init_coda_psdev();
4028c2ecf20Sopenharmony_ci	if ( status ) {
4038c2ecf20Sopenharmony_ci		pr_warn("Problem (%d) in init_coda_psdev\n", status);
4048c2ecf20Sopenharmony_ci		goto out1;
4058c2ecf20Sopenharmony_ci	}
4068c2ecf20Sopenharmony_ci
4078c2ecf20Sopenharmony_ci	status = register_filesystem(&coda_fs_type);
4088c2ecf20Sopenharmony_ci	if (status) {
4098c2ecf20Sopenharmony_ci		pr_warn("failed to register filesystem!\n");
4108c2ecf20Sopenharmony_ci		goto out;
4118c2ecf20Sopenharmony_ci	}
4128c2ecf20Sopenharmony_ci	return 0;
4138c2ecf20Sopenharmony_ciout:
4148c2ecf20Sopenharmony_ci	for (i = 0; i < MAX_CODADEVS; i++)
4158c2ecf20Sopenharmony_ci		device_destroy(coda_psdev_class, MKDEV(CODA_PSDEV_MAJOR, i));
4168c2ecf20Sopenharmony_ci	class_destroy(coda_psdev_class);
4178c2ecf20Sopenharmony_ci	unregister_chrdev(CODA_PSDEV_MAJOR, "coda");
4188c2ecf20Sopenharmony_ci	coda_sysctl_clean();
4198c2ecf20Sopenharmony_ciout1:
4208c2ecf20Sopenharmony_ci	coda_destroy_inodecache();
4218c2ecf20Sopenharmony_ciout2:
4228c2ecf20Sopenharmony_ci	return status;
4238c2ecf20Sopenharmony_ci}
4248c2ecf20Sopenharmony_ci
4258c2ecf20Sopenharmony_cistatic void __exit exit_coda(void)
4268c2ecf20Sopenharmony_ci{
4278c2ecf20Sopenharmony_ci        int err, i;
4288c2ecf20Sopenharmony_ci
4298c2ecf20Sopenharmony_ci	err = unregister_filesystem(&coda_fs_type);
4308c2ecf20Sopenharmony_ci	if (err != 0)
4318c2ecf20Sopenharmony_ci		pr_warn("failed to unregister filesystem\n");
4328c2ecf20Sopenharmony_ci	for (i = 0; i < MAX_CODADEVS; i++)
4338c2ecf20Sopenharmony_ci		device_destroy(coda_psdev_class, MKDEV(CODA_PSDEV_MAJOR, i));
4348c2ecf20Sopenharmony_ci	class_destroy(coda_psdev_class);
4358c2ecf20Sopenharmony_ci	unregister_chrdev(CODA_PSDEV_MAJOR, "coda");
4368c2ecf20Sopenharmony_ci	coda_sysctl_clean();
4378c2ecf20Sopenharmony_ci	coda_destroy_inodecache();
4388c2ecf20Sopenharmony_ci}
4398c2ecf20Sopenharmony_ci
4408c2ecf20Sopenharmony_cimodule_init(init_coda);
4418c2ecf20Sopenharmony_cimodule_exit(exit_coda);
4428c2ecf20Sopenharmony_ci
443