irq-sl28cpld.c 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * sl28cpld interrupt controller driver
  4. *
  5. * Copyright 2020 Kontron Europe GmbH
  6. */
  7. #include <linux/interrupt.h>
  8. #include <linux/kernel.h>
  9. #include <linux/mod_devicetable.h>
  10. #include <linux/module.h>
  11. #include <linux/platform_device.h>
  12. #include <linux/property.h>
  13. #include <linux/regmap.h>
  14. #define INTC_IE 0x00
  15. #define INTC_IP 0x01
  16. static const struct regmap_irq sl28cpld_irqs[] = {
  17. REGMAP_IRQ_REG_LINE(0, 8),
  18. REGMAP_IRQ_REG_LINE(1, 8),
  19. REGMAP_IRQ_REG_LINE(2, 8),
  20. REGMAP_IRQ_REG_LINE(3, 8),
  21. REGMAP_IRQ_REG_LINE(4, 8),
  22. REGMAP_IRQ_REG_LINE(5, 8),
  23. REGMAP_IRQ_REG_LINE(6, 8),
  24. REGMAP_IRQ_REG_LINE(7, 8),
  25. };
  26. struct sl28cpld_intc {
  27. struct regmap *regmap;
  28. struct regmap_irq_chip chip;
  29. struct regmap_irq_chip_data *irq_data;
  30. };
  31. static int sl28cpld_intc_probe(struct platform_device *pdev)
  32. {
  33. struct device *dev = &pdev->dev;
  34. struct sl28cpld_intc *irqchip;
  35. int irq;
  36. u32 base;
  37. int ret;
  38. if (!dev->parent)
  39. return -ENODEV;
  40. irqchip = devm_kzalloc(dev, sizeof(*irqchip), GFP_KERNEL);
  41. if (!irqchip)
  42. return -ENOMEM;
  43. irqchip->regmap = dev_get_regmap(dev->parent, NULL);
  44. if (!irqchip->regmap)
  45. return -ENODEV;
  46. irq = platform_get_irq(pdev, 0);
  47. if (irq < 0)
  48. return irq;
  49. ret = device_property_read_u32(&pdev->dev, "reg", &base);
  50. if (ret)
  51. return -EINVAL;
  52. irqchip->chip.name = "sl28cpld-intc";
  53. irqchip->chip.irqs = sl28cpld_irqs;
  54. irqchip->chip.num_irqs = ARRAY_SIZE(sl28cpld_irqs);
  55. irqchip->chip.num_regs = 1;
  56. irqchip->chip.status_base = base + INTC_IP;
  57. irqchip->chip.mask_base = base + INTC_IE;
  58. irqchip->chip.mask_invert = true;
  59. irqchip->chip.ack_base = base + INTC_IP;
  60. return devm_regmap_add_irq_chip_fwnode(dev, dev_fwnode(dev),
  61. irqchip->regmap, irq,
  62. IRQF_SHARED | IRQF_ONESHOT, 0,
  63. &irqchip->chip,
  64. &irqchip->irq_data);
  65. }
  66. static const struct of_device_id sl28cpld_intc_of_match[] = {
  67. { .compatible = "kontron,sl28cpld-intc" },
  68. {}
  69. };
  70. MODULE_DEVICE_TABLE(of, sl28cpld_intc_of_match);
  71. static struct platform_driver sl28cpld_intc_driver = {
  72. .probe = sl28cpld_intc_probe,
  73. .driver = {
  74. .name = "sl28cpld-intc",
  75. .of_match_table = sl28cpld_intc_of_match,
  76. }
  77. };
  78. module_platform_driver(sl28cpld_intc_driver);
  79. MODULE_DESCRIPTION("sl28cpld Interrupt Controller Driver");
  80. MODULE_AUTHOR("Michael Walle <[email protected]>");
  81. MODULE_LICENSE("GPL");