162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * GPIO driver for the ACCES 104-IDIO-16 family 462306a36Sopenharmony_ci * Copyright (C) 2015 William Breathitt Gray 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * This driver supports the following ACCES devices: 104-IDIO-16, 762306a36Sopenharmony_ci * 104-IDIO-16E, 104-IDO-16, 104-IDIO-8, 104-IDIO-8E, and 104-IDO-8. 862306a36Sopenharmony_ci */ 962306a36Sopenharmony_ci#include <linux/bits.h> 1062306a36Sopenharmony_ci#include <linux/device.h> 1162306a36Sopenharmony_ci#include <linux/err.h> 1262306a36Sopenharmony_ci#include <linux/ioport.h> 1362306a36Sopenharmony_ci#include <linux/irq.h> 1462306a36Sopenharmony_ci#include <linux/isa.h> 1562306a36Sopenharmony_ci#include <linux/kernel.h> 1662306a36Sopenharmony_ci#include <linux/module.h> 1762306a36Sopenharmony_ci#include <linux/moduleparam.h> 1862306a36Sopenharmony_ci#include <linux/regmap.h> 1962306a36Sopenharmony_ci#include <linux/types.h> 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#include "gpio-idio-16.h" 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci#define IDIO_16_EXTENT 8 2462306a36Sopenharmony_ci#define MAX_NUM_IDIO_16 max_num_isa_dev(IDIO_16_EXTENT) 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_cistatic unsigned int base[MAX_NUM_IDIO_16]; 2762306a36Sopenharmony_cistatic unsigned int num_idio_16; 2862306a36Sopenharmony_cimodule_param_hw_array(base, uint, ioport, &num_idio_16, 0); 2962306a36Sopenharmony_ciMODULE_PARM_DESC(base, "ACCES 104-IDIO-16 base addresses"); 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_cistatic unsigned int irq[MAX_NUM_IDIO_16]; 3262306a36Sopenharmony_cistatic unsigned int num_irq; 3362306a36Sopenharmony_cimodule_param_hw_array(irq, uint, irq, &num_irq, 0); 3462306a36Sopenharmony_ciMODULE_PARM_DESC(irq, "ACCES 104-IDIO-16 interrupt line numbers"); 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_cistatic const struct regmap_range idio_16_wr_ranges[] = { 3762306a36Sopenharmony_ci regmap_reg_range(0x0, 0x2), regmap_reg_range(0x4, 0x4), 3862306a36Sopenharmony_ci}; 3962306a36Sopenharmony_cistatic const struct regmap_range idio_16_rd_ranges[] = { 4062306a36Sopenharmony_ci regmap_reg_range(0x1, 0x2), regmap_reg_range(0x5, 0x5), 4162306a36Sopenharmony_ci}; 4262306a36Sopenharmony_cistatic const struct regmap_range idio_16_precious_ranges[] = { 4362306a36Sopenharmony_ci regmap_reg_range(0x2, 0x2), 4462306a36Sopenharmony_ci}; 4562306a36Sopenharmony_cistatic const struct regmap_access_table idio_16_wr_table = { 4662306a36Sopenharmony_ci .yes_ranges = idio_16_wr_ranges, 4762306a36Sopenharmony_ci .n_yes_ranges = ARRAY_SIZE(idio_16_wr_ranges), 4862306a36Sopenharmony_ci}; 4962306a36Sopenharmony_cistatic const struct regmap_access_table idio_16_rd_table = { 5062306a36Sopenharmony_ci .yes_ranges = idio_16_rd_ranges, 5162306a36Sopenharmony_ci .n_yes_ranges = ARRAY_SIZE(idio_16_rd_ranges), 5262306a36Sopenharmony_ci}; 5362306a36Sopenharmony_cistatic const struct regmap_access_table idio_16_precious_table = { 5462306a36Sopenharmony_ci .yes_ranges = idio_16_precious_ranges, 5562306a36Sopenharmony_ci .n_yes_ranges = ARRAY_SIZE(idio_16_precious_ranges), 5662306a36Sopenharmony_ci}; 5762306a36Sopenharmony_cistatic const struct regmap_config idio_16_regmap_config = { 5862306a36Sopenharmony_ci .reg_bits = 8, 5962306a36Sopenharmony_ci .reg_stride = 1, 6062306a36Sopenharmony_ci .val_bits = 8, 6162306a36Sopenharmony_ci .io_port = true, 6262306a36Sopenharmony_ci .wr_table = &idio_16_wr_table, 6362306a36Sopenharmony_ci .rd_table = &idio_16_rd_table, 6462306a36Sopenharmony_ci .volatile_table = &idio_16_rd_table, 6562306a36Sopenharmony_ci .precious_table = &idio_16_precious_table, 6662306a36Sopenharmony_ci .cache_type = REGCACHE_FLAT, 6762306a36Sopenharmony_ci .use_raw_spinlock = true, 6862306a36Sopenharmony_ci}; 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci/* Only input lines (GPIO 16-31) support interrupts */ 7162306a36Sopenharmony_ci#define IDIO_16_REGMAP_IRQ(_id) \ 7262306a36Sopenharmony_ci [16 + _id] = { \ 7362306a36Sopenharmony_ci .mask = BIT(_id), \ 7462306a36Sopenharmony_ci .type = { .types_supported = IRQ_TYPE_EDGE_BOTH }, \ 7562306a36Sopenharmony_ci } 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_cistatic const struct regmap_irq idio_16_regmap_irqs[] = { 7862306a36Sopenharmony_ci IDIO_16_REGMAP_IRQ(0), IDIO_16_REGMAP_IRQ(1), IDIO_16_REGMAP_IRQ(2), /* 0-2 */ 7962306a36Sopenharmony_ci IDIO_16_REGMAP_IRQ(3), IDIO_16_REGMAP_IRQ(4), IDIO_16_REGMAP_IRQ(5), /* 3-5 */ 8062306a36Sopenharmony_ci IDIO_16_REGMAP_IRQ(6), IDIO_16_REGMAP_IRQ(7), IDIO_16_REGMAP_IRQ(8), /* 6-8 */ 8162306a36Sopenharmony_ci IDIO_16_REGMAP_IRQ(9), IDIO_16_REGMAP_IRQ(10), IDIO_16_REGMAP_IRQ(11), /* 9-11 */ 8262306a36Sopenharmony_ci IDIO_16_REGMAP_IRQ(12), IDIO_16_REGMAP_IRQ(13), IDIO_16_REGMAP_IRQ(14), /* 12-14 */ 8362306a36Sopenharmony_ci IDIO_16_REGMAP_IRQ(15), /* 15 */ 8462306a36Sopenharmony_ci}; 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_cistatic int idio_16_probe(struct device *dev, unsigned int id) 8762306a36Sopenharmony_ci{ 8862306a36Sopenharmony_ci const char *const name = dev_name(dev); 8962306a36Sopenharmony_ci struct idio_16_regmap_config config = {}; 9062306a36Sopenharmony_ci void __iomem *regs; 9162306a36Sopenharmony_ci struct regmap *map; 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci if (!devm_request_region(dev, base[id], IDIO_16_EXTENT, name)) { 9462306a36Sopenharmony_ci dev_err(dev, "Unable to lock port addresses (0x%X-0x%X)\n", 9562306a36Sopenharmony_ci base[id], base[id] + IDIO_16_EXTENT); 9662306a36Sopenharmony_ci return -EBUSY; 9762306a36Sopenharmony_ci } 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci regs = devm_ioport_map(dev, base[id], IDIO_16_EXTENT); 10062306a36Sopenharmony_ci if (!regs) 10162306a36Sopenharmony_ci return -ENOMEM; 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_ci map = devm_regmap_init_mmio(dev, regs, &idio_16_regmap_config); 10462306a36Sopenharmony_ci if (IS_ERR(map)) 10562306a36Sopenharmony_ci return dev_err_probe(dev, PTR_ERR(map), "Unable to initialize register map\n"); 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_ci config.parent = dev; 10862306a36Sopenharmony_ci config.map = map; 10962306a36Sopenharmony_ci config.regmap_irqs = idio_16_regmap_irqs; 11062306a36Sopenharmony_ci config.num_regmap_irqs = ARRAY_SIZE(idio_16_regmap_irqs); 11162306a36Sopenharmony_ci config.irq = irq[id]; 11262306a36Sopenharmony_ci config.no_status = true; 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_ci return devm_idio_16_regmap_register(dev, &config); 11562306a36Sopenharmony_ci} 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_cistatic struct isa_driver idio_16_driver = { 11862306a36Sopenharmony_ci .probe = idio_16_probe, 11962306a36Sopenharmony_ci .driver = { 12062306a36Sopenharmony_ci .name = "104-idio-16" 12162306a36Sopenharmony_ci }, 12262306a36Sopenharmony_ci}; 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_cimodule_isa_driver_with_irq(idio_16_driver, num_idio_16, num_irq); 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ciMODULE_AUTHOR("William Breathitt Gray <vilhelm.gray@gmail.com>"); 12762306a36Sopenharmony_ciMODULE_DESCRIPTION("ACCES 104-IDIO-16 GPIO driver"); 12862306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 12962306a36Sopenharmony_ciMODULE_IMPORT_NS(GPIO_IDIO_16); 130