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

📄 mwavedd.c

📁 内核linux2.4.20,可跟rtlinux3.2打补丁 组成实时linux系统,编译内核
💻 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/version.h>#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/proc_fs.h>#include <linux/serial.h>#include <linux/sched.h>#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)#include <linux/spinlock.h>#else#include <asm/spinlock.h>#endif#include <linux/delay.h>#include "smapi.h"#include "mwavedd.h"#include "3780i.h"#include "tp3780i.h"#ifndef __exit#define __exit#endifMODULE_DESCRIPTION("3780i Advanced Communications Processor (Mwave) driver");MODULE_AUTHOR("Mike Sullivan and Paul Schroeder");MODULE_LICENSE("GPL");#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)static int mwave_get_info(char *buf, char **start, off_t offset, int len);#elsestatic int mwave_read_proc(char *buf, char **start, off_t offset, int xlen, int unused);static struct proc_dir_entry mwave_proc = {	0,                      /* unsigned short low_ino */	5,                      /* unsigned short namelen */	"mwave",                /* const char *name */	S_IFREG | S_IRUGO,      /* mode_t mode */	1,                      /* nlink_t nlink */	0,                      /* uid_t uid */	0,                      /* gid_t gid */	0,                      /* unsigned long size */	NULL,                   /* struct inode_operations *ops */	&mwave_read_proc        /* int (*get_info) (...) */};#endif/** 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_PARM(mwave_debug, "i");MODULE_PARM(mwave_3780i_irq, "i");MODULE_PARM(mwave_3780i_io, "i");MODULE_PARM(mwave_uart_irq, "i");MODULE_PARM(mwave_uart_io, "i");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 %x file %x\n",		(int) inode, (int) file);	PRINTK_2(TRACE_MWAVE,		"mwavedd::mwave_open, exit return retval %x\n", retval);	MOD_INC_USE_COUNT;	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 %x file %x\n",		(int) inode, (int) file);	PRINTK_2(TRACE_MWAVE, "mwavedd::mwave_close, exit retval %x\n",		retval);	MOD_DEC_USE_COUNT;	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;	PRINTK_5(TRACE_MWAVE,		"mwavedd::mwave_ioctl, entry inode %x file %x cmd %x arg %x\n",		(int) inode, (int) 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((char *) ioarg, (char *) &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 *pusBuffer = 0;				if( copy_from_user((char *) &rReadData, (char *) ioarg, sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short *) (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,				(void *) pusBuffer, rReadData.ulDataLength, rReadData.usDspAddress);		}			break;			case IOCTL_MW_READ_INST: {			MW_READWRITE rReadData;			unsigned short *pusBuffer = 0;				if( copy_from_user((char *) &rReadData, (char *) ioarg, sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short *) (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 *pusBuffer = 0;				if( copy_from_user((char *) &rWriteData, (char *) ioarg, sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short *) (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 *pusBuffer = 0;				if( copy_from_user((char *) &rWriteData, (char *) ioarg, sizeof(MW_READWRITE)) )				return -EFAULT;			pusBuffer = (unsigned short *) (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 > 16) {				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;	#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)			current->nice = -20;	/* boost to provide priority timing */	#else			current->priority = 0x28;	/* boost to provide priority timing */	#endif				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;			spinlock_t ipc_lock = SPIN_LOCK_UNLOCKED;			unsigned long flags;				PRINTK_3(TRACE_MWAVE,				"mwavedd::mwave_ioctl IOCTL_MW_GET_IPC ipcnum %x, usIntCount %x\n",				ipcnum,				pDrvData->IPCs[ipcnum].usIntCount);			if (ipcnum > 16) {				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) {				PRINTK_2(TRACE_MWAVE,					"mwavedd::mwave_ioctl, thread for ipc %x going to sleep\n",					ipcnum);					spin_lock_irqsave(&ipc_lock, flags);				/* 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 */					spin_unlock_irqrestore(&ipc_lock, flags);					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 */					pDrvData->IPCs[ipcnum].bIsHere = TRUE;

⌨️ 快捷键说明

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