18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * GPIO Driver for Dialog DA9055 PMICs. 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright(c) 2012 Dialog Semiconductor Ltd. 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * Author: David Dajun Chen <dchen@diasemi.com> 88c2ecf20Sopenharmony_ci */ 98c2ecf20Sopenharmony_ci#include <linux/module.h> 108c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 118c2ecf20Sopenharmony_ci#include <linux/gpio/driver.h> 128c2ecf20Sopenharmony_ci 138c2ecf20Sopenharmony_ci#include <linux/mfd/da9055/core.h> 148c2ecf20Sopenharmony_ci#include <linux/mfd/da9055/reg.h> 158c2ecf20Sopenharmony_ci#include <linux/mfd/da9055/pdata.h> 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_ci#define DA9055_VDD_IO 0x0 188c2ecf20Sopenharmony_ci#define DA9055_PUSH_PULL 0x3 198c2ecf20Sopenharmony_ci#define DA9055_ACT_LOW 0x0 208c2ecf20Sopenharmony_ci#define DA9055_GPI 0x1 218c2ecf20Sopenharmony_ci#define DA9055_PORT_MASK 0x3 228c2ecf20Sopenharmony_ci#define DA9055_PORT_SHIFT(offset) (4 * (offset % 2)) 238c2ecf20Sopenharmony_ci 248c2ecf20Sopenharmony_ci#define DA9055_INPUT DA9055_GPI 258c2ecf20Sopenharmony_ci#define DA9055_OUTPUT DA9055_PUSH_PULL 268c2ecf20Sopenharmony_ci#define DA9055_IRQ_GPI0 3 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_cistruct da9055_gpio { 298c2ecf20Sopenharmony_ci struct da9055 *da9055; 308c2ecf20Sopenharmony_ci struct gpio_chip gp; 318c2ecf20Sopenharmony_ci}; 328c2ecf20Sopenharmony_ci 338c2ecf20Sopenharmony_cistatic int da9055_gpio_get(struct gpio_chip *gc, unsigned offset) 348c2ecf20Sopenharmony_ci{ 358c2ecf20Sopenharmony_ci struct da9055_gpio *gpio = gpiochip_get_data(gc); 368c2ecf20Sopenharmony_ci int gpio_direction = 0; 378c2ecf20Sopenharmony_ci int ret; 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_ci /* Get GPIO direction */ 408c2ecf20Sopenharmony_ci ret = da9055_reg_read(gpio->da9055, (offset >> 1) + DA9055_REG_GPIO0_1); 418c2ecf20Sopenharmony_ci if (ret < 0) 428c2ecf20Sopenharmony_ci return ret; 438c2ecf20Sopenharmony_ci 448c2ecf20Sopenharmony_ci gpio_direction = ret & (DA9055_PORT_MASK) << DA9055_PORT_SHIFT(offset); 458c2ecf20Sopenharmony_ci gpio_direction >>= DA9055_PORT_SHIFT(offset); 468c2ecf20Sopenharmony_ci switch (gpio_direction) { 478c2ecf20Sopenharmony_ci case DA9055_INPUT: 488c2ecf20Sopenharmony_ci ret = da9055_reg_read(gpio->da9055, DA9055_REG_STATUS_B); 498c2ecf20Sopenharmony_ci if (ret < 0) 508c2ecf20Sopenharmony_ci return ret; 518c2ecf20Sopenharmony_ci break; 528c2ecf20Sopenharmony_ci case DA9055_OUTPUT: 538c2ecf20Sopenharmony_ci ret = da9055_reg_read(gpio->da9055, DA9055_REG_GPIO_MODE0_2); 548c2ecf20Sopenharmony_ci if (ret < 0) 558c2ecf20Sopenharmony_ci return ret; 568c2ecf20Sopenharmony_ci } 578c2ecf20Sopenharmony_ci 588c2ecf20Sopenharmony_ci return ret & (1 << offset); 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_ci} 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_cistatic void da9055_gpio_set(struct gpio_chip *gc, unsigned offset, int value) 638c2ecf20Sopenharmony_ci{ 648c2ecf20Sopenharmony_ci struct da9055_gpio *gpio = gpiochip_get_data(gc); 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci da9055_reg_update(gpio->da9055, 678c2ecf20Sopenharmony_ci DA9055_REG_GPIO_MODE0_2, 688c2ecf20Sopenharmony_ci 1 << offset, 698c2ecf20Sopenharmony_ci value << offset); 708c2ecf20Sopenharmony_ci} 718c2ecf20Sopenharmony_ci 728c2ecf20Sopenharmony_cistatic int da9055_gpio_direction_input(struct gpio_chip *gc, unsigned offset) 738c2ecf20Sopenharmony_ci{ 748c2ecf20Sopenharmony_ci struct da9055_gpio *gpio = gpiochip_get_data(gc); 758c2ecf20Sopenharmony_ci unsigned char reg_byte; 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_ci reg_byte = (DA9055_ACT_LOW | DA9055_GPI) 788c2ecf20Sopenharmony_ci << DA9055_PORT_SHIFT(offset); 798c2ecf20Sopenharmony_ci 808c2ecf20Sopenharmony_ci return da9055_reg_update(gpio->da9055, (offset >> 1) + 818c2ecf20Sopenharmony_ci DA9055_REG_GPIO0_1, 828c2ecf20Sopenharmony_ci DA9055_PORT_MASK << 838c2ecf20Sopenharmony_ci DA9055_PORT_SHIFT(offset), 848c2ecf20Sopenharmony_ci reg_byte); 858c2ecf20Sopenharmony_ci} 868c2ecf20Sopenharmony_ci 878c2ecf20Sopenharmony_cistatic int da9055_gpio_direction_output(struct gpio_chip *gc, 888c2ecf20Sopenharmony_ci unsigned offset, int value) 898c2ecf20Sopenharmony_ci{ 908c2ecf20Sopenharmony_ci struct da9055_gpio *gpio = gpiochip_get_data(gc); 918c2ecf20Sopenharmony_ci unsigned char reg_byte; 928c2ecf20Sopenharmony_ci int ret; 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci reg_byte = (DA9055_VDD_IO | DA9055_PUSH_PULL) 958c2ecf20Sopenharmony_ci << DA9055_PORT_SHIFT(offset); 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci ret = da9055_reg_update(gpio->da9055, (offset >> 1) + 988c2ecf20Sopenharmony_ci DA9055_REG_GPIO0_1, 998c2ecf20Sopenharmony_ci DA9055_PORT_MASK << 1008c2ecf20Sopenharmony_ci DA9055_PORT_SHIFT(offset), 1018c2ecf20Sopenharmony_ci reg_byte); 1028c2ecf20Sopenharmony_ci if (ret < 0) 1038c2ecf20Sopenharmony_ci return ret; 1048c2ecf20Sopenharmony_ci 1058c2ecf20Sopenharmony_ci da9055_gpio_set(gc, offset, value); 1068c2ecf20Sopenharmony_ci 1078c2ecf20Sopenharmony_ci return 0; 1088c2ecf20Sopenharmony_ci} 1098c2ecf20Sopenharmony_ci 1108c2ecf20Sopenharmony_cistatic int da9055_gpio_to_irq(struct gpio_chip *gc, u32 offset) 1118c2ecf20Sopenharmony_ci{ 1128c2ecf20Sopenharmony_ci struct da9055_gpio *gpio = gpiochip_get_data(gc); 1138c2ecf20Sopenharmony_ci struct da9055 *da9055 = gpio->da9055; 1148c2ecf20Sopenharmony_ci 1158c2ecf20Sopenharmony_ci return regmap_irq_get_virq(da9055->irq_data, 1168c2ecf20Sopenharmony_ci DA9055_IRQ_GPI0 + offset); 1178c2ecf20Sopenharmony_ci} 1188c2ecf20Sopenharmony_ci 1198c2ecf20Sopenharmony_cistatic const struct gpio_chip reference_gp = { 1208c2ecf20Sopenharmony_ci .label = "da9055-gpio", 1218c2ecf20Sopenharmony_ci .owner = THIS_MODULE, 1228c2ecf20Sopenharmony_ci .get = da9055_gpio_get, 1238c2ecf20Sopenharmony_ci .set = da9055_gpio_set, 1248c2ecf20Sopenharmony_ci .direction_input = da9055_gpio_direction_input, 1258c2ecf20Sopenharmony_ci .direction_output = da9055_gpio_direction_output, 1268c2ecf20Sopenharmony_ci .to_irq = da9055_gpio_to_irq, 1278c2ecf20Sopenharmony_ci .can_sleep = true, 1288c2ecf20Sopenharmony_ci .ngpio = 3, 1298c2ecf20Sopenharmony_ci .base = -1, 1308c2ecf20Sopenharmony_ci}; 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_cistatic int da9055_gpio_probe(struct platform_device *pdev) 1338c2ecf20Sopenharmony_ci{ 1348c2ecf20Sopenharmony_ci struct da9055_gpio *gpio; 1358c2ecf20Sopenharmony_ci struct da9055_pdata *pdata; 1368c2ecf20Sopenharmony_ci int ret; 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci gpio = devm_kzalloc(&pdev->dev, sizeof(*gpio), GFP_KERNEL); 1398c2ecf20Sopenharmony_ci if (!gpio) 1408c2ecf20Sopenharmony_ci return -ENOMEM; 1418c2ecf20Sopenharmony_ci 1428c2ecf20Sopenharmony_ci gpio->da9055 = dev_get_drvdata(pdev->dev.parent); 1438c2ecf20Sopenharmony_ci pdata = dev_get_platdata(gpio->da9055->dev); 1448c2ecf20Sopenharmony_ci 1458c2ecf20Sopenharmony_ci gpio->gp = reference_gp; 1468c2ecf20Sopenharmony_ci if (pdata && pdata->gpio_base) 1478c2ecf20Sopenharmony_ci gpio->gp.base = pdata->gpio_base; 1488c2ecf20Sopenharmony_ci 1498c2ecf20Sopenharmony_ci ret = devm_gpiochip_add_data(&pdev->dev, &gpio->gp, gpio); 1508c2ecf20Sopenharmony_ci if (ret < 0) { 1518c2ecf20Sopenharmony_ci dev_err(&pdev->dev, "Could not register gpiochip, %d\n", ret); 1528c2ecf20Sopenharmony_ci return ret; 1538c2ecf20Sopenharmony_ci } 1548c2ecf20Sopenharmony_ci 1558c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, gpio); 1568c2ecf20Sopenharmony_ci 1578c2ecf20Sopenharmony_ci return 0; 1588c2ecf20Sopenharmony_ci} 1598c2ecf20Sopenharmony_ci 1608c2ecf20Sopenharmony_cistatic struct platform_driver da9055_gpio_driver = { 1618c2ecf20Sopenharmony_ci .probe = da9055_gpio_probe, 1628c2ecf20Sopenharmony_ci .driver = { 1638c2ecf20Sopenharmony_ci .name = "da9055-gpio", 1648c2ecf20Sopenharmony_ci }, 1658c2ecf20Sopenharmony_ci}; 1668c2ecf20Sopenharmony_ci 1678c2ecf20Sopenharmony_cistatic int __init da9055_gpio_init(void) 1688c2ecf20Sopenharmony_ci{ 1698c2ecf20Sopenharmony_ci return platform_driver_register(&da9055_gpio_driver); 1708c2ecf20Sopenharmony_ci} 1718c2ecf20Sopenharmony_cisubsys_initcall(da9055_gpio_init); 1728c2ecf20Sopenharmony_ci 1738c2ecf20Sopenharmony_cistatic void __exit da9055_gpio_exit(void) 1748c2ecf20Sopenharmony_ci{ 1758c2ecf20Sopenharmony_ci platform_driver_unregister(&da9055_gpio_driver); 1768c2ecf20Sopenharmony_ci} 1778c2ecf20Sopenharmony_cimodule_exit(da9055_gpio_exit); 1788c2ecf20Sopenharmony_ci 1798c2ecf20Sopenharmony_ciMODULE_AUTHOR("David Dajun Chen <dchen@diasemi.com>"); 1808c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("DA9055 GPIO Device Driver"); 1818c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 1828c2ecf20Sopenharmony_ciMODULE_ALIAS("platform:da9055-gpio"); 183