162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Microsemi/Microchip SoCs serial gpio driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Author: Lars Povlsen <lars.povlsen@microchip.com> 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Copyright (c) 2020 Microchip Technology Inc. and its subsidiaries. 862306a36Sopenharmony_ci */ 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include <linux/bitfield.h> 1162306a36Sopenharmony_ci#include <linux/bits.h> 1262306a36Sopenharmony_ci#include <linux/clk.h> 1362306a36Sopenharmony_ci#include <linux/gpio/driver.h> 1462306a36Sopenharmony_ci#include <linux/io.h> 1562306a36Sopenharmony_ci#include <linux/mfd/ocelot.h> 1662306a36Sopenharmony_ci#include <linux/mod_devicetable.h> 1762306a36Sopenharmony_ci#include <linux/module.h> 1862306a36Sopenharmony_ci#include <linux/platform_device.h> 1962306a36Sopenharmony_ci#include <linux/property.h> 2062306a36Sopenharmony_ci#include <linux/regmap.h> 2162306a36Sopenharmony_ci#include <linux/reset.h> 2262306a36Sopenharmony_ci#include <linux/spinlock.h> 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci#include <linux/pinctrl/pinconf.h> 2562306a36Sopenharmony_ci#include <linux/pinctrl/pinmux.h> 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci#include "core.h" 2862306a36Sopenharmony_ci#include "pinconf.h" 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci#define SGPIO_BITS_PER_WORD 32 3162306a36Sopenharmony_ci#define SGPIO_MAX_BITS 4 3262306a36Sopenharmony_ci#define SGPIO_SRC_BITS 3 /* 3 bit wide field per pin */ 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_cienum { 3562306a36Sopenharmony_ci REG_INPUT_DATA, 3662306a36Sopenharmony_ci REG_PORT_CONFIG, 3762306a36Sopenharmony_ci REG_PORT_ENABLE, 3862306a36Sopenharmony_ci REG_SIO_CONFIG, 3962306a36Sopenharmony_ci REG_SIO_CLOCK, 4062306a36Sopenharmony_ci REG_INT_POLARITY, 4162306a36Sopenharmony_ci REG_INT_TRIGGER, 4262306a36Sopenharmony_ci REG_INT_ACK, 4362306a36Sopenharmony_ci REG_INT_ENABLE, 4462306a36Sopenharmony_ci REG_INT_IDENT, 4562306a36Sopenharmony_ci MAXREG 4662306a36Sopenharmony_ci}; 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_cienum { 4962306a36Sopenharmony_ci SGPIO_ARCH_LUTON, 5062306a36Sopenharmony_ci SGPIO_ARCH_OCELOT, 5162306a36Sopenharmony_ci SGPIO_ARCH_SPARX5, 5262306a36Sopenharmony_ci}; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_cienum { 5562306a36Sopenharmony_ci SGPIO_FLAGS_HAS_IRQ = BIT(0), 5662306a36Sopenharmony_ci}; 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_cistruct sgpio_properties { 5962306a36Sopenharmony_ci int arch; 6062306a36Sopenharmony_ci int flags; 6162306a36Sopenharmony_ci u8 regoff[MAXREG]; 6262306a36Sopenharmony_ci}; 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_ci#define SGPIO_LUTON_AUTO_REPEAT BIT(5) 6562306a36Sopenharmony_ci#define SGPIO_LUTON_PORT_WIDTH GENMASK(3, 2) 6662306a36Sopenharmony_ci#define SGPIO_LUTON_CLK_FREQ GENMASK(11, 0) 6762306a36Sopenharmony_ci#define SGPIO_LUTON_BIT_SOURCE GENMASK(11, 0) 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci#define SGPIO_OCELOT_AUTO_REPEAT BIT(10) 7062306a36Sopenharmony_ci#define SGPIO_OCELOT_SINGLE_SHOT BIT(11) 7162306a36Sopenharmony_ci#define SGPIO_OCELOT_PORT_WIDTH GENMASK(8, 7) 7262306a36Sopenharmony_ci#define SGPIO_OCELOT_CLK_FREQ GENMASK(19, 8) 7362306a36Sopenharmony_ci#define SGPIO_OCELOT_BIT_SOURCE GENMASK(23, 12) 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci#define SGPIO_SPARX5_AUTO_REPEAT BIT(6) 7662306a36Sopenharmony_ci#define SGPIO_SPARX5_SINGLE_SHOT BIT(7) 7762306a36Sopenharmony_ci#define SGPIO_SPARX5_PORT_WIDTH GENMASK(4, 3) 7862306a36Sopenharmony_ci#define SGPIO_SPARX5_CLK_FREQ GENMASK(19, 8) 7962306a36Sopenharmony_ci#define SGPIO_SPARX5_BIT_SOURCE GENMASK(23, 12) 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci#define SGPIO_MASTER_INTR_ENA BIT(0) 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci#define SGPIO_INT_TRG_LEVEL 0 8462306a36Sopenharmony_ci#define SGPIO_INT_TRG_EDGE 1 8562306a36Sopenharmony_ci#define SGPIO_INT_TRG_EDGE_FALL 2 8662306a36Sopenharmony_ci#define SGPIO_INT_TRG_EDGE_RISE 3 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci#define SGPIO_TRG_LEVEL_HIGH 0 8962306a36Sopenharmony_ci#define SGPIO_TRG_LEVEL_LOW 1 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cistatic const struct sgpio_properties properties_luton = { 9262306a36Sopenharmony_ci .arch = SGPIO_ARCH_LUTON, 9362306a36Sopenharmony_ci .regoff = { 0x00, 0x09, 0x29, 0x2a, 0x2b }, 9462306a36Sopenharmony_ci}; 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_cistatic const struct sgpio_properties properties_ocelot = { 9762306a36Sopenharmony_ci .arch = SGPIO_ARCH_OCELOT, 9862306a36Sopenharmony_ci .regoff = { 0x00, 0x06, 0x26, 0x04, 0x05 }, 9962306a36Sopenharmony_ci}; 10062306a36Sopenharmony_ci 10162306a36Sopenharmony_cistatic const struct sgpio_properties properties_sparx5 = { 10262306a36Sopenharmony_ci .arch = SGPIO_ARCH_SPARX5, 10362306a36Sopenharmony_ci .flags = SGPIO_FLAGS_HAS_IRQ, 10462306a36Sopenharmony_ci .regoff = { 0x00, 0x06, 0x26, 0x04, 0x05, 0x2a, 0x32, 0x3a, 0x3e, 0x42 }, 10562306a36Sopenharmony_ci}; 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_cistatic const char * const functions[] = { "gpio" }; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_cistruct sgpio_bank { 11062306a36Sopenharmony_ci struct sgpio_priv *priv; 11162306a36Sopenharmony_ci bool is_input; 11262306a36Sopenharmony_ci struct gpio_chip gpio; 11362306a36Sopenharmony_ci struct pinctrl_desc pctl_desc; 11462306a36Sopenharmony_ci}; 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_cistruct sgpio_priv { 11762306a36Sopenharmony_ci struct device *dev; 11862306a36Sopenharmony_ci struct sgpio_bank in; 11962306a36Sopenharmony_ci struct sgpio_bank out; 12062306a36Sopenharmony_ci u32 bitcount; 12162306a36Sopenharmony_ci u32 ports; 12262306a36Sopenharmony_ci u32 clock; 12362306a36Sopenharmony_ci struct regmap *regs; 12462306a36Sopenharmony_ci const struct sgpio_properties *properties; 12562306a36Sopenharmony_ci spinlock_t lock; 12662306a36Sopenharmony_ci /* protects the config register and single shot mode */ 12762306a36Sopenharmony_ci struct mutex poll_lock; 12862306a36Sopenharmony_ci}; 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_cistruct sgpio_port_addr { 13162306a36Sopenharmony_ci u8 port; 13262306a36Sopenharmony_ci u8 bit; 13362306a36Sopenharmony_ci}; 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_cistatic inline void sgpio_pin_to_addr(struct sgpio_priv *priv, int pin, 13662306a36Sopenharmony_ci struct sgpio_port_addr *addr) 13762306a36Sopenharmony_ci{ 13862306a36Sopenharmony_ci addr->port = pin / priv->bitcount; 13962306a36Sopenharmony_ci addr->bit = pin % priv->bitcount; 14062306a36Sopenharmony_ci} 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_cistatic inline int sgpio_addr_to_pin(struct sgpio_priv *priv, int port, int bit) 14362306a36Sopenharmony_ci{ 14462306a36Sopenharmony_ci return bit + port * priv->bitcount; 14562306a36Sopenharmony_ci} 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_cistatic inline u32 sgpio_get_addr(struct sgpio_priv *priv, u32 rno, u32 off) 14862306a36Sopenharmony_ci{ 14962306a36Sopenharmony_ci return (priv->properties->regoff[rno] + off) * 15062306a36Sopenharmony_ci regmap_get_reg_stride(priv->regs); 15162306a36Sopenharmony_ci} 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_cistatic u32 sgpio_readl(struct sgpio_priv *priv, u32 rno, u32 off) 15462306a36Sopenharmony_ci{ 15562306a36Sopenharmony_ci u32 addr = sgpio_get_addr(priv, rno, off); 15662306a36Sopenharmony_ci u32 val = 0; 15762306a36Sopenharmony_ci int ret; 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci ret = regmap_read(priv->regs, addr, &val); 16062306a36Sopenharmony_ci WARN_ONCE(ret, "error reading sgpio reg %d\n", ret); 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ci return val; 16362306a36Sopenharmony_ci} 16462306a36Sopenharmony_ci 16562306a36Sopenharmony_cistatic void sgpio_writel(struct sgpio_priv *priv, 16662306a36Sopenharmony_ci u32 val, u32 rno, u32 off) 16762306a36Sopenharmony_ci{ 16862306a36Sopenharmony_ci u32 addr = sgpio_get_addr(priv, rno, off); 16962306a36Sopenharmony_ci int ret; 17062306a36Sopenharmony_ci 17162306a36Sopenharmony_ci ret = regmap_write(priv->regs, addr, val); 17262306a36Sopenharmony_ci WARN_ONCE(ret, "error writing sgpio reg %d\n", ret); 17362306a36Sopenharmony_ci} 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_cistatic inline void sgpio_clrsetbits(struct sgpio_priv *priv, 17662306a36Sopenharmony_ci u32 rno, u32 off, u32 clear, u32 set) 17762306a36Sopenharmony_ci{ 17862306a36Sopenharmony_ci u32 addr = sgpio_get_addr(priv, rno, off); 17962306a36Sopenharmony_ci int ret; 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_ci ret = regmap_update_bits(priv->regs, addr, clear | set, set); 18262306a36Sopenharmony_ci WARN_ONCE(ret, "error updating sgpio reg %d\n", ret); 18362306a36Sopenharmony_ci} 18462306a36Sopenharmony_ci 18562306a36Sopenharmony_cistatic inline void sgpio_configure_bitstream(struct sgpio_priv *priv) 18662306a36Sopenharmony_ci{ 18762306a36Sopenharmony_ci int width = priv->bitcount - 1; 18862306a36Sopenharmony_ci u32 clr, set; 18962306a36Sopenharmony_ci 19062306a36Sopenharmony_ci switch (priv->properties->arch) { 19162306a36Sopenharmony_ci case SGPIO_ARCH_LUTON: 19262306a36Sopenharmony_ci clr = SGPIO_LUTON_PORT_WIDTH; 19362306a36Sopenharmony_ci set = SGPIO_LUTON_AUTO_REPEAT | 19462306a36Sopenharmony_ci FIELD_PREP(SGPIO_LUTON_PORT_WIDTH, width); 19562306a36Sopenharmony_ci break; 19662306a36Sopenharmony_ci case SGPIO_ARCH_OCELOT: 19762306a36Sopenharmony_ci clr = SGPIO_OCELOT_PORT_WIDTH; 19862306a36Sopenharmony_ci set = SGPIO_OCELOT_AUTO_REPEAT | 19962306a36Sopenharmony_ci FIELD_PREP(SGPIO_OCELOT_PORT_WIDTH, width); 20062306a36Sopenharmony_ci break; 20162306a36Sopenharmony_ci case SGPIO_ARCH_SPARX5: 20262306a36Sopenharmony_ci clr = SGPIO_SPARX5_PORT_WIDTH; 20362306a36Sopenharmony_ci set = SGPIO_SPARX5_AUTO_REPEAT | 20462306a36Sopenharmony_ci FIELD_PREP(SGPIO_SPARX5_PORT_WIDTH, width); 20562306a36Sopenharmony_ci break; 20662306a36Sopenharmony_ci default: 20762306a36Sopenharmony_ci return; 20862306a36Sopenharmony_ci } 20962306a36Sopenharmony_ci sgpio_clrsetbits(priv, REG_SIO_CONFIG, 0, clr, set); 21062306a36Sopenharmony_ci} 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_cistatic inline void sgpio_configure_clock(struct sgpio_priv *priv, u32 clkfrq) 21362306a36Sopenharmony_ci{ 21462306a36Sopenharmony_ci u32 clr, set; 21562306a36Sopenharmony_ci 21662306a36Sopenharmony_ci switch (priv->properties->arch) { 21762306a36Sopenharmony_ci case SGPIO_ARCH_LUTON: 21862306a36Sopenharmony_ci clr = SGPIO_LUTON_CLK_FREQ; 21962306a36Sopenharmony_ci set = FIELD_PREP(SGPIO_LUTON_CLK_FREQ, clkfrq); 22062306a36Sopenharmony_ci break; 22162306a36Sopenharmony_ci case SGPIO_ARCH_OCELOT: 22262306a36Sopenharmony_ci clr = SGPIO_OCELOT_CLK_FREQ; 22362306a36Sopenharmony_ci set = FIELD_PREP(SGPIO_OCELOT_CLK_FREQ, clkfrq); 22462306a36Sopenharmony_ci break; 22562306a36Sopenharmony_ci case SGPIO_ARCH_SPARX5: 22662306a36Sopenharmony_ci clr = SGPIO_SPARX5_CLK_FREQ; 22762306a36Sopenharmony_ci set = FIELD_PREP(SGPIO_SPARX5_CLK_FREQ, clkfrq); 22862306a36Sopenharmony_ci break; 22962306a36Sopenharmony_ci default: 23062306a36Sopenharmony_ci return; 23162306a36Sopenharmony_ci } 23262306a36Sopenharmony_ci sgpio_clrsetbits(priv, REG_SIO_CLOCK, 0, clr, set); 23362306a36Sopenharmony_ci} 23462306a36Sopenharmony_ci 23562306a36Sopenharmony_cistatic int sgpio_single_shot(struct sgpio_priv *priv) 23662306a36Sopenharmony_ci{ 23762306a36Sopenharmony_ci u32 addr = sgpio_get_addr(priv, REG_SIO_CONFIG, 0); 23862306a36Sopenharmony_ci int ret, ret2; 23962306a36Sopenharmony_ci u32 ctrl; 24062306a36Sopenharmony_ci unsigned int single_shot; 24162306a36Sopenharmony_ci unsigned int auto_repeat; 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_ci switch (priv->properties->arch) { 24462306a36Sopenharmony_ci case SGPIO_ARCH_LUTON: 24562306a36Sopenharmony_ci /* not supported for now */ 24662306a36Sopenharmony_ci return 0; 24762306a36Sopenharmony_ci case SGPIO_ARCH_OCELOT: 24862306a36Sopenharmony_ci single_shot = SGPIO_OCELOT_SINGLE_SHOT; 24962306a36Sopenharmony_ci auto_repeat = SGPIO_OCELOT_AUTO_REPEAT; 25062306a36Sopenharmony_ci break; 25162306a36Sopenharmony_ci case SGPIO_ARCH_SPARX5: 25262306a36Sopenharmony_ci single_shot = SGPIO_SPARX5_SINGLE_SHOT; 25362306a36Sopenharmony_ci auto_repeat = SGPIO_SPARX5_AUTO_REPEAT; 25462306a36Sopenharmony_ci break; 25562306a36Sopenharmony_ci default: 25662306a36Sopenharmony_ci return -EINVAL; 25762306a36Sopenharmony_ci } 25862306a36Sopenharmony_ci 25962306a36Sopenharmony_ci /* 26062306a36Sopenharmony_ci * Trigger immediate burst. This only works when auto repeat is turned 26162306a36Sopenharmony_ci * off. Otherwise, the single shot bit will never be cleared by the 26262306a36Sopenharmony_ci * hardware. Measurements showed that an update might take as long as 26362306a36Sopenharmony_ci * the burst gap. On a LAN9668 this is about 50ms for the largest 26462306a36Sopenharmony_ci * setting. 26562306a36Sopenharmony_ci * After the manual burst, reenable the auto repeat mode again. 26662306a36Sopenharmony_ci */ 26762306a36Sopenharmony_ci mutex_lock(&priv->poll_lock); 26862306a36Sopenharmony_ci ret = regmap_update_bits(priv->regs, addr, single_shot | auto_repeat, 26962306a36Sopenharmony_ci single_shot); 27062306a36Sopenharmony_ci if (ret) 27162306a36Sopenharmony_ci goto out; 27262306a36Sopenharmony_ci 27362306a36Sopenharmony_ci ret = regmap_read_poll_timeout(priv->regs, addr, ctrl, 27462306a36Sopenharmony_ci !(ctrl & single_shot), 100, 60000); 27562306a36Sopenharmony_ci 27662306a36Sopenharmony_ci /* reenable auto repeat mode even if there was an error */ 27762306a36Sopenharmony_ci ret2 = regmap_update_bits(priv->regs, addr, auto_repeat, auto_repeat); 27862306a36Sopenharmony_ciout: 27962306a36Sopenharmony_ci mutex_unlock(&priv->poll_lock); 28062306a36Sopenharmony_ci 28162306a36Sopenharmony_ci return ret ?: ret2; 28262306a36Sopenharmony_ci} 28362306a36Sopenharmony_ci 28462306a36Sopenharmony_cistatic int sgpio_output_set(struct sgpio_priv *priv, 28562306a36Sopenharmony_ci struct sgpio_port_addr *addr, 28662306a36Sopenharmony_ci int value) 28762306a36Sopenharmony_ci{ 28862306a36Sopenharmony_ci unsigned int bit = SGPIO_SRC_BITS * addr->bit; 28962306a36Sopenharmony_ci u32 reg = sgpio_get_addr(priv, REG_PORT_CONFIG, addr->port); 29062306a36Sopenharmony_ci bool changed; 29162306a36Sopenharmony_ci u32 clr, set; 29262306a36Sopenharmony_ci int ret; 29362306a36Sopenharmony_ci 29462306a36Sopenharmony_ci switch (priv->properties->arch) { 29562306a36Sopenharmony_ci case SGPIO_ARCH_LUTON: 29662306a36Sopenharmony_ci clr = FIELD_PREP(SGPIO_LUTON_BIT_SOURCE, BIT(bit)); 29762306a36Sopenharmony_ci set = FIELD_PREP(SGPIO_LUTON_BIT_SOURCE, value << bit); 29862306a36Sopenharmony_ci break; 29962306a36Sopenharmony_ci case SGPIO_ARCH_OCELOT: 30062306a36Sopenharmony_ci clr = FIELD_PREP(SGPIO_OCELOT_BIT_SOURCE, BIT(bit)); 30162306a36Sopenharmony_ci set = FIELD_PREP(SGPIO_OCELOT_BIT_SOURCE, value << bit); 30262306a36Sopenharmony_ci break; 30362306a36Sopenharmony_ci case SGPIO_ARCH_SPARX5: 30462306a36Sopenharmony_ci clr = FIELD_PREP(SGPIO_SPARX5_BIT_SOURCE, BIT(bit)); 30562306a36Sopenharmony_ci set = FIELD_PREP(SGPIO_SPARX5_BIT_SOURCE, value << bit); 30662306a36Sopenharmony_ci break; 30762306a36Sopenharmony_ci default: 30862306a36Sopenharmony_ci return -EINVAL; 30962306a36Sopenharmony_ci } 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_ci ret = regmap_update_bits_check(priv->regs, reg, clr | set, set, 31262306a36Sopenharmony_ci &changed); 31362306a36Sopenharmony_ci if (ret) 31462306a36Sopenharmony_ci return ret; 31562306a36Sopenharmony_ci 31662306a36Sopenharmony_ci if (changed) { 31762306a36Sopenharmony_ci ret = sgpio_single_shot(priv); 31862306a36Sopenharmony_ci if (ret) 31962306a36Sopenharmony_ci return ret; 32062306a36Sopenharmony_ci } 32162306a36Sopenharmony_ci 32262306a36Sopenharmony_ci return 0; 32362306a36Sopenharmony_ci} 32462306a36Sopenharmony_ci 32562306a36Sopenharmony_cistatic int sgpio_output_get(struct sgpio_priv *priv, 32662306a36Sopenharmony_ci struct sgpio_port_addr *addr) 32762306a36Sopenharmony_ci{ 32862306a36Sopenharmony_ci u32 val, portval = sgpio_readl(priv, REG_PORT_CONFIG, addr->port); 32962306a36Sopenharmony_ci unsigned int bit = SGPIO_SRC_BITS * addr->bit; 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_ci switch (priv->properties->arch) { 33262306a36Sopenharmony_ci case SGPIO_ARCH_LUTON: 33362306a36Sopenharmony_ci val = FIELD_GET(SGPIO_LUTON_BIT_SOURCE, portval); 33462306a36Sopenharmony_ci break; 33562306a36Sopenharmony_ci case SGPIO_ARCH_OCELOT: 33662306a36Sopenharmony_ci val = FIELD_GET(SGPIO_OCELOT_BIT_SOURCE, portval); 33762306a36Sopenharmony_ci break; 33862306a36Sopenharmony_ci case SGPIO_ARCH_SPARX5: 33962306a36Sopenharmony_ci val = FIELD_GET(SGPIO_SPARX5_BIT_SOURCE, portval); 34062306a36Sopenharmony_ci break; 34162306a36Sopenharmony_ci default: 34262306a36Sopenharmony_ci val = 0; 34362306a36Sopenharmony_ci break; 34462306a36Sopenharmony_ci } 34562306a36Sopenharmony_ci return !!(val & BIT(bit)); 34662306a36Sopenharmony_ci} 34762306a36Sopenharmony_ci 34862306a36Sopenharmony_cistatic int sgpio_input_get(struct sgpio_priv *priv, 34962306a36Sopenharmony_ci struct sgpio_port_addr *addr) 35062306a36Sopenharmony_ci{ 35162306a36Sopenharmony_ci return !!(sgpio_readl(priv, REG_INPUT_DATA, addr->bit) & BIT(addr->port)); 35262306a36Sopenharmony_ci} 35362306a36Sopenharmony_ci 35462306a36Sopenharmony_cistatic int sgpio_pinconf_get(struct pinctrl_dev *pctldev, 35562306a36Sopenharmony_ci unsigned int pin, unsigned long *config) 35662306a36Sopenharmony_ci{ 35762306a36Sopenharmony_ci struct sgpio_bank *bank = pinctrl_dev_get_drvdata(pctldev); 35862306a36Sopenharmony_ci u32 param = pinconf_to_config_param(*config); 35962306a36Sopenharmony_ci struct sgpio_priv *priv = bank->priv; 36062306a36Sopenharmony_ci struct sgpio_port_addr addr; 36162306a36Sopenharmony_ci int val; 36262306a36Sopenharmony_ci 36362306a36Sopenharmony_ci sgpio_pin_to_addr(priv, pin, &addr); 36462306a36Sopenharmony_ci 36562306a36Sopenharmony_ci switch (param) { 36662306a36Sopenharmony_ci case PIN_CONFIG_INPUT_ENABLE: 36762306a36Sopenharmony_ci val = bank->is_input; 36862306a36Sopenharmony_ci break; 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_ci case PIN_CONFIG_OUTPUT_ENABLE: 37162306a36Sopenharmony_ci val = !bank->is_input; 37262306a36Sopenharmony_ci break; 37362306a36Sopenharmony_ci 37462306a36Sopenharmony_ci case PIN_CONFIG_OUTPUT: 37562306a36Sopenharmony_ci if (bank->is_input) 37662306a36Sopenharmony_ci return -EINVAL; 37762306a36Sopenharmony_ci val = sgpio_output_get(priv, &addr); 37862306a36Sopenharmony_ci break; 37962306a36Sopenharmony_ci 38062306a36Sopenharmony_ci default: 38162306a36Sopenharmony_ci return -ENOTSUPP; 38262306a36Sopenharmony_ci } 38362306a36Sopenharmony_ci 38462306a36Sopenharmony_ci *config = pinconf_to_config_packed(param, val); 38562306a36Sopenharmony_ci 38662306a36Sopenharmony_ci return 0; 38762306a36Sopenharmony_ci} 38862306a36Sopenharmony_ci 38962306a36Sopenharmony_cistatic int sgpio_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin, 39062306a36Sopenharmony_ci unsigned long *configs, unsigned int num_configs) 39162306a36Sopenharmony_ci{ 39262306a36Sopenharmony_ci struct sgpio_bank *bank = pinctrl_dev_get_drvdata(pctldev); 39362306a36Sopenharmony_ci struct sgpio_priv *priv = bank->priv; 39462306a36Sopenharmony_ci struct sgpio_port_addr addr; 39562306a36Sopenharmony_ci int cfg, err = 0; 39662306a36Sopenharmony_ci u32 param, arg; 39762306a36Sopenharmony_ci 39862306a36Sopenharmony_ci sgpio_pin_to_addr(priv, pin, &addr); 39962306a36Sopenharmony_ci 40062306a36Sopenharmony_ci for (cfg = 0; cfg < num_configs; cfg++) { 40162306a36Sopenharmony_ci param = pinconf_to_config_param(configs[cfg]); 40262306a36Sopenharmony_ci arg = pinconf_to_config_argument(configs[cfg]); 40362306a36Sopenharmony_ci 40462306a36Sopenharmony_ci switch (param) { 40562306a36Sopenharmony_ci case PIN_CONFIG_OUTPUT: 40662306a36Sopenharmony_ci if (bank->is_input) 40762306a36Sopenharmony_ci return -EINVAL; 40862306a36Sopenharmony_ci err = sgpio_output_set(priv, &addr, arg); 40962306a36Sopenharmony_ci break; 41062306a36Sopenharmony_ci 41162306a36Sopenharmony_ci default: 41262306a36Sopenharmony_ci err = -ENOTSUPP; 41362306a36Sopenharmony_ci } 41462306a36Sopenharmony_ci } 41562306a36Sopenharmony_ci 41662306a36Sopenharmony_ci return err; 41762306a36Sopenharmony_ci} 41862306a36Sopenharmony_ci 41962306a36Sopenharmony_cistatic const struct pinconf_ops sgpio_confops = { 42062306a36Sopenharmony_ci .is_generic = true, 42162306a36Sopenharmony_ci .pin_config_get = sgpio_pinconf_get, 42262306a36Sopenharmony_ci .pin_config_set = sgpio_pinconf_set, 42362306a36Sopenharmony_ci .pin_config_config_dbg_show = pinconf_generic_dump_config, 42462306a36Sopenharmony_ci}; 42562306a36Sopenharmony_ci 42662306a36Sopenharmony_cistatic int sgpio_get_functions_count(struct pinctrl_dev *pctldev) 42762306a36Sopenharmony_ci{ 42862306a36Sopenharmony_ci return 1; 42962306a36Sopenharmony_ci} 43062306a36Sopenharmony_ci 43162306a36Sopenharmony_cistatic const char *sgpio_get_function_name(struct pinctrl_dev *pctldev, 43262306a36Sopenharmony_ci unsigned int function) 43362306a36Sopenharmony_ci{ 43462306a36Sopenharmony_ci return functions[0]; 43562306a36Sopenharmony_ci} 43662306a36Sopenharmony_ci 43762306a36Sopenharmony_cistatic int sgpio_get_function_groups(struct pinctrl_dev *pctldev, 43862306a36Sopenharmony_ci unsigned int function, 43962306a36Sopenharmony_ci const char *const **groups, 44062306a36Sopenharmony_ci unsigned *const num_groups) 44162306a36Sopenharmony_ci{ 44262306a36Sopenharmony_ci *groups = functions; 44362306a36Sopenharmony_ci *num_groups = ARRAY_SIZE(functions); 44462306a36Sopenharmony_ci 44562306a36Sopenharmony_ci return 0; 44662306a36Sopenharmony_ci} 44762306a36Sopenharmony_ci 44862306a36Sopenharmony_cistatic int sgpio_pinmux_set_mux(struct pinctrl_dev *pctldev, 44962306a36Sopenharmony_ci unsigned int selector, unsigned int group) 45062306a36Sopenharmony_ci{ 45162306a36Sopenharmony_ci return 0; 45262306a36Sopenharmony_ci} 45362306a36Sopenharmony_ci 45462306a36Sopenharmony_cistatic int sgpio_gpio_set_direction(struct pinctrl_dev *pctldev, 45562306a36Sopenharmony_ci struct pinctrl_gpio_range *range, 45662306a36Sopenharmony_ci unsigned int pin, bool input) 45762306a36Sopenharmony_ci{ 45862306a36Sopenharmony_ci struct sgpio_bank *bank = pinctrl_dev_get_drvdata(pctldev); 45962306a36Sopenharmony_ci 46062306a36Sopenharmony_ci return (input == bank->is_input) ? 0 : -EINVAL; 46162306a36Sopenharmony_ci} 46262306a36Sopenharmony_ci 46362306a36Sopenharmony_cistatic int sgpio_gpio_request_enable(struct pinctrl_dev *pctldev, 46462306a36Sopenharmony_ci struct pinctrl_gpio_range *range, 46562306a36Sopenharmony_ci unsigned int offset) 46662306a36Sopenharmony_ci{ 46762306a36Sopenharmony_ci struct sgpio_bank *bank = pinctrl_dev_get_drvdata(pctldev); 46862306a36Sopenharmony_ci struct sgpio_priv *priv = bank->priv; 46962306a36Sopenharmony_ci struct sgpio_port_addr addr; 47062306a36Sopenharmony_ci 47162306a36Sopenharmony_ci sgpio_pin_to_addr(priv, offset, &addr); 47262306a36Sopenharmony_ci 47362306a36Sopenharmony_ci if ((priv->ports & BIT(addr.port)) == 0) { 47462306a36Sopenharmony_ci dev_warn(priv->dev, "Request port %d.%d: Port is not enabled\n", 47562306a36Sopenharmony_ci addr.port, addr.bit); 47662306a36Sopenharmony_ci return -EINVAL; 47762306a36Sopenharmony_ci } 47862306a36Sopenharmony_ci 47962306a36Sopenharmony_ci return 0; 48062306a36Sopenharmony_ci} 48162306a36Sopenharmony_ci 48262306a36Sopenharmony_cistatic const struct pinmux_ops sgpio_pmx_ops = { 48362306a36Sopenharmony_ci .get_functions_count = sgpio_get_functions_count, 48462306a36Sopenharmony_ci .get_function_name = sgpio_get_function_name, 48562306a36Sopenharmony_ci .get_function_groups = sgpio_get_function_groups, 48662306a36Sopenharmony_ci .set_mux = sgpio_pinmux_set_mux, 48762306a36Sopenharmony_ci .gpio_set_direction = sgpio_gpio_set_direction, 48862306a36Sopenharmony_ci .gpio_request_enable = sgpio_gpio_request_enable, 48962306a36Sopenharmony_ci}; 49062306a36Sopenharmony_ci 49162306a36Sopenharmony_cistatic int sgpio_pctl_get_groups_count(struct pinctrl_dev *pctldev) 49262306a36Sopenharmony_ci{ 49362306a36Sopenharmony_ci struct sgpio_bank *bank = pinctrl_dev_get_drvdata(pctldev); 49462306a36Sopenharmony_ci 49562306a36Sopenharmony_ci return bank->pctl_desc.npins; 49662306a36Sopenharmony_ci} 49762306a36Sopenharmony_ci 49862306a36Sopenharmony_cistatic const char *sgpio_pctl_get_group_name(struct pinctrl_dev *pctldev, 49962306a36Sopenharmony_ci unsigned int group) 50062306a36Sopenharmony_ci{ 50162306a36Sopenharmony_ci struct sgpio_bank *bank = pinctrl_dev_get_drvdata(pctldev); 50262306a36Sopenharmony_ci 50362306a36Sopenharmony_ci return bank->pctl_desc.pins[group].name; 50462306a36Sopenharmony_ci} 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_cistatic int sgpio_pctl_get_group_pins(struct pinctrl_dev *pctldev, 50762306a36Sopenharmony_ci unsigned int group, 50862306a36Sopenharmony_ci const unsigned int **pins, 50962306a36Sopenharmony_ci unsigned int *num_pins) 51062306a36Sopenharmony_ci{ 51162306a36Sopenharmony_ci struct sgpio_bank *bank = pinctrl_dev_get_drvdata(pctldev); 51262306a36Sopenharmony_ci 51362306a36Sopenharmony_ci *pins = &bank->pctl_desc.pins[group].number; 51462306a36Sopenharmony_ci *num_pins = 1; 51562306a36Sopenharmony_ci 51662306a36Sopenharmony_ci return 0; 51762306a36Sopenharmony_ci} 51862306a36Sopenharmony_ci 51962306a36Sopenharmony_cistatic const struct pinctrl_ops sgpio_pctl_ops = { 52062306a36Sopenharmony_ci .get_groups_count = sgpio_pctl_get_groups_count, 52162306a36Sopenharmony_ci .get_group_name = sgpio_pctl_get_group_name, 52262306a36Sopenharmony_ci .get_group_pins = sgpio_pctl_get_group_pins, 52362306a36Sopenharmony_ci .dt_node_to_map = pinconf_generic_dt_node_to_map_pin, 52462306a36Sopenharmony_ci .dt_free_map = pinconf_generic_dt_free_map, 52562306a36Sopenharmony_ci}; 52662306a36Sopenharmony_ci 52762306a36Sopenharmony_cistatic int microchip_sgpio_direction_input(struct gpio_chip *gc, unsigned int gpio) 52862306a36Sopenharmony_ci{ 52962306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(gc); 53062306a36Sopenharmony_ci 53162306a36Sopenharmony_ci /* Fixed-position function */ 53262306a36Sopenharmony_ci return bank->is_input ? 0 : -EINVAL; 53362306a36Sopenharmony_ci} 53462306a36Sopenharmony_ci 53562306a36Sopenharmony_cistatic int microchip_sgpio_direction_output(struct gpio_chip *gc, 53662306a36Sopenharmony_ci unsigned int gpio, int value) 53762306a36Sopenharmony_ci{ 53862306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(gc); 53962306a36Sopenharmony_ci struct sgpio_priv *priv = bank->priv; 54062306a36Sopenharmony_ci struct sgpio_port_addr addr; 54162306a36Sopenharmony_ci 54262306a36Sopenharmony_ci /* Fixed-position function */ 54362306a36Sopenharmony_ci if (bank->is_input) 54462306a36Sopenharmony_ci return -EINVAL; 54562306a36Sopenharmony_ci 54662306a36Sopenharmony_ci sgpio_pin_to_addr(priv, gpio, &addr); 54762306a36Sopenharmony_ci 54862306a36Sopenharmony_ci return sgpio_output_set(priv, &addr, value); 54962306a36Sopenharmony_ci} 55062306a36Sopenharmony_ci 55162306a36Sopenharmony_cistatic int microchip_sgpio_get_direction(struct gpio_chip *gc, unsigned int gpio) 55262306a36Sopenharmony_ci{ 55362306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(gc); 55462306a36Sopenharmony_ci 55562306a36Sopenharmony_ci return bank->is_input ? GPIO_LINE_DIRECTION_IN : GPIO_LINE_DIRECTION_OUT; 55662306a36Sopenharmony_ci} 55762306a36Sopenharmony_ci 55862306a36Sopenharmony_cistatic void microchip_sgpio_set_value(struct gpio_chip *gc, 55962306a36Sopenharmony_ci unsigned int gpio, int value) 56062306a36Sopenharmony_ci{ 56162306a36Sopenharmony_ci microchip_sgpio_direction_output(gc, gpio, value); 56262306a36Sopenharmony_ci} 56362306a36Sopenharmony_ci 56462306a36Sopenharmony_cistatic int microchip_sgpio_get_value(struct gpio_chip *gc, unsigned int gpio) 56562306a36Sopenharmony_ci{ 56662306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(gc); 56762306a36Sopenharmony_ci struct sgpio_priv *priv = bank->priv; 56862306a36Sopenharmony_ci struct sgpio_port_addr addr; 56962306a36Sopenharmony_ci 57062306a36Sopenharmony_ci sgpio_pin_to_addr(priv, gpio, &addr); 57162306a36Sopenharmony_ci 57262306a36Sopenharmony_ci return bank->is_input ? sgpio_input_get(priv, &addr) : sgpio_output_get(priv, &addr); 57362306a36Sopenharmony_ci} 57462306a36Sopenharmony_ci 57562306a36Sopenharmony_cistatic int microchip_sgpio_of_xlate(struct gpio_chip *gc, 57662306a36Sopenharmony_ci const struct of_phandle_args *gpiospec, 57762306a36Sopenharmony_ci u32 *flags) 57862306a36Sopenharmony_ci{ 57962306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(gc); 58062306a36Sopenharmony_ci struct sgpio_priv *priv = bank->priv; 58162306a36Sopenharmony_ci int pin; 58262306a36Sopenharmony_ci 58362306a36Sopenharmony_ci /* 58462306a36Sopenharmony_ci * Note that the SGIO pin is defined by *2* numbers, a port 58562306a36Sopenharmony_ci * number between 0 and 31, and a bit index, 0 to 3. 58662306a36Sopenharmony_ci */ 58762306a36Sopenharmony_ci if (gpiospec->args[0] > SGPIO_BITS_PER_WORD || 58862306a36Sopenharmony_ci gpiospec->args[1] > priv->bitcount) 58962306a36Sopenharmony_ci return -EINVAL; 59062306a36Sopenharmony_ci 59162306a36Sopenharmony_ci pin = sgpio_addr_to_pin(priv, gpiospec->args[0], gpiospec->args[1]); 59262306a36Sopenharmony_ci 59362306a36Sopenharmony_ci if (pin > gc->ngpio) 59462306a36Sopenharmony_ci return -EINVAL; 59562306a36Sopenharmony_ci 59662306a36Sopenharmony_ci if (flags) 59762306a36Sopenharmony_ci *flags = gpiospec->args[2]; 59862306a36Sopenharmony_ci 59962306a36Sopenharmony_ci return pin; 60062306a36Sopenharmony_ci} 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_cistatic int microchip_sgpio_get_ports(struct sgpio_priv *priv) 60362306a36Sopenharmony_ci{ 60462306a36Sopenharmony_ci const char *range_property_name = "microchip,sgpio-port-ranges"; 60562306a36Sopenharmony_ci struct device *dev = priv->dev; 60662306a36Sopenharmony_ci u32 range_params[64]; 60762306a36Sopenharmony_ci int i, nranges, ret; 60862306a36Sopenharmony_ci 60962306a36Sopenharmony_ci /* Calculate port mask */ 61062306a36Sopenharmony_ci nranges = device_property_count_u32(dev, range_property_name); 61162306a36Sopenharmony_ci if (nranges < 2 || nranges % 2 || nranges > ARRAY_SIZE(range_params)) { 61262306a36Sopenharmony_ci dev_err(dev, "%s port range: '%s' property\n", 61362306a36Sopenharmony_ci nranges == -EINVAL ? "Missing" : "Invalid", 61462306a36Sopenharmony_ci range_property_name); 61562306a36Sopenharmony_ci return -EINVAL; 61662306a36Sopenharmony_ci } 61762306a36Sopenharmony_ci 61862306a36Sopenharmony_ci ret = device_property_read_u32_array(dev, range_property_name, 61962306a36Sopenharmony_ci range_params, nranges); 62062306a36Sopenharmony_ci if (ret) { 62162306a36Sopenharmony_ci dev_err(dev, "failed to parse '%s' property: %d\n", 62262306a36Sopenharmony_ci range_property_name, ret); 62362306a36Sopenharmony_ci return ret; 62462306a36Sopenharmony_ci } 62562306a36Sopenharmony_ci for (i = 0; i < nranges; i += 2) { 62662306a36Sopenharmony_ci int start, end; 62762306a36Sopenharmony_ci 62862306a36Sopenharmony_ci start = range_params[i]; 62962306a36Sopenharmony_ci end = range_params[i + 1]; 63062306a36Sopenharmony_ci if (start > end || end >= SGPIO_BITS_PER_WORD) { 63162306a36Sopenharmony_ci dev_err(dev, "Ill-formed port-range [%d:%d]\n", 63262306a36Sopenharmony_ci start, end); 63362306a36Sopenharmony_ci } 63462306a36Sopenharmony_ci priv->ports |= GENMASK(end, start); 63562306a36Sopenharmony_ci } 63662306a36Sopenharmony_ci 63762306a36Sopenharmony_ci return 0; 63862306a36Sopenharmony_ci} 63962306a36Sopenharmony_ci 64062306a36Sopenharmony_cistatic void microchip_sgpio_irq_settype(struct irq_data *data, 64162306a36Sopenharmony_ci int type, 64262306a36Sopenharmony_ci int polarity) 64362306a36Sopenharmony_ci{ 64462306a36Sopenharmony_ci struct gpio_chip *chip = irq_data_get_irq_chip_data(data); 64562306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(chip); 64662306a36Sopenharmony_ci unsigned int gpio = irqd_to_hwirq(data); 64762306a36Sopenharmony_ci struct sgpio_port_addr addr; 64862306a36Sopenharmony_ci unsigned long flags; 64962306a36Sopenharmony_ci u32 ena; 65062306a36Sopenharmony_ci 65162306a36Sopenharmony_ci sgpio_pin_to_addr(bank->priv, gpio, &addr); 65262306a36Sopenharmony_ci 65362306a36Sopenharmony_ci spin_lock_irqsave(&bank->priv->lock, flags); 65462306a36Sopenharmony_ci 65562306a36Sopenharmony_ci /* Disable interrupt while changing type */ 65662306a36Sopenharmony_ci ena = sgpio_readl(bank->priv, REG_INT_ENABLE, addr.bit); 65762306a36Sopenharmony_ci sgpio_writel(bank->priv, ena & ~BIT(addr.port), REG_INT_ENABLE, addr.bit); 65862306a36Sopenharmony_ci 65962306a36Sopenharmony_ci /* Type value spread over 2 registers sets: low, high bit */ 66062306a36Sopenharmony_ci sgpio_clrsetbits(bank->priv, REG_INT_TRIGGER, addr.bit, 66162306a36Sopenharmony_ci BIT(addr.port), (!!(type & 0x1)) << addr.port); 66262306a36Sopenharmony_ci sgpio_clrsetbits(bank->priv, REG_INT_TRIGGER, SGPIO_MAX_BITS + addr.bit, 66362306a36Sopenharmony_ci BIT(addr.port), (!!(type & 0x2)) << addr.port); 66462306a36Sopenharmony_ci 66562306a36Sopenharmony_ci if (type == SGPIO_INT_TRG_LEVEL) 66662306a36Sopenharmony_ci sgpio_clrsetbits(bank->priv, REG_INT_POLARITY, addr.bit, 66762306a36Sopenharmony_ci BIT(addr.port), polarity << addr.port); 66862306a36Sopenharmony_ci 66962306a36Sopenharmony_ci /* Possibly re-enable interrupts */ 67062306a36Sopenharmony_ci sgpio_writel(bank->priv, ena, REG_INT_ENABLE, addr.bit); 67162306a36Sopenharmony_ci 67262306a36Sopenharmony_ci spin_unlock_irqrestore(&bank->priv->lock, flags); 67362306a36Sopenharmony_ci} 67462306a36Sopenharmony_ci 67562306a36Sopenharmony_cistatic void microchip_sgpio_irq_setreg(struct irq_data *data, 67662306a36Sopenharmony_ci int reg, 67762306a36Sopenharmony_ci bool clear) 67862306a36Sopenharmony_ci{ 67962306a36Sopenharmony_ci struct gpio_chip *chip = irq_data_get_irq_chip_data(data); 68062306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(chip); 68162306a36Sopenharmony_ci unsigned int gpio = irqd_to_hwirq(data); 68262306a36Sopenharmony_ci struct sgpio_port_addr addr; 68362306a36Sopenharmony_ci 68462306a36Sopenharmony_ci sgpio_pin_to_addr(bank->priv, gpio, &addr); 68562306a36Sopenharmony_ci 68662306a36Sopenharmony_ci if (clear) 68762306a36Sopenharmony_ci sgpio_clrsetbits(bank->priv, reg, addr.bit, BIT(addr.port), 0); 68862306a36Sopenharmony_ci else 68962306a36Sopenharmony_ci sgpio_clrsetbits(bank->priv, reg, addr.bit, 0, BIT(addr.port)); 69062306a36Sopenharmony_ci} 69162306a36Sopenharmony_ci 69262306a36Sopenharmony_cistatic void microchip_sgpio_irq_mask(struct irq_data *data) 69362306a36Sopenharmony_ci{ 69462306a36Sopenharmony_ci struct gpio_chip *chip = irq_data_get_irq_chip_data(data); 69562306a36Sopenharmony_ci 69662306a36Sopenharmony_ci microchip_sgpio_irq_setreg(data, REG_INT_ENABLE, true); 69762306a36Sopenharmony_ci gpiochip_disable_irq(chip, data->hwirq); 69862306a36Sopenharmony_ci} 69962306a36Sopenharmony_ci 70062306a36Sopenharmony_cistatic void microchip_sgpio_irq_unmask(struct irq_data *data) 70162306a36Sopenharmony_ci{ 70262306a36Sopenharmony_ci struct gpio_chip *chip = irq_data_get_irq_chip_data(data); 70362306a36Sopenharmony_ci 70462306a36Sopenharmony_ci gpiochip_enable_irq(chip, data->hwirq); 70562306a36Sopenharmony_ci microchip_sgpio_irq_setreg(data, REG_INT_ENABLE, false); 70662306a36Sopenharmony_ci} 70762306a36Sopenharmony_ci 70862306a36Sopenharmony_cistatic void microchip_sgpio_irq_ack(struct irq_data *data) 70962306a36Sopenharmony_ci{ 71062306a36Sopenharmony_ci struct gpio_chip *chip = irq_data_get_irq_chip_data(data); 71162306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(chip); 71262306a36Sopenharmony_ci unsigned int gpio = irqd_to_hwirq(data); 71362306a36Sopenharmony_ci struct sgpio_port_addr addr; 71462306a36Sopenharmony_ci 71562306a36Sopenharmony_ci sgpio_pin_to_addr(bank->priv, gpio, &addr); 71662306a36Sopenharmony_ci 71762306a36Sopenharmony_ci sgpio_writel(bank->priv, BIT(addr.port), REG_INT_ACK, addr.bit); 71862306a36Sopenharmony_ci} 71962306a36Sopenharmony_ci 72062306a36Sopenharmony_cistatic int microchip_sgpio_irq_set_type(struct irq_data *data, unsigned int type) 72162306a36Sopenharmony_ci{ 72262306a36Sopenharmony_ci switch (type) { 72362306a36Sopenharmony_ci case IRQ_TYPE_EDGE_BOTH: 72462306a36Sopenharmony_ci irq_set_handler_locked(data, handle_edge_irq); 72562306a36Sopenharmony_ci microchip_sgpio_irq_settype(data, SGPIO_INT_TRG_EDGE, 0); 72662306a36Sopenharmony_ci break; 72762306a36Sopenharmony_ci case IRQ_TYPE_EDGE_RISING: 72862306a36Sopenharmony_ci irq_set_handler_locked(data, handle_edge_irq); 72962306a36Sopenharmony_ci microchip_sgpio_irq_settype(data, SGPIO_INT_TRG_EDGE_RISE, 0); 73062306a36Sopenharmony_ci break; 73162306a36Sopenharmony_ci case IRQ_TYPE_EDGE_FALLING: 73262306a36Sopenharmony_ci irq_set_handler_locked(data, handle_edge_irq); 73362306a36Sopenharmony_ci microchip_sgpio_irq_settype(data, SGPIO_INT_TRG_EDGE_FALL, 0); 73462306a36Sopenharmony_ci break; 73562306a36Sopenharmony_ci case IRQ_TYPE_LEVEL_HIGH: 73662306a36Sopenharmony_ci irq_set_handler_locked(data, handle_level_irq); 73762306a36Sopenharmony_ci microchip_sgpio_irq_settype(data, SGPIO_INT_TRG_LEVEL, SGPIO_TRG_LEVEL_HIGH); 73862306a36Sopenharmony_ci break; 73962306a36Sopenharmony_ci case IRQ_TYPE_LEVEL_LOW: 74062306a36Sopenharmony_ci irq_set_handler_locked(data, handle_level_irq); 74162306a36Sopenharmony_ci microchip_sgpio_irq_settype(data, SGPIO_INT_TRG_LEVEL, SGPIO_TRG_LEVEL_LOW); 74262306a36Sopenharmony_ci break; 74362306a36Sopenharmony_ci default: 74462306a36Sopenharmony_ci return -EINVAL; 74562306a36Sopenharmony_ci } 74662306a36Sopenharmony_ci 74762306a36Sopenharmony_ci return 0; 74862306a36Sopenharmony_ci} 74962306a36Sopenharmony_ci 75062306a36Sopenharmony_cistatic const struct irq_chip microchip_sgpio_irqchip = { 75162306a36Sopenharmony_ci .name = "gpio", 75262306a36Sopenharmony_ci .irq_mask = microchip_sgpio_irq_mask, 75362306a36Sopenharmony_ci .irq_ack = microchip_sgpio_irq_ack, 75462306a36Sopenharmony_ci .irq_unmask = microchip_sgpio_irq_unmask, 75562306a36Sopenharmony_ci .irq_set_type = microchip_sgpio_irq_set_type, 75662306a36Sopenharmony_ci .flags = IRQCHIP_IMMUTABLE, 75762306a36Sopenharmony_ci GPIOCHIP_IRQ_RESOURCE_HELPERS, 75862306a36Sopenharmony_ci}; 75962306a36Sopenharmony_ci 76062306a36Sopenharmony_cistatic void sgpio_irq_handler(struct irq_desc *desc) 76162306a36Sopenharmony_ci{ 76262306a36Sopenharmony_ci struct irq_chip *parent_chip = irq_desc_get_chip(desc); 76362306a36Sopenharmony_ci struct gpio_chip *chip = irq_desc_get_handler_data(desc); 76462306a36Sopenharmony_ci struct sgpio_bank *bank = gpiochip_get_data(chip); 76562306a36Sopenharmony_ci struct sgpio_priv *priv = bank->priv; 76662306a36Sopenharmony_ci int bit, port, gpio; 76762306a36Sopenharmony_ci long val; 76862306a36Sopenharmony_ci 76962306a36Sopenharmony_ci for (bit = 0; bit < priv->bitcount; bit++) { 77062306a36Sopenharmony_ci val = sgpio_readl(priv, REG_INT_IDENT, bit); 77162306a36Sopenharmony_ci if (!val) 77262306a36Sopenharmony_ci continue; 77362306a36Sopenharmony_ci 77462306a36Sopenharmony_ci chained_irq_enter(parent_chip, desc); 77562306a36Sopenharmony_ci 77662306a36Sopenharmony_ci for_each_set_bit(port, &val, SGPIO_BITS_PER_WORD) { 77762306a36Sopenharmony_ci gpio = sgpio_addr_to_pin(priv, port, bit); 77862306a36Sopenharmony_ci generic_handle_domain_irq(chip->irq.domain, gpio); 77962306a36Sopenharmony_ci } 78062306a36Sopenharmony_ci 78162306a36Sopenharmony_ci chained_irq_exit(parent_chip, desc); 78262306a36Sopenharmony_ci } 78362306a36Sopenharmony_ci} 78462306a36Sopenharmony_ci 78562306a36Sopenharmony_cistatic int microchip_sgpio_register_bank(struct device *dev, 78662306a36Sopenharmony_ci struct sgpio_priv *priv, 78762306a36Sopenharmony_ci struct fwnode_handle *fwnode, 78862306a36Sopenharmony_ci int bankno) 78962306a36Sopenharmony_ci{ 79062306a36Sopenharmony_ci struct pinctrl_pin_desc *pins; 79162306a36Sopenharmony_ci struct pinctrl_desc *pctl_desc; 79262306a36Sopenharmony_ci struct pinctrl_dev *pctldev; 79362306a36Sopenharmony_ci struct sgpio_bank *bank; 79462306a36Sopenharmony_ci struct gpio_chip *gc; 79562306a36Sopenharmony_ci u32 ngpios; 79662306a36Sopenharmony_ci int i, ret; 79762306a36Sopenharmony_ci 79862306a36Sopenharmony_ci /* Get overall bank struct */ 79962306a36Sopenharmony_ci bank = (bankno == 0) ? &priv->in : &priv->out; 80062306a36Sopenharmony_ci bank->priv = priv; 80162306a36Sopenharmony_ci 80262306a36Sopenharmony_ci if (fwnode_property_read_u32(fwnode, "ngpios", &ngpios)) { 80362306a36Sopenharmony_ci dev_info(dev, "failed to get number of gpios for bank%d\n", 80462306a36Sopenharmony_ci bankno); 80562306a36Sopenharmony_ci ngpios = 64; 80662306a36Sopenharmony_ci } 80762306a36Sopenharmony_ci 80862306a36Sopenharmony_ci priv->bitcount = ngpios / SGPIO_BITS_PER_WORD; 80962306a36Sopenharmony_ci if (priv->bitcount > SGPIO_MAX_BITS) { 81062306a36Sopenharmony_ci dev_err(dev, "Bit width exceeds maximum (%d)\n", 81162306a36Sopenharmony_ci SGPIO_MAX_BITS); 81262306a36Sopenharmony_ci return -EINVAL; 81362306a36Sopenharmony_ci } 81462306a36Sopenharmony_ci 81562306a36Sopenharmony_ci pctl_desc = &bank->pctl_desc; 81662306a36Sopenharmony_ci pctl_desc->name = devm_kasprintf(dev, GFP_KERNEL, "%s-%sput", 81762306a36Sopenharmony_ci dev_name(dev), 81862306a36Sopenharmony_ci bank->is_input ? "in" : "out"); 81962306a36Sopenharmony_ci if (!pctl_desc->name) 82062306a36Sopenharmony_ci return -ENOMEM; 82162306a36Sopenharmony_ci 82262306a36Sopenharmony_ci pctl_desc->pctlops = &sgpio_pctl_ops; 82362306a36Sopenharmony_ci pctl_desc->pmxops = &sgpio_pmx_ops; 82462306a36Sopenharmony_ci pctl_desc->confops = &sgpio_confops; 82562306a36Sopenharmony_ci pctl_desc->owner = THIS_MODULE; 82662306a36Sopenharmony_ci 82762306a36Sopenharmony_ci pins = devm_kzalloc(dev, sizeof(*pins)*ngpios, GFP_KERNEL); 82862306a36Sopenharmony_ci if (!pins) 82962306a36Sopenharmony_ci return -ENOMEM; 83062306a36Sopenharmony_ci 83162306a36Sopenharmony_ci pctl_desc->npins = ngpios; 83262306a36Sopenharmony_ci pctl_desc->pins = pins; 83362306a36Sopenharmony_ci 83462306a36Sopenharmony_ci for (i = 0; i < ngpios; i++) { 83562306a36Sopenharmony_ci struct sgpio_port_addr addr; 83662306a36Sopenharmony_ci 83762306a36Sopenharmony_ci sgpio_pin_to_addr(priv, i, &addr); 83862306a36Sopenharmony_ci 83962306a36Sopenharmony_ci pins[i].number = i; 84062306a36Sopenharmony_ci pins[i].name = devm_kasprintf(dev, GFP_KERNEL, 84162306a36Sopenharmony_ci "SGPIO_%c_p%db%d", 84262306a36Sopenharmony_ci bank->is_input ? 'I' : 'O', 84362306a36Sopenharmony_ci addr.port, addr.bit); 84462306a36Sopenharmony_ci if (!pins[i].name) 84562306a36Sopenharmony_ci return -ENOMEM; 84662306a36Sopenharmony_ci } 84762306a36Sopenharmony_ci 84862306a36Sopenharmony_ci pctldev = devm_pinctrl_register(dev, pctl_desc, bank); 84962306a36Sopenharmony_ci if (IS_ERR(pctldev)) 85062306a36Sopenharmony_ci return dev_err_probe(dev, PTR_ERR(pctldev), "Failed to register pinctrl\n"); 85162306a36Sopenharmony_ci 85262306a36Sopenharmony_ci gc = &bank->gpio; 85362306a36Sopenharmony_ci gc->label = pctl_desc->name; 85462306a36Sopenharmony_ci gc->parent = dev; 85562306a36Sopenharmony_ci gc->fwnode = fwnode; 85662306a36Sopenharmony_ci gc->owner = THIS_MODULE; 85762306a36Sopenharmony_ci gc->get_direction = microchip_sgpio_get_direction; 85862306a36Sopenharmony_ci gc->direction_input = microchip_sgpio_direction_input; 85962306a36Sopenharmony_ci gc->direction_output = microchip_sgpio_direction_output; 86062306a36Sopenharmony_ci gc->get = microchip_sgpio_get_value; 86162306a36Sopenharmony_ci gc->set = microchip_sgpio_set_value; 86262306a36Sopenharmony_ci gc->request = gpiochip_generic_request; 86362306a36Sopenharmony_ci gc->free = gpiochip_generic_free; 86462306a36Sopenharmony_ci gc->of_xlate = microchip_sgpio_of_xlate; 86562306a36Sopenharmony_ci gc->of_gpio_n_cells = 3; 86662306a36Sopenharmony_ci gc->base = -1; 86762306a36Sopenharmony_ci gc->ngpio = ngpios; 86862306a36Sopenharmony_ci gc->can_sleep = !bank->is_input; 86962306a36Sopenharmony_ci 87062306a36Sopenharmony_ci if (bank->is_input && priv->properties->flags & SGPIO_FLAGS_HAS_IRQ) { 87162306a36Sopenharmony_ci int irq; 87262306a36Sopenharmony_ci 87362306a36Sopenharmony_ci irq = fwnode_irq_get(fwnode, 0); 87462306a36Sopenharmony_ci if (irq > 0) { 87562306a36Sopenharmony_ci struct gpio_irq_chip *girq = &gc->irq; 87662306a36Sopenharmony_ci 87762306a36Sopenharmony_ci gpio_irq_chip_set_chip(girq, µchip_sgpio_irqchip); 87862306a36Sopenharmony_ci girq->parent_handler = sgpio_irq_handler; 87962306a36Sopenharmony_ci girq->num_parents = 1; 88062306a36Sopenharmony_ci girq->parents = devm_kcalloc(dev, 1, 88162306a36Sopenharmony_ci sizeof(*girq->parents), 88262306a36Sopenharmony_ci GFP_KERNEL); 88362306a36Sopenharmony_ci if (!girq->parents) 88462306a36Sopenharmony_ci return -ENOMEM; 88562306a36Sopenharmony_ci girq->parents[0] = irq; 88662306a36Sopenharmony_ci girq->default_type = IRQ_TYPE_NONE; 88762306a36Sopenharmony_ci girq->handler = handle_bad_irq; 88862306a36Sopenharmony_ci 88962306a36Sopenharmony_ci /* Disable all individual pins */ 89062306a36Sopenharmony_ci for (i = 0; i < SGPIO_MAX_BITS; i++) 89162306a36Sopenharmony_ci sgpio_writel(priv, 0, REG_INT_ENABLE, i); 89262306a36Sopenharmony_ci /* Master enable */ 89362306a36Sopenharmony_ci sgpio_clrsetbits(priv, REG_SIO_CONFIG, 0, 0, SGPIO_MASTER_INTR_ENA); 89462306a36Sopenharmony_ci } 89562306a36Sopenharmony_ci } 89662306a36Sopenharmony_ci 89762306a36Sopenharmony_ci ret = devm_gpiochip_add_data(dev, gc, bank); 89862306a36Sopenharmony_ci if (ret) 89962306a36Sopenharmony_ci dev_err(dev, "Failed to register: ret %d\n", ret); 90062306a36Sopenharmony_ci 90162306a36Sopenharmony_ci return ret; 90262306a36Sopenharmony_ci} 90362306a36Sopenharmony_ci 90462306a36Sopenharmony_cistatic int microchip_sgpio_probe(struct platform_device *pdev) 90562306a36Sopenharmony_ci{ 90662306a36Sopenharmony_ci int div_clock = 0, ret, port, i, nbanks; 90762306a36Sopenharmony_ci struct device *dev = &pdev->dev; 90862306a36Sopenharmony_ci struct fwnode_handle *fwnode; 90962306a36Sopenharmony_ci struct reset_control *reset; 91062306a36Sopenharmony_ci struct sgpio_priv *priv; 91162306a36Sopenharmony_ci struct clk *clk; 91262306a36Sopenharmony_ci u32 val; 91362306a36Sopenharmony_ci struct regmap_config regmap_config = { 91462306a36Sopenharmony_ci .reg_bits = 32, 91562306a36Sopenharmony_ci .val_bits = 32, 91662306a36Sopenharmony_ci .reg_stride = 4, 91762306a36Sopenharmony_ci }; 91862306a36Sopenharmony_ci 91962306a36Sopenharmony_ci priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); 92062306a36Sopenharmony_ci if (!priv) 92162306a36Sopenharmony_ci return -ENOMEM; 92262306a36Sopenharmony_ci 92362306a36Sopenharmony_ci priv->dev = dev; 92462306a36Sopenharmony_ci spin_lock_init(&priv->lock); 92562306a36Sopenharmony_ci mutex_init(&priv->poll_lock); 92662306a36Sopenharmony_ci 92762306a36Sopenharmony_ci reset = devm_reset_control_get_optional_shared(&pdev->dev, "switch"); 92862306a36Sopenharmony_ci if (IS_ERR(reset)) 92962306a36Sopenharmony_ci return dev_err_probe(dev, PTR_ERR(reset), "Failed to get reset\n"); 93062306a36Sopenharmony_ci reset_control_reset(reset); 93162306a36Sopenharmony_ci 93262306a36Sopenharmony_ci clk = devm_clk_get(dev, NULL); 93362306a36Sopenharmony_ci if (IS_ERR(clk)) 93462306a36Sopenharmony_ci return dev_err_probe(dev, PTR_ERR(clk), "Failed to get clock\n"); 93562306a36Sopenharmony_ci 93662306a36Sopenharmony_ci div_clock = clk_get_rate(clk); 93762306a36Sopenharmony_ci if (device_property_read_u32(dev, "bus-frequency", &priv->clock)) 93862306a36Sopenharmony_ci priv->clock = 12500000; 93962306a36Sopenharmony_ci if (priv->clock == 0 || priv->clock > (div_clock / 2)) { 94062306a36Sopenharmony_ci dev_err(dev, "Invalid frequency %d\n", priv->clock); 94162306a36Sopenharmony_ci return -EINVAL; 94262306a36Sopenharmony_ci } 94362306a36Sopenharmony_ci 94462306a36Sopenharmony_ci priv->regs = ocelot_regmap_from_resource(pdev, 0, ®map_config); 94562306a36Sopenharmony_ci if (IS_ERR(priv->regs)) 94662306a36Sopenharmony_ci return PTR_ERR(priv->regs); 94762306a36Sopenharmony_ci 94862306a36Sopenharmony_ci priv->properties = device_get_match_data(dev); 94962306a36Sopenharmony_ci priv->in.is_input = true; 95062306a36Sopenharmony_ci 95162306a36Sopenharmony_ci /* Get rest of device properties */ 95262306a36Sopenharmony_ci ret = microchip_sgpio_get_ports(priv); 95362306a36Sopenharmony_ci if (ret) 95462306a36Sopenharmony_ci return ret; 95562306a36Sopenharmony_ci 95662306a36Sopenharmony_ci nbanks = device_get_child_node_count(dev); 95762306a36Sopenharmony_ci if (nbanks != 2) { 95862306a36Sopenharmony_ci dev_err(dev, "Must have 2 banks (have %d)\n", nbanks); 95962306a36Sopenharmony_ci return -EINVAL; 96062306a36Sopenharmony_ci } 96162306a36Sopenharmony_ci 96262306a36Sopenharmony_ci i = 0; 96362306a36Sopenharmony_ci device_for_each_child_node(dev, fwnode) { 96462306a36Sopenharmony_ci ret = microchip_sgpio_register_bank(dev, priv, fwnode, i++); 96562306a36Sopenharmony_ci if (ret) { 96662306a36Sopenharmony_ci fwnode_handle_put(fwnode); 96762306a36Sopenharmony_ci return ret; 96862306a36Sopenharmony_ci } 96962306a36Sopenharmony_ci } 97062306a36Sopenharmony_ci 97162306a36Sopenharmony_ci if (priv->in.gpio.ngpio != priv->out.gpio.ngpio) { 97262306a36Sopenharmony_ci dev_err(dev, "Banks must have same GPIO count\n"); 97362306a36Sopenharmony_ci return -ERANGE; 97462306a36Sopenharmony_ci } 97562306a36Sopenharmony_ci 97662306a36Sopenharmony_ci sgpio_configure_bitstream(priv); 97762306a36Sopenharmony_ci 97862306a36Sopenharmony_ci val = max(2U, div_clock / priv->clock); 97962306a36Sopenharmony_ci sgpio_configure_clock(priv, val); 98062306a36Sopenharmony_ci 98162306a36Sopenharmony_ci for (port = 0; port < SGPIO_BITS_PER_WORD; port++) 98262306a36Sopenharmony_ci sgpio_writel(priv, 0, REG_PORT_CONFIG, port); 98362306a36Sopenharmony_ci sgpio_writel(priv, priv->ports, REG_PORT_ENABLE, 0); 98462306a36Sopenharmony_ci 98562306a36Sopenharmony_ci return 0; 98662306a36Sopenharmony_ci} 98762306a36Sopenharmony_ci 98862306a36Sopenharmony_cistatic const struct of_device_id microchip_sgpio_gpio_of_match[] = { 98962306a36Sopenharmony_ci { 99062306a36Sopenharmony_ci .compatible = "microchip,sparx5-sgpio", 99162306a36Sopenharmony_ci .data = &properties_sparx5, 99262306a36Sopenharmony_ci }, { 99362306a36Sopenharmony_ci .compatible = "mscc,luton-sgpio", 99462306a36Sopenharmony_ci .data = &properties_luton, 99562306a36Sopenharmony_ci }, { 99662306a36Sopenharmony_ci .compatible = "mscc,ocelot-sgpio", 99762306a36Sopenharmony_ci .data = &properties_ocelot, 99862306a36Sopenharmony_ci }, { 99962306a36Sopenharmony_ci /* sentinel */ 100062306a36Sopenharmony_ci } 100162306a36Sopenharmony_ci}; 100262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, microchip_sgpio_gpio_of_match); 100362306a36Sopenharmony_ci 100462306a36Sopenharmony_cistatic struct platform_driver microchip_sgpio_pinctrl_driver = { 100562306a36Sopenharmony_ci .driver = { 100662306a36Sopenharmony_ci .name = "pinctrl-microchip-sgpio", 100762306a36Sopenharmony_ci .of_match_table = microchip_sgpio_gpio_of_match, 100862306a36Sopenharmony_ci .suppress_bind_attrs = true, 100962306a36Sopenharmony_ci }, 101062306a36Sopenharmony_ci .probe = microchip_sgpio_probe, 101162306a36Sopenharmony_ci}; 101262306a36Sopenharmony_cimodule_platform_driver(microchip_sgpio_pinctrl_driver); 101362306a36Sopenharmony_ci 101462306a36Sopenharmony_ciMODULE_DESCRIPTION("Microchip SGPIO Pinctrl Driver"); 101562306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 1016