| // 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"); |