struct aspeed_sgpio_pdata {
const u32 pin_mask;
+ const struct aspeed_sgpio_llops *llops;
};
struct aspeed_sgpio {
raw_spinlock_t lock;
void __iomem *base;
int irq;
+ const struct aspeed_sgpio_pdata *pdata;
};
struct aspeed_sgpio_bank {
reg_tolerance,
};
+struct aspeed_sgpio_llops {
+ void (*reg_bit_set)(struct aspeed_sgpio *gpio, unsigned int offset,
+ const enum aspeed_sgpio_reg reg, bool val);
+ bool (*reg_bit_get)(struct aspeed_sgpio *gpio, unsigned int offset,
+ const enum aspeed_sgpio_reg reg);
+ int (*reg_bank_get)(struct aspeed_sgpio *gpio, unsigned int offset,
+ const enum aspeed_sgpio_reg reg);
+};
+
#define GPIO_VAL_VALUE 0x00
#define GPIO_IRQ_ENABLE 0x00
#define GPIO_IRQ_TYPE0 0x04
#define GPIO_IRQ_TYPE2 0x0C
#define GPIO_IRQ_STATUS 0x10
-static void __iomem *bank_reg(struct aspeed_sgpio *gpio,
- const struct aspeed_sgpio_bank *bank,
- const enum aspeed_sgpio_reg reg)
+static void __iomem *aspeed_sgpio_g4_bank_reg(struct aspeed_sgpio *gpio,
+ const struct aspeed_sgpio_bank *bank,
+ const enum aspeed_sgpio_reg reg)
{
switch (reg) {
case reg_val:
static int aspeed_sgpio_get(struct gpio_chip *gc, unsigned int offset)
{
struct aspeed_sgpio *gpio = gpiochip_get_data(gc);
- const struct aspeed_sgpio_bank *bank = to_bank(offset);
enum aspeed_sgpio_reg reg;
int rc = 0;
guard(raw_spinlock_irqsave)(&gpio->lock);
reg = aspeed_sgpio_is_input(offset) ? reg_val : reg_rdata;
- rc = !!(ioread32(bank_reg(gpio, bank, reg)) & GPIO_BIT(offset));
+ rc = gpio->pdata->llops->reg_bit_get(gpio, offset, reg);
return rc;
}
static int sgpio_set_value(struct gpio_chip *gc, unsigned int offset, int val)
{
struct aspeed_sgpio *gpio = gpiochip_get_data(gc);
- const struct aspeed_sgpio_bank *bank = to_bank(offset);
- void __iomem *addr_r, *addr_w;
- u32 reg = 0;
if (aspeed_sgpio_is_input(offset))
return -EINVAL;
- /* Since this is an output, read the cached value from rdata, then
- * update val. */
- addr_r = bank_reg(gpio, bank, reg_rdata);
- addr_w = bank_reg(gpio, bank, reg_val);
-
- reg = ioread32(addr_r);
-
- if (val)
- reg |= GPIO_BIT(offset);
- else
- reg &= ~GPIO_BIT(offset);
-
- iowrite32(reg, addr_w);
+ gpio->pdata->llops->reg_bit_set(gpio, offset, reg_val, val);
return 0;
}
return !!aspeed_sgpio_is_input(offset);
}
-static void irqd_to_aspeed_sgpio_data(struct irq_data *d,
- struct aspeed_sgpio **gpio,
- const struct aspeed_sgpio_bank **bank,
- u32 *bit, int *offset)
-{
- struct aspeed_sgpio *internal;
-
- *offset = irqd_to_hwirq(d);
- internal = irq_data_get_irq_chip_data(d);
- WARN_ON(!internal);
-
- *gpio = internal;
- *bank = to_bank(*offset);
- *bit = GPIO_BIT(*offset);
-}
static void aspeed_sgpio_irq_ack(struct irq_data *d)
{
- const struct aspeed_sgpio_bank *bank;
- struct aspeed_sgpio *gpio;
- void __iomem *status_addr;
- int offset;
- u32 bit;
-
- irqd_to_aspeed_sgpio_data(d, &gpio, &bank, &bit, &offset);
-
- status_addr = bank_reg(gpio, bank, reg_irq_status);
+ struct aspeed_sgpio *gpio = irq_data_get_irq_chip_data(d);
+ int offset = irqd_to_hwirq(d);
guard(raw_spinlock_irqsave)(&gpio->lock);
- iowrite32(bit, status_addr);
+ gpio->pdata->llops->reg_bit_set(gpio, offset, reg_irq_status, 1);
}
static void aspeed_sgpio_irq_set_mask(struct irq_data *d, bool set)
{
- const struct aspeed_sgpio_bank *bank;
- struct aspeed_sgpio *gpio;
- u32 reg, bit;
- void __iomem *addr;
- int offset;
-
- irqd_to_aspeed_sgpio_data(d, &gpio, &bank, &bit, &offset);
- addr = bank_reg(gpio, bank, reg_irq_enable);
+ struct aspeed_sgpio *gpio = irq_data_get_irq_chip_data(d);
+ int offset = irqd_to_hwirq(d);
/* Unmasking the IRQ */
if (set)
- gpiochip_enable_irq(&gpio->chip, irqd_to_hwirq(d));
-
- scoped_guard(raw_spinlock_irqsave, &gpio->lock) {
- reg = ioread32(addr);
- if (set)
- reg |= bit;
- else
- reg &= ~bit;
-
- iowrite32(reg, addr);
+ gpiochip_enable_irq(&gpio->chip, offset);
+ scoped_guard(raw_spinlock_irqsave, &gpio->lock)
+ {
+ gpio->pdata->llops->reg_bit_set(gpio, offset, reg_irq_enable,
+ set);
}
/* Masking the IRQ */
if (!set)
- gpiochip_disable_irq(&gpio->chip, irqd_to_hwirq(d));
-
-
+ gpiochip_disable_irq(&gpio->chip, offset);
}
static void aspeed_sgpio_irq_mask(struct irq_data *d)
u32 type0 = 0;
u32 type1 = 0;
u32 type2 = 0;
- u32 bit, reg;
- const struct aspeed_sgpio_bank *bank;
irq_flow_handler_t handler;
- struct aspeed_sgpio *gpio;
- void __iomem *addr;
- int offset;
-
- irqd_to_aspeed_sgpio_data(d, &gpio, &bank, &bit, &offset);
+ struct aspeed_sgpio *gpio = irq_data_get_irq_chip_data(d);
+ int offset = irqd_to_hwirq(d);
switch (type & IRQ_TYPE_SENSE_MASK) {
case IRQ_TYPE_EDGE_BOTH:
- type2 |= bit;
+ type2 = 1;
fallthrough;
case IRQ_TYPE_EDGE_RISING:
- type0 |= bit;
+ type0 = 1;
fallthrough;
case IRQ_TYPE_EDGE_FALLING:
handler = handle_edge_irq;
break;
case IRQ_TYPE_LEVEL_HIGH:
- type0 |= bit;
+ type0 = 1;
fallthrough;
case IRQ_TYPE_LEVEL_LOW:
- type1 |= bit;
+ type1 = 1;
handler = handle_level_irq;
break;
default:
}
scoped_guard(raw_spinlock_irqsave, &gpio->lock) {
- addr = bank_reg(gpio, bank, reg_irq_type0);
- reg = ioread32(addr);
- reg = (reg & ~bit) | type0;
- iowrite32(reg, addr);
-
- addr = bank_reg(gpio, bank, reg_irq_type1);
- reg = ioread32(addr);
- reg = (reg & ~bit) | type1;
- iowrite32(reg, addr);
-
- addr = bank_reg(gpio, bank, reg_irq_type2);
- reg = ioread32(addr);
- reg = (reg & ~bit) | type2;
- iowrite32(reg, addr);
+ gpio->pdata->llops->reg_bit_set(gpio, offset, reg_irq_type0, type0);
+ gpio->pdata->llops->reg_bit_set(gpio, offset, reg_irq_type1, type1);
+ gpio->pdata->llops->reg_bit_set(gpio, offset, reg_irq_type2, type2);
}
irq_set_handler_locked(d, handler);
chained_irq_enter(ic, desc);
for (i = 0; i < ARRAY_SIZE(aspeed_sgpio_banks); i++) {
- const struct aspeed_sgpio_bank *bank = &aspeed_sgpio_banks[i];
-
- reg = ioread32(bank_reg(data, bank, reg_irq_status));
+ reg = data->pdata->llops->reg_bank_get(data, i << 6, reg_irq_status);
for_each_set_bit(p, ®, 32)
generic_handle_domain_irq(gc->irq.domain, (i * 32 + p) * 2);
static void aspeed_sgpio_irq_print_chip(struct irq_data *d, struct seq_file *p)
{
- const struct aspeed_sgpio_bank *bank;
- struct aspeed_sgpio *gpio;
- u32 bit;
- int offset;
+ struct aspeed_sgpio *gpio = irq_data_get_irq_chip_data(d);
- irqd_to_aspeed_sgpio_data(d, &gpio, &bank, &bit, &offset);
seq_puts(p, dev_name(gpio->dev));
}
/* Apply default IRQ settings */
for (i = 0; i < ARRAY_SIZE(aspeed_sgpio_banks); i++) {
- bank = &aspeed_sgpio_banks[i];
+ bank = &aspeed_sgpio_banks[i];
/* set falling or level-low irq */
iowrite32(0x00000000, bank_reg(gpio, bank, reg_irq_type0));
/* trigger type is edge */
return 0;
}
+static void aspeed_sgpio_g4_reg_bit_set(struct aspeed_sgpio *gpio, unsigned int offset,
+ const enum aspeed_sgpio_reg reg, bool val)
+{
+ const struct aspeed_sgpio_bank *bank = to_bank(offset);
+ void __iomem *addr = aspeed_sgpio_g4_bank_reg(gpio, bank, reg);
+ u32 temp;
+
+ if (reg == reg_val) {
+ /* Since this is an output, read the cached value from rdata, then update val. */
+ addr = aspeed_sgpio_g4_bank_reg(gpio, bank, reg_rdata);
+ temp = ioread32(addr);
+ if (val)
+ temp |= GPIO_BIT(offset);
+ else
+ temp &= ~GPIO_BIT(offset);
+
+ addr = aspeed_sgpio_g4_bank_reg(gpio, bank, reg_val);
+ iowrite32(temp, addr);
+ } else if (reg == reg_irq_status) {
+ if (val)
+ iowrite32(GPIO_BIT(offset), addr);
+ } else {
+ /* When setting other registers, we read from the register itself */
+ temp = ioread32(addr);
+ if (val)
+ temp |= GPIO_BIT(offset);
+ else
+ temp &= ~GPIO_BIT(offset);
+ iowrite32(temp, addr);
+ }
+}
+
+static bool aspeed_sgpio_g4_reg_bit_get(struct aspeed_sgpio *gpio, unsigned int offset,
+ const enum aspeed_sgpio_reg reg)
+{
+ const struct aspeed_sgpio_bank *bank = to_bank(offset);
+ void __iomem *addr = aspeed_sgpio_g4_bank_reg(gpio, bank, reg);
+
+ return !!(ioread32(addr) & GPIO_BIT(offset));
+}
+
+static int aspeed_sgpio_g4_reg_bank_get(struct aspeed_sgpio *gpio, unsigned int offset,
+ const enum aspeed_sgpio_reg reg)
+{
+ const struct aspeed_sgpio_bank *bank = to_bank(offset);
+ void __iomem *addr = aspeed_sgpio_g4_bank_reg(gpio, bank, reg);
+
+ if (reg == reg_irq_status)
+ return ioread32(addr);
+ else
+ return -EOPNOTSUPP;
+}
+
+static const struct aspeed_sgpio_llops aspeed_sgpio_g4_llops = {
+ .reg_bit_set = aspeed_sgpio_g4_reg_bit_set,
+ .reg_bit_get = aspeed_sgpio_g4_reg_bit_get,
+ .reg_bank_get = aspeed_sgpio_g4_reg_bank_get,
+};
+
static const struct aspeed_sgpio_pdata ast2400_sgpio_pdata = {
.pin_mask = GENMASK(9, 6),
+ .llops = &aspeed_sgpio_g4_llops,
};
static int aspeed_sgpio_reset_tolerance(struct gpio_chip *chip,
unsigned int offset, bool enable)
{
struct aspeed_sgpio *gpio = gpiochip_get_data(chip);
- void __iomem *reg;
- u32 val;
-
- reg = bank_reg(gpio, to_bank(offset), reg_tolerance);
guard(raw_spinlock_irqsave)(&gpio->lock);
- val = readl(reg);
-
- if (enable)
- val |= GPIO_BIT(offset);
- else
- val &= ~GPIO_BIT(offset);
-
- writel(val, reg);
+ gpio->pdata->llops->reg_bit_set(gpio, offset, reg_tolerance, enable);
return 0;
}
static const struct aspeed_sgpio_pdata ast2600_sgpiom_pdata = {
.pin_mask = GENMASK(10, 6),
+ .llops = &aspeed_sgpio_g4_llops,
};
static const struct of_device_id aspeed_sgpio_of_table[] = {
static int aspeed_sgpio_probe(struct platform_device *pdev)
{
u32 nr_gpios, sgpio_freq, sgpio_clk_div, gpio_cnt_regval, pin_mask;
- const struct aspeed_sgpio_pdata *pdata;
struct aspeed_sgpio *gpio;
unsigned long apb_freq;
int rc;
gpio->dev = &pdev->dev;
- pdata = device_get_match_data(&pdev->dev);
- if (!pdata)
+ gpio->pdata = device_get_match_data(&pdev->dev);
+ if (!gpio->pdata)
return -EINVAL;
- pin_mask = pdata->pin_mask;
-
+ pin_mask = gpio->pdata->pin_mask;
rc = device_property_read_u32(&pdev->dev, "ngpios", &nr_gpios);
if (rc < 0) {
dev_err(&pdev->dev, "Could not read ngpios property\n");