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

📄 soc_common.c

📁 Linux Kernel 2.6.9 for OMAP1710
💻 C
📖 第 1 页 / 共 2 页
字号:
		(map->flags&MAP_AUTOSZ)?"AUTOSZ ":"",		(map->flags&MAP_0WS)?"0WS ":"",		(map->flags&MAP_WRPROT)?"WRPROT ":"",		(map->flags&MAP_ATTRIB)?"ATTRIB ":"",		(map->flags&MAP_USE_WAIT)?"USE_WAIT ":"");	if (map->map >= MAX_WIN)		return -EINVAL;	if (map->flags & MAP_ACTIVE) {		if (speed == 0)			speed = 300;	} else {		speed = 0;	}	if (map->flags & MAP_ATTRIB) {		res = &skt->res_attr;		skt->spd_attr[map->map] = speed;		skt->spd_mem[map->map] = 0;	} else {		res = &skt->res_mem;		skt->spd_attr[map->map] = 0;		skt->spd_mem[map->map] = speed;	}	skt->ops->set_timing(skt);	map->static_start = res->start + map->card_start;	return 0;}struct bittbl {	unsigned int mask;	const char *name;};static struct bittbl status_bits[] = {	{ SS_WRPROT,		"SS_WRPROT"	},	{ SS_BATDEAD,		"SS_BATDEAD"	},	{ SS_BATWARN,		"SS_BATWARN"	},	{ SS_READY,		"SS_READY"	},	{ SS_DETECT,		"SS_DETECT"	},	{ SS_POWERON,		"SS_POWERON"	},	{ SS_STSCHG,		"SS_STSCHG"	},	{ SS_3VCARD,		"SS_3VCARD"	},	{ SS_XVCARD,		"SS_XVCARD"	},};static struct bittbl conf_bits[] = {	{ SS_PWR_AUTO,		"SS_PWR_AUTO"	},	{ SS_IOCARD,		"SS_IOCARD"	},	{ SS_RESET,		"SS_RESET"	},	{ SS_DMA_MODE,		"SS_DMA_MODE"	},	{ SS_SPKR_ENA,		"SS_SPKR_ENA"	},	{ SS_OUTPUT_ENA,	"SS_OUTPUT_ENA"	},};static voiddump_bits(char **p, const char *prefix, unsigned int val, struct bittbl *bits, int sz){	char *b = *p;	int i;	b += sprintf(b, "%-9s:", prefix);	for (i = 0; i < sz; i++)		if (val & bits[i].mask)			b += sprintf(b, " %s", bits[i].name);	*b++ = '\n';	*p = b;}/* * Implements the /sys/class/pcmcia_socket/??/status file. * * Returns: the number of characters added to the buffer */static ssize_t show_status(struct class_device *class_dev, char *buf){	struct soc_pcmcia_socket *skt =		container_of(class_dev, struct soc_pcmcia_socket, socket.dev);	char *p = buf;	p+=sprintf(p, "slot     : %d\n", skt->nr);	dump_bits(&p, "status", skt->status,		  status_bits, ARRAY_SIZE(status_bits));	dump_bits(&p, "csc_mask", skt->cs_state.csc_mask,		  status_bits, ARRAY_SIZE(status_bits));	dump_bits(&p, "cs_flags", skt->cs_state.flags,		  conf_bits, ARRAY_SIZE(conf_bits));	p+=sprintf(p, "Vcc      : %d\n", skt->cs_state.Vcc);	p+=sprintf(p, "Vpp      : %d\n", skt->cs_state.Vpp);	p+=sprintf(p, "IRQ      : %d (%d)\n", skt->cs_state.io_irq, skt->irq);	if (skt->ops->show_timing)		p+=skt->ops->show_timing(skt, p);	return p-buf;}static CLASS_DEVICE_ATTR(status, S_IRUGO, show_status, NULL);static struct pccard_operations soc_common_pcmcia_operations = {	.init			= soc_common_pcmcia_sock_init,	.suspend		= soc_common_pcmcia_suspend,	.get_status		= soc_common_pcmcia_get_status,	.get_socket		= soc_common_pcmcia_get_socket,	.set_socket		= soc_common_pcmcia_set_socket,	.set_io_map		= soc_common_pcmcia_set_io_map,	.set_mem_map		= soc_common_pcmcia_set_mem_map,};int soc_pcmcia_request_irqs(struct soc_pcmcia_socket *skt,			    struct pcmcia_irqs *irqs, int nr){	int i, res = 0;	for (i = 0; i < nr; i++) {		if (irqs[i].sock != skt->nr)			continue;		res = request_irq(irqs[i].irq, soc_common_pcmcia_interrupt,				  SA_INTERRUPT, irqs[i].str, skt);		if (res)			break;		set_irq_type(irqs[i].irq, IRQT_NOEDGE);	}	if (res) {		printk(KERN_ERR "PCMCIA: request for IRQ%d failed (%d)\n",			irqs[i].irq, res);		while (i--)			if (irqs[i].sock == skt->nr)				free_irq(irqs[i].irq, skt);	}	return res;}EXPORT_SYMBOL(soc_pcmcia_request_irqs);void soc_pcmcia_free_irqs(struct soc_pcmcia_socket *skt,			  struct pcmcia_irqs *irqs, int nr){	int i;	for (i = 0; i < nr; i++)		if (irqs[i].sock == skt->nr)			free_irq(irqs[i].irq, skt);}EXPORT_SYMBOL(soc_pcmcia_free_irqs);void soc_pcmcia_disable_irqs(struct soc_pcmcia_socket *skt,			     struct pcmcia_irqs *irqs, int nr){	int i;	for (i = 0; i < nr; i++)		if (irqs[i].sock == skt->nr)			set_irq_type(irqs[i].irq, IRQT_NOEDGE);}EXPORT_SYMBOL(soc_pcmcia_disable_irqs);void soc_pcmcia_enable_irqs(struct soc_pcmcia_socket *skt,			    struct pcmcia_irqs *irqs, int nr){	int i;	for (i = 0; i < nr; i++)		if (irqs[i].sock == skt->nr) {			set_irq_type(irqs[i].irq, IRQT_RISING);			set_irq_type(irqs[i].irq, IRQT_BOTHEDGE);		}}EXPORT_SYMBOL(soc_pcmcia_enable_irqs);LIST_HEAD(soc_pcmcia_sockets);DECLARE_MUTEX(soc_pcmcia_sockets_lock);static const char *skt_names[] = {	"PCMCIA socket 0",	"PCMCIA socket 1",};struct skt_dev_info {	int nskt;	struct soc_pcmcia_socket skt[0];};#define SKT_DEV_INFO_SIZE(n) \	(sizeof(struct skt_dev_info) + (n)*sizeof(struct soc_pcmcia_socket))#ifdef CONFIG_CPU_FREQstatic intsoc_pcmcia_notifier(struct notifier_block *nb, unsigned long val, void *data){	struct soc_pcmcia_socket *skt;	struct cpufreq_freqs *freqs = data;	int ret = 0;	down(&soc_pcmcia_sockets_lock);	list_for_each_entry(skt, &soc_pcmcia_sockets, node)		if ( skt->ops->frequency_change )			ret += skt->ops->frequency_change(skt, val, freqs);	up(&soc_pcmcia_sockets_lock);	return ret;}static struct notifier_block soc_pcmcia_notifier_block = {	.notifier_call	= soc_pcmcia_notifier};static int soc_pcmcia_cpufreq_register(void){	int ret;	ret = cpufreq_register_notifier(&soc_pcmcia_notifier_block,					CPUFREQ_TRANSITION_NOTIFIER);	if (ret < 0)		printk(KERN_ERR "Unable to register CPU frequency change "				"notifier for PCMCIA (%d)\n", ret);	return ret;}static void soc_pcmcia_cpufreq_unregister(void){	cpufreq_unregister_notifier(&soc_pcmcia_notifier_block, CPUFREQ_TRANSITION_NOTIFIER);}#else#define soc_pcmcia_cpufreq_register()#define soc_pcmcia_cpufreq_unregister()#endifint soc_common_drv_pcmcia_probe(struct device *dev, struct pcmcia_low_level *ops, int first, int nr){	struct skt_dev_info *sinfo;	struct soc_pcmcia_socket *skt;	int ret, i;	down(&soc_pcmcia_sockets_lock);	sinfo = kmalloc(SKT_DEV_INFO_SIZE(nr), GFP_KERNEL);	if (!sinfo) {		ret = -ENOMEM;		goto out;	}	memset(sinfo, 0, SKT_DEV_INFO_SIZE(nr));	sinfo->nskt = nr;	/*	 * Initialise the per-socket structure.	 */	for (i = 0; i < nr; i++) {		skt = &sinfo->skt[i];		skt->socket.ops = &soc_common_pcmcia_operations;		skt->socket.owner = ops->owner;		skt->socket.dev.dev = dev;		init_timer(&skt->poll_timer);		skt->poll_timer.function = soc_common_pcmcia_poll_event;		skt->poll_timer.data = (unsigned long)skt;		skt->poll_timer.expires = jiffies + SOC_PCMCIA_POLL_PERIOD;		skt->nr		= first + i;		skt->irq	= NO_IRQ;		skt->dev	= dev;		skt->ops	= ops;		skt->res_skt.start	= _PCMCIA(skt->nr);		skt->res_skt.end	= _PCMCIA(skt->nr) + PCMCIASp - 1;		skt->res_skt.name	= skt_names[skt->nr];		skt->res_skt.flags	= IORESOURCE_MEM;		ret = request_resource(&iomem_resource, &skt->res_skt);		if (ret)			goto out_err_1;		skt->res_io.start	= _PCMCIAIO(skt->nr);		skt->res_io.end		= _PCMCIAIO(skt->nr) + PCMCIAIOSp - 1;		skt->res_io.name	= "io";		skt->res_io.flags	= IORESOURCE_MEM | IORESOURCE_BUSY;		ret = request_resource(&skt->res_skt, &skt->res_io);		if (ret)			goto out_err_2;		skt->res_mem.start	= _PCMCIAMem(skt->nr);		skt->res_mem.end	= _PCMCIAMem(skt->nr) + PCMCIAMemSp - 1;		skt->res_mem.name	= "memory";		skt->res_mem.flags	= IORESOURCE_MEM;		ret = request_resource(&skt->res_skt, &skt->res_mem);		if (ret)			goto out_err_3;		skt->res_attr.start	= _PCMCIAAttr(skt->nr);		skt->res_attr.end	= _PCMCIAAttr(skt->nr) + PCMCIAAttrSp - 1;		skt->res_attr.name	= "attribute";		skt->res_attr.flags	= IORESOURCE_MEM;		ret = request_resource(&skt->res_skt, &skt->res_attr);		if (ret)			goto out_err_4;		skt->virt_io = ioremap(skt->res_io.start, 0x10000);		if (skt->virt_io == NULL) {			ret = -ENOMEM;			goto out_err_5;		}		if ( list_empty(&soc_pcmcia_sockets) )			soc_pcmcia_cpufreq_register();		list_add(&skt->node, &soc_pcmcia_sockets);		/*		 * We initialize default socket timing here, because		 * we are not guaranteed to see a SetIOMap operation at		 * runtime.		 */		ops->set_timing(skt);		ret = ops->hw_init(skt);		if (ret)			goto out_err_6;		skt->socket.features = SS_CAP_STATIC_MAP|SS_CAP_PCCARD;		skt->socket.irq_mask = 0;		skt->socket.map_size = PAGE_SIZE;		skt->socket.pci_irq = skt->irq;		skt->socket.io_offset = (unsigned long)skt->virt_io;		skt->status = soc_common_pcmcia_skt_state(skt);		ret = pcmcia_register_socket(&skt->socket);		if (ret)			goto out_err_7;		WARN_ON(skt->socket.sock != i);		add_timer(&skt->poll_timer);		class_device_create_file(&skt->socket.dev, &class_device_attr_status);	}	dev_set_drvdata(dev, sinfo);	ret = 0;	goto out;	do {		skt = &sinfo->skt[i];		del_timer_sync(&skt->poll_timer);		pcmcia_unregister_socket(&skt->socket); out_err_7:		flush_scheduled_work();		ops->hw_shutdown(skt); out_err_6: 		list_del(&skt->node);		iounmap(skt->virt_io); out_err_5:		release_resource(&skt->res_attr); out_err_4:		release_resource(&skt->res_mem); out_err_3:		release_resource(&skt->res_io); out_err_2:		release_resource(&skt->res_skt); out_err_1:		i--;	} while (i > 0);	kfree(sinfo); out:	up(&soc_pcmcia_sockets_lock);	return ret;}int soc_common_drv_pcmcia_remove(struct device *dev){	struct skt_dev_info *sinfo = dev_get_drvdata(dev);	int i;	dev_set_drvdata(dev, NULL);	down(&soc_pcmcia_sockets_lock);	for (i = 0; i < sinfo->nskt; i++) {		struct soc_pcmcia_socket *skt = &sinfo->skt[i];		del_timer_sync(&skt->poll_timer);		pcmcia_unregister_socket(&skt->socket);		flush_scheduled_work();		skt->ops->hw_shutdown(skt);		soc_common_pcmcia_config_skt(skt, &dead_socket);		list_del(&skt->node);		iounmap(skt->virt_io);		skt->virt_io = NULL;		release_resource(&skt->res_attr);		release_resource(&skt->res_mem);		release_resource(&skt->res_io);		release_resource(&skt->res_skt);	}	if ( list_empty(&soc_pcmcia_sockets) )		soc_pcmcia_cpufreq_unregister();	up(&soc_pcmcia_sockets_lock);	kfree(sinfo);	return 0;}

⌨️ 快捷键说明

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