2 * Driver for the IMX SNVS ON/OFF Power Key
3 * Copyright (C) 2015 Freescale Semiconductor, Inc. All Rights Reserved.
5 * The code contained herein is licensed under the GNU General Public
6 * License. You may obtain a copy of the GNU General Public License
7 * Version 2 or later at the following locations:
9 * http://www.opensource.org/licenses/gpl-license.html
10 * http://www.gnu.org/copyleft/gpl.html
13 #include <linux/device.h>
14 #include <linux/err.h>
15 #include <linux/init.h>
16 #include <linux/input.h>
17 #include <linux/interrupt.h>
19 #include <linux/jiffies.h>
20 #include <linux/kernel.h>
21 #include <linux/module.h>
23 #include <linux/of_address.h>
24 #include <linux/platform_device.h>
25 #include <linux/mfd/syscon.h>
26 #include <linux/regmap.h>
28 #define SNVS_LPSR_REG 0x4C /* LP Status Register */
29 #define SNVS_LPCR_REG 0x38 /* LP Control Register */
30 #define SNVS_HPSR_REG 0x14
31 #define SNVS_HPSR_BTN BIT(6)
32 #define SNVS_LPSR_SPO BIT(18)
33 #define SNVS_LPCR_DEP_EN BIT(5)
35 #define DEBOUNCE_TIME 30
36 #define REPEAT_INTERVAL 60
38 struct pwrkey_drv_data
{
42 int keystate
; /* 1:pressed */
44 struct timer_list check_timer
;
45 struct input_dev
*input
;
48 static void imx_imx_snvs_check_for_events(struct timer_list
*t
)
50 struct pwrkey_drv_data
*pdata
= from_timer(pdata
, t
, check_timer
);
51 struct input_dev
*input
= pdata
->input
;
54 regmap_read(pdata
->snvs
, SNVS_HPSR_REG
, &state
);
55 state
= state
& SNVS_HPSR_BTN
? 1 : 0;
57 /* only report new event if status changed */
58 if (state
^ pdata
->keystate
) {
59 pdata
->keystate
= state
;
60 input_event(input
, EV_KEY
, pdata
->keycode
, state
);
62 pm_relax(pdata
->input
->dev
.parent
);
65 /* repeat check if pressed long */
67 mod_timer(&pdata
->check_timer
,
68 jiffies
+ msecs_to_jiffies(REPEAT_INTERVAL
));
72 static irqreturn_t
imx_snvs_pwrkey_interrupt(int irq
, void *dev_id
)
74 struct platform_device
*pdev
= dev_id
;
75 struct pwrkey_drv_data
*pdata
= platform_get_drvdata(pdev
);
78 pm_wakeup_event(pdata
->input
->dev
.parent
, 0);
80 regmap_read(pdata
->snvs
, SNVS_LPSR_REG
, &lp_status
);
81 if (lp_status
& SNVS_LPSR_SPO
)
82 mod_timer(&pdata
->check_timer
, jiffies
+ msecs_to_jiffies(DEBOUNCE_TIME
));
84 /* clear SPO status */
85 regmap_write(pdata
->snvs
, SNVS_LPSR_REG
, SNVS_LPSR_SPO
);
90 static void imx_snvs_pwrkey_act(void *pdata
)
92 struct pwrkey_drv_data
*pd
= pdata
;
94 del_timer_sync(&pd
->check_timer
);
97 static int imx_snvs_pwrkey_probe(struct platform_device
*pdev
)
99 struct pwrkey_drv_data
*pdata
= NULL
;
100 struct input_dev
*input
= NULL
;
101 struct device_node
*np
;
104 /* Get SNVS register Page */
105 np
= pdev
->dev
.of_node
;
109 pdata
= devm_kzalloc(&pdev
->dev
, sizeof(*pdata
), GFP_KERNEL
);
113 pdata
->snvs
= syscon_regmap_lookup_by_phandle(np
, "regmap");
114 if (IS_ERR(pdata
->snvs
)) {
115 dev_err(&pdev
->dev
, "Can't get snvs syscon\n");
116 return PTR_ERR(pdata
->snvs
);
119 if (of_property_read_u32(np
, "linux,keycode", &pdata
->keycode
)) {
120 pdata
->keycode
= KEY_POWER
;
121 dev_warn(&pdev
->dev
, "KEY_POWER without setting in dts\n");
124 pdata
->wakeup
= of_property_read_bool(np
, "wakeup-source");
126 pdata
->irq
= platform_get_irq(pdev
, 0);
127 if (pdata
->irq
< 0) {
128 dev_err(&pdev
->dev
, "no irq defined in platform data\n");
132 regmap_update_bits(pdata
->snvs
, SNVS_LPCR_REG
, SNVS_LPCR_DEP_EN
, SNVS_LPCR_DEP_EN
);
134 /* clear the unexpected interrupt before driver ready */
135 regmap_write(pdata
->snvs
, SNVS_LPSR_REG
, SNVS_LPSR_SPO
);
137 timer_setup(&pdata
->check_timer
, imx_imx_snvs_check_for_events
, 0);
139 input
= devm_input_allocate_device(&pdev
->dev
);
141 dev_err(&pdev
->dev
, "failed to allocate the input device\n");
145 input
->name
= pdev
->name
;
146 input
->phys
= "snvs-pwrkey/input0";
147 input
->id
.bustype
= BUS_HOST
;
149 input_set_capability(input
, EV_KEY
, pdata
->keycode
);
151 /* input customer action to cancel release timer */
152 error
= devm_add_action(&pdev
->dev
, imx_snvs_pwrkey_act
, pdata
);
154 dev_err(&pdev
->dev
, "failed to register remove action\n");
158 error
= devm_request_irq(&pdev
->dev
, pdata
->irq
,
159 imx_snvs_pwrkey_interrupt
,
160 0, pdev
->name
, pdev
);
163 dev_err(&pdev
->dev
, "interrupt not available.\n");
167 error
= input_register_device(input
);
169 dev_err(&pdev
->dev
, "failed to register input device\n");
173 pdata
->input
= input
;
174 platform_set_drvdata(pdev
, pdata
);
176 device_init_wakeup(&pdev
->dev
, pdata
->wakeup
);
181 static int __maybe_unused
imx_snvs_pwrkey_suspend(struct device
*dev
)
183 struct platform_device
*pdev
= to_platform_device(dev
);
184 struct pwrkey_drv_data
*pdata
= platform_get_drvdata(pdev
);
186 if (device_may_wakeup(&pdev
->dev
))
187 enable_irq_wake(pdata
->irq
);
192 static int __maybe_unused
imx_snvs_pwrkey_resume(struct device
*dev
)
194 struct platform_device
*pdev
= to_platform_device(dev
);
195 struct pwrkey_drv_data
*pdata
= platform_get_drvdata(pdev
);
197 if (device_may_wakeup(&pdev
->dev
))
198 disable_irq_wake(pdata
->irq
);
203 static const struct of_device_id imx_snvs_pwrkey_ids
[] = {
204 { .compatible
= "fsl,sec-v4.0-pwrkey" },
207 MODULE_DEVICE_TABLE(of
, imx_snvs_pwrkey_ids
);
209 static SIMPLE_DEV_PM_OPS(imx_snvs_pwrkey_pm_ops
, imx_snvs_pwrkey_suspend
,
210 imx_snvs_pwrkey_resume
);
212 static struct platform_driver imx_snvs_pwrkey_driver
= {
214 .name
= "snvs_pwrkey",
215 .pm
= &imx_snvs_pwrkey_pm_ops
,
216 .of_match_table
= imx_snvs_pwrkey_ids
,
218 .probe
= imx_snvs_pwrkey_probe
,
220 module_platform_driver(imx_snvs_pwrkey_driver
);
222 MODULE_AUTHOR("Freescale Semiconductor");
223 MODULE_DESCRIPTION("i.MX snvs power key Driver");
224 MODULE_LICENSE("GPL");