⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 mwavedd.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
/*** mwavedd.c -- mwave device driver*** Written By: Mike Sullivan IBM Corporation** Copyright (C) 1999 IBM Corporation** This program is free software; you can redistribute it and/or modify* it under the terms of the GNU General Public License as published by* the Free Software Foundation; either version 2 of the License, or* (at your option) any later version.** 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.** NO WARRANTY* THE PROGRAM IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR* CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT* LIMITATION, ANY WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT,* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. Each Recipient is* solely responsible for determining the appropriateness of using and* distributing the Program and assumes all risks associated with its* exercise of rights under this Agreement, including but not limited to* the risks and costs of program errors, damage to or loss of data,* programs or equipment, and unavailability or interruption of operations.** DISCLAIMER OF LIABILITY* NEITHER RECIPIENT NOR ANY CONTRIBUTORS SHALL HAVE ANY LIABILITY FOR ANY* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL* DAMAGES (INCLUDING WITHOUT LIMITATION LOST PROFITS), HOWEVER CAUSED AND* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR* TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE* USE OR DISTRIBUTION OF THE PROGRAM OR THE EXERCISE OF ANY RIGHTS GRANTED* HEREUNDER, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES** You should have received a copy of the GNU General Public License* along with this program; if not, write to the Free Software* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA*** 10/23/2000 - Alpha Release*	First release to the public*/#include <linux/module.h>#include <linux/kernel.h>#include <linux/fs.h>#include <linux/init.h>#include <linux/major.h>#include <linux/miscdevice.h>#include <linux/device.h>#include <linux/serial.h>#include <linux/sched.h>#include <linux/spinlock.h>#include <linux/delay.h>#include <linux/serial_8250.h>#include "smapi.h"#include "mwavedd.h"#include "3780i.h"#include "tp3780i.h"MODULE_DESCRIPTION("3780i Advanced Communications Processor (Mwave) driver");MODULE_AUTHOR("Mike Sullivan and Paul Schroeder");MODULE_LICENSE("GPL");/** These parameters support the setting of MWave resources. Note that no* checks are made against other devices (ie. superio) for conflicts.* We'll depend on users using the tpctl utility to do that for now*/int mwave_debug = 0;int mwave_3780i_irq = 0;int mwave_3780i_io = 0;int mwave_uart_irq = 0;int mwave_uart_io = 0;module_param(mwave_debug, int, 0);module_param(mwave_3780i_irq, int, 0);module_param(mwave_3780i_io, int, 0);module_param(mwave_uart_irq, int, 0);module_param(mwave_uart_io, int, 0);static int mwave_open(struct inode *inode, struct file *file);static int mwave_close(struct inode *inode, struct file *file);static int mwave_ioctl(struct inode *inode, struct file *filp,                       unsigned int iocmd, unsigned long ioarg);MWAVE_DEVICE_DATA mwave_s_mdd;static int mwave_open(struct inode *inode, struct file *file){	unsigned int retval = 0;	PRINTK_3(TRACE_MWAVE,		"mwavedd::mwave_open, entry inode %p file %p\n",		 inode, file);	PRINTK_2(TRACE_MWAVE,		"mwavedd::mwave_open, exit return retval %x\n", retval);	return retval;}static int mwave_close(struct inode *inode, struct file *file){	unsigned int retval = 0;	PRINTK_3(TRACE_MWAVE,		"mwavedd::mwave_close, entry inode %p file %p\n",		 inode,  file);	PRINTK_2(TRACE_MWAVE, "mwavedd::mwave_close, exit retval %x\n",		retval);	return retval;}static int mwave_ioctl(struct inode *inode, struct file *file,                       unsigned int iocmd, unsigned long ioarg){	unsigned int retval = 0;	pMWAVE_DEVICE_DATA pDrvData = &mwave_s_mdd;	void __user *arg = (void __user *)ioarg;	PRINTK_5(TRACE_MWAVE,		"mwavedd::mwave_ioctl, entry inode %p file %p cmd %x arg %x\n",		 inode,  file, iocmd, (int) ioarg);	switch (iocmd) {		case IOCTL_MW_RESET:			PRINTK_1(TRACE_MWAVE,				"mwavedd::mwave_ioctl, IOCTL_MW_RESET"				" calling tp3780I_ResetDSP\n");			retval = tp3780I_ResetDSP(&pDrvData->rBDData);			PRINTK_2(TRACE_MWAVE,				"mwavedd::mwave_ioctl, IOCTL_MW_RESET"				" retval %x from tp3780I_ResetDSP\n",				retval);			break;			case IOCTL_MW_RUN:			PRINTK_1(TRACE_MWAVE,				"mwavedd::mwave_ioctl, IOCTL_MW_RUN"				" calling tp3780I_StartDSP\n");			retval = tp3780I_StartDSP(&pDrvData->rBDData);			PRINTK_2(TRACE_MWAVE,				"mwavedd::mwave_ioctl, IOCTL_MW_RUN"				" retval %x from tp3780I_StartDSP\n",				retval);			break;			case IOCTL_MW_DSP_ABILITIES: {			MW_ABILITIES rAbilities;				PRINTK_1(TRACE_MWAVE,				"mwavedd::mwave_ioctl,"				" IOCTL_MW_DSP_ABILITIES calling"				" tp3780I_QueryAbilities\n");			retval = tp3780I_QueryAbilities(&pDrvData->rBDData,					&rAbilities);			PRINTK_2(TRACE_MWAVE,				"mwavedd::mwave_ioctl, IOCTL_MW_DSP_ABILITIES"				" retval %x from tp3780I_QueryAbilities\n",				retval);			if (retval == 0) {				if( copy_to_user(arg, &rAbilities,							sizeof(MW_ABILITIES)) )					return -EFAULT;			}			PRINTK_2(TRACE_MWAVE,				"mwavedd::mwave_ioctl, IOCTL_MW_DSP_ABILITIES"				" exit retval %x\n",				retval);		}			break;			case IOCTL_MW_READ_DATA:		case IOCTL_MW_READCLEAR_DATA: {			MW_READWRITE rReadData;			unsigned short __user *pusBuffer = NULL;				if( copy_from_user(&rReadData, arg,						sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short __user *) (rReadData.pBuf);				PRINTK_4(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_READ_DATA,"				" size %lx, ioarg %lx pusBuffer %p\n",				rReadData.ulDataLength, ioarg, pusBuffer);			retval = tp3780I_ReadWriteDspDStore(&pDrvData->rBDData,					iocmd,					pusBuffer,					rReadData.ulDataLength,					rReadData.usDspAddress);		}			break;			case IOCTL_MW_READ_INST: {			MW_READWRITE rReadData;			unsigned short __user *pusBuffer = NULL;				if( copy_from_user(&rReadData, arg,						sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short __user *) (rReadData.pBuf);				PRINTK_4(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_READ_INST,"				" size %lx, ioarg %lx pusBuffer %p\n",				rReadData.ulDataLength / 2, ioarg,				pusBuffer);			retval = tp3780I_ReadWriteDspDStore(&pDrvData->rBDData,				iocmd, pusBuffer,				rReadData.ulDataLength / 2,				rReadData.usDspAddress);		}			break;			case IOCTL_MW_WRITE_DATA: {			MW_READWRITE rWriteData;			unsigned short __user *pusBuffer = NULL;				if( copy_from_user(&rWriteData, arg,						sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short __user *) (rWriteData.pBuf);				PRINTK_4(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_WRITE_DATA,"				" size %lx, ioarg %lx pusBuffer %p\n",				rWriteData.ulDataLength, ioarg,				pusBuffer);			retval = tp3780I_ReadWriteDspDStore(&pDrvData->rBDData,					iocmd, pusBuffer,					rWriteData.ulDataLength,					rWriteData.usDspAddress);		}			break;			case IOCTL_MW_WRITE_INST: {			MW_READWRITE rWriteData;			unsigned short __user *pusBuffer = NULL;				if( copy_from_user(&rWriteData, arg,						sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short __user *)(rWriteData.pBuf);				PRINTK_4(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_WRITE_INST,"				" size %lx, ioarg %lx pusBuffer %p\n",				rWriteData.ulDataLength, ioarg,				pusBuffer);			retval = tp3780I_ReadWriteDspIStore(&pDrvData->rBDData,					iocmd, pusBuffer,					rWriteData.ulDataLength,					rWriteData.usDspAddress);		}			break;			case IOCTL_MW_REGISTER_IPC: {			unsigned int ipcnum = (unsigned int) ioarg;				PRINTK_3(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_REGISTER_IPC"				" ipcnum %x entry usIntCount %x\n",				ipcnum,				pDrvData->IPCs[ipcnum].usIntCount);				if (ipcnum > ARRAY_SIZE(pDrvData->IPCs)) {				PRINTK_ERROR(KERN_ERR_MWAVE						"mwavedd::mwave_ioctl:"						" IOCTL_MW_REGISTER_IPC:"						" Error: Invalid ipcnum %x\n",						ipcnum);				return -EINVAL;			}			pDrvData->IPCs[ipcnum].bIsHere = FALSE;			pDrvData->IPCs[ipcnum].bIsEnabled = TRUE;				PRINTK_2(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_REGISTER_IPC"				" ipcnum %x exit\n",				ipcnum);		}			break;			case IOCTL_MW_GET_IPC: {			unsigned int ipcnum = (unsigned int) ioarg;				PRINTK_3(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_GET_IPC"				" ipcnum %x, usIntCount %x\n",				ipcnum,				pDrvData->IPCs[ipcnum].usIntCount);			if (ipcnum > ARRAY_SIZE(pDrvData->IPCs)) {				PRINTK_ERROR(KERN_ERR_MWAVE						"mwavedd::mwave_ioctl:"						" IOCTL_MW_GET_IPC: Error:"						" Invalid ipcnum %x\n", ipcnum);				return -EINVAL;			}				if (pDrvData->IPCs[ipcnum].bIsEnabled == TRUE) {				DECLARE_WAITQUEUE(wait, current);				PRINTK_2(TRACE_MWAVE,					"mwavedd::mwave_ioctl, thread for"					" ipc %x going to sleep\n",					ipcnum);				add_wait_queue(&pDrvData->IPCs[ipcnum].ipc_wait_queue, &wait);				pDrvData->IPCs[ipcnum].bIsHere = TRUE;				set_current_state(TASK_INTERRUPTIBLE);				/* check whether an event was signalled by */				/* the interrupt handler while we were gone */				if (pDrvData->IPCs[ipcnum].usIntCount == 1) {	/* first int has occurred (race condition) */					pDrvData->IPCs[ipcnum].usIntCount = 2;	/* first int has been handled */					PRINTK_2(TRACE_MWAVE,						"mwavedd::mwave_ioctl"						" IOCTL_MW_GET_IPC ipcnum %x"						" handling first int\n",						ipcnum);				} else {	/* either 1st int has not yet occurred, or we have already handled the first int */					schedule();					if (pDrvData->IPCs[ipcnum].usIntCount == 1) {						pDrvData->IPCs[ipcnum].usIntCount = 2;					}					PRINTK_2(TRACE_MWAVE,						"mwavedd::mwave_ioctl"						" IOCTL_MW_GET_IPC ipcnum %x"						" woke up and returning to"						" application\n",						ipcnum);				}				pDrvData->IPCs[ipcnum].bIsHere = FALSE;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -