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

📄 ztdynamic.c

📁 This a SOFTWARE pbx DRIVER
💻 C
📖 第 1 页 / 共 2 页
字号:
	struct zt_dynamic *z;	z = dspans;	while(z) {		if (!strcmp(z->dname, zds->driver) &&		    !strcmp(z->addr, zds->addr))			break;		z = z->next;	}	return z;}static struct zt_dynamic_driver *find_driver(char *name){	struct zt_dynamic_driver *ztd;	ztd = drivers;	while(ztd) {		/* here's our driver */		if (!strcmp(name, ztd->name))			break;		ztd = ztd->next;	}	return ztd;}static int destroy_dynamic(ZT_DYNAMIC_SPAN *zds){	unsigned long flags;	struct zt_dynamic *z, *cur, *prev=NULL;	spin_lock_irqsave(&dlock, flags);	z = find_dynamic(zds);	if (!z) {		spin_unlock_irqrestore(&dlock, flags);		return -EINVAL;	}	/* Don't destroy span until it is in use */	if (z->usecount) {		spin_unlock_irqrestore(&dlock, flags);		printk("Attempt to destroy dynamic span while it is in use\n");		return -EBUSY;	}	/* Unlink it */	cur = dspans;	while(cur) {		if (cur == z) {			if (prev)				prev->next = z->next;			else				dspans = z->next;			break;		}		prev = cur;		cur = cur->next;	}	spin_unlock_irqrestore(&dlock, flags);	/* Destroy it */	dynamic_destroy(z);		return 0;}static int ztd_rbsbits(struct zt_chan *chan, int bits){	/* Don't have to do anything */	return 0;}static int ztd_open(struct zt_chan *chan){	struct zt_dynamic *z;	z = chan->span->pvt;	if (z) {		if (z->dead)			return -ENODEV;		z->usecount++;	}#ifndef LINUX26	MOD_INC_USE_COUNT;#endif		return 0;}static int ztd_chanconfig(struct zt_chan *chan, int sigtype){	return 0;}static int ztd_close(struct zt_chan *chan){	struct zt_dynamic *z;	z = chan->span->pvt;	if (z) 		z->usecount--;	if (z->dead && !z->usecount)		dynamic_destroy(z);#ifndef LINUX26	MOD_DEC_USE_COUNT;#endif		return 0;}static int create_dynamic(ZT_DYNAMIC_SPAN *zds){	struct zt_dynamic *z;	struct zt_dynamic_driver *ztd;	unsigned long flags;	int x;	int bufsize;	if (zds->numchans < 1) {		printk("Can't be less than 1 channel (%d)!\n", zds->numchans);		return -EINVAL;	}	if (zds->numchans >= ZT_DYNAMIC_MAX_CHANS) {		printk("Can't create dynamic span with greater than %d channels.  See ztdynamic.c and increase ZT_DYNAMIC_MAX_CHANS\n", zds->numchans);		return -EINVAL;	}	spin_lock_irqsave(&dlock, flags);	z = find_dynamic(zds);	spin_unlock_irqrestore(&dlock, flags);	if (z)		return -EEXIST;	/* XXX There is a silly race here.  We check it doesn't exist, but	       someone could create it between now and then and we'd end up	       with two of them.  We don't want to hold the spinlock	       for *too* long though, especially not if there is a possibility	       of kmalloc.  XXX */	/* Allocate memory */	z = (struct zt_dynamic *)kmalloc(sizeof(struct zt_dynamic), GFP_KERNEL);	if (!z) 		return -ENOMEM;	/* Zero it out */	memset(z, 0, sizeof(struct zt_dynamic));	/* Allocate other memories */	z->chans = kmalloc(sizeof(struct zt_chan) * zds->numchans, GFP_KERNEL);	if (!z->chans) {		dynamic_destroy(z);		return -ENOMEM;	}	/* Zero out channel stuff */	memset(z->chans, 0, sizeof(struct zt_chan) * zds->numchans);	/* Allocate message buffer with sample space and header space */	bufsize = zds->numchans * ZT_CHUNKSIZE + zds->numchans / 4 + 48;	z->msgbuf = kmalloc(bufsize, GFP_KERNEL);	if (!z->msgbuf) {		dynamic_destroy(z);		return -ENOMEM;	}		/* Zero out -- probably not needed but why not */	memset(z->msgbuf, 0, bufsize);	/* Setup parameters properly assuming we're going to be okay. */	strncpy(z->dname, zds->driver, sizeof(z->driver) - 1);	strncpy(z->addr, zds->addr, sizeof(z->addr) - 1);	z->timing = zds->timing;	sprintf(z->span.name, "ZTD/%s/%s", zds->driver, zds->addr);	sprintf(z->span.desc, "Dynamic '%s' span at '%s'", zds->driver, zds->addr);	z->span.channels = zds->numchans;	z->span.pvt = z;	z->span.deflaw = ZT_LAW_MULAW;	z->span.flags |= ZT_FLAG_RBS;	z->span.chans = z->chans;	z->span.rbsbits = ztd_rbsbits;	z->span.open = ztd_open;	z->span.close = ztd_close;	z->span.chanconfig = ztd_chanconfig;	for (x=0;x<zds->numchans;x++) {		sprintf(z->chans[x].name, "ZTD/%s/%s/%d", zds->driver, zds->addr, x+1);		z->chans[x].sigcap = ZT_SIG_EM | ZT_SIG_CLEAR | ZT_SIG_FXSLS |				     ZT_SIG_FXSKS | ZT_SIG_FXSGS | ZT_SIG_FXOLS |				     ZT_SIG_FXOKS | ZT_SIG_FXOGS | ZT_SIG_SF;		z->chans[x].chanpos = x + 1;		z->chans[x].pvt = z;	}		spin_lock_irqsave(&dlock, flags);	ztd = find_driver(zds->driver);	if (!ztd) {		/* Try loading the right module */		char fn[80];		spin_unlock_irqrestore(&dlock, flags);		sprintf(fn, "ztd-%s", zds->driver);		request_module(fn);		spin_lock_irqsave(&dlock, flags);		ztd = find_driver(zds->driver);	}	spin_unlock_irqrestore(&dlock, flags);	/* Another race -- should let the module get unloaded while we	   have it here */	if (!ztd) {		printk("No such driver '%s' for dynamic span\n", zds->driver);		dynamic_destroy(z);		return -EINVAL;	}	/* Create the stuff */	z->pvt = ztd->create(&z->span, z->addr);	if (!z->pvt) {		printk("Driver '%s' (%s) rejected address '%s'\n", ztd->name, ztd->desc, z->addr);		/* Creation failed */		return -EINVAL;	}	/* Remember the driver */	z->driver = ztd;	/* Whee!  We're created.  Now register the span */	if (zt_register(&z->span, 0)) {		printk("Unable to register span '%s'\n", z->span.name);		dynamic_destroy(z);		return -EINVAL;	}	/* Okay, created and registered. add it to the list */	spin_lock_irqsave(&dlock, flags);	z->next = dspans;	dspans = z;	spin_unlock_irqrestore(&dlock, flags);	checkmaster();	/* All done */	return z->span.spanno;}#ifdef ENABLE_TASKLETSstatic void ztd_tasklet(unsigned long data){	taskletrun++;	if (taskletpending) {		taskletexec++;		__ztdynamic_run();	}	taskletpending = 0;}#endifstatic int ztdynamic_ioctl(unsigned int cmd, unsigned long data){	ZT_DYNAMIC_SPAN zds;	int res;	switch(cmd) {	case 0:		/* This is called just before rotation.  If none of our		   spans are pulling timing, then now is the time to process		   them */		if (!hasmaster)			ztdynamic_run();		return 0;	case ZT_DYNAMIC_CREATE:		if (copy_from_user(&zds, (ZT_DYNAMIC_SPAN *)data, sizeof(zds)))			return -EFAULT;		if (debug)			printk("Dynamic Create\n");		res = create_dynamic(&zds);		if (res < 0)			return res;		zds.spanno = res;		/* Let them know the new span number */		if (copy_to_user((ZT_DYNAMIC_SPAN *)data, &zds, sizeof(zds)))			return -EFAULT;		return 0;	case ZT_DYNAMIC_DESTROY:		if (copy_from_user(&zds, (ZT_DYNAMIC_SPAN *)data, sizeof(zds)))			return -EFAULT;		if (debug)			printk("Dynamic Destroy\n");		return destroy_dynamic(&zds);	}	return -ENOTTY;}int zt_dynamic_register(struct zt_dynamic_driver *dri){	unsigned long flags;	int res = 0;	spin_lock_irqsave(&dlock, flags);	if (find_driver(dri->name))		res = -1;	else {		dri->next = drivers;		drivers = dri;	}	spin_unlock_irqrestore(&dlock, flags);	return res;}void zt_dynamic_unregister(struct zt_dynamic_driver *dri){	struct zt_dynamic_driver *cur, *prev=NULL;	struct zt_dynamic *z, *zp, *zn;	unsigned long flags;	spin_lock_irqsave(&dlock, flags);	cur = drivers;	while(cur) {		if (cur == dri) {			if (prev)				prev->next = cur->next;			else				drivers = cur->next;			break;		}		prev = cur;		cur = cur->next;	}	z = dspans;	zp = NULL;	while(z) {		zn = z->next;		if (z->driver == dri) {			/* Unlink */			if (zp)				zp->next = z->next;			else				dspans = z->next;			if (!z->usecount)				dynamic_destroy(z);			else				z->dead = 1;		} else {			zp = z;		}		z = zn;	}	spin_unlock_irqrestore(&dlock, flags);}struct timer_list alarmcheck;static void check_for_red_alarm(unsigned long ignored){	unsigned long flags;	int newalarm;	int alarmchanged = 0;	struct zt_dynamic *z;	spin_lock_irqsave(&dlock, flags);	z = dspans;	while(z) {		newalarm = z->span.alarms & ~ZT_ALARM_RED;		/* If nothing received for a minute, consider that RED ALARM */		if ((jiffies - z->rxjif) > 1000 / HZ) {			newalarm |= ZT_ALARM_RED;			if (z->span.alarms != newalarm) {				z->span.alarms = newalarm;				zt_alarm_notify(&z->span);				alarmchanged++;			}		}		z = z->next;	}	spin_unlock_irqrestore(&dlock, flags);	if (alarmchanged)		checkmaster();	/* Do the next one */	mod_timer(&alarmcheck, jiffies + 1 * HZ);	}int ztdynamic_init(void){	zt_set_dynamic_ioctl(ztdynamic_ioctl);	/* Start process to check for RED ALARM */	init_timer(&alarmcheck);	alarmcheck.expires = 0;	alarmcheck.data = 0;	alarmcheck.function = check_for_red_alarm;	/* Check once per second */	mod_timer(&alarmcheck, jiffies + 1 * HZ);#ifdef ENABLE_TASKLETS	tasklet_init(&ztd_tlet, ztd_tasklet, 0);#endif	printk("Zaptel Dynamic Span support LOADED\n");	return 0;}void ztdynamic_cleanup(void){#ifdef ENABLE_TASKLETS	if (taskletpending) {		tasklet_disable(&ztd_tlet);		tasklet_kill(&ztd_tlet);	}#endif	zt_set_dynamic_ioctl(NULL);	del_timer(&alarmcheck);	printk("Zaptel Dynamic Span support unloaded\n");}#ifdef LINUX26module_param(debug, int, 0600);#elseMODULE_PARM(debug, "i");#endifMODULE_DESCRIPTION("Zaptel Dynamic Span Support");MODULE_AUTHOR("Mark Spencer <markster@linux-support.net>");#ifdef MODULE_LICENSEMODULE_LICENSE("GPL");#endifmodule_init(ztdynamic_init);module_exit(ztdynamic_cleanup);

⌨️ 快捷键说明

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