gpio/omap: fix debounce clock handling
The dbck_enable_mask indicates which all GPIOs within a bank have debounce enabled and dbck is enabled/disabled based upon this. But there is no mechanism to track the dbck state. In order to manage the dbck state we need additional flag and logic so that turning off/on dbck is synchronized with pm_runtime_put/get_sync calls. Signed-off-by: Tarun Kanti DebBarma <tarun.kanti@ti.com> Reviewed-by: Santosh Shilimkar <santosh.shilimkar@ti.com> Reviewed-by: Kevin Hilman <khilman@ti.com> Signed-off-by: Kevin Hilman <khilman@ti.com>
This commit is contained in:
parent
2dc983c565
commit
72f83af998
|
@ -67,6 +67,7 @@ struct gpio_bank {
|
||||||
struct clk *dbck;
|
struct clk *dbck;
|
||||||
u32 mod_usage;
|
u32 mod_usage;
|
||||||
u32 dbck_enable_mask;
|
u32 dbck_enable_mask;
|
||||||
|
bool dbck_enabled;
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
bool is_mpuio;
|
bool is_mpuio;
|
||||||
bool dbck_flag;
|
bool dbck_flag;
|
||||||
|
@ -158,6 +159,22 @@ static inline void _gpio_rmw(void __iomem *base, u32 reg, u32 mask, bool set)
|
||||||
__raw_writel(l, base + reg);
|
__raw_writel(l, base + reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void _gpio_dbck_enable(struct gpio_bank *bank)
|
||||||
|
{
|
||||||
|
if (bank->dbck_enable_mask && !bank->dbck_enabled) {
|
||||||
|
clk_enable(bank->dbck);
|
||||||
|
bank->dbck_enabled = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void _gpio_dbck_disable(struct gpio_bank *bank)
|
||||||
|
{
|
||||||
|
if (bank->dbck_enable_mask && bank->dbck_enabled) {
|
||||||
|
clk_disable(bank->dbck);
|
||||||
|
bank->dbck_enabled = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* _set_gpio_debounce - low level gpio debounce time
|
* _set_gpio_debounce - low level gpio debounce time
|
||||||
* @bank: the gpio bank we're acting upon
|
* @bank: the gpio bank we're acting upon
|
||||||
|
@ -1178,6 +1195,7 @@ static int omap_gpio_runtime_suspend(struct device *dev)
|
||||||
bank->get_context_loss_count(bank->dev);
|
bank->get_context_loss_count(bank->dev);
|
||||||
|
|
||||||
omap_gpio_save_context(bank);
|
omap_gpio_save_context(bank);
|
||||||
|
_gpio_dbck_disable(bank);
|
||||||
spin_unlock_irqrestore(&bank->lock, flags);
|
spin_unlock_irqrestore(&bank->lock, flags);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1192,6 +1210,7 @@ static int omap_gpio_runtime_resume(struct device *dev)
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
spin_lock_irqsave(&bank->lock, flags);
|
spin_lock_irqsave(&bank->lock, flags);
|
||||||
|
_gpio_dbck_enable(bank);
|
||||||
if (!bank->enabled_non_wakeup_gpios || !bank->workaround_enabled) {
|
if (!bank->enabled_non_wakeup_gpios || !bank->workaround_enabled) {
|
||||||
spin_unlock_irqrestore(&bank->lock, flags);
|
spin_unlock_irqrestore(&bank->lock, flags);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1274,16 +1293,11 @@ void omap2_gpio_prepare_for_idle(int pwr_mode)
|
||||||
struct gpio_bank *bank;
|
struct gpio_bank *bank;
|
||||||
|
|
||||||
list_for_each_entry(bank, &omap_gpio_list, node) {
|
list_for_each_entry(bank, &omap_gpio_list, node) {
|
||||||
int j;
|
|
||||||
|
|
||||||
if (!bank->mod_usage || !bank->loses_context)
|
if (!bank->mod_usage || !bank->loses_context)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
bank->power_mode = pwr_mode;
|
bank->power_mode = pwr_mode;
|
||||||
|
|
||||||
for (j = 0; j < hweight_long(bank->dbck_enable_mask); j++)
|
|
||||||
clk_disable(bank->dbck);
|
|
||||||
|
|
||||||
pm_runtime_put_sync_suspend(bank->dev);
|
pm_runtime_put_sync_suspend(bank->dev);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1293,14 +1307,9 @@ void omap2_gpio_resume_after_idle(void)
|
||||||
struct gpio_bank *bank;
|
struct gpio_bank *bank;
|
||||||
|
|
||||||
list_for_each_entry(bank, &omap_gpio_list, node) {
|
list_for_each_entry(bank, &omap_gpio_list, node) {
|
||||||
int j;
|
|
||||||
|
|
||||||
if (!bank->mod_usage || !bank->loses_context)
|
if (!bank->mod_usage || !bank->loses_context)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
for (j = 0; j < hweight_long(bank->dbck_enable_mask); j++)
|
|
||||||
clk_enable(bank->dbck);
|
|
||||||
|
|
||||||
pm_runtime_get_sync(bank->dev);
|
pm_runtime_get_sync(bank->dev);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user