12874c5fdSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
2a3b3ca75SJaewon Kim /*
356bbc99eSKrzysztof Kozlowski  * MAXIM MAX77693/MAX77843 Haptic device driver
4a3b3ca75SJaewon Kim  *
556bbc99eSKrzysztof Kozlowski  * Copyright (C) 2014,2015 Samsung Electronics
6a3b3ca75SJaewon Kim  * Jaewon Kim <jaewon02.kim@samsung.com>
73f5c5920SKrzysztof Kozlowski  * Krzysztof Kozlowski <krzk@kernel.org>
8a3b3ca75SJaewon Kim  *
9a3b3ca75SJaewon Kim  * This program is not provided / owned by Maxim Integrated Products.
10a3b3ca75SJaewon Kim  */
11a3b3ca75SJaewon Kim 
12a3b3ca75SJaewon Kim #include <linux/err.h>
13a3b3ca75SJaewon Kim #include <linux/init.h>
14a3b3ca75SJaewon Kim #include <linux/i2c.h>
15a3b3ca75SJaewon Kim #include <linux/regmap.h>
16a3b3ca75SJaewon Kim #include <linux/input.h>
17a3b3ca75SJaewon Kim #include <linux/module.h>
18a3b3ca75SJaewon Kim #include <linux/platform_device.h>
19a3b3ca75SJaewon Kim #include <linux/pwm.h>
20a3b3ca75SJaewon Kim #include <linux/slab.h>
21a3b3ca75SJaewon Kim #include <linux/workqueue.h>
22a3b3ca75SJaewon Kim #include <linux/regulator/consumer.h>
23a3b3ca75SJaewon Kim #include <linux/mfd/max77693.h>
2461b305cdSKrzysztof Kozlowski #include <linux/mfd/max77693-common.h>
25a3b3ca75SJaewon Kim #include <linux/mfd/max77693-private.h>
2656bbc99eSKrzysztof Kozlowski #include <linux/mfd/max77843-private.h>
27a3b3ca75SJaewon Kim 
28a3b3ca75SJaewon Kim #define MAX_MAGNITUDE_SHIFT	16
29a3b3ca75SJaewon Kim 
30a3b3ca75SJaewon Kim enum max77693_haptic_motor_type {
31a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_ERM = 0,
32a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_LRA,
33a3b3ca75SJaewon Kim };
34a3b3ca75SJaewon Kim 
35a3b3ca75SJaewon Kim enum max77693_haptic_pulse_mode {
36a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_EXTERNAL_MODE = 0,
37a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_INTERNAL_MODE,
38a3b3ca75SJaewon Kim };
39a3b3ca75SJaewon Kim 
40a3b3ca75SJaewon Kim enum max77693_haptic_pwm_divisor {
41a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_PWM_DIVISOR_32 = 0,
42a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_PWM_DIVISOR_64,
43a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_PWM_DIVISOR_128,
44a3b3ca75SJaewon Kim 	MAX77693_HAPTIC_PWM_DIVISOR_256,
45a3b3ca75SJaewon Kim };
46a3b3ca75SJaewon Kim 
47a3b3ca75SJaewon Kim struct max77693_haptic {
486eaa247aSKrzysztof Kozlowski 	enum max77693_types dev_type;
496eaa247aSKrzysztof Kozlowski 
50a3b3ca75SJaewon Kim 	struct regmap *regmap_pmic;
51a3b3ca75SJaewon Kim 	struct regmap *regmap_haptic;
52a3b3ca75SJaewon Kim 	struct device *dev;
53a3b3ca75SJaewon Kim 	struct input_dev *input_dev;
54a3b3ca75SJaewon Kim 	struct pwm_device *pwm_dev;
55a3b3ca75SJaewon Kim 	struct regulator *motor_reg;
56a3b3ca75SJaewon Kim 
57a3b3ca75SJaewon Kim 	bool enabled;
58a3b3ca75SJaewon Kim 	bool suspend_state;
59a3b3ca75SJaewon Kim 	unsigned int magnitude;
60a3b3ca75SJaewon Kim 	unsigned int pwm_duty;
61a3b3ca75SJaewon Kim 	enum max77693_haptic_motor_type type;
62a3b3ca75SJaewon Kim 	enum max77693_haptic_pulse_mode mode;
63a3b3ca75SJaewon Kim 
64a3b3ca75SJaewon Kim 	struct work_struct work;
65a3b3ca75SJaewon Kim };
66a3b3ca75SJaewon Kim 
max77693_haptic_set_duty_cycle(struct max77693_haptic * haptic)67a3b3ca75SJaewon Kim static int max77693_haptic_set_duty_cycle(struct max77693_haptic *haptic)
68a3b3ca75SJaewon Kim {
69ad4e9fecSBoris Brezillon 	struct pwm_args pargs;
70ad4e9fecSBoris Brezillon 	int delta;
71a3b3ca75SJaewon Kim 	int error;
72a3b3ca75SJaewon Kim 
73ad4e9fecSBoris Brezillon 	pwm_get_args(haptic->pwm_dev, &pargs);
74ad4e9fecSBoris Brezillon 	delta = (pargs.period + haptic->pwm_duty) / 2;
75ad4e9fecSBoris Brezillon 	error = pwm_config(haptic->pwm_dev, delta, pargs.period);
76a3b3ca75SJaewon Kim 	if (error) {
77a3b3ca75SJaewon Kim 		dev_err(haptic->dev, "failed to configure pwm: %d\n", error);
78a3b3ca75SJaewon Kim 		return error;
79a3b3ca75SJaewon Kim 	}
80a3b3ca75SJaewon Kim 
81a3b3ca75SJaewon Kim 	return 0;
82a3b3ca75SJaewon Kim }
83a3b3ca75SJaewon Kim 
max77843_haptic_bias(struct max77693_haptic * haptic,bool on)8456bbc99eSKrzysztof Kozlowski static int max77843_haptic_bias(struct max77693_haptic *haptic, bool on)
8556bbc99eSKrzysztof Kozlowski {
8656bbc99eSKrzysztof Kozlowski 	int error;
8756bbc99eSKrzysztof Kozlowski 
8856bbc99eSKrzysztof Kozlowski 	if (haptic->dev_type != TYPE_MAX77843)
8956bbc99eSKrzysztof Kozlowski 		return 0;
9056bbc99eSKrzysztof Kozlowski 
9156bbc99eSKrzysztof Kozlowski 	error = regmap_update_bits(haptic->regmap_haptic,
9256bbc99eSKrzysztof Kozlowski 				   MAX77843_SYS_REG_MAINCTRL1,
9356bbc99eSKrzysztof Kozlowski 				   MAX77843_MAINCTRL1_BIASEN_MASK,
9456bbc99eSKrzysztof Kozlowski 				   on << MAINCTRL1_BIASEN_SHIFT);
9556bbc99eSKrzysztof Kozlowski 	if (error) {
9656bbc99eSKrzysztof Kozlowski 		dev_err(haptic->dev, "failed to %s bias: %d\n",
9756bbc99eSKrzysztof Kozlowski 			on ? "enable" : "disable", error);
9856bbc99eSKrzysztof Kozlowski 		return error;
9956bbc99eSKrzysztof Kozlowski 	}
10056bbc99eSKrzysztof Kozlowski 
10156bbc99eSKrzysztof Kozlowski 	return 0;
10256bbc99eSKrzysztof Kozlowski }
10356bbc99eSKrzysztof Kozlowski 
max77693_haptic_configure(struct max77693_haptic * haptic,bool enable)104a3b3ca75SJaewon Kim static int max77693_haptic_configure(struct max77693_haptic *haptic,
105a3b3ca75SJaewon Kim 				     bool enable)
106a3b3ca75SJaewon Kim {
1076eaa247aSKrzysztof Kozlowski 	unsigned int value, config_reg;
108a3b3ca75SJaewon Kim 	int error;
109a3b3ca75SJaewon Kim 
1106eaa247aSKrzysztof Kozlowski 	switch (haptic->dev_type) {
1116eaa247aSKrzysztof Kozlowski 	case TYPE_MAX77693:
112a3b3ca75SJaewon Kim 		value = ((haptic->type << MAX77693_CONFIG2_MODE) |
113a3b3ca75SJaewon Kim 			(enable << MAX77693_CONFIG2_MEN) |
114a3b3ca75SJaewon Kim 			(haptic->mode << MAX77693_CONFIG2_HTYP) |
115b3d8ba74SKrzysztof Kozlowski 			MAX77693_HAPTIC_PWM_DIVISOR_128);
1166eaa247aSKrzysztof Kozlowski 		config_reg = MAX77693_HAPTIC_REG_CONFIG2;
1176eaa247aSKrzysztof Kozlowski 		break;
11856bbc99eSKrzysztof Kozlowski 	case TYPE_MAX77843:
11956bbc99eSKrzysztof Kozlowski 		value = (haptic->type << MCONFIG_MODE_SHIFT) |
12056bbc99eSKrzysztof Kozlowski 			(enable << MCONFIG_MEN_SHIFT) |
12156bbc99eSKrzysztof Kozlowski 			MAX77693_HAPTIC_PWM_DIVISOR_128;
12256bbc99eSKrzysztof Kozlowski 		config_reg = MAX77843_HAP_REG_MCONFIG;
12356bbc99eSKrzysztof Kozlowski 		break;
1246eaa247aSKrzysztof Kozlowski 	default:
1256eaa247aSKrzysztof Kozlowski 		return -EINVAL;
1266eaa247aSKrzysztof Kozlowski 	}
127a3b3ca75SJaewon Kim 
128a3b3ca75SJaewon Kim 	error = regmap_write(haptic->regmap_haptic,
1296eaa247aSKrzysztof Kozlowski 			     config_reg, value);
130a3b3ca75SJaewon Kim 	if (error) {
131a3b3ca75SJaewon Kim 		dev_err(haptic->dev,
132a3b3ca75SJaewon Kim 			"failed to update haptic config: %d\n", error);
133a3b3ca75SJaewon Kim 		return error;
134a3b3ca75SJaewon Kim 	}
135a3b3ca75SJaewon Kim 
136a3b3ca75SJaewon Kim 	return 0;
137a3b3ca75SJaewon Kim }
138a3b3ca75SJaewon Kim 
max77693_haptic_lowsys(struct max77693_haptic * haptic,bool enable)139a3b3ca75SJaewon Kim static int max77693_haptic_lowsys(struct max77693_haptic *haptic, bool enable)
140a3b3ca75SJaewon Kim {
141a3b3ca75SJaewon Kim 	int error;
142a3b3ca75SJaewon Kim 
14356bbc99eSKrzysztof Kozlowski 	if (haptic->dev_type != TYPE_MAX77693)
14456bbc99eSKrzysztof Kozlowski 		return 0;
14556bbc99eSKrzysztof Kozlowski 
146a3b3ca75SJaewon Kim 	error = regmap_update_bits(haptic->regmap_pmic,
147a3b3ca75SJaewon Kim 				   MAX77693_PMIC_REG_LSCNFG,
148a3b3ca75SJaewon Kim 				   MAX77693_PMIC_LOW_SYS_MASK,
149a3b3ca75SJaewon Kim 				   enable << MAX77693_PMIC_LOW_SYS_SHIFT);
150a3b3ca75SJaewon Kim 	if (error) {
151a3b3ca75SJaewon Kim 		dev_err(haptic->dev, "cannot update pmic regmap: %d\n", error);
152a3b3ca75SJaewon Kim 		return error;
153a3b3ca75SJaewon Kim 	}
154a3b3ca75SJaewon Kim 
155a3b3ca75SJaewon Kim 	return 0;
156a3b3ca75SJaewon Kim }
157a3b3ca75SJaewon Kim 
max77693_haptic_enable(struct max77693_haptic * haptic)158a3b3ca75SJaewon Kim static void max77693_haptic_enable(struct max77693_haptic *haptic)
159a3b3ca75SJaewon Kim {
160a3b3ca75SJaewon Kim 	int error;
161a3b3ca75SJaewon Kim 
162a3b3ca75SJaewon Kim 	if (haptic->enabled)
163a3b3ca75SJaewon Kim 		return;
164a3b3ca75SJaewon Kim 
165a3b3ca75SJaewon Kim 	error = pwm_enable(haptic->pwm_dev);
166a3b3ca75SJaewon Kim 	if (error) {
167a3b3ca75SJaewon Kim 		dev_err(haptic->dev,
168a3b3ca75SJaewon Kim 			"failed to enable haptic pwm device: %d\n", error);
169a3b3ca75SJaewon Kim 		return;
170a3b3ca75SJaewon Kim 	}
171a3b3ca75SJaewon Kim 
172a3b3ca75SJaewon Kim 	error = max77693_haptic_lowsys(haptic, true);
173a3b3ca75SJaewon Kim 	if (error)
174a3b3ca75SJaewon Kim 		goto err_enable_lowsys;
175a3b3ca75SJaewon Kim 
176a3b3ca75SJaewon Kim 	error = max77693_haptic_configure(haptic, true);
177a3b3ca75SJaewon Kim 	if (error)
178a3b3ca75SJaewon Kim 		goto err_enable_config;
179a3b3ca75SJaewon Kim 
180a3b3ca75SJaewon Kim 	haptic->enabled = true;
181a3b3ca75SJaewon Kim 
182a3b3ca75SJaewon Kim 	return;
183a3b3ca75SJaewon Kim 
184a3b3ca75SJaewon Kim err_enable_config:
185a3b3ca75SJaewon Kim 	max77693_haptic_lowsys(haptic, false);
186a3b3ca75SJaewon Kim err_enable_lowsys:
187a3b3ca75SJaewon Kim 	pwm_disable(haptic->pwm_dev);
188a3b3ca75SJaewon Kim }
189a3b3ca75SJaewon Kim 
max77693_haptic_disable(struct max77693_haptic * haptic)190a3b3ca75SJaewon Kim static void max77693_haptic_disable(struct max77693_haptic *haptic)
191a3b3ca75SJaewon Kim {
192a3b3ca75SJaewon Kim 	int error;
193a3b3ca75SJaewon Kim 
194135d916fSJaewon Kim 	if (!haptic->enabled)
195a3b3ca75SJaewon Kim 		return;
196a3b3ca75SJaewon Kim 
197a3b3ca75SJaewon Kim 	error = max77693_haptic_configure(haptic, false);
198a3b3ca75SJaewon Kim 	if (error)
199a3b3ca75SJaewon Kim 		return;
200a3b3ca75SJaewon Kim 
201a3b3ca75SJaewon Kim 	error = max77693_haptic_lowsys(haptic, false);
202a3b3ca75SJaewon Kim 	if (error)
203a3b3ca75SJaewon Kim 		goto err_disable_lowsys;
204a3b3ca75SJaewon Kim 
205a3b3ca75SJaewon Kim 	pwm_disable(haptic->pwm_dev);
206a3b3ca75SJaewon Kim 	haptic->enabled = false;
207a3b3ca75SJaewon Kim 
208a3b3ca75SJaewon Kim 	return;
209a3b3ca75SJaewon Kim 
210a3b3ca75SJaewon Kim err_disable_lowsys:
211a3b3ca75SJaewon Kim 	max77693_haptic_configure(haptic, true);
212a3b3ca75SJaewon Kim }
213a3b3ca75SJaewon Kim 
max77693_haptic_play_work(struct work_struct * work)214a3b3ca75SJaewon Kim static void max77693_haptic_play_work(struct work_struct *work)
215a3b3ca75SJaewon Kim {
216a3b3ca75SJaewon Kim 	struct max77693_haptic *haptic =
217a3b3ca75SJaewon Kim 			container_of(work, struct max77693_haptic, work);
218a3b3ca75SJaewon Kim 	int error;
219a3b3ca75SJaewon Kim 
220a3b3ca75SJaewon Kim 	error = max77693_haptic_set_duty_cycle(haptic);
221a3b3ca75SJaewon Kim 	if (error) {
222a3b3ca75SJaewon Kim 		dev_err(haptic->dev, "failed to set duty cycle: %d\n", error);
223a3b3ca75SJaewon Kim 		return;
224a3b3ca75SJaewon Kim 	}
225a3b3ca75SJaewon Kim 
226a3b3ca75SJaewon Kim 	if (haptic->magnitude)
227a3b3ca75SJaewon Kim 		max77693_haptic_enable(haptic);
228a3b3ca75SJaewon Kim 	else
229a3b3ca75SJaewon Kim 		max77693_haptic_disable(haptic);
230a3b3ca75SJaewon Kim }
231a3b3ca75SJaewon Kim 
max77693_haptic_play_effect(struct input_dev * dev,void * data,struct ff_effect * effect)232a3b3ca75SJaewon Kim static int max77693_haptic_play_effect(struct input_dev *dev, void *data,
233a3b3ca75SJaewon Kim 				       struct ff_effect *effect)
234a3b3ca75SJaewon Kim {
235a3b3ca75SJaewon Kim 	struct max77693_haptic *haptic = input_get_drvdata(dev);
236ad4e9fecSBoris Brezillon 	struct pwm_args pargs;
237fbefc5e7SDmitry Torokhov 	u64 period_mag_multi;
238a3b3ca75SJaewon Kim 
239a3b3ca75SJaewon Kim 	haptic->magnitude = effect->u.rumble.strong_magnitude;
240a3b3ca75SJaewon Kim 	if (!haptic->magnitude)
241a3b3ca75SJaewon Kim 		haptic->magnitude = effect->u.rumble.weak_magnitude;
242a3b3ca75SJaewon Kim 
243a3b3ca75SJaewon Kim 	/*
244a3b3ca75SJaewon Kim 	 * The magnitude comes from force-feedback interface.
245a3b3ca75SJaewon Kim 	 * The formula to convert magnitude to pwm_duty as follows:
246a3b3ca75SJaewon Kim 	 * - pwm_duty = (magnitude * pwm_period) / MAX_MAGNITUDE(0xFFFF)
247a3b3ca75SJaewon Kim 	 */
248ad4e9fecSBoris Brezillon 	pwm_get_args(haptic->pwm_dev, &pargs);
249ad4e9fecSBoris Brezillon 	period_mag_multi = (u64)pargs.period * haptic->magnitude;
250a3b3ca75SJaewon Kim 	haptic->pwm_duty = (unsigned int)(period_mag_multi >>
251a3b3ca75SJaewon Kim 						MAX_MAGNITUDE_SHIFT);
252a3b3ca75SJaewon Kim 
253a3b3ca75SJaewon Kim 	schedule_work(&haptic->work);
254a3b3ca75SJaewon Kim 
255a3b3ca75SJaewon Kim 	return 0;
256a3b3ca75SJaewon Kim }
257a3b3ca75SJaewon Kim 
max77693_haptic_open(struct input_dev * dev)258a3b3ca75SJaewon Kim static int max77693_haptic_open(struct input_dev *dev)
259a3b3ca75SJaewon Kim {
260a3b3ca75SJaewon Kim 	struct max77693_haptic *haptic = input_get_drvdata(dev);
261a3b3ca75SJaewon Kim 	int error;
262a3b3ca75SJaewon Kim 
26356bbc99eSKrzysztof Kozlowski 	error = max77843_haptic_bias(haptic, true);
26456bbc99eSKrzysztof Kozlowski 	if (error)
26556bbc99eSKrzysztof Kozlowski 		return error;
26656bbc99eSKrzysztof Kozlowski 
267a3b3ca75SJaewon Kim 	error = regulator_enable(haptic->motor_reg);
268a3b3ca75SJaewon Kim 	if (error) {
269a3b3ca75SJaewon Kim 		dev_err(haptic->dev,
270a3b3ca75SJaewon Kim 			"failed to enable regulator: %d\n", error);
271a3b3ca75SJaewon Kim 		return error;
272a3b3ca75SJaewon Kim 	}
273a3b3ca75SJaewon Kim 
274a3b3ca75SJaewon Kim 	return 0;
275a3b3ca75SJaewon Kim }
276a3b3ca75SJaewon Kim 
max77693_haptic_close(struct input_dev * dev)277a3b3ca75SJaewon Kim static void max77693_haptic_close(struct input_dev *dev)
278a3b3ca75SJaewon Kim {
279a3b3ca75SJaewon Kim 	struct max77693_haptic *haptic = input_get_drvdata(dev);
280a3b3ca75SJaewon Kim 	int error;
281a3b3ca75SJaewon Kim 
282a3b3ca75SJaewon Kim 	cancel_work_sync(&haptic->work);
283a3b3ca75SJaewon Kim 	max77693_haptic_disable(haptic);
284a3b3ca75SJaewon Kim 
285a3b3ca75SJaewon Kim 	error = regulator_disable(haptic->motor_reg);
286a3b3ca75SJaewon Kim 	if (error)
287a3b3ca75SJaewon Kim 		dev_err(haptic->dev,
288a3b3ca75SJaewon Kim 			"failed to disable regulator: %d\n", error);
28956bbc99eSKrzysztof Kozlowski 
29056bbc99eSKrzysztof Kozlowski 	max77843_haptic_bias(haptic, false);
291a3b3ca75SJaewon Kim }
292a3b3ca75SJaewon Kim 
max77693_haptic_probe(struct platform_device * pdev)293a3b3ca75SJaewon Kim static int max77693_haptic_probe(struct platform_device *pdev)
294a3b3ca75SJaewon Kim {
295a3b3ca75SJaewon Kim 	struct max77693_dev *max77693 = dev_get_drvdata(pdev->dev.parent);
296a3b3ca75SJaewon Kim 	struct max77693_haptic *haptic;
297a3b3ca75SJaewon Kim 	int error;
298a3b3ca75SJaewon Kim 
299a3b3ca75SJaewon Kim 	haptic = devm_kzalloc(&pdev->dev, sizeof(*haptic), GFP_KERNEL);
300a3b3ca75SJaewon Kim 	if (!haptic)
301a3b3ca75SJaewon Kim 		return -ENOMEM;
302a3b3ca75SJaewon Kim 
303a3b3ca75SJaewon Kim 	haptic->regmap_pmic = max77693->regmap;
304a3b3ca75SJaewon Kim 	haptic->dev = &pdev->dev;
305a3b3ca75SJaewon Kim 	haptic->type = MAX77693_HAPTIC_LRA;
306a3b3ca75SJaewon Kim 	haptic->mode = MAX77693_HAPTIC_EXTERNAL_MODE;
307a3b3ca75SJaewon Kim 	haptic->suspend_state = false;
308a3b3ca75SJaewon Kim 
3096eaa247aSKrzysztof Kozlowski 	/* Variant-specific init */
3106eaa247aSKrzysztof Kozlowski 	haptic->dev_type = platform_get_device_id(pdev)->driver_data;
3116eaa247aSKrzysztof Kozlowski 	switch (haptic->dev_type) {
3126eaa247aSKrzysztof Kozlowski 	case TYPE_MAX77693:
3136eaa247aSKrzysztof Kozlowski 		haptic->regmap_haptic = max77693->regmap_haptic;
3146eaa247aSKrzysztof Kozlowski 		break;
31556bbc99eSKrzysztof Kozlowski 	case TYPE_MAX77843:
31656bbc99eSKrzysztof Kozlowski 		haptic->regmap_haptic = max77693->regmap;
31756bbc99eSKrzysztof Kozlowski 		break;
3186eaa247aSKrzysztof Kozlowski 	default:
3196eaa247aSKrzysztof Kozlowski 		dev_err(&pdev->dev, "unsupported device type: %u\n",
3206eaa247aSKrzysztof Kozlowski 			haptic->dev_type);
3216eaa247aSKrzysztof Kozlowski 		return -EINVAL;
3226eaa247aSKrzysztof Kozlowski 	}
3236eaa247aSKrzysztof Kozlowski 
324a3b3ca75SJaewon Kim 	INIT_WORK(&haptic->work, max77693_haptic_play_work);
325a3b3ca75SJaewon Kim 
326a3b3ca75SJaewon Kim 	/* Get pwm and regulatot for haptic device */
327a3b3ca75SJaewon Kim 	haptic->pwm_dev = devm_pwm_get(&pdev->dev, NULL);
328a3b3ca75SJaewon Kim 	if (IS_ERR(haptic->pwm_dev)) {
329a3b3ca75SJaewon Kim 		dev_err(&pdev->dev, "failed to get pwm device\n");
330a3b3ca75SJaewon Kim 		return PTR_ERR(haptic->pwm_dev);
331a3b3ca75SJaewon Kim 	}
332a3b3ca75SJaewon Kim 
333ad4e9fecSBoris Brezillon 	/*
334ad4e9fecSBoris Brezillon 	 * FIXME: pwm_apply_args() should be removed when switching to the
335ad4e9fecSBoris Brezillon 	 * atomic PWM API.
336ad4e9fecSBoris Brezillon 	 */
337ad4e9fecSBoris Brezillon 	pwm_apply_args(haptic->pwm_dev);
338ad4e9fecSBoris Brezillon 
339a3b3ca75SJaewon Kim 	haptic->motor_reg = devm_regulator_get(&pdev->dev, "haptic");
340a3b3ca75SJaewon Kim 	if (IS_ERR(haptic->motor_reg)) {
341a3b3ca75SJaewon Kim 		dev_err(&pdev->dev, "failed to get regulator\n");
342a3b3ca75SJaewon Kim 		return PTR_ERR(haptic->motor_reg);
343a3b3ca75SJaewon Kim 	}
344a3b3ca75SJaewon Kim 
345a3b3ca75SJaewon Kim 	/* Initialize input device for haptic device */
346a3b3ca75SJaewon Kim 	haptic->input_dev = devm_input_allocate_device(&pdev->dev);
347a3b3ca75SJaewon Kim 	if (!haptic->input_dev) {
348a3b3ca75SJaewon Kim 		dev_err(&pdev->dev, "failed to allocate input device\n");
349a3b3ca75SJaewon Kim 		return -ENOMEM;
350a3b3ca75SJaewon Kim 	}
351a3b3ca75SJaewon Kim 
352a3b3ca75SJaewon Kim 	haptic->input_dev->name = "max77693-haptic";
353a3b3ca75SJaewon Kim 	haptic->input_dev->id.version = 1;
354a3b3ca75SJaewon Kim 	haptic->input_dev->dev.parent = &pdev->dev;
355a3b3ca75SJaewon Kim 	haptic->input_dev->open = max77693_haptic_open;
356a3b3ca75SJaewon Kim 	haptic->input_dev->close = max77693_haptic_close;
357a3b3ca75SJaewon Kim 	input_set_drvdata(haptic->input_dev, haptic);
358a3b3ca75SJaewon Kim 	input_set_capability(haptic->input_dev, EV_FF, FF_RUMBLE);
359a3b3ca75SJaewon Kim 
360a3b3ca75SJaewon Kim 	error = input_ff_create_memless(haptic->input_dev, NULL,
361a3b3ca75SJaewon Kim 				max77693_haptic_play_effect);
362a3b3ca75SJaewon Kim 	if (error) {
363a3b3ca75SJaewon Kim 		dev_err(&pdev->dev, "failed to create force-feedback\n");
364a3b3ca75SJaewon Kim 		return error;
365a3b3ca75SJaewon Kim 	}
366a3b3ca75SJaewon Kim 
367a3b3ca75SJaewon Kim 	error = input_register_device(haptic->input_dev);
368a3b3ca75SJaewon Kim 	if (error) {
369a3b3ca75SJaewon Kim 		dev_err(&pdev->dev, "failed to register input device\n");
370a3b3ca75SJaewon Kim 		return error;
371a3b3ca75SJaewon Kim 	}
372a3b3ca75SJaewon Kim 
373a3b3ca75SJaewon Kim 	platform_set_drvdata(pdev, haptic);
374a3b3ca75SJaewon Kim 
375a3b3ca75SJaewon Kim 	return 0;
376a3b3ca75SJaewon Kim }
377a3b3ca75SJaewon Kim 
max77693_haptic_suspend(struct device * dev)378*1e302399SJonathan Cameron static int max77693_haptic_suspend(struct device *dev)
379a3b3ca75SJaewon Kim {
380a3b3ca75SJaewon Kim 	struct platform_device *pdev = to_platform_device(dev);
381a3b3ca75SJaewon Kim 	struct max77693_haptic *haptic = platform_get_drvdata(pdev);
382a3b3ca75SJaewon Kim 
383a3b3ca75SJaewon Kim 	if (haptic->enabled) {
384a3b3ca75SJaewon Kim 		max77693_haptic_disable(haptic);
385a3b3ca75SJaewon Kim 		haptic->suspend_state = true;
386a3b3ca75SJaewon Kim 	}
387a3b3ca75SJaewon Kim 
388a3b3ca75SJaewon Kim 	return 0;
389a3b3ca75SJaewon Kim }
390a3b3ca75SJaewon Kim 
max77693_haptic_resume(struct device * dev)391*1e302399SJonathan Cameron static int max77693_haptic_resume(struct device *dev)
392a3b3ca75SJaewon Kim {
393a3b3ca75SJaewon Kim 	struct platform_device *pdev = to_platform_device(dev);
394a3b3ca75SJaewon Kim 	struct max77693_haptic *haptic = platform_get_drvdata(pdev);
395a3b3ca75SJaewon Kim 
396a3b3ca75SJaewon Kim 	if (haptic->suspend_state) {
397a3b3ca75SJaewon Kim 		max77693_haptic_enable(haptic);
398a3b3ca75SJaewon Kim 		haptic->suspend_state = false;
399a3b3ca75SJaewon Kim 	}
400a3b3ca75SJaewon Kim 
401a3b3ca75SJaewon Kim 	return 0;
402a3b3ca75SJaewon Kim }
403a3b3ca75SJaewon Kim 
404*1e302399SJonathan Cameron static DEFINE_SIMPLE_DEV_PM_OPS(max77693_haptic_pm_ops,
405*1e302399SJonathan Cameron 				max77693_haptic_suspend,
406*1e302399SJonathan Cameron 				max77693_haptic_resume);
407a3b3ca75SJaewon Kim 
4086eaa247aSKrzysztof Kozlowski static const struct platform_device_id max77693_haptic_id[] = {
4096eaa247aSKrzysztof Kozlowski 	{ "max77693-haptic", TYPE_MAX77693 },
41056bbc99eSKrzysztof Kozlowski 	{ "max77843-haptic", TYPE_MAX77843 },
4116eaa247aSKrzysztof Kozlowski 	{},
4126eaa247aSKrzysztof Kozlowski };
4136eaa247aSKrzysztof Kozlowski MODULE_DEVICE_TABLE(platform, max77693_haptic_id);
4146eaa247aSKrzysztof Kozlowski 
415a3b3ca75SJaewon Kim static struct platform_driver max77693_haptic_driver = {
416a3b3ca75SJaewon Kim 	.driver		= {
417a3b3ca75SJaewon Kim 		.name	= "max77693-haptic",
418*1e302399SJonathan Cameron 		.pm	= pm_sleep_ptr(&max77693_haptic_pm_ops),
419a3b3ca75SJaewon Kim 	},
420a3b3ca75SJaewon Kim 	.probe		= max77693_haptic_probe,
4216eaa247aSKrzysztof Kozlowski 	.id_table	= max77693_haptic_id,
422a3b3ca75SJaewon Kim };
423a3b3ca75SJaewon Kim module_platform_driver(max77693_haptic_driver);
424a3b3ca75SJaewon Kim 
425a3b3ca75SJaewon Kim MODULE_AUTHOR("Jaewon Kim <jaewon02.kim@samsung.com>");
4263f5c5920SKrzysztof Kozlowski MODULE_AUTHOR("Krzysztof Kozlowski <krzk@kernel.org>");
42756bbc99eSKrzysztof Kozlowski MODULE_DESCRIPTION("MAXIM 77693/77843 Haptic driver");
428a3b3ca75SJaewon Kim MODULE_LICENSE("GPL");
429