// SPDX-License-Identifier: GPL-2.0-only // Copyright (C) 2018 ROHM Semiconductors #include <linux/gpio/driver.h> #include <linux/mfd/rohm-bd71828.h> #include <linux/module.h> #include <linux/platform_device.h> #include <linux/regmap.h> #define GPIO_OUT_REG(off) (BD71828_REG_GPIO_CTRL1 + (off)) #define HALL_GPIO_OFFSET 3 struct bd71828_gpio { struct regmap *regmap; struct device *dev; struct gpio_chip gpio; }; static void bd71828_gpio_set(struct gpio_chip *chip, unsigned int offset, int value) { int ret; struct bd71828_gpio *bdgpio = gpiochip_get_data(chip); u8 val = (value) ? BD71828_GPIO_OUT_HI : BD71828_GPIO_OUT_LO; /* * The HALL input pin can only be used as input. If this is the pin * we are dealing with - then we are done */ if (offset == HALL_GPIO_OFFSET) return; ret = regmap_update_bits(bdgpio->regmap, GPIO_OUT_REG(offset), BD71828_GPIO_OUT_MASK, val); if (ret) dev_err(bdgpio->dev, "Could not set gpio to %d\n", value); } static int bd71828_gpio_get(struct gpio_chip *chip, unsigned int offset) { int ret; unsigned int val; struct bd71828_gpio *bdgpio = gpiochip_get_data(chip); if (offset == HALL_GPIO_OFFSET) ret = regmap_read(bdgpio->regmap, BD71828_REG_IO_STAT, &val); else ret = regmap_read(bdgpio->regmap, GPIO_OUT_REG(offset), &val); if (!ret) ret = (val & BD71828_GPIO_OUT_MASK); return ret; } static int bd71828_gpio_set_config(struct gpio_chip *chip, unsigned int offset, unsigned long config) { struct bd71828_gpio *bdgpio = gpiochip_get_data(chip); if (offset == HALL_GPIO_OFFSET) return -ENOTSUPP; switch (pinconf_to_config_param(config)) { case PIN_CONFIG_DRIVE_OPEN_DRAIN: return regmap_update_bits(bdgpio->regmap, GPIO_OUT_REG(offset), BD71828_GPIO_DRIVE_MASK, BD71828_GPIO_OPEN_DRAIN); case PIN_CONFIG_DRIVE_PUSH_PULL: return regmap_update_bits(bdgpio->regmap, GPIO_OUT_REG(offset), BD71828_GPIO_DRIVE_MASK, BD71828_GPIO_PUSH_PULL); default: break; } return -ENOTSUPP; } static int bd71828_get_direction(struct gpio_chip *chip, unsigned int offset) { /* * Pin usage is selected by OTP data. We can't read it runtime. Hence * we trust that if the pin is not excluded by "gpio-reserved-ranges" * the OTP configuration is set to OUT. (Other pins but HALL input pin * on BD71828 can't really be used for general purpose input - input * states are used for specific cases like regulator control or * PMIC_ON_REQ. */ if (offset == HALL_GPIO_OFFSET) return GPIO_LINE_DIRECTION_IN; return GPIO_LINE_DIRECTION_OUT; } static int bd71828_probe(struct platform_device *pdev) { struct device *dev = &pdev->dev; struct bd71828_gpio *bdgpio; bdgpio = devm_kzalloc(dev, sizeof(*bdgpio), GFP_KERNEL); if (!bdgpio) return -ENOMEM; bdgpio->dev = dev; bdgpio->gpio.parent = dev->parent; bdgpio->gpio.label = "bd71828-gpio"; bdgpio->gpio.owner = THIS_MODULE; bdgpio->gpio.get_direction = bd71828_get_direction; bdgpio->gpio.set_config = bd71828_gpio_set_config; bdgpio->gpio.can_sleep = true; bdgpio->gpio.get = bd71828_gpio_get; bdgpio->gpio.set = bd71828_gpio_set; bdgpio->gpio.base = -1; /* * See if we need some implementation to mark some PINs as * not controllable based on DT info or if core can handle * "gpio-reserved-ranges" and exclude them from control */ bdgpio->gpio.ngpio = 4; bdgpio->regmap = dev_get_regmap(dev->parent, NULL); if (!bdgpio->regmap) return -ENODEV; return devm_gpiochip_add_data(dev, &bdgpio->gpio, bdgpio); } static struct platform_driver bd71828_gpio = { .driver = { .name = "bd71828-gpio" }, .probe = bd71828_probe, }; module_platform_driver(bd71828_gpio); MODULE_AUTHOR("Matti Vaittinen <matti.vaittinen@fi.rohmeurope.com>"); MODULE_DESCRIPTION("BD71828 voltage regulator driver"); MODULE_LICENSE("GPL"); MODULE_ALIAS("platform:bd71828-gpio");