| // SPDX-License-Identifier: GPL-2.0-only | 
 | /* | 
 |  * Kontron PLD GPIO driver | 
 |  * | 
 |  * Copyright (c) 2010-2013 Kontron Europe GmbH | 
 |  * Author: Michael Brunner <michael.brunner@kontron.com> | 
 |  */ | 
 |  | 
 | #include <linux/init.h> | 
 | #include <linux/kernel.h> | 
 | #include <linux/module.h> | 
 | #include <linux/bitops.h> | 
 | #include <linux/errno.h> | 
 | #include <linux/platform_device.h> | 
 | #include <linux/gpio/driver.h> | 
 | #include <linux/mfd/kempld.h> | 
 |  | 
 | #define KEMPLD_GPIO_MAX_NUM		16 | 
 | #define KEMPLD_GPIO_MASK(x)		(BIT((x) % 8)) | 
 | #define KEMPLD_GPIO_DIR_NUM(x)		(0x40 + (x) / 8) | 
 | #define KEMPLD_GPIO_LVL_NUM(x)		(0x42 + (x) / 8) | 
 | #define KEMPLD_GPIO_EVT_LVL_EDGE	0x46 | 
 | #define KEMPLD_GPIO_IEN			0x4A | 
 |  | 
 | struct kempld_gpio_data { | 
 | 	struct gpio_chip		chip; | 
 | 	struct kempld_device_data	*pld; | 
 | }; | 
 |  | 
 | /* | 
 |  * Set or clear GPIO bit | 
 |  * kempld_get_mutex must be called prior to calling this function. | 
 |  */ | 
 | static void kempld_gpio_bitop(struct kempld_device_data *pld, | 
 | 			      u8 reg, u8 bit, u8 val) | 
 | { | 
 | 	u8 status; | 
 |  | 
 | 	status = kempld_read8(pld, reg); | 
 | 	if (val) | 
 | 		status |= KEMPLD_GPIO_MASK(bit); | 
 | 	else | 
 | 		status &= ~KEMPLD_GPIO_MASK(bit); | 
 | 	kempld_write8(pld, reg, status); | 
 | } | 
 |  | 
 | static int kempld_gpio_get_bit(struct kempld_device_data *pld, u8 reg, u8 bit) | 
 | { | 
 | 	u8 status; | 
 |  | 
 | 	kempld_get_mutex(pld); | 
 | 	status = kempld_read8(pld, reg); | 
 | 	kempld_release_mutex(pld); | 
 |  | 
 | 	return !!(status & KEMPLD_GPIO_MASK(bit)); | 
 | } | 
 |  | 
 | static int kempld_gpio_get(struct gpio_chip *chip, unsigned offset) | 
 | { | 
 | 	struct kempld_gpio_data *gpio = gpiochip_get_data(chip); | 
 | 	struct kempld_device_data *pld = gpio->pld; | 
 |  | 
 | 	return !!kempld_gpio_get_bit(pld, KEMPLD_GPIO_LVL_NUM(offset), offset); | 
 | } | 
 |  | 
 | static void kempld_gpio_set(struct gpio_chip *chip, unsigned offset, int value) | 
 | { | 
 | 	struct kempld_gpio_data *gpio = gpiochip_get_data(chip); | 
 | 	struct kempld_device_data *pld = gpio->pld; | 
 |  | 
 | 	kempld_get_mutex(pld); | 
 | 	kempld_gpio_bitop(pld, KEMPLD_GPIO_LVL_NUM(offset), offset, value); | 
 | 	kempld_release_mutex(pld); | 
 | } | 
 |  | 
 | static int kempld_gpio_direction_input(struct gpio_chip *chip, unsigned offset) | 
 | { | 
 | 	struct kempld_gpio_data *gpio = gpiochip_get_data(chip); | 
 | 	struct kempld_device_data *pld = gpio->pld; | 
 |  | 
 | 	kempld_get_mutex(pld); | 
 | 	kempld_gpio_bitop(pld, KEMPLD_GPIO_DIR_NUM(offset), offset, 0); | 
 | 	kempld_release_mutex(pld); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int kempld_gpio_direction_output(struct gpio_chip *chip, unsigned offset, | 
 | 					int value) | 
 | { | 
 | 	struct kempld_gpio_data *gpio = gpiochip_get_data(chip); | 
 | 	struct kempld_device_data *pld = gpio->pld; | 
 |  | 
 | 	kempld_get_mutex(pld); | 
 | 	kempld_gpio_bitop(pld, KEMPLD_GPIO_LVL_NUM(offset), offset, value); | 
 | 	kempld_gpio_bitop(pld, KEMPLD_GPIO_DIR_NUM(offset), offset, 1); | 
 | 	kempld_release_mutex(pld); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int kempld_gpio_get_direction(struct gpio_chip *chip, unsigned offset) | 
 | { | 
 | 	struct kempld_gpio_data *gpio = gpiochip_get_data(chip); | 
 | 	struct kempld_device_data *pld = gpio->pld; | 
 |  | 
 | 	if (kempld_gpio_get_bit(pld, KEMPLD_GPIO_DIR_NUM(offset), offset)) | 
 | 		return GPIO_LINE_DIRECTION_OUT; | 
 |  | 
 | 	return GPIO_LINE_DIRECTION_IN; | 
 | } | 
 |  | 
 | static int kempld_gpio_pincount(struct kempld_device_data *pld) | 
 | { | 
 | 	u16 evt, evt_back; | 
 |  | 
 | 	kempld_get_mutex(pld); | 
 |  | 
 | 	/* Backup event register as it might be already initialized */ | 
 | 	evt_back = kempld_read16(pld, KEMPLD_GPIO_EVT_LVL_EDGE); | 
 | 	/* Clear event register */ | 
 | 	kempld_write16(pld, KEMPLD_GPIO_EVT_LVL_EDGE, 0x0000); | 
 | 	/* Read back event register */ | 
 | 	evt = kempld_read16(pld, KEMPLD_GPIO_EVT_LVL_EDGE); | 
 | 	/* Restore event register */ | 
 | 	kempld_write16(pld, KEMPLD_GPIO_EVT_LVL_EDGE, evt_back); | 
 |  | 
 | 	kempld_release_mutex(pld); | 
 |  | 
 | 	return evt ? __ffs(evt) : 16; | 
 | } | 
 |  | 
 | static int kempld_gpio_probe(struct platform_device *pdev) | 
 | { | 
 | 	struct device *dev = &pdev->dev; | 
 | 	struct kempld_device_data *pld = dev_get_drvdata(dev->parent); | 
 | 	struct kempld_platform_data *pdata = dev_get_platdata(pld->dev); | 
 | 	struct kempld_gpio_data *gpio; | 
 | 	struct gpio_chip *chip; | 
 | 	int ret; | 
 |  | 
 | 	if (pld->info.spec_major < 2) { | 
 | 		dev_err(dev, | 
 | 			"Driver only supports GPIO devices compatible to PLD spec. rev. 2.0 or higher\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	gpio = devm_kzalloc(dev, sizeof(*gpio), GFP_KERNEL); | 
 | 	if (!gpio) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	gpio->pld = pld; | 
 |  | 
 | 	platform_set_drvdata(pdev, gpio); | 
 |  | 
 | 	chip = &gpio->chip; | 
 | 	chip->label = "gpio-kempld"; | 
 | 	chip->owner = THIS_MODULE; | 
 | 	chip->parent = dev; | 
 | 	chip->can_sleep = true; | 
 | 	if (pdata && pdata->gpio_base) | 
 | 		chip->base = pdata->gpio_base; | 
 | 	else | 
 | 		chip->base = -1; | 
 | 	chip->direction_input = kempld_gpio_direction_input; | 
 | 	chip->direction_output = kempld_gpio_direction_output; | 
 | 	chip->get_direction = kempld_gpio_get_direction; | 
 | 	chip->get = kempld_gpio_get; | 
 | 	chip->set = kempld_gpio_set; | 
 | 	chip->ngpio = kempld_gpio_pincount(pld); | 
 | 	if (chip->ngpio == 0) { | 
 | 		dev_err(dev, "No GPIO pins detected\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	ret = devm_gpiochip_add_data(dev, chip, gpio); | 
 | 	if (ret) { | 
 | 		dev_err(dev, "Could not register GPIO chip\n"); | 
 | 		return ret; | 
 | 	} | 
 |  | 
 | 	dev_info(dev, "GPIO functionality initialized with %d pins\n", | 
 | 		 chip->ngpio); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static struct platform_driver kempld_gpio_driver = { | 
 | 	.driver = { | 
 | 		.name = "kempld-gpio", | 
 | 	}, | 
 | 	.probe		= kempld_gpio_probe, | 
 | }; | 
 |  | 
 | module_platform_driver(kempld_gpio_driver); | 
 |  | 
 | MODULE_DESCRIPTION("KEM PLD GPIO Driver"); | 
 | MODULE_AUTHOR("Michael Brunner <michael.brunner@kontron.com>"); | 
 | MODULE_LICENSE("GPL"); | 
 | MODULE_ALIAS("platform:kempld-gpio"); |