123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314 |
- #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),
- };
- 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[] = {
- {
-
- 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);
-
- 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;
- }
-
- 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");
|