162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Loongson-2F/3A/3B GPIO Support 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (c) 2008 Richard Liu, STMicroelectronics <richard.liu@st.com> 662306a36Sopenharmony_ci * Copyright (c) 2008-2010 Arnaud Patard <apatard@mandriva.com> 762306a36Sopenharmony_ci * Copyright (c) 2013 Hongbing Hu <huhb@lemote.com> 862306a36Sopenharmony_ci * Copyright (c) 2014 Huacai Chen <chenhc@lemote.com> 962306a36Sopenharmony_ci */ 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include <linux/kernel.h> 1262306a36Sopenharmony_ci#include <linux/init.h> 1362306a36Sopenharmony_ci#include <linux/module.h> 1462306a36Sopenharmony_ci#include <linux/spinlock.h> 1562306a36Sopenharmony_ci#include <linux/err.h> 1662306a36Sopenharmony_ci#include <linux/gpio/driver.h> 1762306a36Sopenharmony_ci#include <linux/platform_device.h> 1862306a36Sopenharmony_ci#include <linux/bitops.h> 1962306a36Sopenharmony_ci#include <asm/types.h> 2062306a36Sopenharmony_ci#include <loongson.h> 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ci#define STLS2F_N_GPIO 4 2362306a36Sopenharmony_ci#define STLS3A_N_GPIO 16 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci#ifdef CONFIG_CPU_LOONGSON64 2662306a36Sopenharmony_ci#define LOONGSON_N_GPIO STLS3A_N_GPIO 2762306a36Sopenharmony_ci#else 2862306a36Sopenharmony_ci#define LOONGSON_N_GPIO STLS2F_N_GPIO 2962306a36Sopenharmony_ci#endif 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_ci/* 3262306a36Sopenharmony_ci * Offset into the register where we read lines, we write them from offset 0. 3362306a36Sopenharmony_ci * This offset is the only thing that stand between us and using 3462306a36Sopenharmony_ci * GPIO_GENERIC. 3562306a36Sopenharmony_ci */ 3662306a36Sopenharmony_ci#define LOONGSON_GPIO_IN_OFFSET 16 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_cistatic DEFINE_SPINLOCK(gpio_lock); 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_cistatic int loongson_gpio_get_value(struct gpio_chip *chip, unsigned gpio) 4162306a36Sopenharmony_ci{ 4262306a36Sopenharmony_ci u32 val; 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci spin_lock(&gpio_lock); 4562306a36Sopenharmony_ci val = LOONGSON_GPIODATA; 4662306a36Sopenharmony_ci spin_unlock(&gpio_lock); 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci return !!(val & BIT(gpio + LOONGSON_GPIO_IN_OFFSET)); 4962306a36Sopenharmony_ci} 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_cistatic void loongson_gpio_set_value(struct gpio_chip *chip, 5262306a36Sopenharmony_ci unsigned gpio, int value) 5362306a36Sopenharmony_ci{ 5462306a36Sopenharmony_ci u32 val; 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci spin_lock(&gpio_lock); 5762306a36Sopenharmony_ci val = LOONGSON_GPIODATA; 5862306a36Sopenharmony_ci if (value) 5962306a36Sopenharmony_ci val |= BIT(gpio); 6062306a36Sopenharmony_ci else 6162306a36Sopenharmony_ci val &= ~BIT(gpio); 6262306a36Sopenharmony_ci LOONGSON_GPIODATA = val; 6362306a36Sopenharmony_ci spin_unlock(&gpio_lock); 6462306a36Sopenharmony_ci} 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_cistatic int loongson_gpio_direction_input(struct gpio_chip *chip, unsigned gpio) 6762306a36Sopenharmony_ci{ 6862306a36Sopenharmony_ci u32 temp; 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci spin_lock(&gpio_lock); 7162306a36Sopenharmony_ci temp = LOONGSON_GPIOIE; 7262306a36Sopenharmony_ci temp |= BIT(gpio); 7362306a36Sopenharmony_ci LOONGSON_GPIOIE = temp; 7462306a36Sopenharmony_ci spin_unlock(&gpio_lock); 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ci return 0; 7762306a36Sopenharmony_ci} 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_cistatic int loongson_gpio_direction_output(struct gpio_chip *chip, 8062306a36Sopenharmony_ci unsigned gpio, int level) 8162306a36Sopenharmony_ci{ 8262306a36Sopenharmony_ci u32 temp; 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_ci loongson_gpio_set_value(chip, gpio, level); 8562306a36Sopenharmony_ci spin_lock(&gpio_lock); 8662306a36Sopenharmony_ci temp = LOONGSON_GPIOIE; 8762306a36Sopenharmony_ci temp &= ~BIT(gpio); 8862306a36Sopenharmony_ci LOONGSON_GPIOIE = temp; 8962306a36Sopenharmony_ci spin_unlock(&gpio_lock); 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_ci return 0; 9262306a36Sopenharmony_ci} 9362306a36Sopenharmony_ci 9462306a36Sopenharmony_cistatic int loongson_gpio_probe(struct platform_device *pdev) 9562306a36Sopenharmony_ci{ 9662306a36Sopenharmony_ci struct gpio_chip *gc; 9762306a36Sopenharmony_ci struct device *dev = &pdev->dev; 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci gc = devm_kzalloc(dev, sizeof(*gc), GFP_KERNEL); 10062306a36Sopenharmony_ci if (!gc) 10162306a36Sopenharmony_ci return -ENOMEM; 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_ci gc->label = "loongson-gpio-chip"; 10462306a36Sopenharmony_ci gc->base = 0; 10562306a36Sopenharmony_ci gc->ngpio = LOONGSON_N_GPIO; 10662306a36Sopenharmony_ci gc->get = loongson_gpio_get_value; 10762306a36Sopenharmony_ci gc->set = loongson_gpio_set_value; 10862306a36Sopenharmony_ci gc->direction_input = loongson_gpio_direction_input; 10962306a36Sopenharmony_ci gc->direction_output = loongson_gpio_direction_output; 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci return gpiochip_add_data(gc, NULL); 11262306a36Sopenharmony_ci} 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_cistatic struct platform_driver loongson_gpio_driver = { 11562306a36Sopenharmony_ci .driver = { 11662306a36Sopenharmony_ci .name = "loongson-gpio", 11762306a36Sopenharmony_ci }, 11862306a36Sopenharmony_ci .probe = loongson_gpio_probe, 11962306a36Sopenharmony_ci}; 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_cistatic int __init loongson_gpio_setup(void) 12262306a36Sopenharmony_ci{ 12362306a36Sopenharmony_ci struct platform_device *pdev; 12462306a36Sopenharmony_ci int ret; 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ci ret = platform_driver_register(&loongson_gpio_driver); 12762306a36Sopenharmony_ci if (ret) { 12862306a36Sopenharmony_ci pr_err("error registering loongson GPIO driver\n"); 12962306a36Sopenharmony_ci return ret; 13062306a36Sopenharmony_ci } 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci pdev = platform_device_register_simple("loongson-gpio", -1, NULL, 0); 13362306a36Sopenharmony_ci return PTR_ERR_OR_ZERO(pdev); 13462306a36Sopenharmony_ci} 13562306a36Sopenharmony_cipostcore_initcall(loongson_gpio_setup); 136