|  | // SPDX-License-Identifier: GPL-2.0+ | 
|  | /* | 
|  | * TQ-Systems PLD MFD core driver, based on vendor driver by | 
|  | * Vadim V.Vlasov <vvlasov@dev.rtsoft.ru> | 
|  | * | 
|  | * Copyright (c) 2015 TQ-Systems GmbH | 
|  | * Copyright (c) 2019 Andrew Lunn <andrew@lunn.ch> | 
|  | */ | 
|  |  | 
|  | #include <linux/delay.h> | 
|  | #include <linux/dmi.h> | 
|  | #include <linux/i2c.h> | 
|  | #include <linux/io.h> | 
|  | #include <linux/mfd/core.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/platform_data/i2c-ocores.h> | 
|  | #include <linux/platform_device.h> | 
|  |  | 
|  | #define TQMX86_IOBASE	0x180 | 
|  | #define TQMX86_IOSIZE	0x20 | 
|  | #define TQMX86_IOBASE_I2C	0x1a0 | 
|  | #define TQMX86_IOSIZE_I2C	0xa | 
|  | #define TQMX86_IOBASE_WATCHDOG	0x18b | 
|  | #define TQMX86_IOSIZE_WATCHDOG	0x2 | 
|  | #define TQMX86_IOBASE_GPIO	0x18d | 
|  | #define TQMX86_IOSIZE_GPIO	0x4 | 
|  |  | 
|  | #define TQMX86_REG_BOARD_ID	0x00 | 
|  | #define TQMX86_REG_BOARD_ID_E38M	1 | 
|  | #define TQMX86_REG_BOARD_ID_50UC	2 | 
|  | #define TQMX86_REG_BOARD_ID_E38C	3 | 
|  | #define TQMX86_REG_BOARD_ID_60EB	4 | 
|  | #define TQMX86_REG_BOARD_ID_E39MS	5 | 
|  | #define TQMX86_REG_BOARD_ID_E39C1	6 | 
|  | #define TQMX86_REG_BOARD_ID_E39C2	7 | 
|  | #define TQMX86_REG_BOARD_ID_70EB	8 | 
|  | #define TQMX86_REG_BOARD_ID_80UC	9 | 
|  | #define TQMX86_REG_BOARD_ID_110EB	11 | 
|  | #define TQMX86_REG_BOARD_ID_E40M	12 | 
|  | #define TQMX86_REG_BOARD_ID_E40S	13 | 
|  | #define TQMX86_REG_BOARD_ID_E40C1	14 | 
|  | #define TQMX86_REG_BOARD_ID_E40C2	15 | 
|  | #define TQMX86_REG_BOARD_REV	0x01 | 
|  | #define TQMX86_REG_IO_EXT_INT	0x06 | 
|  | #define TQMX86_REG_IO_EXT_INT_NONE		0 | 
|  | #define TQMX86_REG_IO_EXT_INT_7			1 | 
|  | #define TQMX86_REG_IO_EXT_INT_9			2 | 
|  | #define TQMX86_REG_IO_EXT_INT_12		3 | 
|  | #define TQMX86_REG_IO_EXT_INT_MASK		0x3 | 
|  | #define TQMX86_REG_IO_EXT_INT_GPIO_SHIFT	4 | 
|  | #define TQMX86_REG_SAUC		0x17 | 
|  |  | 
|  | #define TQMX86_REG_I2C_DETECT	0x1a7 | 
|  | #define TQMX86_REG_I2C_DETECT_SOFT		0xa5 | 
|  |  | 
|  | static uint gpio_irq; | 
|  | module_param(gpio_irq, uint, 0); | 
|  | MODULE_PARM_DESC(gpio_irq, "GPIO IRQ number (7, 9, 12)"); | 
|  |  | 
|  | static const struct resource tqmx_i2c_soft_resources[] = { | 
|  | DEFINE_RES_IO(TQMX86_IOBASE_I2C, TQMX86_IOSIZE_I2C), | 
|  | }; | 
|  |  | 
|  | static const struct resource tqmx_watchdog_resources[] = { | 
|  | DEFINE_RES_IO(TQMX86_IOBASE_WATCHDOG, TQMX86_IOSIZE_WATCHDOG), | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * The IRQ resource must be first, since it is updated with the | 
|  | * configured IRQ in the probe function. | 
|  | */ | 
|  | static struct resource tqmx_gpio_resources[] = { | 
|  | DEFINE_RES_IRQ(0), | 
|  | DEFINE_RES_IO(TQMX86_IOBASE_GPIO, TQMX86_IOSIZE_GPIO), | 
|  | }; | 
|  |  | 
|  | static struct i2c_board_info tqmx86_i2c_devices[] = { | 
|  | { | 
|  | /* 4K EEPROM at 0x50 */ | 
|  | I2C_BOARD_INFO("24c32", 0x50), | 
|  | }, | 
|  | }; | 
|  |  | 
|  | static struct ocores_i2c_platform_data ocores_platform_data = { | 
|  | .num_devices = ARRAY_SIZE(tqmx86_i2c_devices), | 
|  | .devices = tqmx86_i2c_devices, | 
|  | }; | 
|  |  | 
|  | static const struct mfd_cell tqmx86_i2c_soft_dev[] = { | 
|  | { | 
|  | .name = "ocores-i2c", | 
|  | .platform_data = &ocores_platform_data, | 
|  | .pdata_size = sizeof(ocores_platform_data), | 
|  | .resources = tqmx_i2c_soft_resources, | 
|  | .num_resources = ARRAY_SIZE(tqmx_i2c_soft_resources), | 
|  | }, | 
|  | }; | 
|  |  | 
|  | static const struct mfd_cell tqmx86_devs[] = { | 
|  | { | 
|  | .name = "tqmx86-wdt", | 
|  | .resources = tqmx_watchdog_resources, | 
|  | .num_resources = ARRAY_SIZE(tqmx_watchdog_resources), | 
|  | .ignore_resource_conflicts = true, | 
|  | }, | 
|  | { | 
|  | .name = "tqmx86-gpio", | 
|  | .resources = tqmx_gpio_resources, | 
|  | .num_resources = ARRAY_SIZE(tqmx_gpio_resources), | 
|  | .ignore_resource_conflicts = true, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | static const char *tqmx86_board_id_to_name(u8 board_id, u8 sauc) | 
|  | { | 
|  | switch (board_id) { | 
|  | case TQMX86_REG_BOARD_ID_E38M: | 
|  | return "TQMxE38M"; | 
|  | case TQMX86_REG_BOARD_ID_50UC: | 
|  | return "TQMx50UC"; | 
|  | case TQMX86_REG_BOARD_ID_E38C: | 
|  | return "TQMxE38C"; | 
|  | case TQMX86_REG_BOARD_ID_60EB: | 
|  | return "TQMx60EB"; | 
|  | case TQMX86_REG_BOARD_ID_E39MS: | 
|  | return (sauc == 0xff) ? "TQMxE39M" : "TQMxE39S"; | 
|  | case TQMX86_REG_BOARD_ID_E39C1: | 
|  | return "TQMxE39C1"; | 
|  | case TQMX86_REG_BOARD_ID_E39C2: | 
|  | return "TQMxE39C2"; | 
|  | case TQMX86_REG_BOARD_ID_70EB: | 
|  | return "TQMx70EB"; | 
|  | case TQMX86_REG_BOARD_ID_80UC: | 
|  | return "TQMx80UC"; | 
|  | case TQMX86_REG_BOARD_ID_110EB: | 
|  | return "TQMx110EB"; | 
|  | case TQMX86_REG_BOARD_ID_E40M: | 
|  | return "TQMxE40M"; | 
|  | case TQMX86_REG_BOARD_ID_E40S: | 
|  | return "TQMxE40S"; | 
|  | case TQMX86_REG_BOARD_ID_E40C1: | 
|  | return "TQMxE40C1"; | 
|  | case TQMX86_REG_BOARD_ID_E40C2: | 
|  | return "TQMxE40C2"; | 
|  | default: | 
|  | return "Unknown"; | 
|  | } | 
|  | } | 
|  |  | 
|  | static int tqmx86_board_id_to_clk_rate(struct device *dev, u8 board_id) | 
|  | { | 
|  | switch (board_id) { | 
|  | case TQMX86_REG_BOARD_ID_50UC: | 
|  | case TQMX86_REG_BOARD_ID_60EB: | 
|  | case TQMX86_REG_BOARD_ID_70EB: | 
|  | case TQMX86_REG_BOARD_ID_80UC: | 
|  | case TQMX86_REG_BOARD_ID_110EB: | 
|  | case TQMX86_REG_BOARD_ID_E40M: | 
|  | case TQMX86_REG_BOARD_ID_E40S: | 
|  | case TQMX86_REG_BOARD_ID_E40C1: | 
|  | case TQMX86_REG_BOARD_ID_E40C2: | 
|  | return 24000; | 
|  | case TQMX86_REG_BOARD_ID_E39MS: | 
|  | case TQMX86_REG_BOARD_ID_E39C1: | 
|  | case TQMX86_REG_BOARD_ID_E39C2: | 
|  | return 25000; | 
|  | case TQMX86_REG_BOARD_ID_E38M: | 
|  | case TQMX86_REG_BOARD_ID_E38C: | 
|  | return 33000; | 
|  | default: | 
|  | dev_warn(dev, "unknown board %d, assuming 24MHz LPC clock\n", | 
|  | board_id); | 
|  | return 24000; | 
|  | } | 
|  | } | 
|  |  | 
|  | static int tqmx86_probe(struct platform_device *pdev) | 
|  | { | 
|  | u8 board_id, sauc, rev, i2c_det, io_ext_int_val; | 
|  | struct device *dev = &pdev->dev; | 
|  | u8 gpio_irq_cfg, readback; | 
|  | const char *board_name; | 
|  | void __iomem *io_base; | 
|  | int err; | 
|  |  | 
|  | switch (gpio_irq) { | 
|  | case 0: | 
|  | gpio_irq_cfg = TQMX86_REG_IO_EXT_INT_NONE; | 
|  | break; | 
|  | case 7: | 
|  | gpio_irq_cfg = TQMX86_REG_IO_EXT_INT_7; | 
|  | break; | 
|  | case 9: | 
|  | gpio_irq_cfg = TQMX86_REG_IO_EXT_INT_9; | 
|  | break; | 
|  | case 12: | 
|  | gpio_irq_cfg = TQMX86_REG_IO_EXT_INT_12; | 
|  | break; | 
|  | default: | 
|  | pr_err("tqmx86: Invalid GPIO IRQ (%d)\n", gpio_irq); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | io_base = devm_ioport_map(dev, TQMX86_IOBASE, TQMX86_IOSIZE); | 
|  | if (!io_base) | 
|  | return -ENOMEM; | 
|  |  | 
|  | board_id = ioread8(io_base + TQMX86_REG_BOARD_ID); | 
|  | sauc = ioread8(io_base + TQMX86_REG_SAUC); | 
|  | board_name = tqmx86_board_id_to_name(board_id, sauc); | 
|  | rev = ioread8(io_base + TQMX86_REG_BOARD_REV); | 
|  |  | 
|  | dev_info(dev, | 
|  | "Found %s - Board ID %d, PCB Revision %d, PLD Revision %d\n", | 
|  | board_name, board_id, rev >> 4, rev & 0xf); | 
|  |  | 
|  | /* | 
|  | * The I2C_DETECT register is in the range assigned to the I2C driver | 
|  | * later, so we don't extend TQMX86_IOSIZE. Use inb() for this one-off | 
|  | * access instead of ioport_map + unmap. | 
|  | */ | 
|  | i2c_det = inb(TQMX86_REG_I2C_DETECT); | 
|  |  | 
|  | if (gpio_irq_cfg) { | 
|  | io_ext_int_val = | 
|  | gpio_irq_cfg << TQMX86_REG_IO_EXT_INT_GPIO_SHIFT; | 
|  | iowrite8(io_ext_int_val, io_base + TQMX86_REG_IO_EXT_INT); | 
|  | readback = ioread8(io_base + TQMX86_REG_IO_EXT_INT); | 
|  | if (readback != io_ext_int_val) { | 
|  | dev_warn(dev, "GPIO interrupts not supported.\n"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | /* Assumes the IRQ resource is first. */ | 
|  | tqmx_gpio_resources[0].start = gpio_irq; | 
|  | } else { | 
|  | tqmx_gpio_resources[0].flags = 0; | 
|  | } | 
|  |  | 
|  | ocores_platform_data.clock_khz = tqmx86_board_id_to_clk_rate(dev, board_id); | 
|  |  | 
|  | if (i2c_det == TQMX86_REG_I2C_DETECT_SOFT) { | 
|  | err = devm_mfd_add_devices(dev, PLATFORM_DEVID_NONE, | 
|  | tqmx86_i2c_soft_dev, | 
|  | ARRAY_SIZE(tqmx86_i2c_soft_dev), | 
|  | NULL, 0, NULL); | 
|  | if (err) | 
|  | return err; | 
|  | } | 
|  |  | 
|  | return devm_mfd_add_devices(dev, PLATFORM_DEVID_NONE, | 
|  | tqmx86_devs, | 
|  | ARRAY_SIZE(tqmx86_devs), | 
|  | NULL, 0, NULL); | 
|  | } | 
|  |  | 
|  | static int tqmx86_create_platform_device(const struct dmi_system_id *id) | 
|  | { | 
|  | struct platform_device *pdev; | 
|  | int err; | 
|  |  | 
|  | pdev = platform_device_alloc("tqmx86", -1); | 
|  | if (!pdev) | 
|  | return -ENOMEM; | 
|  |  | 
|  | err = platform_device_add(pdev); | 
|  | if (err) | 
|  | platform_device_put(pdev); | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | static const struct dmi_system_id tqmx86_dmi_table[] __initconst = { | 
|  | { | 
|  | .ident = "TQMX86", | 
|  | .matches = { | 
|  | DMI_MATCH(DMI_SYS_VENDOR, "TQ-Group"), | 
|  | DMI_MATCH(DMI_PRODUCT_NAME, "TQMx"), | 
|  | }, | 
|  | .callback = tqmx86_create_platform_device, | 
|  | }, | 
|  | { | 
|  | .ident = "TQMX86", | 
|  | .matches = { | 
|  | DMI_MATCH(DMI_SYS_VENDOR, "TQ-Systems"), | 
|  | DMI_MATCH(DMI_PRODUCT_NAME, "TQMx"), | 
|  | }, | 
|  | .callback = tqmx86_create_platform_device, | 
|  | }, | 
|  | {} | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(dmi, tqmx86_dmi_table); | 
|  |  | 
|  | static struct platform_driver tqmx86_driver = { | 
|  | .driver		= { | 
|  | .name	= "tqmx86", | 
|  | }, | 
|  | .probe		= tqmx86_probe, | 
|  | }; | 
|  |  | 
|  | static int __init tqmx86_init(void) | 
|  | { | 
|  | if (!dmi_check_system(tqmx86_dmi_table)) | 
|  | return -ENODEV; | 
|  |  | 
|  | return platform_driver_register(&tqmx86_driver); | 
|  | } | 
|  |  | 
|  | module_init(tqmx86_init); | 
|  |  | 
|  | MODULE_DESCRIPTION("TQMx86 PLD Core Driver"); | 
|  | MODULE_AUTHOR("Andrew Lunn <andrew@lunn.ch>"); | 
|  | MODULE_LICENSE("GPL"); | 
|  | MODULE_ALIAS("platform:tqmx86"); |