📄 soc_common.c
字号:
(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 + -