|
9 | 9 | #include <linux/input.h> |
10 | 10 | #include <linux/interrupt.h> |
11 | 11 | #include <linux/kernel.h> |
| 12 | +#include <linux/ktime.h> |
12 | 13 | #include <linux/log2.h> |
13 | 14 | #include <linux/module.h> |
14 | 15 | #include <linux/of.h> |
|
20 | 21 |
|
21 | 22 | #define PON_REV2 0x01 |
22 | 23 |
|
| 24 | +#define PON_SUBTYPE 0x05 |
| 25 | + |
| 26 | +#define PON_SUBTYPE_PRIMARY 0x01 |
| 27 | +#define PON_SUBTYPE_SECONDARY 0x02 |
| 28 | +#define PON_SUBTYPE_1REG 0x03 |
| 29 | +#define PON_SUBTYPE_GEN2_PRIMARY 0x04 |
| 30 | +#define PON_SUBTYPE_GEN2_SECONDARY 0x05 |
| 31 | +#define PON_SUBTYPE_GEN3_PBS 0x08 |
| 32 | +#define PON_SUBTYPE_GEN3_HLOS 0x09 |
| 33 | + |
23 | 34 | #define PON_RT_STS 0x10 |
24 | 35 | #define PON_KPDPWR_N_SET BIT(0) |
25 | 36 | #define PON_RESIN_N_SET BIT(1) |
@@ -60,9 +71,12 @@ struct pm8941_pwrkey { |
60 | 71 | struct input_dev *input; |
61 | 72 |
|
62 | 73 | unsigned int revision; |
| 74 | + unsigned int subtype; |
63 | 75 | struct notifier_block reboot_notifier; |
64 | 76 |
|
65 | 77 | u32 code; |
| 78 | + u32 sw_debounce_time_us; |
| 79 | + ktime_t sw_debounce_end_time; |
66 | 80 | const struct pm8941_data *data; |
67 | 81 | }; |
68 | 82 |
|
@@ -132,20 +146,66 @@ static irqreturn_t pm8941_pwrkey_irq(int irq, void *_data) |
132 | 146 | { |
133 | 147 | struct pm8941_pwrkey *pwrkey = _data; |
134 | 148 | unsigned int sts; |
135 | | - int error; |
| 149 | + int err; |
| 150 | + |
| 151 | + if (pwrkey->sw_debounce_time_us) { |
| 152 | + if (ktime_before(ktime_get(), pwrkey->sw_debounce_end_time)) { |
| 153 | + dev_dbg(pwrkey->dev, |
| 154 | + "ignoring key event received before debounce end %llu us\n", |
| 155 | + pwrkey->sw_debounce_end_time); |
| 156 | + return IRQ_HANDLED; |
| 157 | + } |
| 158 | + } |
136 | 159 |
|
137 | | - error = regmap_read(pwrkey->regmap, |
138 | | - pwrkey->baseaddr + PON_RT_STS, &sts); |
139 | | - if (error) |
| 160 | + err = regmap_read(pwrkey->regmap, pwrkey->baseaddr + PON_RT_STS, &sts); |
| 161 | + if (err) |
140 | 162 | return IRQ_HANDLED; |
141 | 163 |
|
142 | | - input_report_key(pwrkey->input, pwrkey->code, |
143 | | - sts & pwrkey->data->status_bit); |
| 164 | + sts &= pwrkey->data->status_bit; |
| 165 | + |
| 166 | + if (pwrkey->sw_debounce_time_us && !sts) |
| 167 | + pwrkey->sw_debounce_end_time = ktime_add_us(ktime_get(), |
| 168 | + pwrkey->sw_debounce_time_us); |
| 169 | + |
| 170 | + input_report_key(pwrkey->input, pwrkey->code, sts); |
144 | 171 | input_sync(pwrkey->input); |
145 | 172 |
|
146 | 173 | return IRQ_HANDLED; |
147 | 174 | } |
148 | 175 |
|
| 176 | +static int pm8941_pwrkey_sw_debounce_init(struct pm8941_pwrkey *pwrkey) |
| 177 | +{ |
| 178 | + unsigned int val, addr, mask; |
| 179 | + int error; |
| 180 | + |
| 181 | + if (pwrkey->data->has_pon_pbs && !pwrkey->pon_pbs_baseaddr) { |
| 182 | + dev_err(pwrkey->dev, |
| 183 | + "PON_PBS address missing, can't read HW debounce time\n"); |
| 184 | + return 0; |
| 185 | + } |
| 186 | + |
| 187 | + if (pwrkey->pon_pbs_baseaddr) |
| 188 | + addr = pwrkey->pon_pbs_baseaddr + PON_DBC_CTL; |
| 189 | + else |
| 190 | + addr = pwrkey->baseaddr + PON_DBC_CTL; |
| 191 | + error = regmap_read(pwrkey->regmap, addr, &val); |
| 192 | + if (error) |
| 193 | + return error; |
| 194 | + |
| 195 | + if (pwrkey->subtype >= PON_SUBTYPE_GEN2_PRIMARY) |
| 196 | + mask = 0xf; |
| 197 | + else |
| 198 | + mask = 0x7; |
| 199 | + |
| 200 | + pwrkey->sw_debounce_time_us = |
| 201 | + 2 * USEC_PER_SEC / (1 << (mask - (val & mask))); |
| 202 | + |
| 203 | + dev_dbg(pwrkey->dev, "SW debounce time = %u us\n", |
| 204 | + pwrkey->sw_debounce_time_us); |
| 205 | + |
| 206 | + return 0; |
| 207 | +} |
| 208 | + |
149 | 209 | static int __maybe_unused pm8941_pwrkey_suspend(struct device *dev) |
150 | 210 | { |
151 | 211 | struct pm8941_pwrkey *pwrkey = dev_get_drvdata(dev); |
@@ -238,6 +298,13 @@ static int pm8941_pwrkey_probe(struct platform_device *pdev) |
238 | 298 | return error; |
239 | 299 | } |
240 | 300 |
|
| 301 | + error = regmap_read(pwrkey->regmap, pwrkey->baseaddr + PON_SUBTYPE, |
| 302 | + &pwrkey->subtype); |
| 303 | + if (error) { |
| 304 | + dev_err(&pdev->dev, "failed to read subtype: %d\n", error); |
| 305 | + return error; |
| 306 | + } |
| 307 | + |
241 | 308 | error = of_property_read_u32(pdev->dev.of_node, "linux,code", |
242 | 309 | &pwrkey->code); |
243 | 310 | if (error) { |
@@ -272,6 +339,10 @@ static int pm8941_pwrkey_probe(struct platform_device *pdev) |
272 | 339 | } |
273 | 340 | } |
274 | 341 |
|
| 342 | + error = pm8941_pwrkey_sw_debounce_init(pwrkey); |
| 343 | + if (error) |
| 344 | + return error; |
| 345 | + |
275 | 346 | if (pwrkey->data->pull_up_bit) { |
276 | 347 | error = regmap_update_bits(pwrkey->regmap, |
277 | 348 | pwrkey->baseaddr + PON_PULL_CTL, |
|
0 commit comments