2019-06-01 16:08:37 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2015-03-10 15:02:19 +08:00
|
|
|
/*
|
|
|
|
* GPIO driver for AMD
|
|
|
|
*
|
|
|
|
* Copyright (c) 2014,2015 AMD Corporation.
|
|
|
|
* Authors: Ken Xue <Ken.Xue@amd.com>
|
|
|
|
* Wu, Jeff <Jeff.Wu@amd.com>
|
|
|
|
*
|
2017-05-03 14:29:11 +08:00
|
|
|
* Contact Information: Nehal Shah <Nehal-bakulchandra.Shah@amd.com>
|
|
|
|
* Shyam Sundar S K <Shyam-sundar.S-k@amd.com>
|
2015-03-10 15:02:19 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/bug.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/compiler.h>
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/log2.h>
|
|
|
|
#include <linux/io.h>
|
2018-09-13 19:58:21 +08:00
|
|
|
#include <linux/gpio/driver.h>
|
2015-03-10 15:02:19 +08:00
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <linux/mutex.h>
|
|
|
|
#include <linux/acpi.h>
|
|
|
|
#include <linux/seq_file.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/bitops.h>
|
|
|
|
#include <linux/pinctrl/pinconf.h>
|
|
|
|
#include <linux/pinctrl/pinconf-generic.h>
|
|
|
|
|
2017-09-11 14:11:56 +08:00
|
|
|
#include "core.h"
|
2015-03-10 15:02:19 +08:00
|
|
|
#include "pinctrl-utils.h"
|
|
|
|
#include "pinctrl-amd.h"
|
|
|
|
|
2018-02-17 03:12:43 +08:00
|
|
|
static int amd_gpio_get_direction(struct gpio_chip *gc, unsigned offset)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
u32 pin_reg;
|
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
|
|
|
|
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
|
|
|
pin_reg = readl(gpio_dev->base + offset * 4);
|
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
|
|
|
|
2020-02-14 21:57:12 +08:00
|
|
|
if (pin_reg & BIT(OUTPUT_ENABLE_OFF))
|
|
|
|
return GPIO_LINE_DIRECTION_OUT;
|
|
|
|
|
|
|
|
return GPIO_LINE_DIRECTION_IN;
|
2018-02-17 03:12:43 +08:00
|
|
|
}
|
|
|
|
|
2015-03-10 15:02:19 +08:00
|
|
|
static int amd_gpio_direction_input(struct gpio_chip *gc, unsigned offset)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
u32 pin_reg;
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + offset * 4);
|
|
|
|
pin_reg &= ~BIT(OUTPUT_ENABLE_OFF);
|
|
|
|
writel(pin_reg, gpio_dev->base + offset * 4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_gpio_direction_output(struct gpio_chip *gc, unsigned offset,
|
|
|
|
int value)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned long flags;
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + offset * 4);
|
|
|
|
pin_reg |= BIT(OUTPUT_ENABLE_OFF);
|
|
|
|
if (value)
|
|
|
|
pin_reg |= BIT(OUTPUT_VALUE_OFF);
|
|
|
|
else
|
|
|
|
pin_reg &= ~BIT(OUTPUT_VALUE_OFF);
|
|
|
|
writel(pin_reg, gpio_dev->base + offset * 4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_gpio_get_value(struct gpio_chip *gc, unsigned offset)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned long flags;
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + offset * 4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
return !!(pin_reg & BIT(PIN_STS_OFF));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void amd_gpio_set_value(struct gpio_chip *gc, unsigned offset, int value)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned long flags;
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + offset * 4);
|
|
|
|
if (value)
|
|
|
|
pin_reg |= BIT(OUTPUT_VALUE_OFF);
|
|
|
|
else
|
|
|
|
pin_reg &= ~BIT(OUTPUT_VALUE_OFF);
|
|
|
|
writel(pin_reg, gpio_dev->base + offset * 4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_gpio_set_debounce(struct gpio_chip *gc, unsigned offset,
|
|
|
|
unsigned debounce)
|
|
|
|
{
|
|
|
|
u32 time;
|
2015-03-27 17:44:26 +08:00
|
|
|
u32 pin_reg;
|
|
|
|
int ret = 0;
|
2015-03-10 15:02:19 +08:00
|
|
|
unsigned long flags;
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + offset * 4);
|
|
|
|
|
|
|
|
if (debounce) {
|
|
|
|
pin_reg |= DB_TYPE_REMOVE_GLITCH << DB_CNTRL_OFF;
|
|
|
|
pin_reg &= ~DB_TMR_OUT_MASK;
|
|
|
|
/*
|
|
|
|
Debounce Debounce Timer Max
|
|
|
|
TmrLarge TmrOutUnit Unit Debounce
|
|
|
|
Time
|
|
|
|
0 0 61 usec (2 RtcClk) 976 usec
|
|
|
|
0 1 244 usec (8 RtcClk) 3.9 msec
|
|
|
|
1 0 15.6 msec (512 RtcClk) 250 msec
|
|
|
|
1 1 62.5 msec (2048 RtcClk) 1 sec
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (debounce < 61) {
|
|
|
|
pin_reg |= 1;
|
|
|
|
pin_reg &= ~BIT(DB_TMR_OUT_UNIT_OFF);
|
|
|
|
pin_reg &= ~BIT(DB_TMR_LARGE_OFF);
|
|
|
|
} else if (debounce < 976) {
|
|
|
|
time = debounce / 61;
|
|
|
|
pin_reg |= time & DB_TMR_OUT_MASK;
|
|
|
|
pin_reg &= ~BIT(DB_TMR_OUT_UNIT_OFF);
|
|
|
|
pin_reg &= ~BIT(DB_TMR_LARGE_OFF);
|
|
|
|
} else if (debounce < 3900) {
|
|
|
|
time = debounce / 244;
|
|
|
|
pin_reg |= time & DB_TMR_OUT_MASK;
|
|
|
|
pin_reg |= BIT(DB_TMR_OUT_UNIT_OFF);
|
|
|
|
pin_reg &= ~BIT(DB_TMR_LARGE_OFF);
|
|
|
|
} else if (debounce < 250000) {
|
2020-11-06 07:19:10 +08:00
|
|
|
time = debounce / 15625;
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg |= time & DB_TMR_OUT_MASK;
|
|
|
|
pin_reg &= ~BIT(DB_TMR_OUT_UNIT_OFF);
|
|
|
|
pin_reg |= BIT(DB_TMR_LARGE_OFF);
|
|
|
|
} else if (debounce < 1000000) {
|
|
|
|
time = debounce / 62500;
|
|
|
|
pin_reg |= time & DB_TMR_OUT_MASK;
|
|
|
|
pin_reg |= BIT(DB_TMR_OUT_UNIT_OFF);
|
|
|
|
pin_reg |= BIT(DB_TMR_LARGE_OFF);
|
|
|
|
} else {
|
2020-11-06 07:19:09 +08:00
|
|
|
pin_reg &= ~(DB_CNTRl_MASK << DB_CNTRL_OFF);
|
2015-03-27 17:44:26 +08:00
|
|
|
ret = -EINVAL;
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
pin_reg &= ~BIT(DB_TMR_OUT_UNIT_OFF);
|
|
|
|
pin_reg &= ~BIT(DB_TMR_LARGE_OFF);
|
|
|
|
pin_reg &= ~DB_TMR_OUT_MASK;
|
2020-11-06 07:19:09 +08:00
|
|
|
pin_reg &= ~(DB_CNTRl_MASK << DB_CNTRL_OFF);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
writel(pin_reg, gpio_dev->base + offset * 4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2015-03-27 17:44:26 +08:00
|
|
|
return ret;
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
2017-01-23 20:34:34 +08:00
|
|
|
static int amd_gpio_set_config(struct gpio_chip *gc, unsigned offset,
|
|
|
|
unsigned long config)
|
|
|
|
{
|
|
|
|
u32 debounce;
|
|
|
|
|
|
|
|
if (pinconf_to_config_param(config) != PIN_CONFIG_INPUT_DEBOUNCE)
|
|
|
|
return -ENOTSUPP;
|
|
|
|
|
|
|
|
debounce = pinconf_to_config_argument(config);
|
|
|
|
return amd_gpio_set_debounce(gc, offset, debounce);
|
|
|
|
}
|
|
|
|
|
2015-03-10 15:02:19 +08:00
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
|
|
static void amd_gpio_dbg_show(struct seq_file *s, struct gpio_chip *gc)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
2020-11-06 07:19:11 +08:00
|
|
|
u32 db_cntrl;
|
2015-03-10 15:02:19 +08:00
|
|
|
unsigned long flags;
|
|
|
|
unsigned int bank, i, pin_num;
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2020-11-06 07:19:11 +08:00
|
|
|
bool tmr_out_unit;
|
|
|
|
unsigned int time;
|
|
|
|
unsigned int unit;
|
|
|
|
bool tmr_large;
|
|
|
|
|
2015-03-10 15:02:19 +08:00
|
|
|
char *level_trig;
|
|
|
|
char *active_level;
|
|
|
|
char *interrupt_enable;
|
|
|
|
char *interrupt_mask;
|
|
|
|
char *wake_cntrl0;
|
|
|
|
char *wake_cntrl1;
|
|
|
|
char *wake_cntrl2;
|
|
|
|
char *pin_sts;
|
|
|
|
char *pull_up_sel;
|
|
|
|
char *pull_up_enable;
|
|
|
|
char *pull_down_enable;
|
|
|
|
char *output_value;
|
|
|
|
char *output_enable;
|
2020-11-06 07:19:11 +08:00
|
|
|
char debounce_value[40];
|
|
|
|
char *debounce_enable;
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2016-12-06 14:47:48 +08:00
|
|
|
for (bank = 0; bank < gpio_dev->hwbank_num; bank++) {
|
2015-03-10 15:02:19 +08:00
|
|
|
seq_printf(s, "GPIO bank%d\t", bank);
|
|
|
|
|
|
|
|
switch (bank) {
|
|
|
|
case 0:
|
|
|
|
i = 0;
|
|
|
|
pin_num = AMD_GPIO_PINS_BANK0;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
i = 64;
|
|
|
|
pin_num = AMD_GPIO_PINS_BANK1 + i;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
i = 128;
|
|
|
|
pin_num = AMD_GPIO_PINS_BANK2 + i;
|
|
|
|
break;
|
2016-12-06 14:47:48 +08:00
|
|
|
case 3:
|
|
|
|
i = 192;
|
|
|
|
pin_num = AMD_GPIO_PINS_BANK3 + i;
|
|
|
|
break;
|
2017-01-03 16:18:58 +08:00
|
|
|
default:
|
|
|
|
/* Illegal bank number, ignore */
|
|
|
|
continue;
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
for (; i < pin_num; i++) {
|
|
|
|
seq_printf(s, "pin%d\t", i);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + i * 4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
if (pin_reg & BIT(INTERRUPT_ENABLE_OFF)) {
|
2018-07-17 09:07:41 +08:00
|
|
|
u8 level = (pin_reg >> ACTIVE_LEVEL_OFF) &
|
|
|
|
ACTIVE_LEVEL_MASK;
|
2015-03-10 15:02:19 +08:00
|
|
|
interrupt_enable = "interrupt is enabled|";
|
|
|
|
|
2018-07-17 09:07:41 +08:00
|
|
|
if (level == ACTIVE_LEVEL_HIGH)
|
2015-03-10 15:02:19 +08:00
|
|
|
active_level = "Active high|";
|
2018-07-17 09:07:41 +08:00
|
|
|
else if (level == ACTIVE_LEVEL_LOW)
|
|
|
|
active_level = "Active low|";
|
|
|
|
else if (!(pin_reg & BIT(LEVEL_TRIG_OFF)) &&
|
|
|
|
level == ACTIVE_LEVEL_BOTH)
|
2015-03-10 15:02:19 +08:00
|
|
|
active_level = "Active on both|";
|
|
|
|
else
|
2016-11-23 21:44:47 +08:00
|
|
|
active_level = "Unknown Active level|";
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
if (pin_reg & BIT(LEVEL_TRIG_OFF))
|
|
|
|
level_trig = "Level trigger|";
|
|
|
|
else
|
|
|
|
level_trig = "Edge trigger|";
|
|
|
|
|
|
|
|
} else {
|
|
|
|
interrupt_enable =
|
|
|
|
"interrupt is disabled|";
|
|
|
|
active_level = " ";
|
|
|
|
level_trig = " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pin_reg & BIT(INTERRUPT_MASK_OFF))
|
|
|
|
interrupt_mask =
|
|
|
|
"interrupt is unmasked|";
|
|
|
|
else
|
|
|
|
interrupt_mask =
|
|
|
|
"interrupt is masked|";
|
|
|
|
|
2016-12-06 14:47:48 +08:00
|
|
|
if (pin_reg & BIT(WAKE_CNTRL_OFF_S0I3))
|
2015-03-10 15:02:19 +08:00
|
|
|
wake_cntrl0 = "enable wakeup in S0i3 state|";
|
|
|
|
else
|
|
|
|
wake_cntrl0 = "disable wakeup in S0i3 state|";
|
|
|
|
|
2016-12-06 14:47:48 +08:00
|
|
|
if (pin_reg & BIT(WAKE_CNTRL_OFF_S3))
|
2015-03-10 15:02:19 +08:00
|
|
|
wake_cntrl1 = "enable wakeup in S3 state|";
|
|
|
|
else
|
|
|
|
wake_cntrl1 = "disable wakeup in S3 state|";
|
|
|
|
|
2016-12-06 14:47:48 +08:00
|
|
|
if (pin_reg & BIT(WAKE_CNTRL_OFF_S4))
|
2015-03-10 15:02:19 +08:00
|
|
|
wake_cntrl2 = "enable wakeup in S4/S5 state|";
|
|
|
|
else
|
|
|
|
wake_cntrl2 = "disable wakeup in S4/S5 state|";
|
|
|
|
|
|
|
|
if (pin_reg & BIT(PULL_UP_ENABLE_OFF)) {
|
|
|
|
pull_up_enable = "pull-up is enabled|";
|
|
|
|
if (pin_reg & BIT(PULL_UP_SEL_OFF))
|
|
|
|
pull_up_sel = "8k pull-up|";
|
|
|
|
else
|
|
|
|
pull_up_sel = "4k pull-up|";
|
|
|
|
} else {
|
|
|
|
pull_up_enable = "pull-up is disabled|";
|
|
|
|
pull_up_sel = " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pin_reg & BIT(PULL_DOWN_ENABLE_OFF))
|
|
|
|
pull_down_enable = "pull-down is enabled|";
|
|
|
|
else
|
|
|
|
pull_down_enable = "Pull-down is disabled|";
|
|
|
|
|
|
|
|
if (pin_reg & BIT(OUTPUT_ENABLE_OFF)) {
|
|
|
|
pin_sts = " ";
|
|
|
|
output_enable = "output is enabled|";
|
|
|
|
if (pin_reg & BIT(OUTPUT_VALUE_OFF))
|
|
|
|
output_value = "output is high|";
|
|
|
|
else
|
|
|
|
output_value = "output is low|";
|
|
|
|
} else {
|
|
|
|
output_enable = "output is disabled|";
|
|
|
|
output_value = " ";
|
|
|
|
|
|
|
|
if (pin_reg & BIT(PIN_STS_OFF))
|
|
|
|
pin_sts = "input is high|";
|
|
|
|
else
|
|
|
|
pin_sts = "input is low|";
|
|
|
|
}
|
|
|
|
|
2020-11-06 07:19:11 +08:00
|
|
|
db_cntrl = (DB_CNTRl_MASK << DB_CNTRL_OFF) & pin_reg;
|
|
|
|
if (db_cntrl) {
|
|
|
|
tmr_out_unit = pin_reg & BIT(DB_TMR_OUT_UNIT_OFF);
|
|
|
|
tmr_large = pin_reg & BIT(DB_TMR_LARGE_OFF);
|
|
|
|
time = pin_reg & DB_TMR_OUT_MASK;
|
|
|
|
if (tmr_large) {
|
|
|
|
if (tmr_out_unit)
|
|
|
|
unit = 62500;
|
|
|
|
else
|
|
|
|
unit = 15625;
|
|
|
|
} else {
|
|
|
|
if (tmr_out_unit)
|
|
|
|
unit = 244;
|
|
|
|
else
|
|
|
|
unit = 61;
|
|
|
|
}
|
|
|
|
if ((DB_TYPE_REMOVE_GLITCH << DB_CNTRL_OFF) == db_cntrl)
|
|
|
|
debounce_enable = "debouncing filter (high and low) enabled|";
|
|
|
|
else if ((DB_TYPE_PRESERVE_LOW_GLITCH << DB_CNTRL_OFF) == db_cntrl)
|
|
|
|
debounce_enable = "debouncing filter (low) enabled|";
|
|
|
|
else
|
|
|
|
debounce_enable = "debouncing filter (high) enabled|";
|
|
|
|
|
|
|
|
snprintf(debounce_value, sizeof(debounce_value),
|
|
|
|
"debouncing timeout is %u (us)|", time * unit);
|
|
|
|
} else {
|
|
|
|
debounce_enable = "debouncing filter disabled|";
|
|
|
|
snprintf(debounce_value, sizeof(debounce_value), " ");
|
|
|
|
}
|
|
|
|
|
2015-03-10 15:02:19 +08:00
|
|
|
seq_printf(s, "%s %s %s %s %s %s\n"
|
2020-11-06 07:19:11 +08:00
|
|
|
" %s %s %s %s %s %s %s %s %s 0x%x\n",
|
2015-03-10 15:02:19 +08:00
|
|
|
level_trig, active_level, interrupt_enable,
|
|
|
|
interrupt_mask, wake_cntrl0, wake_cntrl1,
|
|
|
|
wake_cntrl2, pin_sts, pull_up_sel,
|
|
|
|
pull_up_enable, pull_down_enable,
|
2020-11-06 07:19:11 +08:00
|
|
|
output_value, output_enable,
|
|
|
|
debounce_enable, debounce_value, pin_reg);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define amd_gpio_dbg_show NULL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void amd_gpio_irq_enable(struct irq_data *d)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned long flags;
|
|
|
|
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + (d->hwirq)*4);
|
|
|
|
pin_reg |= BIT(INTERRUPT_ENABLE_OFF);
|
|
|
|
pin_reg |= BIT(INTERRUPT_MASK_OFF);
|
|
|
|
writel(pin_reg, gpio_dev->base + (d->hwirq)*4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void amd_gpio_irq_disable(struct irq_data *d)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned long flags;
|
|
|
|
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + (d->hwirq)*4);
|
|
|
|
pin_reg &= ~BIT(INTERRUPT_ENABLE_OFF);
|
|
|
|
pin_reg &= ~BIT(INTERRUPT_MASK_OFF);
|
|
|
|
writel(pin_reg, gpio_dev->base + (d->hwirq)*4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void amd_gpio_irq_mask(struct irq_data *d)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned long flags;
|
|
|
|
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + (d->hwirq)*4);
|
|
|
|
pin_reg &= ~BIT(INTERRUPT_MASK_OFF);
|
|
|
|
writel(pin_reg, gpio_dev->base + (d->hwirq)*4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void amd_gpio_irq_unmask(struct irq_data *d)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned long flags;
|
|
|
|
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + (d->hwirq)*4);
|
|
|
|
pin_reg |= BIT(INTERRUPT_MASK_OFF);
|
|
|
|
writel(pin_reg, gpio_dev->base + (d->hwirq)*4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void amd_gpio_irq_eoi(struct irq_data *d)
|
|
|
|
{
|
|
|
|
u32 reg;
|
|
|
|
unsigned long flags;
|
|
|
|
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
reg = readl(gpio_dev->base + WAKE_INT_MASTER_REG);
|
|
|
|
reg |= EOI_MASK;
|
|
|
|
writel(reg, gpio_dev->base + WAKE_INT_MASTER_REG);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_gpio_irq_set_type(struct irq_data *d, unsigned int type)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
2018-09-23 03:58:26 +08:00
|
|
|
u32 pin_reg, pin_reg_irq_en, mask;
|
2020-06-27 05:10:26 +08:00
|
|
|
unsigned long flags;
|
2015-03-10 15:02:19 +08:00
|
|
|
struct gpio_chip *gc = irq_data_get_irq_chip_data(d);
|
2015-12-08 16:21:38 +08:00
|
|
|
struct amd_gpio *gpio_dev = gpiochip_get_data(gc);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + (d->hwirq)*4);
|
|
|
|
|
|
|
|
switch (type & IRQ_TYPE_SENSE_MASK) {
|
|
|
|
case IRQ_TYPE_EDGE_RISING:
|
|
|
|
pin_reg &= ~BIT(LEVEL_TRIG_OFF);
|
|
|
|
pin_reg &= ~(ACTIVE_LEVEL_MASK << ACTIVE_LEVEL_OFF);
|
|
|
|
pin_reg |= ACTIVE_HIGH << ACTIVE_LEVEL_OFF;
|
2015-06-23 21:52:47 +08:00
|
|
|
irq_set_handler_locked(d, handle_edge_irq);
|
2015-03-10 15:02:19 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case IRQ_TYPE_EDGE_FALLING:
|
|
|
|
pin_reg &= ~BIT(LEVEL_TRIG_OFF);
|
|
|
|
pin_reg &= ~(ACTIVE_LEVEL_MASK << ACTIVE_LEVEL_OFF);
|
|
|
|
pin_reg |= ACTIVE_LOW << ACTIVE_LEVEL_OFF;
|
2015-06-23 21:52:47 +08:00
|
|
|
irq_set_handler_locked(d, handle_edge_irq);
|
2015-03-10 15:02:19 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case IRQ_TYPE_EDGE_BOTH:
|
|
|
|
pin_reg &= ~BIT(LEVEL_TRIG_OFF);
|
|
|
|
pin_reg &= ~(ACTIVE_LEVEL_MASK << ACTIVE_LEVEL_OFF);
|
|
|
|
pin_reg |= BOTH_EADGE << ACTIVE_LEVEL_OFF;
|
2015-06-23 21:52:47 +08:00
|
|
|
irq_set_handler_locked(d, handle_edge_irq);
|
2015-03-10 15:02:19 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case IRQ_TYPE_LEVEL_HIGH:
|
|
|
|
pin_reg |= LEVEL_TRIGGER << LEVEL_TRIG_OFF;
|
|
|
|
pin_reg &= ~(ACTIVE_LEVEL_MASK << ACTIVE_LEVEL_OFF);
|
|
|
|
pin_reg |= ACTIVE_HIGH << ACTIVE_LEVEL_OFF;
|
2015-06-23 21:52:47 +08:00
|
|
|
irq_set_handler_locked(d, handle_level_irq);
|
2015-03-10 15:02:19 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case IRQ_TYPE_LEVEL_LOW:
|
|
|
|
pin_reg |= LEVEL_TRIGGER << LEVEL_TRIG_OFF;
|
|
|
|
pin_reg &= ~(ACTIVE_LEVEL_MASK << ACTIVE_LEVEL_OFF);
|
|
|
|
pin_reg |= ACTIVE_LOW << ACTIVE_LEVEL_OFF;
|
2015-06-23 21:52:47 +08:00
|
|
|
irq_set_handler_locked(d, handle_level_irq);
|
2015-03-10 15:02:19 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case IRQ_TYPE_NONE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
dev_err(&gpio_dev->pdev->dev, "Invalid type value\n");
|
|
|
|
ret = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
pin_reg |= CLR_INTR_STAT << INTERRUPT_STS_OFF;
|
2018-09-23 03:58:26 +08:00
|
|
|
/*
|
|
|
|
* If WAKE_INT_MASTER_REG.MaskStsEn is set, a software write to the
|
|
|
|
* debounce registers of any GPIO will block wake/interrupt status
|
2019-01-05 05:49:12 +08:00
|
|
|
* generation for *all* GPIOs for a length of time that depends on
|
2018-09-23 03:58:26 +08:00
|
|
|
* WAKE_INT_MASTER_REG.MaskStsLength[11:0]. During this period the
|
|
|
|
* INTERRUPT_ENABLE bit will read as 0.
|
|
|
|
*
|
|
|
|
* We temporarily enable irq for the GPIO whose configuration is
|
|
|
|
* changing, and then wait for it to read back as 1 to know when
|
|
|
|
* debounce has settled and then disable the irq again.
|
|
|
|
* We do this polling with the spinlock held to ensure other GPIO
|
|
|
|
* access routines do not read an incorrect value for the irq enable
|
|
|
|
* bit of other GPIOs. We keep the GPIO masked while polling to avoid
|
|
|
|
* spurious irqs, and disable the irq again after polling.
|
|
|
|
*/
|
|
|
|
mask = BIT(INTERRUPT_ENABLE_OFF);
|
|
|
|
pin_reg_irq_en = pin_reg;
|
|
|
|
pin_reg_irq_en |= mask;
|
|
|
|
pin_reg_irq_en &= ~BIT(INTERRUPT_MASK_OFF);
|
|
|
|
writel(pin_reg_irq_en, gpio_dev->base + (d->hwirq)*4);
|
|
|
|
while ((readl(gpio_dev->base + (d->hwirq)*4) & mask) != mask)
|
|
|
|
continue;
|
2015-03-10 15:02:19 +08:00
|
|
|
writel(pin_reg, gpio_dev->base + (d->hwirq)*4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void amd_irq_ack(struct irq_data *d)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* based on HW design,there is no need to ack HW
|
|
|
|
* before handle current irq. But this routine is
|
|
|
|
* necessary for handle_edge_irq
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct irq_chip amd_gpio_irqchip = {
|
|
|
|
.name = "amd_gpio",
|
|
|
|
.irq_ack = amd_irq_ack,
|
|
|
|
.irq_enable = amd_gpio_irq_enable,
|
|
|
|
.irq_disable = amd_gpio_irq_disable,
|
|
|
|
.irq_mask = amd_gpio_irq_mask,
|
|
|
|
.irq_unmask = amd_gpio_irq_unmask,
|
|
|
|
.irq_eoi = amd_gpio_irq_eoi,
|
|
|
|
.irq_set_type = amd_gpio_irq_set_type,
|
2016-12-06 14:47:48 +08:00
|
|
|
.flags = IRQCHIP_SKIP_SET_WAKE,
|
2015-03-10 15:02:19 +08:00
|
|
|
};
|
|
|
|
|
2017-05-24 05:23:32 +08:00
|
|
|
#define PIN_IRQ_PENDING (BIT(INTERRUPT_STS_OFF) | BIT(WAKE_STS_OFF))
|
|
|
|
|
|
|
|
static irqreturn_t amd_gpio_irq_handler(int irq, void *dev_id)
|
2015-03-10 15:02:19 +08:00
|
|
|
{
|
2017-05-24 05:23:32 +08:00
|
|
|
struct amd_gpio *gpio_dev = dev_id;
|
|
|
|
struct gpio_chip *gc = &gpio_dev->gc;
|
|
|
|
irqreturn_t ret = IRQ_NONE;
|
|
|
|
unsigned int i, irqnr;
|
2015-03-10 15:02:19 +08:00
|
|
|
unsigned long flags;
|
2019-10-22 23:11:54 +08:00
|
|
|
u32 __iomem *regs;
|
|
|
|
u32 regval;
|
2017-05-24 05:23:32 +08:00
|
|
|
u64 status, mask;
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-05-24 05:23:32 +08:00
|
|
|
/* Read the wake status */
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2017-05-24 05:23:32 +08:00
|
|
|
status = readl(gpio_dev->base + WAKE_INT_STATUS_REG1);
|
|
|
|
status <<= 32;
|
|
|
|
status |= readl(gpio_dev->base + WAKE_INT_STATUS_REG0);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-05-24 05:23:32 +08:00
|
|
|
/* Bit 0-45 contain the relevant status bits */
|
|
|
|
status &= (1ULL << 46) - 1;
|
|
|
|
regs = gpio_dev->base;
|
|
|
|
for (mask = 1, irqnr = 0; status; mask <<= 1, regs += 4, irqnr += 4) {
|
|
|
|
if (!(status & mask))
|
|
|
|
continue;
|
|
|
|
status &= ~mask;
|
|
|
|
|
|
|
|
/* Each status bit covers four pins */
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
regval = readl(regs + i);
|
2018-07-17 08:57:18 +08:00
|
|
|
if (!(regval & PIN_IRQ_PENDING) ||
|
|
|
|
!(regval & BIT(INTERRUPT_MASK_OFF)))
|
2017-05-24 05:23:32 +08:00
|
|
|
continue;
|
2017-11-08 02:15:47 +08:00
|
|
|
irq = irq_find_mapping(gc->irq.domain, irqnr + i);
|
pinctrl: amd: disable spurious-firing GPIO IRQs
When cold-booting Asus X434DA, GPIO 7 is found to be already configured
as an interrupt, and the GPIO level is found to be in a state that
causes the interrupt to fire.
As soon as pinctrl-amd probes, this interrupt fires and invokes
amd_gpio_irq_handler(). The IRQ is acked, but no GPIO-IRQ handler was
invoked, so the GPIO level being unchanged just causes another interrupt
to fire again immediately after.
This results in an interrupt storm causing this platform to hang
during boot, right after pinctrl-amd is probed.
Detect this situation and disable the GPIO interrupt when this happens.
This enables the affected platform to boot as normal. GPIO 7 actually is
the I2C touchpad interrupt line, and later on, i2c-multitouch loads and
re-enables this interrupt when it is ready to handle it.
Instead of this approach, I considered disabling all GPIO interrupts at
probe time, however that seems a little risky, and I also confirmed that
Windows does not seem to have this behaviour: the same 41 GPIO IRQs are
enabled under both Linux and Windows, which is a far larger collection
than the GPIOs referenced by the DSDT on this platform.
Signed-off-by: Daniel Drake <drake@endlessm.com>
Link: https://lore.kernel.org/r/20190814090540.7152-1-drake@endlessm.com
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
2019-08-14 17:05:40 +08:00
|
|
|
if (irq != 0)
|
|
|
|
generic_handle_irq(irq);
|
pinctrl/amd: fix masking of GPIO interrupts
On Asus laptop models X505BA, X505BP, X542BA and X542BP, the i2c-hid
touchpad (using a GPIO for interrupts) becomes unresponsive after a
few minutes of usage, or after placing two fingers on the touchpad,
which seems to have the effect of queuing up a large amount of input
data to be transferred.
When the touchpad is in unresponsive state, we observed that the GPIO
level-triggered interrupt is still at it's active level, however the
pinctrl-amd driver is not receiving/dispatching more interrupts at this
point.
After the initial interrupt arrives, amd_gpio_irq_mask() is called
however we then see amd_gpio_irq_handler() being called repeatedly for
the same irq; the interrupt mask is not taking effect because of the
following sequence of events:
- amd_gpio_irq_handler fires, reads and caches pin reg
- amd_gpio_irq_handler calls generic_handle_irq()
- During IRQ handling, amd_gpio_irq_mask() is called and modifies pin reg
- amd_gpio_irq_handler clears interrupt by writing cached value
The stale cached value written at the final stage undoes the masking.
Fix this by re-reading the register before clearing the interrupt.
I also spotted that the interrupt-clearing code can race against
amd_gpio_irq_mask() / amd_gpio_irq_unmask(), so add locking there.
Presumably this race was leading to the loss of interrupts.
After these changes, the touchpad appears to be working fine.
Signed-off-by: Daniel Drake <drake@endlessm.com>
Acked-by: Shah, Nehal-bakulchandra <Nehal-Bakulchandra.shah@amd.com>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
2017-10-02 12:00:54 +08:00
|
|
|
|
|
|
|
/* Clear interrupt.
|
|
|
|
* We must read the pin register again, in case the
|
|
|
|
* value was changed while executing
|
|
|
|
* generic_handle_irq() above.
|
pinctrl: amd: disable spurious-firing GPIO IRQs
When cold-booting Asus X434DA, GPIO 7 is found to be already configured
as an interrupt, and the GPIO level is found to be in a state that
causes the interrupt to fire.
As soon as pinctrl-amd probes, this interrupt fires and invokes
amd_gpio_irq_handler(). The IRQ is acked, but no GPIO-IRQ handler was
invoked, so the GPIO level being unchanged just causes another interrupt
to fire again immediately after.
This results in an interrupt storm causing this platform to hang
during boot, right after pinctrl-amd is probed.
Detect this situation and disable the GPIO interrupt when this happens.
This enables the affected platform to boot as normal. GPIO 7 actually is
the I2C touchpad interrupt line, and later on, i2c-multitouch loads and
re-enables this interrupt when it is ready to handle it.
Instead of this approach, I considered disabling all GPIO interrupts at
probe time, however that seems a little risky, and I also confirmed that
Windows does not seem to have this behaviour: the same 41 GPIO IRQs are
enabled under both Linux and Windows, which is a far larger collection
than the GPIOs referenced by the DSDT on this platform.
Signed-off-by: Daniel Drake <drake@endlessm.com>
Link: https://lore.kernel.org/r/20190814090540.7152-1-drake@endlessm.com
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
2019-08-14 17:05:40 +08:00
|
|
|
* If we didn't find a mapping for the interrupt,
|
|
|
|
* disable it in order to avoid a system hang caused
|
|
|
|
* by an interrupt storm.
|
pinctrl/amd: fix masking of GPIO interrupts
On Asus laptop models X505BA, X505BP, X542BA and X542BP, the i2c-hid
touchpad (using a GPIO for interrupts) becomes unresponsive after a
few minutes of usage, or after placing two fingers on the touchpad,
which seems to have the effect of queuing up a large amount of input
data to be transferred.
When the touchpad is in unresponsive state, we observed that the GPIO
level-triggered interrupt is still at it's active level, however the
pinctrl-amd driver is not receiving/dispatching more interrupts at this
point.
After the initial interrupt arrives, amd_gpio_irq_mask() is called
however we then see amd_gpio_irq_handler() being called repeatedly for
the same irq; the interrupt mask is not taking effect because of the
following sequence of events:
- amd_gpio_irq_handler fires, reads and caches pin reg
- amd_gpio_irq_handler calls generic_handle_irq()
- During IRQ handling, amd_gpio_irq_mask() is called and modifies pin reg
- amd_gpio_irq_handler clears interrupt by writing cached value
The stale cached value written at the final stage undoes the masking.
Fix this by re-reading the register before clearing the interrupt.
I also spotted that the interrupt-clearing code can race against
amd_gpio_irq_mask() / amd_gpio_irq_unmask(), so add locking there.
Presumably this race was leading to the loss of interrupts.
After these changes, the touchpad appears to be working fine.
Signed-off-by: Daniel Drake <drake@endlessm.com>
Acked-by: Shah, Nehal-bakulchandra <Nehal-Bakulchandra.shah@amd.com>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
2017-10-02 12:00:54 +08:00
|
|
|
*/
|
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
|
|
|
regval = readl(regs + i);
|
pinctrl: amd: disable spurious-firing GPIO IRQs
When cold-booting Asus X434DA, GPIO 7 is found to be already configured
as an interrupt, and the GPIO level is found to be in a state that
causes the interrupt to fire.
As soon as pinctrl-amd probes, this interrupt fires and invokes
amd_gpio_irq_handler(). The IRQ is acked, but no GPIO-IRQ handler was
invoked, so the GPIO level being unchanged just causes another interrupt
to fire again immediately after.
This results in an interrupt storm causing this platform to hang
during boot, right after pinctrl-amd is probed.
Detect this situation and disable the GPIO interrupt when this happens.
This enables the affected platform to boot as normal. GPIO 7 actually is
the I2C touchpad interrupt line, and later on, i2c-multitouch loads and
re-enables this interrupt when it is ready to handle it.
Instead of this approach, I considered disabling all GPIO interrupts at
probe time, however that seems a little risky, and I also confirmed that
Windows does not seem to have this behaviour: the same 41 GPIO IRQs are
enabled under both Linux and Windows, which is a far larger collection
than the GPIOs referenced by the DSDT on this platform.
Signed-off-by: Daniel Drake <drake@endlessm.com>
Link: https://lore.kernel.org/r/20190814090540.7152-1-drake@endlessm.com
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
2019-08-14 17:05:40 +08:00
|
|
|
if (irq == 0) {
|
|
|
|
regval &= ~BIT(INTERRUPT_ENABLE_OFF);
|
|
|
|
dev_dbg(&gpio_dev->pdev->dev,
|
|
|
|
"Disabling spurious GPIO IRQ %d\n",
|
|
|
|
irqnr + i);
|
|
|
|
}
|
2017-05-24 05:23:32 +08:00
|
|
|
writel(regval, regs + i);
|
pinctrl/amd: fix masking of GPIO interrupts
On Asus laptop models X505BA, X505BP, X542BA and X542BP, the i2c-hid
touchpad (using a GPIO for interrupts) becomes unresponsive after a
few minutes of usage, or after placing two fingers on the touchpad,
which seems to have the effect of queuing up a large amount of input
data to be transferred.
When the touchpad is in unresponsive state, we observed that the GPIO
level-triggered interrupt is still at it's active level, however the
pinctrl-amd driver is not receiving/dispatching more interrupts at this
point.
After the initial interrupt arrives, amd_gpio_irq_mask() is called
however we then see amd_gpio_irq_handler() being called repeatedly for
the same irq; the interrupt mask is not taking effect because of the
following sequence of events:
- amd_gpio_irq_handler fires, reads and caches pin reg
- amd_gpio_irq_handler calls generic_handle_irq()
- During IRQ handling, amd_gpio_irq_mask() is called and modifies pin reg
- amd_gpio_irq_handler clears interrupt by writing cached value
The stale cached value written at the final stage undoes the masking.
Fix this by re-reading the register before clearing the interrupt.
I also spotted that the interrupt-clearing code can race against
amd_gpio_irq_mask() / amd_gpio_irq_unmask(), so add locking there.
Presumably this race was leading to the loss of interrupts.
After these changes, the touchpad appears to be working fine.
Signed-off-by: Daniel Drake <drake@endlessm.com>
Acked-by: Shah, Nehal-bakulchandra <Nehal-Bakulchandra.shah@amd.com>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
2017-10-02 12:00:54 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2017-05-24 05:23:32 +08:00
|
|
|
ret = IRQ_HANDLED;
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-24 05:23:32 +08:00
|
|
|
/* Signal EOI to the GPIO unit */
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2017-05-24 05:23:32 +08:00
|
|
|
regval = readl(gpio_dev->base + WAKE_INT_MASTER_REG);
|
|
|
|
regval |= EOI_MASK;
|
|
|
|
writel(regval, gpio_dev->base + WAKE_INT_MASTER_REG);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-05-24 05:23:32 +08:00
|
|
|
return ret;
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_get_groups_count(struct pinctrl_dev *pctldev)
|
|
|
|
{
|
|
|
|
struct amd_gpio *gpio_dev = pinctrl_dev_get_drvdata(pctldev);
|
|
|
|
|
|
|
|
return gpio_dev->ngroups;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *amd_get_group_name(struct pinctrl_dev *pctldev,
|
|
|
|
unsigned group)
|
|
|
|
{
|
|
|
|
struct amd_gpio *gpio_dev = pinctrl_dev_get_drvdata(pctldev);
|
|
|
|
|
|
|
|
return gpio_dev->groups[group].name;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_get_group_pins(struct pinctrl_dev *pctldev,
|
|
|
|
unsigned group,
|
|
|
|
const unsigned **pins,
|
|
|
|
unsigned *num_pins)
|
|
|
|
{
|
|
|
|
struct amd_gpio *gpio_dev = pinctrl_dev_get_drvdata(pctldev);
|
|
|
|
|
|
|
|
*pins = gpio_dev->groups[group].pins;
|
|
|
|
*num_pins = gpio_dev->groups[group].npins;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct pinctrl_ops amd_pinctrl_ops = {
|
|
|
|
.get_groups_count = amd_get_groups_count,
|
|
|
|
.get_group_name = amd_get_group_name,
|
|
|
|
.get_group_pins = amd_get_group_pins,
|
|
|
|
#ifdef CONFIG_OF
|
|
|
|
.dt_node_to_map = pinconf_generic_dt_node_to_map_group,
|
2016-03-31 19:44:42 +08:00
|
|
|
.dt_free_map = pinctrl_utils_free_map,
|
2015-03-10 15:02:19 +08:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
static int amd_pinconf_get(struct pinctrl_dev *pctldev,
|
|
|
|
unsigned int pin,
|
|
|
|
unsigned long *config)
|
|
|
|
{
|
|
|
|
u32 pin_reg;
|
|
|
|
unsigned arg;
|
|
|
|
unsigned long flags;
|
|
|
|
struct amd_gpio *gpio_dev = pinctrl_dev_get_drvdata(pctldev);
|
|
|
|
enum pin_config_param param = pinconf_to_config_param(*config);
|
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
pin_reg = readl(gpio_dev->base + pin*4);
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
switch (param) {
|
|
|
|
case PIN_CONFIG_INPUT_DEBOUNCE:
|
|
|
|
arg = pin_reg & DB_TMR_OUT_MASK;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PIN_CONFIG_BIAS_PULL_DOWN:
|
|
|
|
arg = (pin_reg >> PULL_DOWN_ENABLE_OFF) & BIT(0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PIN_CONFIG_BIAS_PULL_UP:
|
|
|
|
arg = (pin_reg >> PULL_UP_SEL_OFF) & (BIT(0) | BIT(1));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PIN_CONFIG_DRIVE_STRENGTH:
|
|
|
|
arg = (pin_reg >> DRV_STRENGTH_SEL_OFF) & DRV_STRENGTH_SEL_MASK;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
dev_err(&gpio_dev->pdev->dev, "Invalid config param %04x\n",
|
|
|
|
param);
|
|
|
|
return -ENOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
*config = pinconf_to_config_packed(param, arg);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
|
|
|
|
unsigned long *configs, unsigned num_configs)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
u32 arg;
|
2015-03-27 17:44:26 +08:00
|
|
|
int ret = 0;
|
|
|
|
u32 pin_reg;
|
2015-03-10 15:02:19 +08:00
|
|
|
unsigned long flags;
|
|
|
|
enum pin_config_param param;
|
|
|
|
struct amd_gpio *gpio_dev = pinctrl_dev_get_drvdata(pctldev);
|
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_irqsave(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
for (i = 0; i < num_configs; i++) {
|
|
|
|
param = pinconf_to_config_param(configs[i]);
|
|
|
|
arg = pinconf_to_config_argument(configs[i]);
|
|
|
|
pin_reg = readl(gpio_dev->base + pin*4);
|
|
|
|
|
|
|
|
switch (param) {
|
|
|
|
case PIN_CONFIG_INPUT_DEBOUNCE:
|
|
|
|
pin_reg &= ~DB_TMR_OUT_MASK;
|
|
|
|
pin_reg |= arg & DB_TMR_OUT_MASK;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PIN_CONFIG_BIAS_PULL_DOWN:
|
|
|
|
pin_reg &= ~BIT(PULL_DOWN_ENABLE_OFF);
|
|
|
|
pin_reg |= (arg & BIT(0)) << PULL_DOWN_ENABLE_OFF;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PIN_CONFIG_BIAS_PULL_UP:
|
|
|
|
pin_reg &= ~BIT(PULL_UP_SEL_OFF);
|
|
|
|
pin_reg |= (arg & BIT(0)) << PULL_UP_SEL_OFF;
|
|
|
|
pin_reg &= ~BIT(PULL_UP_ENABLE_OFF);
|
|
|
|
pin_reg |= ((arg>>1) & BIT(0)) << PULL_UP_ENABLE_OFF;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PIN_CONFIG_DRIVE_STRENGTH:
|
|
|
|
pin_reg &= ~(DRV_STRENGTH_SEL_MASK
|
|
|
|
<< DRV_STRENGTH_SEL_OFF);
|
|
|
|
pin_reg |= (arg & DRV_STRENGTH_SEL_MASK)
|
|
|
|
<< DRV_STRENGTH_SEL_OFF;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
dev_err(&gpio_dev->pdev->dev,
|
|
|
|
"Invalid config param %04x\n", param);
|
2015-03-27 17:44:26 +08:00
|
|
|
ret = -ENOTSUPP;
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
writel(pin_reg, gpio_dev->base + pin*4);
|
|
|
|
}
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_unlock_irqrestore(&gpio_dev->lock, flags);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2015-03-27 17:44:26 +08:00
|
|
|
return ret;
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_pinconf_group_get(struct pinctrl_dev *pctldev,
|
|
|
|
unsigned int group,
|
|
|
|
unsigned long *config)
|
|
|
|
{
|
|
|
|
const unsigned *pins;
|
|
|
|
unsigned npins;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = amd_get_group_pins(pctldev, group, &pins, &npins);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (amd_pinconf_get(pctldev, pins[0], config))
|
|
|
|
return -ENOTSUPP;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_pinconf_group_set(struct pinctrl_dev *pctldev,
|
|
|
|
unsigned group, unsigned long *configs,
|
|
|
|
unsigned num_configs)
|
|
|
|
{
|
|
|
|
const unsigned *pins;
|
|
|
|
unsigned npins;
|
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
ret = amd_get_group_pins(pctldev, group, &pins, &npins);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
for (i = 0; i < npins; i++) {
|
|
|
|
if (amd_pinconf_set(pctldev, pins[i], configs, num_configs))
|
|
|
|
return -ENOTSUPP;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct pinconf_ops amd_pinconf_ops = {
|
|
|
|
.pin_config_get = amd_pinconf_get,
|
|
|
|
.pin_config_set = amd_pinconf_set,
|
|
|
|
.pin_config_group_get = amd_pinconf_group_get,
|
|
|
|
.pin_config_group_set = amd_pinconf_group_set,
|
|
|
|
};
|
|
|
|
|
2017-09-11 14:11:56 +08:00
|
|
|
#ifdef CONFIG_PM_SLEEP
|
|
|
|
static bool amd_gpio_should_save(struct amd_gpio *gpio_dev, unsigned int pin)
|
|
|
|
{
|
|
|
|
const struct pin_desc *pd = pin_desc_get(gpio_dev->pctrl, pin);
|
|
|
|
|
|
|
|
if (!pd)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Only restore the pin if it is actually in use by the kernel (or
|
|
|
|
* by userspace).
|
|
|
|
*/
|
|
|
|
if (pd->mux_owner || pd->gpio_owner ||
|
|
|
|
gpiochip_line_is_irq(&gpio_dev->gc, pin))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-09-14 00:15:01 +08:00
|
|
|
static int amd_gpio_suspend(struct device *dev)
|
2017-09-11 14:11:56 +08:00
|
|
|
{
|
2018-10-22 04:00:30 +08:00
|
|
|
struct amd_gpio *gpio_dev = dev_get_drvdata(dev);
|
2017-09-11 14:11:56 +08:00
|
|
|
struct pinctrl_desc *desc = gpio_dev->pctrl->desc;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < desc->npins; i++) {
|
|
|
|
int pin = desc->pins[i].number;
|
|
|
|
|
|
|
|
if (!amd_gpio_should_save(gpio_dev, pin))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
gpio_dev->saved_regs[i] = readl(gpio_dev->base + pin*4);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-09-14 00:15:01 +08:00
|
|
|
static int amd_gpio_resume(struct device *dev)
|
2017-09-11 14:11:56 +08:00
|
|
|
{
|
2018-10-22 04:00:30 +08:00
|
|
|
struct amd_gpio *gpio_dev = dev_get_drvdata(dev);
|
2017-09-11 14:11:56 +08:00
|
|
|
struct pinctrl_desc *desc = gpio_dev->pctrl->desc;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < desc->npins; i++) {
|
|
|
|
int pin = desc->pins[i].number;
|
|
|
|
|
|
|
|
if (!amd_gpio_should_save(gpio_dev, pin))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
writel(gpio_dev->saved_regs[i], gpio_dev->base + pin*4);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct dev_pm_ops amd_gpio_pm_ops = {
|
|
|
|
SET_LATE_SYSTEM_SLEEP_PM_OPS(amd_gpio_suspend,
|
|
|
|
amd_gpio_resume)
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2015-03-10 15:02:19 +08:00
|
|
|
static struct pinctrl_desc amd_pinctrl_desc = {
|
|
|
|
.pins = kerncz_pins,
|
|
|
|
.npins = ARRAY_SIZE(kerncz_pins),
|
|
|
|
.pctlops = &amd_pinctrl_ops,
|
|
|
|
.confops = &amd_pinconf_ops,
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int amd_gpio_probe(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
2015-03-27 17:44:26 +08:00
|
|
|
int irq_base;
|
2015-03-10 15:02:19 +08:00
|
|
|
struct resource *res;
|
|
|
|
struct amd_gpio *gpio_dev;
|
2020-07-22 18:15:45 +08:00
|
|
|
struct gpio_irq_chip *girq;
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
gpio_dev = devm_kzalloc(&pdev->dev,
|
|
|
|
sizeof(struct amd_gpio), GFP_KERNEL);
|
|
|
|
if (!gpio_dev)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2017-03-10 00:22:04 +08:00
|
|
|
raw_spin_lock_init(&gpio_dev->lock);
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
|
|
if (!res) {
|
|
|
|
dev_err(&pdev->dev, "Failed to get gpio io resource.\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2020-01-06 16:43:50 +08:00
|
|
|
gpio_dev->base = devm_ioremap(&pdev->dev, res->start,
|
2015-03-10 15:02:19 +08:00
|
|
|
resource_size(res));
|
2016-02-06 22:56:36 +08:00
|
|
|
if (!gpio_dev->base)
|
|
|
|
return -ENOMEM;
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
irq_base = platform_get_irq(pdev, 0);
|
2019-07-31 02:15:33 +08:00
|
|
|
if (irq_base < 0)
|
2017-08-10 00:09:33 +08:00
|
|
|
return irq_base;
|
2015-03-10 15:02:19 +08:00
|
|
|
|
2017-09-11 14:11:56 +08:00
|
|
|
#ifdef CONFIG_PM_SLEEP
|
|
|
|
gpio_dev->saved_regs = devm_kcalloc(&pdev->dev, amd_pinctrl_desc.npins,
|
|
|
|
sizeof(*gpio_dev->saved_regs),
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!gpio_dev->saved_regs)
|
|
|
|
return -ENOMEM;
|
|
|
|
#endif
|
|
|
|
|
2015-03-10 15:02:19 +08:00
|
|
|
gpio_dev->pdev = pdev;
|
2018-02-17 03:12:43 +08:00
|
|
|
gpio_dev->gc.get_direction = amd_gpio_get_direction;
|
2015-03-10 15:02:19 +08:00
|
|
|
gpio_dev->gc.direction_input = amd_gpio_direction_input;
|
|
|
|
gpio_dev->gc.direction_output = amd_gpio_direction_output;
|
|
|
|
gpio_dev->gc.get = amd_gpio_get_value;
|
|
|
|
gpio_dev->gc.set = amd_gpio_set_value;
|
2017-01-23 20:34:34 +08:00
|
|
|
gpio_dev->gc.set_config = amd_gpio_set_config;
|
2015-03-10 15:02:19 +08:00
|
|
|
gpio_dev->gc.dbg_show = amd_gpio_dbg_show;
|
|
|
|
|
2016-12-06 14:47:48 +08:00
|
|
|
gpio_dev->gc.base = -1;
|
2015-03-10 15:02:19 +08:00
|
|
|
gpio_dev->gc.label = pdev->name;
|
|
|
|
gpio_dev->gc.owner = THIS_MODULE;
|
2015-11-04 16:56:26 +08:00
|
|
|
gpio_dev->gc.parent = &pdev->dev;
|
2016-12-06 14:47:48 +08:00
|
|
|
gpio_dev->gc.ngpio = resource_size(res) / 4;
|
2015-03-10 15:02:19 +08:00
|
|
|
#if defined(CONFIG_OF_GPIO)
|
|
|
|
gpio_dev->gc.of_node = pdev->dev.of_node;
|
|
|
|
#endif
|
|
|
|
|
2016-12-06 14:47:48 +08:00
|
|
|
gpio_dev->hwbank_num = gpio_dev->gc.ngpio / 64;
|
2015-03-10 15:02:19 +08:00
|
|
|
gpio_dev->groups = kerncz_groups;
|
|
|
|
gpio_dev->ngroups = ARRAY_SIZE(kerncz_groups);
|
|
|
|
|
|
|
|
amd_pinctrl_desc.name = dev_name(&pdev->dev);
|
2016-02-24 17:14:07 +08:00
|
|
|
gpio_dev->pctrl = devm_pinctrl_register(&pdev->dev, &amd_pinctrl_desc,
|
|
|
|
gpio_dev);
|
2015-06-09 12:01:16 +08:00
|
|
|
if (IS_ERR(gpio_dev->pctrl)) {
|
2015-03-10 15:02:19 +08:00
|
|
|
dev_err(&pdev->dev, "Couldn't register pinctrl driver\n");
|
2015-06-09 12:01:16 +08:00
|
|
|
return PTR_ERR(gpio_dev->pctrl);
|
2015-03-10 15:02:19 +08:00
|
|
|
}
|
|
|
|
|
2020-07-22 18:15:45 +08:00
|
|
|
girq = &gpio_dev->gc.irq;
|
|
|
|
girq->chip = &amd_gpio_irqchip;
|
|
|
|
/* This will let us handle the parent IRQ in the driver */
|
|
|
|
girq->parent_handler = NULL;
|
|
|
|
girq->num_parents = 0;
|
|
|
|
girq->parents = NULL;
|
|
|
|
girq->default_type = IRQ_TYPE_NONE;
|
|
|
|
girq->handler = handle_simple_irq;
|
|
|
|
|
2015-12-08 16:21:38 +08:00
|
|
|
ret = gpiochip_add_data(&gpio_dev->gc, gpio_dev);
|
2015-03-10 15:02:19 +08:00
|
|
|
if (ret)
|
2016-02-24 17:14:07 +08:00
|
|
|
return ret;
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
ret = gpiochip_add_pin_range(&gpio_dev->gc, dev_name(&pdev->dev),
|
2016-12-06 14:47:48 +08:00
|
|
|
0, 0, gpio_dev->gc.ngpio);
|
2015-03-10 15:02:19 +08:00
|
|
|
if (ret) {
|
|
|
|
dev_err(&pdev->dev, "Failed to add pin range\n");
|
|
|
|
goto out2;
|
|
|
|
}
|
|
|
|
|
2019-04-04 21:16:26 +08:00
|
|
|
ret = devm_request_irq(&pdev->dev, irq_base, amd_gpio_irq_handler,
|
|
|
|
IRQF_SHARED, KBUILD_MODNAME, gpio_dev);
|
2017-05-24 05:23:32 +08:00
|
|
|
if (ret)
|
|
|
|
goto out2;
|
|
|
|
|
2015-03-10 15:02:19 +08:00
|
|
|
platform_set_drvdata(pdev, gpio_dev);
|
|
|
|
|
|
|
|
dev_dbg(&pdev->dev, "amd gpio driver loaded\n");
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
out2:
|
|
|
|
gpiochip_remove(&gpio_dev->gc);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int amd_gpio_remove(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct amd_gpio *gpio_dev;
|
|
|
|
|
|
|
|
gpio_dev = platform_get_drvdata(pdev);
|
|
|
|
|
|
|
|
gpiochip_remove(&gpio_dev->gc);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-07-13 22:49:30 +08:00
|
|
|
#ifdef CONFIG_ACPI
|
2015-03-10 15:02:19 +08:00
|
|
|
static const struct acpi_device_id amd_gpio_acpi_match[] = {
|
|
|
|
{ "AMD0030", 0 },
|
2016-03-11 10:58:42 +08:00
|
|
|
{ "AMDI0030", 0},
|
2015-03-10 15:02:19 +08:00
|
|
|
{ },
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(acpi, amd_gpio_acpi_match);
|
2020-07-13 22:49:30 +08:00
|
|
|
#endif
|
2015-03-10 15:02:19 +08:00
|
|
|
|
|
|
|
static struct platform_driver amd_gpio_driver = {
|
|
|
|
.driver = {
|
|
|
|
.name = "amd_gpio",
|
|
|
|
.acpi_match_table = ACPI_PTR(amd_gpio_acpi_match),
|
2017-09-11 14:11:56 +08:00
|
|
|
#ifdef CONFIG_PM_SLEEP
|
|
|
|
.pm = &amd_gpio_pm_ops,
|
|
|
|
#endif
|
2015-03-10 15:02:19 +08:00
|
|
|
},
|
|
|
|
.probe = amd_gpio_probe,
|
|
|
|
.remove = amd_gpio_remove,
|
|
|
|
};
|
|
|
|
|
|
|
|
module_platform_driver(amd_gpio_driver);
|
|
|
|
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
|
|
MODULE_AUTHOR("Ken Xue <Ken.Xue@amd.com>, Jeff Wu <Jeff.Wu@amd.com>");
|
|
|
|
MODULE_DESCRIPTION("AMD GPIO pinctrl driver");
|