| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Copyright (C) KEBA Industrial Automation Gmbh 2024
- *
- * Driver for KEBA system FPGA
- *
- * The KEBA system FPGA implements various devices. This driver registers
- * auxiliary devices for every device within the FPGA.
- */
- #include <linux/device.h>
- #include <linux/i2c.h>
- #include <linux/misc/keba.h>
- #include <linux/module.h>
- #include <linux/mtd/partitions.h>
- #include <linux/nvmem-consumer.h>
- #include <linux/nvmem-provider.h>
- #include <linux/pci.h>
- #include <linux/spi/flash.h>
- #include <linux/spi/spi.h>
- #define CP500 "cp500"
- #define PCI_VENDOR_ID_KEBA 0xCEBA
- #define PCI_DEVICE_ID_KEBA_CP035 0x2706
- #define PCI_DEVICE_ID_KEBA_CP505 0x2703
- #define PCI_DEVICE_ID_KEBA_CP520 0x2696
- #define CP500_SYS_BAR 0
- #define CP500_ECM_BAR 1
- /* BAR 0 registers */
- #define CP500_VERSION_REG 0x00
- #define CP500_RECONFIG_REG 0x11 /* upper 8-bits of STARTUP register */
- #define CP500_PRESENT_REG 0x20
- #define CP500_AXI_REG 0x40
- /* Bits in BUILD_REG */
- #define CP500_BUILD_TEST 0x8000 /* FPGA test version */
- /* Bits in RECONFIG_REG */
- #define CP500_RECFG_REQ 0x01 /* reconfigure FPGA on next reset */
- /* Bits in PRESENT_REG */
- #define CP500_PRESENT_FAN0 0x01
- /* MSIX */
- #define CP500_AXI_MSIX 3
- #define CP500_RFB_UART_MSIX 4
- #define CP500_DEBUG_UART_MSIX 5
- #define CP500_SI1_UART_MSIX 6
- #define CP500_NUM_MSIX 8
- #define CP500_NUM_MSIX_NO_MMI 2
- #define CP500_NUM_MSIX_NO_AXI 3
- /* EEPROM */
- #define CP500_EEPROM_DA_OFFSET 0x016F
- #define CP500_EEPROM_DA_ESC_TYPE_MASK 0x01
- #define CP500_EEPROM_ESC_LAN9252 0x00
- #define CP500_EEPROM_ESC_ET1100 0x01
- #define CP500_EEPROM_CPU_NAME "cpu_eeprom"
- #define CP500_EEPROM_CPU_OFFSET 0
- #define CP500_EEPROM_CPU_SIZE 3072
- #define CP500_EEPROM_USER_NAME "user_eeprom"
- #define CP500_EEPROM_USER_OFFSET 3072
- #define CP500_EEPROM_USER_SIZE 1024
- /* SPI flash running at full speed */
- #define CP500_FLASH_HZ (33 * 1000 * 1000)
- /* LAN9252 */
- #define CP500_LAN9252_HZ (10 * 1000 * 1000)
- #define CP500_IS_CP035(dev) ((dev)->pci_dev->device == PCI_DEVICE_ID_KEBA_CP035)
- #define CP500_IS_CP505(dev) ((dev)->pci_dev->device == PCI_DEVICE_ID_KEBA_CP505)
- #define CP500_IS_CP520(dev) ((dev)->pci_dev->device == PCI_DEVICE_ID_KEBA_CP520)
- struct cp500_dev_info {
- off_t offset;
- size_t size;
- unsigned int msix;
- };
- struct cp500_devs {
- struct cp500_dev_info startup;
- struct cp500_dev_info spi;
- struct cp500_dev_info i2c;
- struct cp500_dev_info fan;
- struct cp500_dev_info batt;
- struct cp500_dev_info uart0_rfb;
- struct cp500_dev_info uart1_dbg;
- struct cp500_dev_info uart2_si1;
- };
- /* list of devices within FPGA of CP035 family (CP035, CP056, CP057) */
- static struct cp500_devs cp035_devices = {
- .startup = { 0x0000, SZ_4K },
- .spi = { 0x1000, SZ_4K },
- .i2c = { 0x4000, SZ_4K },
- .fan = { 0x9000, SZ_4K },
- .batt = { 0xA000, SZ_4K },
- .uart0_rfb = { 0xB000, SZ_4K, CP500_RFB_UART_MSIX },
- .uart2_si1 = { 0xD000, SZ_4K, CP500_SI1_UART_MSIX },
- };
- /* list of devices within FPGA of CP505 family (CP503, CP505, CP507) */
- static struct cp500_devs cp505_devices = {
- .startup = { 0x0000, SZ_4K },
- .spi = { 0x4000, SZ_4K },
- .i2c = { 0x5000, SZ_4K },
- .fan = { 0x9000, SZ_4K },
- .batt = { 0xA000, SZ_4K },
- .uart0_rfb = { 0xB000, SZ_4K, CP500_RFB_UART_MSIX },
- .uart2_si1 = { 0xD000, SZ_4K, CP500_SI1_UART_MSIX },
- };
- /* list of devices within FPGA of CP520 family (CP520, CP530) */
- static struct cp500_devs cp520_devices = {
- .startup = { 0x0000, SZ_4K },
- .spi = { 0x4000, SZ_4K },
- .i2c = { 0x5000, SZ_4K },
- .fan = { 0x8000, SZ_4K },
- .batt = { 0x9000, SZ_4K },
- .uart0_rfb = { 0xC000, SZ_4K, CP500_RFB_UART_MSIX },
- .uart1_dbg = { 0xD000, SZ_4K, CP500_DEBUG_UART_MSIX },
- };
- struct cp500_nvmem {
- struct nvmem_device *base_nvmem;
- unsigned int offset;
- struct nvmem_device *nvmem;
- };
- struct cp500 {
- struct pci_dev *pci_dev;
- struct cp500_devs *devs;
- int msix_num;
- struct {
- int major;
- int minor;
- int build;
- } version;
- struct notifier_block nvmem_notifier;
- atomic_t nvmem_notified;
- /* system FPGA BAR */
- resource_size_t sys_hwbase;
- struct keba_spi_auxdev *spi;
- struct keba_i2c_auxdev *i2c;
- struct keba_fan_auxdev *fan;
- struct keba_batt_auxdev *batt;
- struct keba_uart_auxdev *uart0_rfb;
- struct keba_uart_auxdev *uart1_dbg;
- struct keba_uart_auxdev *uart2_si1;
- /* ECM EtherCAT BAR */
- resource_size_t ecm_hwbase;
- /* NVMEM devices */
- struct cp500_nvmem nvmem_cpu;
- struct cp500_nvmem nvmem_user;
- void __iomem *system_startup_addr;
- };
- /* I2C devices */
- #define CP500_EEPROM_ADDR 0x50
- static struct i2c_board_info cp500_i2c_info[] = {
- { /* temperature sensor */
- I2C_BOARD_INFO("emc1403", 0x4c),
- },
- { /*
- * CPU EEPROM
- * CP035 family: CPU board
- * CP505 family: bridge board
- * CP520 family: carrier board
- */
- I2C_BOARD_INFO("24c32", CP500_EEPROM_ADDR),
- },
- { /* interface board EEPROM */
- I2C_BOARD_INFO("24c32", CP500_EEPROM_ADDR + 1),
- },
- { /*
- * EEPROM (optional)
- * CP505 family: CPU board
- * CP520 family: MMI board
- */
- I2C_BOARD_INFO("24c32", CP500_EEPROM_ADDR + 2),
- },
- { /* extension module 0 EEPROM (optional) */
- I2C_BOARD_INFO("24c32", CP500_EEPROM_ADDR + 3),
- },
- { /* extension module 1 EEPROM (optional) */
- I2C_BOARD_INFO("24c32", CP500_EEPROM_ADDR + 4),
- },
- { /* extension module 2 EEPROM (optional) */
- I2C_BOARD_INFO("24c32", CP500_EEPROM_ADDR + 5),
- },
- { /* extension module 3 EEPROM (optional) */
- I2C_BOARD_INFO("24c32", CP500_EEPROM_ADDR + 6),
- }
- };
- /* SPI devices */
- static struct mtd_partition cp500_partitions[] = {
- {
- .name = "system-flash-parts",
- .size = MTDPART_SIZ_FULL,
- .offset = 0,
- .mask_flags = 0
- }
- };
- static const struct flash_platform_data cp500_w25q32 = {
- .type = "w25q32",
- .name = "system-flash",
- .parts = cp500_partitions,
- .nr_parts = ARRAY_SIZE(cp500_partitions),
- };
- static const struct flash_platform_data cp500_m25p16 = {
- .type = "m25p16",
- .name = "system-flash",
- .parts = cp500_partitions,
- .nr_parts = ARRAY_SIZE(cp500_partitions),
- };
- static struct spi_board_info cp500_spi_info[] = {
- { /* system FPGA configuration bitstream flash */
- .modalias = "m25p80",
- .platform_data = &cp500_m25p16,
- .max_speed_hz = CP500_FLASH_HZ,
- .chip_select = 0,
- .mode = SPI_MODE_3,
- }, { /* LAN9252 EtherCAT slave controller */
- .modalias = "lan9252",
- .platform_data = NULL,
- .max_speed_hz = CP500_LAN9252_HZ,
- .chip_select = 1,
- .mode = SPI_MODE_3,
- }
- };
- static ssize_t cp500_get_fpga_version(struct cp500 *cp500, char *buf,
- size_t max_len)
- {
- int n;
- if (CP500_IS_CP035(cp500))
- n = scnprintf(buf, max_len, "CP035");
- else if (CP500_IS_CP505(cp500))
- n = scnprintf(buf, max_len, "CP505");
- else
- n = scnprintf(buf, max_len, "CP500");
- n += scnprintf(buf + n, max_len - n, "_FPGA_%d.%02d",
- cp500->version.major, cp500->version.minor);
- /* test versions have test bit set */
- if (cp500->version.build & CP500_BUILD_TEST)
- n += scnprintf(buf + n, max_len - n, "Test%d",
- cp500->version.build & ~CP500_BUILD_TEST);
- n += scnprintf(buf + n, max_len - n, "\n");
- return n;
- }
- static ssize_t version_show(struct device *dev, struct device_attribute *attr,
- char *buf)
- {
- struct cp500 *cp500 = dev_get_drvdata(dev);
- return cp500_get_fpga_version(cp500, buf, PAGE_SIZE);
- }
- static DEVICE_ATTR_RO(version);
- static ssize_t keep_cfg_show(struct device *dev, struct device_attribute *attr,
- char *buf)
- {
- struct cp500 *cp500 = dev_get_drvdata(dev);
- unsigned long keep_cfg = 1;
- /*
- * FPGA configuration stream is kept during reset when RECONFIG bit is
- * zero
- */
- if (ioread8(cp500->system_startup_addr + CP500_RECONFIG_REG) &
- CP500_RECFG_REQ)
- keep_cfg = 0;
- return sysfs_emit(buf, "%lu\n", keep_cfg);
- }
- static ssize_t keep_cfg_store(struct device *dev, struct device_attribute *attr,
- const char *buf, size_t count)
- {
- struct cp500 *cp500 = dev_get_drvdata(dev);
- unsigned long keep_cfg;
- if (kstrtoul(buf, 10, &keep_cfg) < 0)
- return -EINVAL;
- /*
- * In normal operation "keep_cfg" is "1". This means that the FPGA keeps
- * its configuration stream during a reset.
- * In case of a firmware update of the FPGA, the configuration stream
- * needs to be reloaded. This can be done without a powercycle by
- * writing a "0" into the "keep_cfg" attribute. After a reset/reboot th
- * new configuration stream will be loaded.
- */
- if (keep_cfg)
- iowrite8(0, cp500->system_startup_addr + CP500_RECONFIG_REG);
- else
- iowrite8(CP500_RECFG_REQ,
- cp500->system_startup_addr + CP500_RECONFIG_REG);
- return count;
- }
- static DEVICE_ATTR_RW(keep_cfg);
- static struct attribute *cp500_attrs[] = {
- &dev_attr_version.attr,
- &dev_attr_keep_cfg.attr,
- NULL
- };
- ATTRIBUTE_GROUPS(cp500);
- static void cp500_i2c_release(struct device *dev)
- {
- struct keba_i2c_auxdev *i2c =
- container_of(dev, struct keba_i2c_auxdev, auxdev.dev);
- kfree(i2c);
- }
- static int cp500_register_i2c(struct cp500 *cp500)
- {
- int ret;
- cp500->i2c = kzalloc_obj(*cp500->i2c);
- if (!cp500->i2c)
- return -ENOMEM;
- cp500->i2c->auxdev.name = "i2c";
- cp500->i2c->auxdev.id = 0;
- cp500->i2c->auxdev.dev.release = cp500_i2c_release;
- cp500->i2c->auxdev.dev.parent = &cp500->pci_dev->dev;
- cp500->i2c->io = (struct resource) {
- /* I2C register area */
- .start = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->i2c.offset,
- .end = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->i2c.offset +
- cp500->devs->i2c.size - 1,
- .flags = IORESOURCE_MEM,
- };
- cp500->i2c->info_size = ARRAY_SIZE(cp500_i2c_info);
- cp500->i2c->info = cp500_i2c_info;
- ret = auxiliary_device_init(&cp500->i2c->auxdev);
- if (ret) {
- kfree(cp500->i2c);
- cp500->i2c = NULL;
- return ret;
- }
- ret = __auxiliary_device_add(&cp500->i2c->auxdev, "keba");
- if (ret) {
- auxiliary_device_uninit(&cp500->i2c->auxdev);
- cp500->i2c = NULL;
- return ret;
- }
- return 0;
- }
- static void cp500_spi_release(struct device *dev)
- {
- struct keba_spi_auxdev *spi =
- container_of(dev, struct keba_spi_auxdev, auxdev.dev);
- kfree(spi);
- }
- static int cp500_register_spi(struct cp500 *cp500, u8 esc_type)
- {
- int info_size;
- int ret;
- cp500->spi = kzalloc_obj(*cp500->spi);
- if (!cp500->spi)
- return -ENOMEM;
- if (CP500_IS_CP035(cp500))
- cp500_spi_info[0].platform_data = &cp500_w25q32;
- if (esc_type == CP500_EEPROM_ESC_LAN9252)
- info_size = ARRAY_SIZE(cp500_spi_info);
- else
- info_size = ARRAY_SIZE(cp500_spi_info) - 1;
- cp500->spi->auxdev.name = "spi";
- cp500->spi->auxdev.id = 0;
- cp500->spi->auxdev.dev.release = cp500_spi_release;
- cp500->spi->auxdev.dev.parent = &cp500->pci_dev->dev;
- cp500->spi->io = (struct resource) {
- /* SPI register area */
- .start = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->spi.offset,
- .end = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->spi.offset +
- cp500->devs->spi.size - 1,
- .flags = IORESOURCE_MEM,
- };
- cp500->spi->info_size = info_size;
- cp500->spi->info = cp500_spi_info;
- ret = auxiliary_device_init(&cp500->spi->auxdev);
- if (ret) {
- kfree(cp500->spi);
- cp500->spi = NULL;
- return ret;
- }
- ret = __auxiliary_device_add(&cp500->spi->auxdev, "keba");
- if (ret) {
- auxiliary_device_uninit(&cp500->spi->auxdev);
- cp500->spi = NULL;
- return ret;
- }
- return 0;
- }
- static void cp500_fan_release(struct device *dev)
- {
- struct keba_fan_auxdev *fan =
- container_of(dev, struct keba_fan_auxdev, auxdev.dev);
- kfree(fan);
- }
- static int cp500_register_fan(struct cp500 *cp500)
- {
- int ret;
- cp500->fan = kzalloc_obj(*cp500->fan);
- if (!cp500->fan)
- return -ENOMEM;
- cp500->fan->auxdev.name = "fan";
- cp500->fan->auxdev.id = 0;
- cp500->fan->auxdev.dev.release = cp500_fan_release;
- cp500->fan->auxdev.dev.parent = &cp500->pci_dev->dev;
- cp500->fan->io = (struct resource) {
- /* fan register area */
- .start = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->fan.offset,
- .end = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->fan.offset +
- cp500->devs->fan.size - 1,
- .flags = IORESOURCE_MEM,
- };
- ret = auxiliary_device_init(&cp500->fan->auxdev);
- if (ret) {
- kfree(cp500->fan);
- cp500->fan = NULL;
- return ret;
- }
- ret = __auxiliary_device_add(&cp500->fan->auxdev, "keba");
- if (ret) {
- auxiliary_device_uninit(&cp500->fan->auxdev);
- cp500->fan = NULL;
- return ret;
- }
- return 0;
- }
- static void cp500_batt_release(struct device *dev)
- {
- struct keba_batt_auxdev *fan =
- container_of(dev, struct keba_batt_auxdev, auxdev.dev);
- kfree(fan);
- }
- static int cp500_register_batt(struct cp500 *cp500)
- {
- int ret;
- cp500->batt = kzalloc_obj(*cp500->batt);
- if (!cp500->batt)
- return -ENOMEM;
- cp500->batt->auxdev.name = "batt";
- cp500->batt->auxdev.id = 0;
- cp500->batt->auxdev.dev.release = cp500_batt_release;
- cp500->batt->auxdev.dev.parent = &cp500->pci_dev->dev;
- cp500->batt->io = (struct resource) {
- /* battery register area */
- .start = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->batt.offset,
- .end = (resource_size_t) cp500->sys_hwbase +
- cp500->devs->batt.offset +
- cp500->devs->batt.size - 1,
- .flags = IORESOURCE_MEM,
- };
- ret = auxiliary_device_init(&cp500->batt->auxdev);
- if (ret) {
- kfree(cp500->batt);
- cp500->batt = NULL;
- return ret;
- }
- ret = __auxiliary_device_add(&cp500->batt->auxdev, "keba");
- if (ret) {
- auxiliary_device_uninit(&cp500->batt->auxdev);
- cp500->batt = NULL;
- return ret;
- }
- return 0;
- }
- static void cp500_uart_release(struct device *dev)
- {
- struct keba_uart_auxdev *uart =
- container_of(dev, struct keba_uart_auxdev, auxdev.dev);
- kfree(uart);
- }
- static int cp500_register_uart(struct cp500 *cp500,
- struct keba_uart_auxdev **uart, const char *name,
- struct cp500_dev_info *info, unsigned int irq)
- {
- int ret;
- *uart = kzalloc_obj(**uart);
- if (!*uart)
- return -ENOMEM;
- (*uart)->auxdev.name = name;
- (*uart)->auxdev.id = 0;
- (*uart)->auxdev.dev.release = cp500_uart_release;
- (*uart)->auxdev.dev.parent = &cp500->pci_dev->dev;
- (*uart)->io = (struct resource) {
- /* UART register area */
- .start = (resource_size_t) cp500->sys_hwbase + info->offset,
- .end = (resource_size_t) cp500->sys_hwbase + info->offset +
- info->size - 1,
- .flags = IORESOURCE_MEM,
- };
- (*uart)->irq = irq;
- ret = auxiliary_device_init(&(*uart)->auxdev);
- if (ret) {
- kfree(*uart);
- *uart = NULL;
- return ret;
- }
- ret = __auxiliary_device_add(&(*uart)->auxdev, "keba");
- if (ret) {
- auxiliary_device_uninit(&(*uart)->auxdev);
- *uart = NULL;
- return ret;
- }
- return 0;
- }
- static int cp500_nvmem_read(void *priv, unsigned int offset, void *val,
- size_t bytes)
- {
- struct cp500_nvmem *nvmem = priv;
- int ret;
- ret = nvmem_device_read(nvmem->base_nvmem, nvmem->offset + offset,
- bytes, val);
- if (ret != bytes)
- return ret;
- return 0;
- }
- static int cp500_nvmem_write(void *priv, unsigned int offset, void *val,
- size_t bytes)
- {
- struct cp500_nvmem *nvmem = priv;
- int ret;
- ret = nvmem_device_write(nvmem->base_nvmem, nvmem->offset + offset,
- bytes, val);
- if (ret != bytes)
- return ret;
- return 0;
- }
- static int cp500_nvmem_register(struct cp500 *cp500,
- struct nvmem_device *base_nvmem)
- {
- struct device *dev = &cp500->pci_dev->dev;
- struct nvmem_config nvmem_config = {};
- struct nvmem_device *tmp;
- /*
- * The main EEPROM of CP500 devices is logically split into two EEPROMs.
- * The first logical EEPROM with 3 kB contains the type label which is
- * programmed during production of the device. The second logical EEPROM
- * with 1 kB is not programmed during production and can be used for
- * arbitrary user data.
- */
- nvmem_config.dev = dev;
- nvmem_config.owner = THIS_MODULE;
- nvmem_config.id = NVMEM_DEVID_NONE;
- nvmem_config.type = NVMEM_TYPE_EEPROM;
- nvmem_config.root_only = true;
- nvmem_config.reg_read = cp500_nvmem_read;
- nvmem_config.reg_write = cp500_nvmem_write;
- cp500->nvmem_cpu.base_nvmem = base_nvmem;
- cp500->nvmem_cpu.offset = CP500_EEPROM_CPU_OFFSET;
- nvmem_config.name = CP500_EEPROM_CPU_NAME;
- nvmem_config.size = CP500_EEPROM_CPU_SIZE;
- nvmem_config.priv = &cp500->nvmem_cpu;
- tmp = nvmem_register(&nvmem_config);
- if (IS_ERR(tmp))
- return PTR_ERR(tmp);
- cp500->nvmem_cpu.nvmem = tmp;
- cp500->nvmem_user.base_nvmem = base_nvmem;
- cp500->nvmem_user.offset = CP500_EEPROM_USER_OFFSET;
- nvmem_config.name = CP500_EEPROM_USER_NAME;
- nvmem_config.size = CP500_EEPROM_USER_SIZE;
- nvmem_config.priv = &cp500->nvmem_user;
- tmp = nvmem_register(&nvmem_config);
- if (IS_ERR(tmp)) {
- nvmem_unregister(cp500->nvmem_cpu.nvmem);
- cp500->nvmem_cpu.nvmem = NULL;
- return PTR_ERR(tmp);
- }
- cp500->nvmem_user.nvmem = tmp;
- return 0;
- }
- static void cp500_nvmem_unregister(struct cp500 *cp500)
- {
- int notified;
- if (cp500->nvmem_user.nvmem) {
- nvmem_unregister(cp500->nvmem_user.nvmem);
- cp500->nvmem_user.nvmem = NULL;
- }
- if (cp500->nvmem_cpu.nvmem) {
- nvmem_unregister(cp500->nvmem_cpu.nvmem);
- cp500->nvmem_cpu.nvmem = NULL;
- }
- /* CPU and user nvmem use the same base_nvmem, put only once */
- notified = atomic_read(&cp500->nvmem_notified);
- if (notified)
- nvmem_device_put(cp500->nvmem_cpu.base_nvmem);
- }
- static int cp500_nvmem_match(struct device *dev, const void *data)
- {
- const struct cp500 *cp500 = data;
- struct i2c_client *client;
- /* match only CPU EEPROM below the cp500 device */
- dev = dev->parent;
- client = i2c_verify_client(dev);
- if (!client || client->addr != CP500_EEPROM_ADDR)
- return 0;
- while ((dev = dev->parent))
- if (dev == &cp500->pci_dev->dev)
- return 1;
- return 0;
- }
- static int cp500_nvmem(struct notifier_block *nb, unsigned long action,
- void *data)
- {
- struct nvmem_device *nvmem;
- struct cp500 *cp500;
- struct device *dev;
- int notified;
- u8 esc_type;
- int ret;
- if (action != NVMEM_ADD)
- return NOTIFY_DONE;
- cp500 = container_of(nb, struct cp500, nvmem_notifier);
- dev = &cp500->pci_dev->dev;
- /* process CPU EEPROM content only once */
- notified = atomic_read(&cp500->nvmem_notified);
- if (notified)
- return NOTIFY_DONE;
- nvmem = nvmem_device_find(cp500, cp500_nvmem_match);
- if (IS_ERR_OR_NULL(nvmem))
- return NOTIFY_DONE;
- if (!atomic_try_cmpxchg_relaxed(&cp500->nvmem_notified, ¬ified, 1)) {
- nvmem_device_put(nvmem);
- return NOTIFY_DONE;
- }
- ret = cp500_nvmem_register(cp500, nvmem);
- if (ret)
- return ret;
- ret = nvmem_device_read(nvmem, CP500_EEPROM_DA_OFFSET, sizeof(esc_type),
- (void *)&esc_type);
- if (ret != sizeof(esc_type)) {
- dev_warn(dev, "Failed to read device assembly!\n");
- return NOTIFY_DONE;
- }
- esc_type &= CP500_EEPROM_DA_ESC_TYPE_MASK;
- if (cp500_register_spi(cp500, esc_type))
- dev_warn(dev, "Failed to register SPI!\n");
- return NOTIFY_OK;
- }
- static void cp500_register_auxiliary_devs(struct cp500 *cp500)
- {
- struct device *dev = &cp500->pci_dev->dev;
- u8 present = ioread8(cp500->system_startup_addr + CP500_PRESENT_REG);
- if (cp500_register_i2c(cp500))
- dev_warn(dev, "Failed to register I2C!\n");
- if (present & CP500_PRESENT_FAN0)
- if (cp500_register_fan(cp500))
- dev_warn(dev, "Failed to register fan!\n");
- if (cp500_register_batt(cp500))
- dev_warn(dev, "Failed to register battery!\n");
- if (cp500->devs->uart0_rfb.size &&
- cp500->devs->uart0_rfb.msix < cp500->msix_num) {
- int irq = pci_irq_vector(cp500->pci_dev,
- cp500->devs->uart0_rfb.msix);
- if (cp500_register_uart(cp500, &cp500->uart0_rfb, "rs485-uart",
- &cp500->devs->uart0_rfb, irq))
- dev_warn(dev, "Failed to register RFB UART!\n");
- }
- if (cp500->devs->uart1_dbg.size &&
- cp500->devs->uart1_dbg.msix < cp500->msix_num) {
- int irq = pci_irq_vector(cp500->pci_dev,
- cp500->devs->uart1_dbg.msix);
- if (cp500_register_uart(cp500, &cp500->uart1_dbg, "rs232-uart",
- &cp500->devs->uart1_dbg, irq))
- dev_warn(dev, "Failed to register debug UART!\n");
- }
- if (cp500->devs->uart2_si1.size &&
- cp500->devs->uart2_si1.msix < cp500->msix_num) {
- int irq = pci_irq_vector(cp500->pci_dev,
- cp500->devs->uart2_si1.msix);
- if (cp500_register_uart(cp500, &cp500->uart2_si1, "uart",
- &cp500->devs->uart2_si1, irq))
- dev_warn(dev, "Failed to register SI1 UART!\n");
- }
- }
- static void cp500_unregister_dev(struct auxiliary_device *auxdev)
- {
- auxiliary_device_delete(auxdev);
- auxiliary_device_uninit(auxdev);
- }
- static void cp500_unregister_auxiliary_devs(struct cp500 *cp500)
- {
- if (cp500->spi) {
- cp500_unregister_dev(&cp500->spi->auxdev);
- cp500->spi = NULL;
- }
- if (cp500->i2c) {
- cp500_unregister_dev(&cp500->i2c->auxdev);
- cp500->i2c = NULL;
- }
- if (cp500->fan) {
- cp500_unregister_dev(&cp500->fan->auxdev);
- cp500->fan = NULL;
- }
- if (cp500->batt) {
- cp500_unregister_dev(&cp500->batt->auxdev);
- cp500->batt = NULL;
- }
- if (cp500->uart0_rfb) {
- cp500_unregister_dev(&cp500->uart0_rfb->auxdev);
- cp500->uart0_rfb = NULL;
- }
- if (cp500->uart1_dbg) {
- cp500_unregister_dev(&cp500->uart1_dbg->auxdev);
- cp500->uart1_dbg = NULL;
- }
- if (cp500->uart2_si1) {
- cp500_unregister_dev(&cp500->uart2_si1->auxdev);
- cp500->uart2_si1 = NULL;
- }
- }
- static irqreturn_t cp500_axi_handler(int irq, void *dev)
- {
- struct cp500 *cp500 = dev;
- u32 axi_address = ioread32(cp500->system_startup_addr + CP500_AXI_REG);
- /*
- * FPGA signals AXI response error, print AXI address to indicate which
- * IP core was affected
- */
- dev_err(&cp500->pci_dev->dev, "AXI response error at 0x%08x\n",
- axi_address);
- return IRQ_HANDLED;
- }
- static int cp500_enable(struct cp500 *cp500)
- {
- int axi_irq = -1;
- int ret;
- if (cp500->msix_num > CP500_NUM_MSIX_NO_AXI) {
- axi_irq = pci_irq_vector(cp500->pci_dev, CP500_AXI_MSIX);
- ret = request_irq(axi_irq, cp500_axi_handler, 0,
- CP500, cp500);
- if (ret != 0) {
- dev_err(&cp500->pci_dev->dev,
- "Failed to register AXI response error!\n");
- return ret;
- }
- }
- return 0;
- }
- static void cp500_disable(struct cp500 *cp500)
- {
- int axi_irq;
- if (cp500->msix_num > CP500_NUM_MSIX_NO_AXI) {
- axi_irq = pci_irq_vector(cp500->pci_dev, CP500_AXI_MSIX);
- free_irq(axi_irq, cp500);
- }
- }
- static int cp500_probe(struct pci_dev *pci_dev, const struct pci_device_id *id)
- {
- struct device *dev = &pci_dev->dev;
- struct resource startup;
- struct cp500 *cp500;
- u32 cp500_vers;
- char buf[64];
- int ret;
- cp500 = devm_kzalloc(dev, sizeof(*cp500), GFP_KERNEL);
- if (!cp500)
- return -ENOMEM;
- cp500->pci_dev = pci_dev;
- cp500->sys_hwbase = pci_resource_start(pci_dev, CP500_SYS_BAR);
- cp500->ecm_hwbase = pci_resource_start(pci_dev, CP500_ECM_BAR);
- if (!cp500->sys_hwbase || !cp500->ecm_hwbase)
- return -ENODEV;
- if (CP500_IS_CP035(cp500))
- cp500->devs = &cp035_devices;
- else if (CP500_IS_CP505(cp500))
- cp500->devs = &cp505_devices;
- else if (CP500_IS_CP520(cp500))
- cp500->devs = &cp520_devices;
- else
- return -ENODEV;
- ret = pci_enable_device(pci_dev);
- if (ret)
- return ret;
- pci_set_master(pci_dev);
- startup = *pci_resource_n(pci_dev, CP500_SYS_BAR);
- startup.end = startup.start + cp500->devs->startup.size - 1;
- cp500->system_startup_addr = devm_ioremap_resource(&pci_dev->dev,
- &startup);
- if (IS_ERR(cp500->system_startup_addr)) {
- ret = PTR_ERR(cp500->system_startup_addr);
- goto out_disable;
- }
- cp500->msix_num = pci_alloc_irq_vectors(pci_dev, CP500_NUM_MSIX_NO_MMI,
- CP500_NUM_MSIX, PCI_IRQ_MSIX);
- if (cp500->msix_num < CP500_NUM_MSIX_NO_MMI) {
- dev_err(&pci_dev->dev,
- "Hardware does not support enough MSI-X interrupts\n");
- ret = -ENODEV;
- goto out_disable;
- }
- cp500_vers = ioread32(cp500->system_startup_addr + CP500_VERSION_REG);
- cp500->version.major = (cp500_vers & 0xff);
- cp500->version.minor = (cp500_vers >> 8) & 0xff;
- cp500->version.build = (cp500_vers >> 16) & 0xffff;
- cp500_get_fpga_version(cp500, buf, sizeof(buf));
- dev_info(&pci_dev->dev, "FPGA version %s", buf);
- pci_set_drvdata(pci_dev, cp500);
- cp500->nvmem_notifier.notifier_call = cp500_nvmem;
- ret = nvmem_register_notifier(&cp500->nvmem_notifier);
- if (ret != 0)
- goto out_free_irq;
- ret = cp500_enable(cp500);
- if (ret != 0)
- goto out_unregister_nvmem;
- cp500_register_auxiliary_devs(cp500);
- return 0;
- out_unregister_nvmem:
- nvmem_unregister_notifier(&cp500->nvmem_notifier);
- out_free_irq:
- pci_free_irq_vectors(pci_dev);
- out_disable:
- pci_clear_master(pci_dev);
- pci_disable_device(pci_dev);
- return ret;
- }
- static void cp500_remove(struct pci_dev *pci_dev)
- {
- struct cp500 *cp500 = pci_get_drvdata(pci_dev);
- /*
- * unregister CPU and user nvmem and put base_nvmem before parent
- * auxiliary device of base_nvmem is unregistered
- */
- nvmem_unregister_notifier(&cp500->nvmem_notifier);
- cp500_nvmem_unregister(cp500);
- cp500_unregister_auxiliary_devs(cp500);
- cp500_disable(cp500);
- pci_set_drvdata(pci_dev, 0);
- pci_free_irq_vectors(pci_dev);
- pci_clear_master(pci_dev);
- pci_disable_device(pci_dev);
- }
- static struct pci_device_id cp500_ids[] = {
- { PCI_DEVICE(PCI_VENDOR_ID_KEBA, PCI_DEVICE_ID_KEBA_CP035) },
- { PCI_DEVICE(PCI_VENDOR_ID_KEBA, PCI_DEVICE_ID_KEBA_CP505) },
- { PCI_DEVICE(PCI_VENDOR_ID_KEBA, PCI_DEVICE_ID_KEBA_CP520) },
- { }
- };
- MODULE_DEVICE_TABLE(pci, cp500_ids);
- static struct pci_driver cp500_driver = {
- .name = CP500,
- .id_table = cp500_ids,
- .probe = cp500_probe,
- .remove = cp500_remove,
- .dev_groups = cp500_groups,
- };
- module_pci_driver(cp500_driver);
- MODULE_AUTHOR("Gerhard Engleder <eg@keba.com>");
- MODULE_DESCRIPTION("KEBA CP500 system FPGA driver");
- MODULE_LICENSE("GPL");
|