123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * regmap based generic GPIO driver
- *
- * Copyright 2020 Michael Walle <[email protected]>
- */
- #include <linux/gpio/driver.h>
- #include <linux/gpio/regmap.h>
- #include <linux/kernel.h>
- #include <linux/module.h>
- #include <linux/regmap.h>
- struct gpio_regmap {
- struct device *parent;
- struct regmap *regmap;
- struct gpio_chip gpio_chip;
- int reg_stride;
- int ngpio_per_reg;
- unsigned int reg_dat_base;
- unsigned int reg_set_base;
- unsigned int reg_clr_base;
- unsigned int reg_dir_in_base;
- unsigned int reg_dir_out_base;
- int (*reg_mask_xlate)(struct gpio_regmap *gpio, unsigned int base,
- unsigned int offset, unsigned int *reg,
- unsigned int *mask);
- void *driver_data;
- };
- static unsigned int gpio_regmap_addr(unsigned int addr)
- {
- if (addr == GPIO_REGMAP_ADDR_ZERO)
- return 0;
- return addr;
- }
- static int gpio_regmap_simple_xlate(struct gpio_regmap *gpio,
- unsigned int base, unsigned int offset,
- unsigned int *reg, unsigned int *mask)
- {
- unsigned int line = offset % gpio->ngpio_per_reg;
- unsigned int stride = offset / gpio->ngpio_per_reg;
- *reg = base + stride * gpio->reg_stride;
- *mask = BIT(line);
- return 0;
- }
- static int gpio_regmap_get(struct gpio_chip *chip, unsigned int offset)
- {
- struct gpio_regmap *gpio = gpiochip_get_data(chip);
- unsigned int base, val, reg, mask;
- int ret;
- /* we might not have an output register if we are input only */
- if (gpio->reg_dat_base)
- base = gpio_regmap_addr(gpio->reg_dat_base);
- else
- base = gpio_regmap_addr(gpio->reg_set_base);
- ret = gpio->reg_mask_xlate(gpio, base, offset, ®, &mask);
- if (ret)
- return ret;
- ret = regmap_read(gpio->regmap, reg, &val);
- if (ret)
- return ret;
- return !!(val & mask);
- }
- static void gpio_regmap_set(struct gpio_chip *chip, unsigned int offset,
- int val)
- {
- struct gpio_regmap *gpio = gpiochip_get_data(chip);
- unsigned int base = gpio_regmap_addr(gpio->reg_set_base);
- unsigned int reg, mask;
- gpio->reg_mask_xlate(gpio, base, offset, ®, &mask);
- if (val)
- regmap_update_bits(gpio->regmap, reg, mask, mask);
- else
- regmap_update_bits(gpio->regmap, reg, mask, 0);
- }
- static void gpio_regmap_set_with_clear(struct gpio_chip *chip,
- unsigned int offset, int val)
- {
- struct gpio_regmap *gpio = gpiochip_get_data(chip);
- unsigned int base, reg, mask;
- if (val)
- base = gpio_regmap_addr(gpio->reg_set_base);
- else
- base = gpio_regmap_addr(gpio->reg_clr_base);
- gpio->reg_mask_xlate(gpio, base, offset, ®, &mask);
- regmap_write(gpio->regmap, reg, mask);
- }
- static int gpio_regmap_get_direction(struct gpio_chip *chip,
- unsigned int offset)
- {
- struct gpio_regmap *gpio = gpiochip_get_data(chip);
- unsigned int base, val, reg, mask;
- int invert, ret;
- if (gpio->reg_dir_out_base) {
- base = gpio_regmap_addr(gpio->reg_dir_out_base);
- invert = 0;
- } else if (gpio->reg_dir_in_base) {
- base = gpio_regmap_addr(gpio->reg_dir_in_base);
- invert = 1;
- } else {
- return -EOPNOTSUPP;
- }
- ret = gpio->reg_mask_xlate(gpio, base, offset, ®, &mask);
- if (ret)
- return ret;
- ret = regmap_read(gpio->regmap, reg, &val);
- if (ret)
- return ret;
- if (!!(val & mask) ^ invert)
- return GPIO_LINE_DIRECTION_OUT;
- else
- return GPIO_LINE_DIRECTION_IN;
- }
- static int gpio_regmap_set_direction(struct gpio_chip *chip,
- unsigned int offset, bool output)
- {
- struct gpio_regmap *gpio = gpiochip_get_data(chip);
- unsigned int base, val, reg, mask;
- int invert, ret;
- if (gpio->reg_dir_out_base) {
- base = gpio_regmap_addr(gpio->reg_dir_out_base);
- invert = 0;
- } else if (gpio->reg_dir_in_base) {
- base = gpio_regmap_addr(gpio->reg_dir_in_base);
- invert = 1;
- } else {
- return -EOPNOTSUPP;
- }
- ret = gpio->reg_mask_xlate(gpio, base, offset, ®, &mask);
- if (ret)
- return ret;
- if (invert)
- val = output ? 0 : mask;
- else
- val = output ? mask : 0;
- return regmap_update_bits(gpio->regmap, reg, mask, val);
- }
- static int gpio_regmap_direction_input(struct gpio_chip *chip,
- unsigned int offset)
- {
- return gpio_regmap_set_direction(chip, offset, false);
- }
- static int gpio_regmap_direction_output(struct gpio_chip *chip,
- unsigned int offset, int value)
- {
- gpio_regmap_set(chip, offset, value);
- return gpio_regmap_set_direction(chip, offset, true);
- }
- void *gpio_regmap_get_drvdata(struct gpio_regmap *gpio)
- {
- return gpio->driver_data;
- }
- EXPORT_SYMBOL_GPL(gpio_regmap_get_drvdata);
- /**
- * gpio_regmap_register() - Register a generic regmap GPIO controller
- * @config: configuration for gpio_regmap
- *
- * Return: A pointer to the registered gpio_regmap or ERR_PTR error value.
- */
- struct gpio_regmap *gpio_regmap_register(const struct gpio_regmap_config *config)
- {
- struct gpio_regmap *gpio;
- struct gpio_chip *chip;
- int ret;
- if (!config->parent)
- return ERR_PTR(-EINVAL);
- if (!config->ngpio)
- return ERR_PTR(-EINVAL);
- /* we need at least one */
- if (!config->reg_dat_base && !config->reg_set_base)
- return ERR_PTR(-EINVAL);
- /* if we have a direction register we need both input and output */
- if ((config->reg_dir_out_base || config->reg_dir_in_base) &&
- (!config->reg_dat_base || !config->reg_set_base))
- return ERR_PTR(-EINVAL);
- /* we don't support having both registers simultaneously for now */
- if (config->reg_dir_out_base && config->reg_dir_in_base)
- return ERR_PTR(-EINVAL);
- gpio = kzalloc(sizeof(*gpio), GFP_KERNEL);
- if (!gpio)
- return ERR_PTR(-ENOMEM);
- gpio->parent = config->parent;
- gpio->driver_data = config->drvdata;
- gpio->regmap = config->regmap;
- gpio->ngpio_per_reg = config->ngpio_per_reg;
- gpio->reg_stride = config->reg_stride;
- gpio->reg_mask_xlate = config->reg_mask_xlate;
- gpio->reg_dat_base = config->reg_dat_base;
- gpio->reg_set_base = config->reg_set_base;
- gpio->reg_clr_base = config->reg_clr_base;
- gpio->reg_dir_in_base = config->reg_dir_in_base;
- gpio->reg_dir_out_base = config->reg_dir_out_base;
- /* if not set, assume there is only one register */
- if (!gpio->ngpio_per_reg)
- gpio->ngpio_per_reg = config->ngpio;
- /* if not set, assume they are consecutive */
- if (!gpio->reg_stride)
- gpio->reg_stride = 1;
- if (!gpio->reg_mask_xlate)
- gpio->reg_mask_xlate = gpio_regmap_simple_xlate;
- chip = &gpio->gpio_chip;
- chip->parent = config->parent;
- chip->fwnode = config->fwnode;
- chip->base = -1;
- chip->ngpio = config->ngpio;
- chip->names = config->names;
- chip->label = config->label ?: dev_name(config->parent);
- /*
- * If our regmap is fast_io we should probably set can_sleep to false.
- * Right now, the regmap doesn't save this property, nor is there any
- * access function for it.
- * The only regmap type which uses fast_io is regmap-mmio. For now,
- * assume a safe default of true here.
- */
- chip->can_sleep = true;
- chip->get = gpio_regmap_get;
- if (gpio->reg_set_base && gpio->reg_clr_base)
- chip->set = gpio_regmap_set_with_clear;
- else if (gpio->reg_set_base)
- chip->set = gpio_regmap_set;
- if (gpio->reg_dir_in_base || gpio->reg_dir_out_base) {
- chip->get_direction = gpio_regmap_get_direction;
- chip->direction_input = gpio_regmap_direction_input;
- chip->direction_output = gpio_regmap_direction_output;
- }
- ret = gpiochip_add_data(chip, gpio);
- if (ret < 0)
- goto err_free_gpio;
- if (config->irq_domain) {
- ret = gpiochip_irqchip_add_domain(chip, config->irq_domain);
- if (ret)
- goto err_remove_gpiochip;
- }
- return gpio;
- err_remove_gpiochip:
- gpiochip_remove(chip);
- err_free_gpio:
- kfree(gpio);
- return ERR_PTR(ret);
- }
- EXPORT_SYMBOL_GPL(gpio_regmap_register);
- /**
- * gpio_regmap_unregister() - Unregister a generic regmap GPIO controller
- * @gpio: gpio_regmap device to unregister
- */
- void gpio_regmap_unregister(struct gpio_regmap *gpio)
- {
- gpiochip_remove(&gpio->gpio_chip);
- kfree(gpio);
- }
- EXPORT_SYMBOL_GPL(gpio_regmap_unregister);
- static void devm_gpio_regmap_unregister(void *res)
- {
- gpio_regmap_unregister(res);
- }
- /**
- * devm_gpio_regmap_register() - resource managed gpio_regmap_register()
- * @dev: device that is registering this GPIO device
- * @config: configuration for gpio_regmap
- *
- * Managed gpio_regmap_register(). For generic regmap GPIO device registered by
- * this function, gpio_regmap_unregister() is automatically called on driver
- * detach. See gpio_regmap_register() for more information.
- *
- * Return: A pointer to the registered gpio_regmap or ERR_PTR error value.
- */
- struct gpio_regmap *devm_gpio_regmap_register(struct device *dev,
- const struct gpio_regmap_config *config)
- {
- struct gpio_regmap *gpio;
- int ret;
- gpio = gpio_regmap_register(config);
- if (IS_ERR(gpio))
- return gpio;
- ret = devm_add_action_or_reset(dev, devm_gpio_regmap_unregister, gpio);
- if (ret)
- return ERR_PTR(ret);
- return gpio;
- }
- EXPORT_SYMBOL_GPL(devm_gpio_regmap_register);
- MODULE_AUTHOR("Michael Walle <[email protected]>");
- MODULE_DESCRIPTION("GPIO generic regmap driver core");
- MODULE_LICENSE("GPL");
|