aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDan Murphy <dmurphy@ti.com>2014-08-17 12:24:26 -0400
committerDmitry Torokhov <dmitry.torokhov@gmail.com>2014-08-20 01:19:28 -0400
commit7132fe4f568721cbd5d9bce5a8a71556e9bc45b4 (patch)
tree15fd0ff732e3c03fed8a94a905cb88a9378ce4f1
parent3361a97601f243f1842bee6ca709e399f47b2ce3 (diff)
Input: drv260x - add TI drv260x haptics driver
Add the TI drv260x haptics/vibrator driver. This device uses the input force feedback to produce a wave form to driver an ERM or LRA actuator device. The initial driver supports the devices real time playback mode. But the device has additional wave patterns in ROM. This functionality will be added in future patchsets. Product data sheet is located here: http://www.ti.com/product/drv2605 Signed-off-by: Dan Murphy <dmurphy@ti.com> Signed-off-by: Dmitry Torokhov <dmitry.torokhov@gmail.com>
-rw-r--r--Documentation/devicetree/bindings/input/ti,drv260x.txt51
-rw-r--r--drivers/input/misc/Kconfig11
-rw-r--r--drivers/input/misc/Makefile1
-rw-r--r--drivers/input/misc/drv260x.c738
-rw-r--r--include/dt-bindings/input/ti-drv260x.h36
-rw-r--r--include/linux/platform_data/drv260x-pdata.h28
6 files changed, 865 insertions, 0 deletions
diff --git a/Documentation/devicetree/bindings/input/ti,drv260x.txt b/Documentation/devicetree/bindings/input/ti,drv260x.txt
new file mode 100644
index 000000000000..a9c8519eb9de
--- /dev/null
+++ b/Documentation/devicetree/bindings/input/ti,drv260x.txt
@@ -0,0 +1,51 @@
1Texas Instruments - drv260x Haptics driver family
2
3The drv260x family serial control bus communicates through I2C protocols
4
5Required properties:
6 - compatible - One of:
7 "ti,drv2604" - DRV2604
8 "ti,drv2605" - DRV2605
9 "ti,drv2605l" - DRV2605L
10 - reg - I2C slave address
11 - vbat-supply - Required supply regulator
12 - mode - Power up mode of the chip (defined in include/dt-bindings/input/ti-drv260x.h)
13 DRV260X_LRA_MODE - Linear Resonance Actuator mode (Piezoelectric)
14 DRV260X_LRA_NO_CAL_MODE - This is a LRA Mode but there is no calibration
15 sequence during init. And the device is configured for real
16 time playback mode (RTP mode).
17 DRV260X_ERM_MODE - Eccentric Rotating Mass mode (Rotary vibrator)
18 - library-sel - These are ROM based waveforms pre-programmed into the IC.
19 This should be set to set the library to use at power up.
20 (defined in include/dt-bindings/input/ti-drv260x.h)
21 DRV260X_LIB_EMPTY - Do not use a pre-programmed library
22 DRV260X_ERM_LIB_A - Pre-programmed Library
23 DRV260X_ERM_LIB_B - Pre-programmed Library
24 DRV260X_ERM_LIB_C - Pre-programmed Library
25 DRV260X_ERM_LIB_D - Pre-programmed Library
26 DRV260X_ERM_LIB_E - Pre-programmed Library
27 DRV260X_ERM_LIB_F - Pre-programmed Library
28 DRV260X_LIB_LRA - Pre-programmed LRA Library
29
30Optional properties:
31 - enable-gpio - gpio pin to enable/disable the device.
32 - vib-rated-mv - The rated voltage of the actuator in millivolts.
33 If this is not set then the value will be defaulted to
34 3.2 v.
35 - vib-overdrive-mv - The overdrive voltage of the actuator in millivolts.
36 If this is not set then the value will be defaulted to
37 3.2 v.
38Example:
39
40drv2605l: drv2605l@5a {
41 compatible = "ti,drv2605l";
42 reg = <0x5a>;
43 enable-gpio = <&gpio1 28 GPIO_ACTIVE_HIGH>;
44 mode = <DRV260X_LRA_MODE>;
45 library-sel = <DRV260X_LIB_LRA>;
46 vib-rated-mv = <3200>;
47 vib-overdriver-mv = <3200>;
48};
49
50For more product information please see the link below:
51http://www.ti.com/product/drv2605
diff --git a/drivers/input/misc/Kconfig b/drivers/input/misc/Kconfig
index 2ff4425a893b..41d0ae62ba05 100644
--- a/drivers/input/misc/Kconfig
+++ b/drivers/input/misc/Kconfig
@@ -676,4 +676,15 @@ config INPUT_SOC_BUTTON_ARRAY
676 To compile this driver as a module, choose M here: the 676 To compile this driver as a module, choose M here: the
677 module will be called soc_button_array. 677 module will be called soc_button_array.
678 678
679config INPUT_DRV260X_HAPTICS
680 tristate "TI DRV260X haptics support"
681 depends on INPUT && I2C && GPIOLIB
682 select INPUT_FF_MEMLESS
683 select REGMAP_I2C
684 help
685 Say Y to enable support for the TI DRV260X haptics driver.
686
687 To compile this driver as a module, choose M here: the
688 module will be called drv260x-haptics.
689
679endif 690endif
diff --git a/drivers/input/misc/Makefile b/drivers/input/misc/Makefile
index 4955ad322a01..cd4bc2d3474f 100644
--- a/drivers/input/misc/Makefile
+++ b/drivers/input/misc/Makefile
@@ -26,6 +26,7 @@ obj-$(CONFIG_INPUT_COBALT_BTNS) += cobalt_btns.o
26obj-$(CONFIG_INPUT_DA9052_ONKEY) += da9052_onkey.o 26obj-$(CONFIG_INPUT_DA9052_ONKEY) += da9052_onkey.o
27obj-$(CONFIG_INPUT_DA9055_ONKEY) += da9055_onkey.o 27obj-$(CONFIG_INPUT_DA9055_ONKEY) += da9055_onkey.o
28obj-$(CONFIG_INPUT_DM355EVM) += dm355evm_keys.o 28obj-$(CONFIG_INPUT_DM355EVM) += dm355evm_keys.o
29obj-$(CONFIG_INPUT_DRV260X_HAPTICS) += drv260x.o
29obj-$(CONFIG_INPUT_GP2A) += gp2ap002a00f.o 30obj-$(CONFIG_INPUT_GP2A) += gp2ap002a00f.o
30obj-$(CONFIG_INPUT_GPIO_BEEPER) += gpio-beeper.o 31obj-$(CONFIG_INPUT_GPIO_BEEPER) += gpio-beeper.o
31obj-$(CONFIG_INPUT_GPIO_TILT_POLLED) += gpio_tilt_polled.o 32obj-$(CONFIG_INPUT_GPIO_TILT_POLLED) += gpio_tilt_polled.o
diff --git a/drivers/input/misc/drv260x.c b/drivers/input/misc/drv260x.c
new file mode 100644
index 000000000000..e90e3b8189f7
--- /dev/null
+++ b/drivers/input/misc/drv260x.c
@@ -0,0 +1,738 @@
1/*
2 * DRV260X haptics driver family
3 *
4 * Author: Dan Murphy <dmurphy@ti.com>
5 *
6 * Copyright: (C) 2014 Texas Instruments, Inc.
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
11 *
12 * This program is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 */
17
18#include <linux/i2c.h>
19#include <linux/input.h>
20#include <linux/module.h>
21#include <linux/of_gpio.h>
22#include <linux/platform_device.h>
23#include <linux/regmap.h>
24#include <linux/slab.h>
25#include <linux/delay.h>
26#include <linux/gpio/consumer.h>
27#include <linux/regulator/consumer.h>
28
29#include <dt-bindings/input/ti-drv260x.h>
30#include <linux/platform_data/drv260x-pdata.h>
31
32#define DRV260X_STATUS 0x0
33#define DRV260X_MODE 0x1
34#define DRV260X_RT_PB_IN 0x2
35#define DRV260X_LIB_SEL 0x3
36#define DRV260X_WV_SEQ_1 0x4
37#define DRV260X_WV_SEQ_2 0x5
38#define DRV260X_WV_SEQ_3 0x6
39#define DRV260X_WV_SEQ_4 0x7
40#define DRV260X_WV_SEQ_5 0x8
41#define DRV260X_WV_SEQ_6 0x9
42#define DRV260X_WV_SEQ_7 0xa
43#define DRV260X_WV_SEQ_8 0xb
44#define DRV260X_GO 0xc
45#define DRV260X_OVERDRIVE_OFF 0xd
46#define DRV260X_SUSTAIN_P_OFF 0xe
47#define DRV260X_SUSTAIN_N_OFF 0xf
48#define DRV260X_BRAKE_OFF 0x10
49#define DRV260X_A_TO_V_CTRL 0x11
50#define DRV260X_A_TO_V_MIN_INPUT 0x12
51#define DRV260X_A_TO_V_MAX_INPUT 0x13
52#define DRV260X_A_TO_V_MIN_OUT 0x14
53#define DRV260X_A_TO_V_MAX_OUT 0x15
54#define DRV260X_RATED_VOLT 0x16
55#define DRV260X_OD_CLAMP_VOLT 0x17
56#define DRV260X_CAL_COMP 0x18
57#define DRV260X_CAL_BACK_EMF 0x19
58#define DRV260X_FEEDBACK_CTRL 0x1a
59#define DRV260X_CTRL1 0x1b
60#define DRV260X_CTRL2 0x1c
61#define DRV260X_CTRL3 0x1d
62#define DRV260X_CTRL4 0x1e
63#define DRV260X_CTRL5 0x1f
64#define DRV260X_LRA_LOOP_PERIOD 0x20
65#define DRV260X_VBAT_MON 0x21
66#define DRV260X_LRA_RES_PERIOD 0x22
67#define DRV260X_MAX_REG 0x23
68
69#define DRV260X_ALLOWED_R_BYTES 25
70#define DRV260X_ALLOWED_W_BYTES 2
71#define DRV260X_MAX_RW_RETRIES 5
72#define DRV260X_I2C_RETRY_DELAY 10
73
74#define DRV260X_GO_BIT 0x01
75
76/* Library Selection */
77#define DRV260X_LIB_SEL_MASK 0x07
78#define DRV260X_LIB_SEL_RAM 0x0
79#define DRV260X_LIB_SEL_OD 0x1
80#define DRV260X_LIB_SEL_40_60 0x2
81#define DRV260X_LIB_SEL_60_80 0x3
82#define DRV260X_LIB_SEL_100_140 0x4
83#define DRV260X_LIB_SEL_140_PLUS 0x5
84
85#define DRV260X_LIB_SEL_HIZ_MASK 0x10
86#define DRV260X_LIB_SEL_HIZ_EN 0x01
87#define DRV260X_LIB_SEL_HIZ_DIS 0
88
89/* Mode register */
90#define DRV260X_STANDBY (1 << 6)
91#define DRV260X_STANDBY_MASK 0x40
92#define DRV260X_INTERNAL_TRIGGER 0x00
93#define DRV260X_EXT_TRIGGER_EDGE 0x01
94#define DRV260X_EXT_TRIGGER_LEVEL 0x02
95#define DRV260X_PWM_ANALOG_IN 0x03
96#define DRV260X_AUDIOHAPTIC 0x04
97#define DRV260X_RT_PLAYBACK 0x05
98#define DRV260X_DIAGNOSTICS 0x06
99#define DRV260X_AUTO_CAL 0x07
100
101/* Audio to Haptics Control */
102#define DRV260X_AUDIO_HAPTICS_PEAK_10MS (0 << 2)
103#define DRV260X_AUDIO_HAPTICS_PEAK_20MS (1 << 2)
104#define DRV260X_AUDIO_HAPTICS_PEAK_30MS (2 << 2)
105#define DRV260X_AUDIO_HAPTICS_PEAK_40MS (3 << 2)
106
107#define DRV260X_AUDIO_HAPTICS_FILTER_100HZ 0x00
108#define DRV260X_AUDIO_HAPTICS_FILTER_125HZ 0x01
109#define DRV260X_AUDIO_HAPTICS_FILTER_150HZ 0x02
110#define DRV260X_AUDIO_HAPTICS_FILTER_200HZ 0x03
111
112/* Min/Max Input/Output Voltages */
113#define DRV260X_AUDIO_HAPTICS_MIN_IN_VOLT 0x19
114#define DRV260X_AUDIO_HAPTICS_MAX_IN_VOLT 0x64
115#define DRV260X_AUDIO_HAPTICS_MIN_OUT_VOLT 0x19
116#define DRV260X_AUDIO_HAPTICS_MAX_OUT_VOLT 0xFF
117
118/* Feedback register */
119#define DRV260X_FB_REG_ERM_MODE 0x7f
120#define DRV260X_FB_REG_LRA_MODE (1 << 7)
121
122#define DRV260X_BRAKE_FACTOR_MASK 0x1f
123#define DRV260X_BRAKE_FACTOR_2X (1 << 0)
124#define DRV260X_BRAKE_FACTOR_3X (2 << 4)
125#define DRV260X_BRAKE_FACTOR_4X (3 << 4)
126#define DRV260X_BRAKE_FACTOR_6X (4 << 4)
127#define DRV260X_BRAKE_FACTOR_8X (5 << 4)
128#define DRV260X_BRAKE_FACTOR_16 (6 << 4)
129#define DRV260X_BRAKE_FACTOR_DIS (7 << 4)
130
131#define DRV260X_LOOP_GAIN_LOW 0xf3
132#define DRV260X_LOOP_GAIN_MED (1 << 2)
133#define DRV260X_LOOP_GAIN_HIGH (2 << 2)
134#define DRV260X_LOOP_GAIN_VERY_HIGH (3 << 2)
135
136#define DRV260X_BEMF_GAIN_0 0xfc
137#define DRV260X_BEMF_GAIN_1 (1 << 0)
138#define DRV260X_BEMF_GAIN_2 (2 << 0)
139#define DRV260X_BEMF_GAIN_3 (3 << 0)
140
141/* Control 1 register */
142#define DRV260X_AC_CPLE_EN (1 << 5)
143#define DRV260X_STARTUP_BOOST (1 << 7)
144
145/* Control 2 register */
146
147#define DRV260X_IDISS_TIME_45 0
148#define DRV260X_IDISS_TIME_75 (1 << 0)
149#define DRV260X_IDISS_TIME_150 (1 << 1)
150#define DRV260X_IDISS_TIME_225 0x03
151
152#define DRV260X_BLANK_TIME_45 (0 << 2)
153#define DRV260X_BLANK_TIME_75 (1 << 2)
154#define DRV260X_BLANK_TIME_150 (2 << 2)
155#define DRV260X_BLANK_TIME_225 (3 << 2)
156
157#define DRV260X_SAMP_TIME_150 (0 << 4)
158#define DRV260X_SAMP_TIME_200 (1 << 4)
159#define DRV260X_SAMP_TIME_250 (2 << 4)
160#define DRV260X_SAMP_TIME_300 (3 << 4)
161
162#define DRV260X_BRAKE_STABILIZER (1 << 6)
163#define DRV260X_UNIDIR_IN (0 << 7)
164#define DRV260X_BIDIR_IN (1 << 7)
165
166/* Control 3 Register */
167#define DRV260X_LRA_OPEN_LOOP (1 << 0)
168#define DRV260X_ANANLOG_IN (1 << 1)
169#define DRV260X_LRA_DRV_MODE (1 << 2)
170#define DRV260X_RTP_UNSIGNED_DATA (1 << 3)
171#define DRV260X_SUPPLY_COMP_DIS (1 << 4)
172#define DRV260X_ERM_OPEN_LOOP (1 << 5)
173#define DRV260X_NG_THRESH_0 (0 << 6)
174#define DRV260X_NG_THRESH_2 (1 << 6)
175#define DRV260X_NG_THRESH_4 (2 << 6)
176#define DRV260X_NG_THRESH_8 (3 << 6)
177
178/* Control 4 Register */
179#define DRV260X_AUTOCAL_TIME_150MS (0 << 4)
180#define DRV260X_AUTOCAL_TIME_250MS (1 << 4)
181#define DRV260X_AUTOCAL_TIME_500MS (2 << 4)
182#define DRV260X_AUTOCAL_TIME_1000MS (3 << 4)
183
184/**
185 * struct drv260x_data -
186 * @input_dev - Pointer to the input device
187 * @client - Pointer to the I2C client
188 * @regmap - Register map of the device
189 * @work - Work item used to off load the enable/disable of the vibration
190 * @enable_gpio - Pointer to the gpio used for enable/disabling
191 * @regulator - Pointer to the regulator for the IC
192 * @magnitude - Magnitude of the vibration event
193 * @mode - The operating mode of the IC (LRA_NO_CAL, ERM or LRA)
194 * @library - The vibration library to be used
195 * @rated_voltage - The rated_voltage of the actuator
196 * @overdriver_voltage - The over drive voltage of the actuator
197**/
198struct drv260x_data {
199 struct input_dev *input_dev;
200 struct i2c_client *client;
201 struct regmap *regmap;
202 struct work_struct work;
203 struct gpio_desc *enable_gpio;
204 struct regulator *regulator;
205 u32 magnitude;
206 u32 mode;
207 u32 library;
208 int rated_voltage;
209 int overdrive_voltage;
210};
211
212static struct reg_default drv260x_reg_defs[] = {
213 { DRV260X_STATUS, 0xe0 },
214 { DRV260X_MODE, 0x40 },
215 { DRV260X_RT_PB_IN, 0x00 },
216 { DRV260X_LIB_SEL, 0x00 },
217 { DRV260X_WV_SEQ_1, 0x01 },
218 { DRV260X_WV_SEQ_2, 0x00 },
219 { DRV260X_WV_SEQ_3, 0x00 },
220 { DRV260X_WV_SEQ_4, 0x00 },
221 { DRV260X_WV_SEQ_5, 0x00 },
222 { DRV260X_WV_SEQ_6, 0x00 },
223 { DRV260X_WV_SEQ_7, 0x00 },
224 { DRV260X_WV_SEQ_8, 0x00 },
225 { DRV260X_GO, 0x00 },
226 { DRV260X_OVERDRIVE_OFF, 0x00 },
227 { DRV260X_SUSTAIN_P_OFF, 0x00 },
228 { DRV260X_SUSTAIN_N_OFF, 0x00 },
229 { DRV260X_BRAKE_OFF, 0x00 },
230 { DRV260X_A_TO_V_CTRL, 0x05 },
231 { DRV260X_A_TO_V_MIN_INPUT, 0x19 },
232 { DRV260X_A_TO_V_MAX_INPUT, 0xff },
233 { DRV260X_A_TO_V_MIN_OUT, 0x19 },
234 { DRV260X_A_TO_V_MAX_OUT, 0xff },
235 { DRV260X_RATED_VOLT, 0x3e },
236 { DRV260X_OD_CLAMP_VOLT, 0x8c },
237 { DRV260X_CAL_COMP, 0x0c },
238 { DRV260X_CAL_BACK_EMF, 0x6c },
239 { DRV260X_FEEDBACK_CTRL, 0x36 },
240 { DRV260X_CTRL1, 0x93 },
241 { DRV260X_CTRL2, 0xfa },
242 { DRV260X_CTRL3, 0xa0 },
243 { DRV260X_CTRL4, 0x20 },
244 { DRV260X_CTRL5, 0x80 },
245 { DRV260X_LRA_LOOP_PERIOD, 0x33 },
246 { DRV260X_VBAT_MON, 0x00 },
247 { DRV260X_LRA_RES_PERIOD, 0x00 },
248};
249
250#define DRV260X_DEF_RATED_VOLT 0x90
251#define DRV260X_DEF_OD_CLAMP_VOLT 0x90
252
253/**
254 * Rated and Overdriver Voltages:
255 * Calculated using the formula r = v * 255 / 5.6
256 * where r is what will be written to the register
257 * and v is the rated or overdriver voltage of the actuator
258 **/
259static int drv260x_calculate_voltage(unsigned int voltage)
260{
261 return (voltage * 255 / 5600);
262}
263
264static void drv260x_worker(struct work_struct *work)
265{
266 struct drv260x_data *haptics = container_of(work, struct drv260x_data, work);
267 int error;
268
269 gpiod_set_value(haptics->enable_gpio, 1);
270 /* Data sheet says to wait 250us before trying to communicate */
271 udelay(250);
272
273 error = regmap_write(haptics->regmap,
274 DRV260X_MODE, DRV260X_RT_PLAYBACK);
275 if (error) {
276 dev_err(&haptics->client->dev,
277 "Failed to write set mode: %d\n", error);
278 } else {
279 error = regmap_write(haptics->regmap,
280 DRV260X_RT_PB_IN, haptics->magnitude);
281 if (error)
282 dev_err(&haptics->client->dev,
283 "Failed to set magnitude: %d\n", error);
284 }
285}
286
287static int drv260x_haptics_play(struct input_dev *input, void *data,
288 struct ff_effect *effect)
289{
290 struct drv260x_data *haptics = input_get_drvdata(input);
291
292 haptics->mode = DRV260X_LRA_NO_CAL_MODE;
293
294 if (effect->u.rumble.strong_magnitude > 0)
295 haptics->magnitude = effect->u.rumble.strong_magnitude;
296 else if (effect->u.rumble.weak_magnitude > 0)
297 haptics->magnitude = effect->u.rumble.weak_magnitude;
298 else
299 haptics->magnitude = 0;
300
301 schedule_work(&haptics->work);
302
303 return 0;
304}
305
306static void drv260x_close(struct input_dev *input)
307{
308 struct drv260x_data *haptics = input_get_drvdata(input);
309 int error;
310
311 cancel_work_sync(&haptics->work);
312
313 error = regmap_write(haptics->regmap, DRV260X_MODE, DRV260X_STANDBY);
314 if (error)
315 dev_err(&haptics->client->dev,
316 "Failed to enter standby mode: %d\n", error);
317
318 gpiod_set_value(haptics->enable_gpio, 0);
319}
320
321static const struct reg_default drv260x_lra_cal_regs[] = {
322 { DRV260X_MODE, DRV260X_AUTO_CAL },
323 { DRV260X_CTRL3, DRV260X_NG_THRESH_2 },
324 { DRV260X_FEEDBACK_CTRL, DRV260X_FB_REG_LRA_MODE |
325 DRV260X_BRAKE_FACTOR_4X | DRV260X_LOOP_GAIN_HIGH },
326};
327
328static const struct reg_default drv260x_lra_init_regs[] = {
329 { DRV260X_MODE, DRV260X_RT_PLAYBACK },
330 { DRV260X_A_TO_V_CTRL, DRV260X_AUDIO_HAPTICS_PEAK_20MS |
331 DRV260X_AUDIO_HAPTICS_FILTER_125HZ },
332 { DRV260X_A_TO_V_MIN_INPUT, DRV260X_AUDIO_HAPTICS_MIN_IN_VOLT },
333 { DRV260X_A_TO_V_MAX_INPUT, DRV260X_AUDIO_HAPTICS_MAX_IN_VOLT },
334 { DRV260X_A_TO_V_MIN_OUT, DRV260X_AUDIO_HAPTICS_MIN_OUT_VOLT },
335 { DRV260X_A_TO_V_MAX_OUT, DRV260X_AUDIO_HAPTICS_MAX_OUT_VOLT },
336 { DRV260X_FEEDBACK_CTRL, DRV260X_FB_REG_LRA_MODE |
337 DRV260X_BRAKE_FACTOR_2X | DRV260X_LOOP_GAIN_MED |
338 DRV260X_BEMF_GAIN_3 },
339 { DRV260X_CTRL1, DRV260X_STARTUP_BOOST },
340 { DRV260X_CTRL2, DRV260X_SAMP_TIME_250 },
341 { DRV260X_CTRL3, DRV260X_NG_THRESH_2 | DRV260X_ANANLOG_IN },
342 { DRV260X_CTRL4, DRV260X_AUTOCAL_TIME_500MS },
343};
344
345static const struct reg_default drv260x_erm_cal_regs[] = {
346 { DRV260X_MODE, DRV260X_AUTO_CAL },
347 { DRV260X_A_TO_V_MIN_INPUT, DRV260X_AUDIO_HAPTICS_MIN_IN_VOLT },
348 { DRV260X_A_TO_V_MAX_INPUT, DRV260X_AUDIO_HAPTICS_MAX_IN_VOLT },
349 { DRV260X_A_TO_V_MIN_OUT, DRV260X_AUDIO_HAPTICS_MIN_OUT_VOLT },
350 { DRV260X_A_TO_V_MAX_OUT, DRV260X_AUDIO_HAPTICS_MAX_OUT_VOLT },
351 { DRV260X_FEEDBACK_CTRL, DRV260X_BRAKE_FACTOR_3X |
352 DRV260X_LOOP_GAIN_MED | DRV260X_BEMF_GAIN_2 },
353 { DRV260X_CTRL1, DRV260X_STARTUP_BOOST },
354 { DRV260X_CTRL2, DRV260X_SAMP_TIME_250 | DRV260X_BLANK_TIME_75 |
355 DRV260X_IDISS_TIME_75 },
356 { DRV260X_CTRL3, DRV260X_NG_THRESH_2 | DRV260X_ERM_OPEN_LOOP },
357 { DRV260X_CTRL4, DRV260X_AUTOCAL_TIME_500MS },
358};
359
360static int drv260x_init(struct drv260x_data *haptics)
361{
362 int error;
363 unsigned int cal_buf;
364
365 error = regmap_write(haptics->regmap,
366 DRV260X_RATED_VOLT, haptics->rated_voltage);
367 if (error) {
368 dev_err(&haptics->client->dev,
369 "Failed to write DRV260X_RATED_VOLT register: %d\n",
370 error);
371 return error;
372 }
373
374 error = regmap_write(haptics->regmap,
375 DRV260X_OD_CLAMP_VOLT, haptics->overdrive_voltage);
376 if (error) {
377 dev_err(&haptics->client->dev,
378 "Failed to write DRV260X_OD_CLAMP_VOLT register: %d\n",
379 error);
380 return error;
381 }
382
383 switch (haptics->mode) {
384 case DRV260X_LRA_MODE:
385 error = regmap_register_patch(haptics->regmap,
386 drv260x_lra_cal_regs,
387 ARRAY_SIZE(drv260x_lra_cal_regs));
388 if (error) {
389 dev_err(&haptics->client->dev,
390 "Failed to write LRA calibration registers: %d\n",
391 error);
392 return error;
393 }
394
395 break;
396
397 case DRV260X_ERM_MODE:
398 error = regmap_register_patch(haptics->regmap,
399 drv260x_erm_cal_regs,
400 ARRAY_SIZE(drv260x_erm_cal_regs));
401 if (error) {
402 dev_err(&haptics->client->dev,
403 "Failed to write ERM calibration registers: %d\n",
404 error);
405 return error;
406 }
407
408 error = regmap_update_bits(haptics->regmap, DRV260X_LIB_SEL,
409 DRV260X_LIB_SEL_MASK,
410 haptics->library);
411 if (error) {
412 dev_err(&haptics->client->dev,
413 "Failed to write DRV260X_LIB_SEL register: %d\n",
414 error);
415 return error;
416 }
417
418 break;
419
420 default:
421 error = regmap_register_patch(haptics->regmap,
422 drv260x_lra_init_regs,
423 ARRAY_SIZE(drv260x_lra_init_regs));
424 if (error) {
425 dev_err(&haptics->client->dev,
426 "Failed to write LRA init registers: %d\n",
427 error);
428 return error;
429 }
430
431 error = regmap_update_bits(haptics->regmap, DRV260X_LIB_SEL,
432 DRV260X_LIB_SEL_MASK,
433 haptics->library);
434 if (error) {
435 dev_err(&haptics->client->dev,
436 "Failed to write DRV260X_LIB_SEL register: %d\n",
437 error);
438 return error;
439 }
440
441 /* No need to set GO bit here */
442 return 0;
443 }
444
445 error = regmap_write(haptics->regmap, DRV260X_GO, DRV260X_GO_BIT);
446 if (error) {
447 dev_err(&haptics->client->dev,
448 "Failed to write GO register: %d\n",
449 error);
450 return error;
451 }
452
453 do {
454 error = regmap_read(haptics->regmap, DRV260X_GO, &cal_buf);
455 if (error) {
456 dev_err(&haptics->client->dev,
457 "Failed to read GO register: %d\n",
458 error);
459 return error;
460 }
461 } while (cal_buf == DRV260X_GO_BIT);
462
463 return 0;
464}
465
466static const struct regmap_config drv260x_regmap_config = {
467 .reg_bits = 8,
468 .val_bits = 8,
469
470 .max_register = DRV260X_MAX_REG,
471 .reg_defaults = drv260x_reg_defs,
472 .num_reg_defaults = ARRAY_SIZE(drv260x_reg_defs),
473 .cache_type = REGCACHE_NONE,
474};
475
476#ifdef CONFIG_OF
477static int drv260x_parse_dt(struct device *dev,
478 struct drv260x_data *haptics)
479{
480 struct device_node *np = dev->of_node;
481 unsigned int voltage;
482 int error;
483
484 error = of_property_read_u32(np, "mode", &haptics->mode);
485 if (error) {
486 dev_err(dev, "%s: No entry for mode\n", __func__);
487 return error;
488 }
489
490 error = of_property_read_u32(np, "library-sel", &haptics->library);
491 if (error) {
492 dev_err(dev, "%s: No entry for library selection\n",
493 __func__);
494 return error;
495 }
496
497 error = of_property_read_u32(np, "vib-rated-mv", &voltage);
498 if (!error)
499 haptics->rated_voltage = drv260x_calculate_voltage(voltage);
500
501
502 error = of_property_read_u32(np, "vib-overdrive-mv", &voltage);
503 if (!error)
504 haptics->overdrive_voltage = drv260x_calculate_voltage(voltage);
505
506 return 0;
507}
508#else
509static inline int drv260x_parse_dt(struct device *dev,
510 struct drv260x_data *haptics)
511{
512 dev_err(dev, "no platform data defined\n");
513
514 return -EINVAL;
515}
516#endif
517
518static int drv260x_probe(struct i2c_client *client,
519 const struct i2c_device_id *id)
520{
521 const struct drv260x_platform_data *pdata = dev_get_platdata(&client->dev);
522 struct drv260x_data *haptics;
523 int error;
524
525 haptics = devm_kzalloc(&client->dev, sizeof(*haptics), GFP_KERNEL);
526 if (!haptics)
527 return -ENOMEM;
528
529 haptics->rated_voltage = DRV260X_DEF_OD_CLAMP_VOLT;
530 haptics->rated_voltage = DRV260X_DEF_RATED_VOLT;
531
532 if (pdata) {
533 haptics->mode = pdata->mode;
534 haptics->library = pdata->library_selection;
535 if (pdata->vib_overdrive_voltage)
536 haptics->overdrive_voltage = drv260x_calculate_voltage(pdata->vib_overdrive_voltage);
537 if (pdata->vib_rated_voltage)
538 haptics->rated_voltage = drv260x_calculate_voltage(pdata->vib_rated_voltage);
539 } else if (client->dev.of_node) {
540 error = drv260x_parse_dt(&client->dev, haptics);
541 if (error)
542 return error;
543 } else {
544 dev_err(&client->dev, "Platform data not set\n");
545 return -ENODEV;
546 }
547
548
549 if (haptics->mode < DRV260X_LRA_MODE ||
550 haptics->mode > DRV260X_ERM_MODE) {
551 dev_err(&client->dev,
552 "Vibrator mode is invalid: %i\n",
553 haptics->mode);
554 return -EINVAL;
555 }
556
557 if (haptics->library < DRV260X_LIB_EMPTY ||
558 haptics->library > DRV260X_ERM_LIB_F) {
559 dev_err(&client->dev,
560 "Library value is invalid: %i\n", haptics->library);
561 return -EINVAL;
562 }
563
564 if (haptics->mode == DRV260X_LRA_MODE &&
565 haptics->library != DRV260X_LIB_EMPTY &&
566 haptics->library != DRV260X_LIB_LRA) {
567 dev_err(&client->dev,
568 "LRA Mode with ERM Library mismatch\n");
569 return -EINVAL;
570 }
571
572 haptics->regulator = devm_regulator_get(&client->dev, "vbat");
573 if (IS_ERR(haptics->regulator)) {
574 error = PTR_ERR(haptics->regulator);
575 dev_err(&client->dev,
576 "unable to get regulator, error: %d\n", error);
577 return error;
578 }
579
580 haptics->enable_gpio = devm_gpiod_get(&client->dev, "enable");
581 if (IS_ERR(haptics->enable_gpio)) {
582 error = PTR_ERR(haptics->enable_gpio);
583 if (error != -ENOENT && error != -ENOSYS)
584 return error;
585 haptics->enable_gpio = NULL;
586 } else {
587 gpiod_direction_output(haptics->enable_gpio, 1);
588 }
589
590 haptics->input_dev = devm_input_allocate_device(&client->dev);
591 if (!haptics->input_dev) {
592 dev_err(&client->dev, "Failed to allocate input device\n");
593 return -ENOMEM;
594 }
595
596 haptics->input_dev->name = "drv260x:haptics";
597 haptics->input_dev->dev.parent = client->dev.parent;
598 haptics->input_dev->close = drv260x_close;
599 input_set_drvdata(haptics->input_dev, haptics);
600 input_set_capability(haptics->input_dev, EV_FF, FF_RUMBLE);
601
602 error = input_ff_create_memless(haptics->input_dev, NULL,
603 drv260x_haptics_play);
604 if (error) {
605 dev_err(&client->dev, "input_ff_create() failed: %d\n",
606 error);
607 return error;
608 }
609
610 INIT_WORK(&haptics->work, drv260x_worker);
611
612 haptics->client = client;
613 i2c_set_clientdata(client, haptics);
614
615 haptics->regmap = devm_regmap_init_i2c(client, &drv260x_regmap_config);
616 if (IS_ERR(haptics->regmap)) {
617 error = PTR_ERR(haptics->regmap);
618 dev_err(&client->dev, "Failed to allocate register map: %d\n",
619 error);
620 return error;
621 }
622
623 error = drv260x_init(haptics);
624 if (error) {
625 dev_err(&client->dev, "Device init failed: %d\n", error);
626 return error;
627 }
628
629 error = input_register_device(haptics->input_dev);
630 if (error) {
631 dev_err(&client->dev, "couldn't register input device: %d\n",
632 error);
633 return error;
634 }
635
636 return 0;
637}
638
639#ifdef CONFIG_PM_SLEEP
640static int drv260x_suspend(struct device *dev)
641{
642 struct drv260x_data *haptics = dev_get_drvdata(dev);
643 int ret = 0;
644
645 mutex_lock(&haptics->input_dev->mutex);
646
647 if (haptics->input_dev->users) {
648 ret = regmap_update_bits(haptics->regmap,
649 DRV260X_MODE,
650 DRV260X_STANDBY_MASK,
651 DRV260X_STANDBY);
652 if (ret) {
653 dev_err(dev, "Failed to set standby mode\n");
654 goto out;
655 }
656
657 gpiod_set_value(haptics->enable_gpio, 0);
658
659 ret = regulator_disable(haptics->regulator);
660 if (ret) {
661 dev_err(dev, "Failed to disable regulator\n");
662 regmap_update_bits(haptics->regmap,
663 DRV260X_MODE,
664 DRV260X_STANDBY_MASK, 0);
665 }
666 }
667out:
668 mutex_unlock(&haptics->input_dev->mutex);
669 return ret;
670}
671
672static int drv260x_resume(struct device *dev)
673{
674 struct drv260x_data *haptics = dev_get_drvdata(dev);
675 int ret = 0;
676
677 mutex_lock(&haptics->input_dev->mutex);
678
679 if (haptics->input_dev->users) {
680 ret = regulator_enable(haptics->regulator);
681 if (ret) {
682 dev_err(dev, "Failed to enable regulator\n");
683 goto out;
684 }
685
686 ret = regmap_update_bits(haptics->regmap,
687 DRV260X_MODE,
688 DRV260X_STANDBY_MASK, 0);
689 if (ret) {
690 dev_err(dev, "Failed to unset standby mode\n");
691 regulator_disable(haptics->regulator);
692 goto out;
693 }
694
695 gpiod_set_value(haptics->enable_gpio, 1);
696 }
697
698out:
699 mutex_unlock(&haptics->input_dev->mutex);
700 return ret;
701}
702#endif
703
704static SIMPLE_DEV_PM_OPS(drv260x_pm_ops, drv260x_suspend, drv260x_resume);
705
706static const struct i2c_device_id drv260x_id[] = {
707 { "drv2605l", 0 },
708 { }
709};
710MODULE_DEVICE_TABLE(i2c, drv260x_id);
711
712#ifdef CONFIG_OF
713static const struct of_device_id drv260x_of_match[] = {
714 { .compatible = "ti,drv2604", },
715 { .compatible = "ti,drv2604l", },
716 { .compatible = "ti,drv2605", },
717 { .compatible = "ti,drv2605l", },
718 { }
719};
720MODULE_DEVICE_TABLE(of, drv260x_of_match);
721#endif
722
723static struct i2c_driver drv260x_driver = {
724 .probe = drv260x_probe,
725 .driver = {
726 .name = "drv260x-haptics",
727 .owner = THIS_MODULE,
728 .of_match_table = of_match_ptr(drv260x_of_match),
729 .pm = &drv260x_pm_ops,
730 },
731 .id_table = drv260x_id,
732};
733module_i2c_driver(drv260x_driver);
734
735MODULE_ALIAS("platform:drv260x-haptics");
736MODULE_DESCRIPTION("TI DRV260x haptics driver");
737MODULE_LICENSE("GPL");
738MODULE_AUTHOR("Dan Murphy <dmurphy@ti.com>");
diff --git a/include/dt-bindings/input/ti-drv260x.h b/include/dt-bindings/input/ti-drv260x.h
new file mode 100644
index 000000000000..2626e6d9f707
--- /dev/null
+++ b/include/dt-bindings/input/ti-drv260x.h
@@ -0,0 +1,36 @@
1/*
2 * DRV260X haptics driver family
3 *
4 * Author: Dan Murphy <dmurphy@ti.com>
5 *
6 * Copyright: (C) 2014 Texas Instruments, Inc.
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
11 *
12 * This program is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 */
17
18#ifndef _DT_BINDINGS_TI_DRV260X_H
19#define _DT_BINDINGS_TI_DRV260X_H
20
21/* Calibration Types */
22#define DRV260X_LRA_MODE 0x00
23#define DRV260X_LRA_NO_CAL_MODE 0x01
24#define DRV260X_ERM_MODE 0x02
25
26/* Library Selection */
27#define DRV260X_LIB_EMPTY 0x00
28#define DRV260X_ERM_LIB_A 0x01
29#define DRV260X_ERM_LIB_B 0x02
30#define DRV260X_ERM_LIB_C 0x03
31#define DRV260X_ERM_LIB_D 0x04
32#define DRV260X_ERM_LIB_E 0x05
33#define DRV260X_LIB_LRA 0x06
34#define DRV260X_ERM_LIB_F 0x07
35
36#endif
diff --git a/include/linux/platform_data/drv260x-pdata.h b/include/linux/platform_data/drv260x-pdata.h
new file mode 100644
index 000000000000..0a03b0944411
--- /dev/null
+++ b/include/linux/platform_data/drv260x-pdata.h
@@ -0,0 +1,28 @@
1/*
2 * Platform data for DRV260X haptics driver family
3 *
4 * Author: Dan Murphy <dmurphy@ti.com>
5 *
6 * Copyright: (C) 2014 Texas Instruments, Inc.
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
11 *
12 * This program is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 */
17
18#ifndef _LINUX_DRV260X_PDATA_H
19#define _LINUX_DRV260X_PDATA_H
20
21struct drv260x_platform_data {
22 u32 library_selection;
23 u32 mode;
24 u32 vib_rated_voltage;
25 u32 vib_overdrive_voltage;
26};
27
28#endif