xref: /linux/drivers/mfd/cs40l50-core.c (revision 76d9b92e68f2bb55890f935c5143f4fef97a935d)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * CS40L50 Advanced Haptic Driver with waveform memory,
4  * integrated DSP, and closed-loop algorithms
5  *
6  * Copyright 2024 Cirrus Logic, Inc.
7  *
8  * Author: James Ogletree <james.ogletree@cirrus.com>
9  */
10 
11 #include <linux/firmware/cirrus/cs_dsp.h>
12 #include <linux/firmware/cirrus/wmfw.h>
13 #include <linux/mfd/core.h>
14 #include <linux/mfd/cs40l50.h>
15 #include <linux/pm_runtime.h>
16 #include <linux/regulator/consumer.h>
17 
18 static const struct mfd_cell cs40l50_devs[] = {
19 	{ .name = "cs40l50-codec", },
20 	{ .name = "cs40l50-vibra", },
21 };
22 
23 const struct regmap_config cs40l50_regmap = {
24 	.reg_bits =		32,
25 	.reg_stride =		4,
26 	.val_bits =		32,
27 	.reg_format_endian =	REGMAP_ENDIAN_BIG,
28 	.val_format_endian =	REGMAP_ENDIAN_BIG,
29 };
30 EXPORT_SYMBOL_GPL(cs40l50_regmap);
31 
32 static const char * const cs40l50_supplies[] = {
33 	"vdd-io",
34 };
35 
36 static const struct regmap_irq cs40l50_reg_irqs[] = {
37 	REGMAP_IRQ_REG(CS40L50_DSP_QUEUE_IRQ, CS40L50_IRQ1_INT_2_OFFSET,
38 		       CS40L50_DSP_QUEUE_MASK),
39 	REGMAP_IRQ_REG(CS40L50_AMP_SHORT_IRQ, CS40L50_IRQ1_INT_1_OFFSET,
40 		       CS40L50_AMP_SHORT_MASK),
41 	REGMAP_IRQ_REG(CS40L50_TEMP_ERR_IRQ, CS40L50_IRQ1_INT_8_OFFSET,
42 		       CS40L50_TEMP_ERR_MASK),
43 	REGMAP_IRQ_REG(CS40L50_BST_UVP_IRQ, CS40L50_IRQ1_INT_9_OFFSET,
44 		       CS40L50_BST_UVP_MASK),
45 	REGMAP_IRQ_REG(CS40L50_BST_SHORT_IRQ, CS40L50_IRQ1_INT_9_OFFSET,
46 		       CS40L50_BST_SHORT_MASK),
47 	REGMAP_IRQ_REG(CS40L50_BST_ILIMIT_IRQ, CS40L50_IRQ1_INT_9_OFFSET,
48 		       CS40L50_BST_ILIMIT_MASK),
49 	REGMAP_IRQ_REG(CS40L50_UVLO_VDDBATT_IRQ, CS40L50_IRQ1_INT_10_OFFSET,
50 		       CS40L50_UVLO_VDDBATT_MASK),
51 	REGMAP_IRQ_REG(CS40L50_GLOBAL_ERROR_IRQ, CS40L50_IRQ1_INT_18_OFFSET,
52 		       CS40L50_GLOBAL_ERROR_MASK),
53 };
54 
55 static struct regmap_irq_chip cs40l50_irq_chip = {
56 	.name =		"cs40l50",
57 	.status_base =	CS40L50_IRQ1_INT_1,
58 	.mask_base =	CS40L50_IRQ1_MASK_1,
59 	.ack_base =	CS40L50_IRQ1_INT_1,
60 	.num_regs =	22,
61 	.irqs =		cs40l50_reg_irqs,
62 	.num_irqs =	ARRAY_SIZE(cs40l50_reg_irqs),
63 	.runtime_pm =	true,
64 };
65 
66 int cs40l50_dsp_write(struct device *dev, struct regmap *regmap, u32 val)
67 {
68 	int i, ret;
69 	u32 ack;
70 
71 	/* Device NAKs if hibernating, so optionally retry */
72 	for (i = 0; i < CS40L50_DSP_TIMEOUT_COUNT; i++) {
73 		ret = regmap_write(regmap, CS40L50_DSP_QUEUE, val);
74 		if (!ret)
75 			break;
76 
77 		usleep_range(CS40L50_DSP_POLL_US, CS40L50_DSP_POLL_US + 100);
78 	}
79 
80 	/* If the write never took place, no need to check for the ACK */
81 	if (i == CS40L50_DSP_TIMEOUT_COUNT) {
82 		dev_err(dev, "Timed out writing %#X to DSP: %d\n", val, ret);
83 		return ret;
84 	}
85 
86 	ret = regmap_read_poll_timeout(regmap, CS40L50_DSP_QUEUE, ack, !ack,
87 				       CS40L50_DSP_POLL_US,
88 				       CS40L50_DSP_POLL_US * CS40L50_DSP_TIMEOUT_COUNT);
89 	if (ret)
90 		dev_err(dev, "DSP failed to ACK %#X: %d\n", val, ret);
91 
92 	return ret;
93 }
94 EXPORT_SYMBOL_GPL(cs40l50_dsp_write);
95 
96 static const struct cs_dsp_region cs40l50_dsp_regions[] = {
97 	{ .type = WMFW_HALO_PM_PACKED, .base = CS40L50_PMEM_0 },
98 	{ .type = WMFW_HALO_XM_PACKED, .base = CS40L50_XMEM_PACKED_0 },
99 	{ .type = WMFW_HALO_YM_PACKED, .base = CS40L50_YMEM_PACKED_0 },
100 	{ .type = WMFW_ADSP2_XM, .base = CS40L50_XMEM_UNPACKED24_0 },
101 	{ .type = WMFW_ADSP2_YM, .base = CS40L50_YMEM_UNPACKED24_0 },
102 };
103 
104 static const struct reg_sequence cs40l50_internal_vamp_config[] = {
105 	{ CS40L50_BST_LPMODE_SEL, CS40L50_DCM_LOW_POWER },
106 	{ CS40L50_BLOCK_ENABLES2, CS40L50_OVERTEMP_WARN },
107 };
108 
109 static const struct reg_sequence cs40l50_irq_mask_override[] = {
110 	{ CS40L50_IRQ1_MASK_2, CS40L50_IRQ_MASK_2_OVERRIDE },
111 	{ CS40L50_IRQ1_MASK_20, CS40L50_IRQ_MASK_20_OVERRIDE },
112 };
113 
114 static int cs40l50_wseq_init(struct cs40l50 *cs40l50)
115 {
116 	struct cs_dsp *dsp = &cs40l50->dsp;
117 
118 	cs40l50->wseqs[CS40L50_STANDBY].ctl = cs_dsp_get_ctl(dsp, "STANDBY_SEQUENCE",
119 							     WMFW_ADSP2_XM,
120 							     CS40L50_PM_ALGO);
121 	if (!cs40l50->wseqs[CS40L50_STANDBY].ctl) {
122 		dev_err(cs40l50->dev, "Control not found for standby sequence\n");
123 		return -ENOENT;
124 	}
125 
126 	cs40l50->wseqs[CS40L50_ACTIVE].ctl = cs_dsp_get_ctl(dsp, "ACTIVE_SEQUENCE",
127 							    WMFW_ADSP2_XM,
128 							    CS40L50_PM_ALGO);
129 	if (!cs40l50->wseqs[CS40L50_ACTIVE].ctl) {
130 		dev_err(cs40l50->dev, "Control not found for active sequence\n");
131 		return -ENOENT;
132 	}
133 
134 	cs40l50->wseqs[CS40L50_PWR_ON].ctl = cs_dsp_get_ctl(dsp, "PM_PWR_ON_SEQ",
135 							    WMFW_ADSP2_XM,
136 							    CS40L50_PM_ALGO);
137 	if (!cs40l50->wseqs[CS40L50_PWR_ON].ctl) {
138 		dev_err(cs40l50->dev, "Control not found for power-on sequence\n");
139 		return -ENOENT;
140 	}
141 
142 	return cs_dsp_wseq_init(&cs40l50->dsp, cs40l50->wseqs, ARRAY_SIZE(cs40l50->wseqs));
143 }
144 
145 static int cs40l50_dsp_config(struct cs40l50 *cs40l50)
146 {
147 	int ret;
148 
149 	/* Configure internal V_AMP supply */
150 	ret = regmap_multi_reg_write(cs40l50->regmap, cs40l50_internal_vamp_config,
151 				     ARRAY_SIZE(cs40l50_internal_vamp_config));
152 	if (ret)
153 		return ret;
154 
155 	ret = cs_dsp_wseq_multi_write(&cs40l50->dsp, &cs40l50->wseqs[CS40L50_PWR_ON],
156 				      cs40l50_internal_vamp_config, CS_DSP_WSEQ_FULL,
157 				      ARRAY_SIZE(cs40l50_internal_vamp_config), false);
158 	if (ret)
159 		return ret;
160 
161 	/* Override firmware defaults for IRQ masks */
162 	ret = regmap_multi_reg_write(cs40l50->regmap, cs40l50_irq_mask_override,
163 				     ARRAY_SIZE(cs40l50_irq_mask_override));
164 	if (ret)
165 		return ret;
166 
167 	return cs_dsp_wseq_multi_write(&cs40l50->dsp, &cs40l50->wseqs[CS40L50_PWR_ON],
168 				       cs40l50_irq_mask_override, CS_DSP_WSEQ_FULL,
169 				       ARRAY_SIZE(cs40l50_irq_mask_override), false);
170 }
171 
172 static int cs40l50_dsp_post_run(struct cs_dsp *dsp)
173 {
174 	struct cs40l50 *cs40l50 = container_of(dsp, struct cs40l50, dsp);
175 	int ret;
176 
177 	ret = cs40l50_wseq_init(cs40l50);
178 	if (ret)
179 		return ret;
180 
181 	ret = cs40l50_dsp_config(cs40l50);
182 	if (ret) {
183 		dev_err(cs40l50->dev, "Failed to configure DSP: %d\n", ret);
184 		return ret;
185 	}
186 
187 	ret = devm_mfd_add_devices(cs40l50->dev, PLATFORM_DEVID_NONE, cs40l50_devs,
188 				   ARRAY_SIZE(cs40l50_devs), NULL, 0, NULL);
189 	if (ret)
190 		dev_err(cs40l50->dev, "Failed to add child devices: %d\n", ret);
191 
192 	return ret;
193 }
194 
195 static const struct cs_dsp_client_ops client_ops = {
196 	.post_run = cs40l50_dsp_post_run,
197 };
198 
199 static void cs40l50_dsp_remove(void *data)
200 {
201 	cs_dsp_remove(data);
202 }
203 
204 static int cs40l50_dsp_init(struct cs40l50 *cs40l50)
205 {
206 	int ret;
207 
208 	cs40l50->dsp.num = 1;
209 	cs40l50->dsp.type = WMFW_HALO;
210 	cs40l50->dsp.dev = cs40l50->dev;
211 	cs40l50->dsp.regmap = cs40l50->regmap;
212 	cs40l50->dsp.base = CS40L50_CORE_BASE;
213 	cs40l50->dsp.base_sysinfo = CS40L50_SYS_INFO_ID;
214 	cs40l50->dsp.mem = cs40l50_dsp_regions;
215 	cs40l50->dsp.num_mems = ARRAY_SIZE(cs40l50_dsp_regions);
216 	cs40l50->dsp.no_core_startstop = true;
217 	cs40l50->dsp.client_ops = &client_ops;
218 
219 	ret = cs_dsp_halo_init(&cs40l50->dsp);
220 	if (ret)
221 		return ret;
222 
223 	return devm_add_action_or_reset(cs40l50->dev, cs40l50_dsp_remove,
224 					&cs40l50->dsp);
225 }
226 
227 static int cs40l50_reset_dsp(struct cs40l50 *cs40l50)
228 {
229 	int ret;
230 
231 	mutex_lock(&cs40l50->lock);
232 
233 	if (cs40l50->dsp.running)
234 		cs_dsp_stop(&cs40l50->dsp);
235 
236 	if (cs40l50->dsp.booted)
237 		cs_dsp_power_down(&cs40l50->dsp);
238 
239 	ret = cs40l50_dsp_write(cs40l50->dev, cs40l50->regmap, CS40L50_SHUTDOWN);
240 	if (ret)
241 		goto err_mutex;
242 
243 	ret = cs_dsp_power_up(&cs40l50->dsp, cs40l50->fw, "cs40l50.wmfw",
244 			      cs40l50->bin, "cs40l50.bin", "cs40l50");
245 	if (ret)
246 		goto err_mutex;
247 
248 	ret = cs40l50_dsp_write(cs40l50->dev, cs40l50->regmap, CS40L50_SYSTEM_RESET);
249 	if (ret)
250 		goto err_mutex;
251 
252 	ret = cs40l50_dsp_write(cs40l50->dev, cs40l50->regmap, CS40L50_PREVENT_HIBER);
253 	if (ret)
254 		goto err_mutex;
255 
256 	ret = cs_dsp_run(&cs40l50->dsp);
257 err_mutex:
258 	mutex_unlock(&cs40l50->lock);
259 
260 	return ret;
261 }
262 
263 static void cs40l50_dsp_power_down(void *data)
264 {
265 	cs_dsp_power_down(data);
266 }
267 
268 static void cs40l50_dsp_stop(void *data)
269 {
270 	cs_dsp_stop(data);
271 }
272 
273 static void cs40l50_dsp_bringup(const struct firmware *bin, void *context)
274 {
275 	struct cs40l50 *cs40l50 = context;
276 	u32 nwaves;
277 	int ret;
278 
279 	/* Wavetable is optional; bringup DSP regardless */
280 	cs40l50->bin = bin;
281 
282 	ret = cs40l50_reset_dsp(cs40l50);
283 	if (ret) {
284 		dev_err(cs40l50->dev, "Failed to reset DSP: %d\n", ret);
285 		goto err_fw;
286 	}
287 
288 	ret = regmap_read(cs40l50->regmap, CS40L50_NUM_WAVES, &nwaves);
289 	if (ret)
290 		goto err_fw;
291 
292 	dev_info(cs40l50->dev, "%u RAM effects loaded\n", nwaves);
293 
294 	/* Add teardown actions for first-time bringup */
295 	ret = devm_add_action_or_reset(cs40l50->dev, cs40l50_dsp_power_down,
296 				       &cs40l50->dsp);
297 	if (ret) {
298 		dev_err(cs40l50->dev, "Failed to add power down action: %d\n", ret);
299 		goto err_fw;
300 	}
301 
302 	ret = devm_add_action_or_reset(cs40l50->dev, cs40l50_dsp_stop, &cs40l50->dsp);
303 	if (ret)
304 		dev_err(cs40l50->dev, "Failed to add stop action: %d\n", ret);
305 err_fw:
306 	release_firmware(cs40l50->bin);
307 	release_firmware(cs40l50->fw);
308 }
309 
310 static void cs40l50_request_firmware(const struct firmware *fw, void *context)
311 {
312 	struct cs40l50 *cs40l50 = context;
313 	int ret;
314 
315 	if (!fw) {
316 		dev_err(cs40l50->dev, "No firmware file found\n");
317 		return;
318 	}
319 
320 	cs40l50->fw = fw;
321 
322 	ret = request_firmware_nowait(THIS_MODULE, FW_ACTION_UEVENT, CS40L50_WT,
323 				      cs40l50->dev, GFP_KERNEL, cs40l50,
324 				      cs40l50_dsp_bringup);
325 	if (ret) {
326 		dev_err(cs40l50->dev, "Failed to request %s: %d\n", CS40L50_WT, ret);
327 		release_firmware(cs40l50->fw);
328 	}
329 }
330 
331 struct cs40l50_irq {
332 	const char *name;
333 	int virq;
334 };
335 
336 static struct cs40l50_irq cs40l50_irqs[] = {
337 	{ "DSP", },
338 	{ "Global", },
339 	{ "Boost UVLO", },
340 	{ "Boost current limit", },
341 	{ "Boost short", },
342 	{ "Boost undervolt", },
343 	{ "Overtemp", },
344 	{ "Amp short", },
345 };
346 
347 static const struct reg_sequence cs40l50_err_rls[] = {
348 	{ CS40L50_ERR_RLS, CS40L50_GLOBAL_ERR_RLS_SET },
349 	{ CS40L50_ERR_RLS, CS40L50_GLOBAL_ERR_RLS_CLEAR },
350 };
351 
352 static irqreturn_t cs40l50_hw_err(int irq, void *data)
353 {
354 	struct cs40l50 *cs40l50 = data;
355 	int ret = 0, i;
356 
357 	mutex_lock(&cs40l50->lock);
358 
359 	/* Log hardware interrupt and execute error release sequence */
360 	for (i = 1; i < ARRAY_SIZE(cs40l50_irqs); i++) {
361 		if (cs40l50_irqs[i].virq == irq) {
362 			dev_err(cs40l50->dev, "%s error\n", cs40l50_irqs[i].name);
363 			ret = regmap_multi_reg_write(cs40l50->regmap, cs40l50_err_rls,
364 						     ARRAY_SIZE(cs40l50_err_rls));
365 			break;
366 		}
367 	}
368 
369 	mutex_unlock(&cs40l50->lock);
370 	return IRQ_RETVAL(!ret);
371 }
372 
373 static irqreturn_t cs40l50_dsp_queue(int irq, void *data)
374 {
375 	struct cs40l50 *cs40l50 = data;
376 	u32 rd_ptr, val, wt_ptr;
377 	int ret = 0;
378 
379 	mutex_lock(&cs40l50->lock);
380 
381 	/* Read from DSP queue, log, and update read pointer */
382 	while (!ret) {
383 		ret = regmap_read(cs40l50->regmap, CS40L50_DSP_QUEUE_WT, &wt_ptr);
384 		if (ret)
385 			break;
386 
387 		ret = regmap_read(cs40l50->regmap, CS40L50_DSP_QUEUE_RD, &rd_ptr);
388 		if (ret)
389 			break;
390 
391 		/* Check if queue is empty */
392 		if (wt_ptr == rd_ptr)
393 			break;
394 
395 		ret = regmap_read(cs40l50->regmap, rd_ptr, &val);
396 		if (ret)
397 			break;
398 
399 		dev_dbg(cs40l50->dev, "DSP payload: %#X", val);
400 
401 		rd_ptr += sizeof(u32);
402 
403 		if (rd_ptr > CS40L50_DSP_QUEUE_END)
404 			rd_ptr = CS40L50_DSP_QUEUE_BASE;
405 
406 		ret = regmap_write(cs40l50->regmap, CS40L50_DSP_QUEUE_RD, rd_ptr);
407 	}
408 
409 	mutex_unlock(&cs40l50->lock);
410 
411 	return IRQ_RETVAL(!ret);
412 }
413 
414 static int cs40l50_irq_init(struct cs40l50 *cs40l50)
415 {
416 	int ret, i, virq;
417 
418 	ret = devm_regmap_add_irq_chip(cs40l50->dev, cs40l50->regmap, cs40l50->irq,
419 				       IRQF_ONESHOT | IRQF_SHARED, 0,
420 				       &cs40l50_irq_chip, &cs40l50->irq_data);
421 	if (ret) {
422 		dev_err(cs40l50->dev, "Failed adding IRQ chip\n");
423 		return ret;
424 	}
425 
426 	for (i = 0; i < ARRAY_SIZE(cs40l50_irqs); i++) {
427 		virq = regmap_irq_get_virq(cs40l50->irq_data, i);
428 		if (virq < 0) {
429 			dev_err(cs40l50->dev, "Failed getting virq for %s\n",
430 				cs40l50_irqs[i].name);
431 			return virq;
432 		}
433 
434 		cs40l50_irqs[i].virq = virq;
435 
436 		/* Handle DSP and hardware interrupts separately */
437 		ret = devm_request_threaded_irq(cs40l50->dev, virq, NULL,
438 						i ? cs40l50_hw_err : cs40l50_dsp_queue,
439 						IRQF_ONESHOT | IRQF_SHARED,
440 						cs40l50_irqs[i].name, cs40l50);
441 		if (ret) {
442 			return dev_err_probe(cs40l50->dev, ret,
443 					     "Failed requesting %s IRQ\n",
444 					     cs40l50_irqs[i].name);
445 		}
446 	}
447 
448 	return 0;
449 }
450 
451 static int cs40l50_get_model(struct cs40l50 *cs40l50)
452 {
453 	int ret;
454 
455 	ret = regmap_read(cs40l50->regmap, CS40L50_DEVID, &cs40l50->devid);
456 	if (ret)
457 		return ret;
458 
459 	if (cs40l50->devid != CS40L50_DEVID_A)
460 		return -EINVAL;
461 
462 	ret = regmap_read(cs40l50->regmap, CS40L50_REVID, &cs40l50->revid);
463 	if (ret)
464 		return ret;
465 
466 	if (cs40l50->revid < CS40L50_REVID_B0)
467 		return -EINVAL;
468 
469 	dev_dbg(cs40l50->dev, "Cirrus Logic CS40L50 rev. %02X\n", cs40l50->revid);
470 
471 	return 0;
472 }
473 
474 static int cs40l50_pm_runtime_setup(struct device *dev)
475 {
476 	int ret;
477 
478 	pm_runtime_set_autosuspend_delay(dev, CS40L50_AUTOSUSPEND_MS);
479 	pm_runtime_use_autosuspend(dev);
480 	pm_runtime_get_noresume(dev);
481 	ret = pm_runtime_set_active(dev);
482 	if (ret)
483 		return ret;
484 
485 	return devm_pm_runtime_enable(dev);
486 }
487 
488 int cs40l50_probe(struct cs40l50 *cs40l50)
489 {
490 	struct device *dev = cs40l50->dev;
491 	int ret;
492 
493 	mutex_init(&cs40l50->lock);
494 
495 	cs40l50->reset_gpio = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_HIGH);
496 	if (IS_ERR(cs40l50->reset_gpio))
497 		return dev_err_probe(dev, PTR_ERR(cs40l50->reset_gpio),
498 				     "Failed getting reset GPIO\n");
499 
500 	ret = devm_regulator_bulk_get_enable(dev, ARRAY_SIZE(cs40l50_supplies),
501 					     cs40l50_supplies);
502 	if (ret)
503 		return dev_err_probe(dev, ret, "Failed getting supplies\n");
504 
505 	/* Ensure minimum reset pulse width */
506 	usleep_range(CS40L50_RESET_PULSE_US, CS40L50_RESET_PULSE_US + 100);
507 
508 	gpiod_set_value_cansleep(cs40l50->reset_gpio, 0);
509 
510 	/* Wait for control port to be ready */
511 	usleep_range(CS40L50_CP_READY_US, CS40L50_CP_READY_US + 100);
512 
513 	ret = cs40l50_get_model(cs40l50);
514 	if (ret)
515 		return dev_err_probe(dev, ret, "Failed to get part number\n");
516 
517 	ret = cs40l50_dsp_init(cs40l50);
518 	if (ret)
519 		return dev_err_probe(dev, ret, "Failed to initialize DSP\n");
520 
521 	ret = cs40l50_pm_runtime_setup(dev);
522 	if (ret)
523 		return dev_err_probe(dev, ret, "Failed to initialize runtime PM\n");
524 
525 	ret = cs40l50_irq_init(cs40l50);
526 	if (ret)
527 		return ret;
528 
529 	ret = request_firmware_nowait(THIS_MODULE, FW_ACTION_UEVENT, CS40L50_FW,
530 				      dev, GFP_KERNEL, cs40l50, cs40l50_request_firmware);
531 	if (ret)
532 		return dev_err_probe(dev, ret, "Failed to request %s\n", CS40L50_FW);
533 
534 	pm_runtime_mark_last_busy(dev);
535 	pm_runtime_put_autosuspend(dev);
536 
537 	return 0;
538 }
539 EXPORT_SYMBOL_GPL(cs40l50_probe);
540 
541 int cs40l50_remove(struct cs40l50 *cs40l50)
542 {
543 	gpiod_set_value_cansleep(cs40l50->reset_gpio, 1);
544 
545 	return 0;
546 }
547 EXPORT_SYMBOL_GPL(cs40l50_remove);
548 
549 static int cs40l50_runtime_suspend(struct device *dev)
550 {
551 	struct cs40l50 *cs40l50 = dev_get_drvdata(dev);
552 
553 	return regmap_write(cs40l50->regmap, CS40L50_DSP_QUEUE, CS40L50_ALLOW_HIBER);
554 }
555 
556 static int cs40l50_runtime_resume(struct device *dev)
557 {
558 	struct cs40l50 *cs40l50 = dev_get_drvdata(dev);
559 
560 	return cs40l50_dsp_write(dev, cs40l50->regmap, CS40L50_PREVENT_HIBER);
561 }
562 
563 EXPORT_GPL_DEV_PM_OPS(cs40l50_pm_ops) = {
564 	RUNTIME_PM_OPS(cs40l50_runtime_suspend, cs40l50_runtime_resume, NULL)
565 };
566 
567 MODULE_DESCRIPTION("CS40L50 Advanced Haptic Driver");
568 MODULE_AUTHOR("James Ogletree, Cirrus Logic Inc. <james.ogletree@cirrus.com>");
569 MODULE_LICENSE("GPL");
570 MODULE_IMPORT_NS(FW_CS_DSP);
571