diff --git a/drivers/char/ipmi/ipmi_si_intf.c b/drivers/char/ipmi/ipmi_si_intf.c index 4b832b96d02b..d0a0a5d9e5ff 100644 --- a/drivers/char/ipmi/ipmi_si_intf.c +++ b/drivers/char/ipmi/ipmi_si_intf.c @@ -173,9 +173,6 @@ struct smi_info { * IPMI */ struct si_sm_io io; - int (*io_setup)(struct smi_info *info); - void (*io_cleanup)(struct smi_info *info); - unsigned int io_size; /* * Per-OEM handler, called from handle_flags(). Returns 1 @@ -1488,48 +1485,48 @@ static void port_outl(const struct si_sm_io *io, unsigned int offset, outl(b << io->regshift, addr+(offset * io->regspacing)); } -static void port_cleanup(struct smi_info *info) +static void port_cleanup(struct si_sm_io *io) { - unsigned int addr = info->io.addr_data; + unsigned int addr = io->addr_data; int idx; if (addr) { - for (idx = 0; idx < info->io_size; idx++) - release_region(addr + idx * info->io.regspacing, - info->io.regsize); + for (idx = 0; idx < io->io_size; idx++) + release_region(addr + idx * io->regspacing, + io->regsize); } } -static int port_setup(struct smi_info *info) +static int port_setup(struct si_sm_io *io) { - unsigned int addr = info->io.addr_data; + unsigned int addr = io->addr_data; int idx; if (!addr) return -ENODEV; - info->io_cleanup = port_cleanup; + io->io_cleanup = port_cleanup; /* * Figure out the actual inb/inw/inl/etc routine to use based * upon the register size. */ - switch (info->io.regsize) { + switch (io->regsize) { case 1: - info->io.inputb = port_inb; - info->io.outputb = port_outb; + io->inputb = port_inb; + io->outputb = port_outb; break; case 2: - info->io.inputb = port_inw; - info->io.outputb = port_outw; + io->inputb = port_inw; + io->outputb = port_outw; break; case 4: - info->io.inputb = port_inl; - info->io.outputb = port_outl; + io->inputb = port_inl; + io->outputb = port_outl; break; default: - dev_warn(info->io.dev, "Invalid register size: %d\n", - info->io.regsize); + dev_warn(io->dev, "Invalid register size: %d\n", + io->regsize); return -EINVAL; } @@ -1539,13 +1536,13 @@ static int port_setup(struct smi_info *info) * entire I/O region. Therefore we must register each I/O * port separately. */ - for (idx = 0; idx < info->io_size; idx++) { - if (request_region(addr + idx * info->io.regspacing, - info->io.regsize, DEVICE_NAME) == NULL) { + for (idx = 0; idx < io->io_size; idx++) { + if (request_region(addr + idx * io->regspacing, + io->regsize, DEVICE_NAME) == NULL) { /* Undo allocations */ while (idx--) - release_region(addr + idx * info->io.regspacing, - info->io.regsize); + release_region(addr + idx * io->regspacing, + io->regsize); return -EIO; } } @@ -1604,60 +1601,60 @@ static void mem_outq(const struct si_sm_io *io, unsigned int offset, } #endif -static void mem_region_cleanup(struct smi_info *info, int num) +static void mem_region_cleanup(struct si_sm_io *io, int num) { - unsigned long addr = info->io.addr_data; + unsigned long addr = io->addr_data; int idx; for (idx = 0; idx < num; idx++) - release_mem_region(addr + idx * info->io.regspacing, - info->io.regsize); + release_mem_region(addr + idx * io->regspacing, + io->regsize); } -static void mem_cleanup(struct smi_info *info) +static void mem_cleanup(struct si_sm_io *io) { - if (info->io.addr) { - iounmap(info->io.addr); - mem_region_cleanup(info, info->io_size); + if (io->addr) { + iounmap(io->addr); + mem_region_cleanup(io, io->io_size); } } -static int mem_setup(struct smi_info *info) +static int mem_setup(struct si_sm_io *io) { - unsigned long addr = info->io.addr_data; + unsigned long addr = io->addr_data; int mapsize, idx; if (!addr) return -ENODEV; - info->io_cleanup = mem_cleanup; + io->io_cleanup = mem_cleanup; /* * Figure out the actual readb/readw/readl/etc routine to use based * upon the register size. */ - switch (info->io.regsize) { + switch (io->regsize) { case 1: - info->io.inputb = intf_mem_inb; - info->io.outputb = intf_mem_outb; + io->inputb = intf_mem_inb; + io->outputb = intf_mem_outb; break; case 2: - info->io.inputb = intf_mem_inw; - info->io.outputb = intf_mem_outw; + io->inputb = intf_mem_inw; + io->outputb = intf_mem_outw; break; case 4: - info->io.inputb = intf_mem_inl; - info->io.outputb = intf_mem_outl; + io->inputb = intf_mem_inl; + io->outputb = intf_mem_outl; break; #ifdef readq case 8: - info->io.inputb = mem_inq; - info->io.outputb = mem_outq; + io->inputb = mem_inq; + io->outputb = mem_outq; break; #endif default: - dev_warn(info->io.dev, "Invalid register size: %d\n", - info->io.regsize); + dev_warn(io->dev, "Invalid register size: %d\n", + io->regsize); return -EINVAL; } @@ -1667,11 +1664,11 @@ static int mem_setup(struct smi_info *info) * entire region. Therefore we must request each register * separately. */ - for (idx = 0; idx < info->io_size; idx++) { - if (request_mem_region(addr + idx * info->io.regspacing, - info->io.regsize, DEVICE_NAME) == NULL) { + for (idx = 0; idx < io->io_size; idx++) { + if (request_mem_region(addr + idx * io->regspacing, + io->regsize, DEVICE_NAME) == NULL) { /* Undo allocations */ - mem_region_cleanup(info, idx); + mem_region_cleanup(io, idx); return -EIO; } } @@ -1683,11 +1680,11 @@ static int mem_setup(struct smi_info *info) * between the first address to the end of the last full * register. */ - mapsize = ((info->io_size * info->io.regspacing) - - (info->io.regspacing - info->io.regsize)); - info->io.addr = ioremap(addr, mapsize); - if (info->io.addr == NULL) { - mem_region_cleanup(info, info->io_size); + mapsize = ((io->io_size * io->regspacing) + - (io->regspacing - io->regsize)); + io->addr = ioremap(addr, mapsize); + if (io->addr == NULL) { + mem_region_cleanup(io, io->io_size); return -EIO; } return 0; @@ -1903,10 +1900,6 @@ static int hotmod_handler(const char *val, struct kernel_param *kp) info->io.si_type = si_type; info->io.addr_data = addr; info->io.addr_type = addr_space; - if (addr_space == IPMI_MEM_ADDR_SPACE) - info->io_setup = mem_setup; - else - info->io_setup = port_setup; info->io.addr = NULL; info->io.regspacing = regspacing; @@ -1987,12 +1980,10 @@ static int hardcode_find_bmc(void) if (ports[i]) { /* An I/O port */ - info->io_setup = port_setup; info->io.addr_data = ports[i]; info->io.addr_type = IPMI_IO_ADDR_SPACE; } else if (addrs[i]) { /* A memory port */ - info->io_setup = mem_setup; info->io.addr_data = addrs[i]; info->io.addr_type = IPMI_MEM_ADDR_SPACE; } else { @@ -2192,10 +2183,8 @@ static int try_init_spmi(struct SPMITable *spmi) info->io.regshift = spmi->addr.bit_offset; if (spmi->addr.space_id == ACPI_ADR_SPACE_SYSTEM_MEMORY) { - info->io_setup = mem_setup; info->io.addr_type = IPMI_MEM_ADDR_SPACE; } else if (spmi->addr.space_id == ACPI_ADR_SPACE_SYSTEM_IO) { - info->io_setup = port_setup; info->io.addr_type = IPMI_IO_ADDR_SPACE; } else { kfree(info); @@ -2248,14 +2237,11 @@ ipmi_get_info_from_resources(struct platform_device *pdev, res = platform_get_resource(pdev, IORESOURCE_IO, 0); if (res) { - info->io_setup = port_setup; info->io.addr_type = IPMI_IO_ADDR_SPACE; } else { res = platform_get_resource(pdev, IORESOURCE_MEM, 0); - if (res) { - info->io_setup = mem_setup; + if (res) info->io.addr_type = IPMI_MEM_ADDR_SPACE; - } } if (!res) { dev_err(&pdev->dev, "no I/O or memory address\n"); @@ -2389,13 +2375,13 @@ static int ipmi_pci_probe_regspacing(struct smi_info *info) info->io.regsize = DEFAULT_REGSIZE; info->io.regshift = 0; - info->io_size = 2; + info->io.io_size = 2; info->handlers = &kcs_smi_handlers; /* detect 1, 4, 16byte spacing */ for (regspacing = DEFAULT_REGSPACING; regspacing <= 16;) { info->io.regspacing = regspacing; - if (info->io_setup(info)) { + if (info->io.io_setup(&info->io)) { dev_err(info->io.dev, "Could not setup I/O space\n"); return DEFAULT_REGSPACING; @@ -2404,7 +2390,7 @@ static int ipmi_pci_probe_regspacing(struct smi_info *info) info->io.outputb(&info->io, 1, 0x10); /* read status back */ status = info->io.inputb(&info->io, 1); - info->io_cleanup(info); + info->io.io_cleanup(&info->io); if (status) return regspacing; regspacing *= 4; @@ -2456,13 +2442,10 @@ static int ipmi_pci_probe(struct pci_dev *pdev, info->io.addr_source_cleanup = ipmi_pci_cleanup; info->io.addr_source_data = pdev; - if (pci_resource_flags(pdev, 0) & IORESOURCE_IO) { - info->io_setup = port_setup; + if (pci_resource_flags(pdev, 0) & IORESOURCE_IO) info->io.addr_type = IPMI_IO_ADDR_SPACE; - } else { - info->io_setup = mem_setup; + else info->io.addr_type = IPMI_MEM_ADDR_SPACE; - } info->io.addr_data = pci_resource_start(pdev, 0); info->io.regspacing = ipmi_pci_probe_regspacing(info); @@ -2577,13 +2560,10 @@ static int of_ipmi_probe(struct platform_device *pdev) info->io.addr_source = SI_DEVICETREE; info->io.irq_setup = ipmi_std_irq_setup; - if (resource.flags & IORESOURCE_IO) { - info->io_setup = port_setup; - info->io.addr_type = IPMI_IO_ADDR_SPACE; - } else { - info->io_setup = mem_setup; - info->io.addr_type = IPMI_MEM_ADDR_SPACE; - } + if (resource.flags & IORESOURCE_IO) + info->io.addr_type = IPMI_IO_ADDR_SPACE; + else + info->io.addr_type = IPMI_MEM_ADDR_SPACE; info->io.addr_data = resource.start; @@ -2794,7 +2774,6 @@ static int __init ipmi_parisc_probe(struct parisc_device *dev) info->io.si_type = SI_KCS; info->io.addr_source = SI_DEVICETREE; - info->io_setup = mem_setup; info->io.addr_type = IPMI_MEM_ADDR_SPACE; info->io.addr_data = dev->hpa.start; info->io.regsize = 1; @@ -3419,6 +3398,16 @@ int ipmi_si_add_smi(struct smi_info *new_smi) int rv = 0; struct smi_info *dup; + if (!new_smi->io.io_setup) { + if (new_smi->io.addr_type == IPMI_IO_ADDR_SPACE) { + new_smi->io.io_setup = port_setup; + } else if (new_smi->io.addr_type == IPMI_MEM_ADDR_SPACE) { + new_smi->io.io_setup = mem_setup; + } else { + return -EINVAL; + } + } + mutex_lock(&smi_infos_lock); dup = find_dup_si(new_smi); if (dup) { @@ -3522,11 +3511,11 @@ static int try_smi_init(struct smi_info *new_smi) rv = -ENOMEM; goto out_err; } - new_smi->io_size = new_smi->handlers->init_data(new_smi->si_sm, - &new_smi->io); + new_smi->io.io_size = new_smi->handlers->init_data(new_smi->si_sm, + &new_smi->io); /* Now that we know the I/O size, we can set up the I/O. */ - rv = new_smi->io_setup(new_smi); + rv = new_smi->io.io_setup(&new_smi->io); if (rv) { dev_err(new_smi->io.dev, "Could not set up I/O space\n"); goto out_err; @@ -3679,9 +3668,9 @@ out_err: new_smi->io.addr_source_cleanup(&new_smi->io); new_smi->io.addr_source_cleanup = NULL; } - if (new_smi->io_cleanup) { - new_smi->io_cleanup(new_smi); - new_smi->io_cleanup = NULL; + if (new_smi->io.io_cleanup) { + new_smi->io.io_cleanup(&new_smi->io); + new_smi->io.io_cleanup = NULL; } if (new_smi->pdev) { @@ -3861,8 +3850,8 @@ static void cleanup_one_si(struct smi_info *to_clean) if (to_clean->io.addr_source_cleanup) to_clean->io.addr_source_cleanup(&to_clean->io); - if (to_clean->io_cleanup) - to_clean->io_cleanup(to_clean); + if (to_clean->io.io_cleanup) + to_clean->io.io_cleanup(&to_clean->io); if (to_clean->pdev) platform_device_unregister(to_clean->pdev); diff --git a/drivers/char/ipmi/ipmi_si_sm.h b/drivers/char/ipmi/ipmi_si_sm.h index bf3f50cede44..9df77c664908 100644 --- a/drivers/char/ipmi/ipmi_si_sm.h +++ b/drivers/char/ipmi/ipmi_si_sm.h @@ -71,6 +71,10 @@ struct si_sm_io { void (*addr_source_cleanup)(struct si_sm_io *io); void *addr_source_data; + int (*io_setup)(struct si_sm_io *info); + void (*io_cleanup)(struct si_sm_io *info); + unsigned int io_size; + int irq; int (*irq_setup)(struct si_sm_io *io); void *irq_handler_data;