// SPDX-License-Identifier: GPL-2.0 /* * CS40L50 Advanced Haptic Driver with waveform memory, * integrated DSP, and closed-loop algorithms * * Copyright 2024 Cirrus Logic, Inc. * * Author: James Ogletree */ #include #include #include #include #include #include static const struct mfd_cell cs40l50_devs[] = { { .name = "cs40l50-codec", }, { .name = "cs40l50-vibra", }, }; const struct regmap_config cs40l50_regmap = { .reg_bits = 32, .reg_stride = 4, .val_bits = 32, .reg_format_endian = REGMAP_ENDIAN_BIG, .val_format_endian = REGMAP_ENDIAN_BIG, }; EXPORT_SYMBOL_GPL(cs40l50_regmap); static const char * const cs40l50_supplies[] = { "vdd-io", }; static const struct regmap_irq cs40l50_reg_irqs[] = { REGMAP_IRQ_REG(CS40L50_DSP_QUEUE_IRQ, CS40L50_IRQ1_INT_2_OFFSET, CS40L50_DSP_QUEUE_MASK), REGMAP_IRQ_REG(CS40L50_AMP_SHORT_IRQ, CS40L50_IRQ1_INT_1_OFFSET, CS40L50_AMP_SHORT_MASK), REGMAP_IRQ_REG(CS40L50_TEMP_ERR_IRQ, CS40L50_IRQ1_INT_8_OFFSET, CS40L50_TEMP_ERR_MASK), REGMAP_IRQ_REG(CS40L50_BST_UVP_IRQ, CS40L50_IRQ1_INT_9_OFFSET, CS40L50_BST_UVP_MASK), REGMAP_IRQ_REG(CS40L50_BST_SHORT_IRQ, CS40L50_IRQ1_INT_9_OFFSET, CS40L50_BST_SHORT_MASK), REGMAP_IRQ_REG(CS40L50_BST_ILIMIT_IRQ, CS40L50_IRQ1_INT_9_OFFSET, CS40L50_BST_ILIMIT_MASK), REGMAP_IRQ_REG(CS40L50_UVLO_VDDBATT_IRQ, CS40L50_IRQ1_INT_10_OFFSET, CS40L50_UVLO_VDDBATT_MASK), REGMAP_IRQ_REG(CS40L50_GLOBAL_ERROR_IRQ, CS40L50_IRQ1_INT_18_OFFSET, CS40L50_GLOBAL_ERROR_MASK), }; static struct regmap_irq_chip cs40l50_irq_chip = { .name = "cs40l50", .status_base = CS40L50_IRQ1_INT_1, .mask_base = CS40L50_IRQ1_MASK_1, .ack_base = CS40L50_IRQ1_INT_1, .num_regs = 22, .irqs = cs40l50_reg_irqs, .num_irqs = ARRAY_SIZE(cs40l50_reg_irqs), .runtime_pm = true, }; int cs40l50_dsp_write(struct device *dev, struct regmap *regmap, u32 val) { int i, ret; u32 ack; /* Device NAKs if hibernating, so optionally retry */ for (i = 0; i < CS40L50_DSP_TIMEOUT_COUNT; i++) { ret = regmap_write(regmap, CS40L50_DSP_QUEUE, val); if (!ret) break; usleep_range(CS40L50_DSP_POLL_US, CS40L50_DSP_POLL_US + 100); } /* If the write never took place, no need to check for the ACK */ if (i == CS40L50_DSP_TIMEOUT_COUNT) { dev_err(dev, "Timed out writing %#X to DSP: %d\n", val, ret); return ret; } ret = regmap_read_poll_timeout(regmap, CS40L50_DSP_QUEUE, ack, !ack, CS40L50_DSP_POLL_US, CS40L50_DSP_POLL_US * CS40L50_DSP_TIMEOUT_COUNT); if (ret) dev_err(dev, "DSP failed to ACK %#X: %d\n", val, ret); return ret; } EXPORT_SYMBOL_GPL(cs40l50_dsp_write); static const struct cs_dsp_region cs40l50_dsp_regions[] = { { .type = WMFW_HALO_PM_PACKED, .base = CS40L50_PMEM_0 }, { .type = WMFW_HALO_XM_PACKED, .base = CS40L50_XMEM_PACKED_0 }, { .type = WMFW_HALO_YM_PACKED, .base = CS40L50_YMEM_PACKED_0 }, { .type = WMFW_ADSP2_XM, .base = CS40L50_XMEM_UNPACKED24_0 }, { .type = WMFW_ADSP2_YM, .base = CS40L50_YMEM_UNPACKED24_0 }, }; static const struct reg_sequence cs40l50_internal_vamp_config[] = { { CS40L50_BST_LPMODE_SEL, CS40L50_DCM_LOW_POWER }, { CS40L50_BLOCK_ENABLES2, CS40L50_OVERTEMP_WARN }, }; static const struct reg_sequence cs40l50_irq_mask_override[] = { { CS40L50_IRQ1_MASK_2, CS40L50_IRQ_MASK_2_OVERRIDE }, { CS40L50_IRQ1_MASK_20, CS40L50_IRQ_MASK_20_OVERRIDE }, }; static int cs40l50_wseq_init(struct cs40l50 *cs40l50) { struct cs_dsp *dsp = &cs40l50->dsp; cs40l50->wseqs[CS40L50_STANDBY].ctl = cs_dsp_get_ctl(dsp, "STANDBY_SEQUENCE", WMFW_ADSP2_XM, CS40L50_PM_ALGO); if (!cs40l50->wseqs[CS40L50_STANDBY].ctl) { dev_err(cs40l50->dev, "Control not found for standby sequence\n"); return -ENOENT; } cs40l50->wseqs[CS40L50_ACTIVE].ctl = cs_dsp_get_ctl(dsp, "ACTIVE_SEQUENCE", WMFW_ADSP2_XM, CS40L50_PM_ALGO); if (!cs40l50->wseqs[CS40L50_ACTIVE].ctl) { dev_err(cs40l50->dev, "Control not found for active sequence\n"); return -ENOENT; } cs40l50->wseqs[CS40L50_PWR_ON].ctl = cs_dsp_get_ctl(dsp, "PM_PWR_ON_SEQ", WMFW_ADSP2_XM, CS40L50_PM_ALGO); if (!cs40l50->wseqs[CS40L50_PWR_ON].ctl) { dev_err(cs40l50->dev, "Control not found for power-on sequence\n"); return -ENOENT; } return cs_dsp_wseq_init(&cs40l50->dsp, cs40l50->wseqs, ARRAY_SIZE(cs40l50->wseqs)); } static int cs40l50_dsp_config(struct cs40l50 *cs40l50) { int ret; /* Configure internal V_AMP supply */ ret = regmap_multi_reg_write(cs40l50->regmap, cs40l50_internal_vamp_config, ARRAY_SIZE(cs40l50_internal_vamp_config)); if (ret) return ret; ret = cs_dsp_wseq_multi_write(&cs40l50->dsp, &cs40l50->wseqs[CS40L50_PWR_ON], cs40l50_internal_vamp_config, CS_DSP_WSEQ_FULL, ARRAY_SIZE(cs40l50_internal_vamp_config), false); if (ret) return ret; /* Override firmware defaults for IRQ masks */ ret = regmap_multi_reg_write(cs40l50->regmap, cs40l50_irq_mask_override, ARRAY_SIZE(cs40l50_irq_mask_override)); if (ret) return ret; return cs_dsp_wseq_multi_write(&cs40l50->dsp, &cs40l50->wseqs[CS40L50_PWR_ON], cs40l50_irq_mask_override, CS_DSP_WSEQ_FULL, ARRAY_SIZE(cs40l50_irq_mask_override), false); } static int cs40l50_dsp_post_run(struct cs_dsp *dsp) { struct cs40l50 *cs40l50 = container_of(dsp, struct cs40l50, dsp); int ret; ret = cs40l50_wseq_init(cs40l50); if (ret) return ret; ret = cs40l50_dsp_config(cs40l50); if (ret) { dev_err(cs40l50->dev, "Failed to configure DSP: %d\n", ret); return ret; } ret = devm_mfd_add_devices(cs40l50->dev, PLATFORM_DEVID_NONE, cs40l50_devs, ARRAY_SIZE(cs40l50_devs), NULL, 0, NULL); if (ret) dev_err(cs40l50->dev, "Failed to add child devices: %d\n", ret); return ret; } static const struct cs_dsp_client_ops client_ops = { .post_run = cs40l50_dsp_post_run, }; static void cs40l50_dsp_remove(void *data) { cs_dsp_remove(data); } static int cs40l50_dsp_init(struct cs40l50 *cs40l50) { int ret; cs40l50->dsp.num = 1; cs40l50->dsp.type = WMFW_HALO; cs40l50->dsp.dev = cs40l50->dev; cs40l50->dsp.regmap = cs40l50->regmap; cs40l50->dsp.base = CS40L50_CORE_BASE; cs40l50->dsp.base_sysinfo = CS40L50_SYS_INFO_ID; cs40l50->dsp.mem = cs40l50_dsp_regions; cs40l50->dsp.num_mems = ARRAY_SIZE(cs40l50_dsp_regions); cs40l50->dsp.no_core_startstop = true; cs40l50->dsp.client_ops = &client_ops; ret = cs_dsp_halo_init(&cs40l50->dsp); if (ret) return ret; return devm_add_action_or_reset(cs40l50->dev, cs40l50_dsp_remove, &cs40l50->dsp); } static int cs40l50_reset_dsp(struct cs40l50 *cs40l50) { int ret; mutex_lock(&cs40l50->lock); if (cs40l50->dsp.running) cs_dsp_stop(&cs40l50->dsp); if (cs40l50->dsp.booted) cs_dsp_power_down(&cs40l50->dsp); ret = cs40l50_dsp_write(cs40l50->dev, cs40l50->regmap, CS40L50_SHUTDOWN); if (ret) goto err_mutex; ret = cs_dsp_power_up(&cs40l50->dsp, cs40l50->fw, "cs40l50.wmfw", cs40l50->bin, "cs40l50.bin", "cs40l50"); if (ret) goto err_mutex; ret = cs40l50_dsp_write(cs40l50->dev, cs40l50->regmap, CS40L50_SYSTEM_RESET); if (ret) goto err_mutex; ret = cs40l50_dsp_write(cs40l50->dev, cs40l50->regmap, CS40L50_PREVENT_HIBER); if (ret) goto err_mutex; ret = cs_dsp_run(&cs40l50->dsp); err_mutex: mutex_unlock(&cs40l50->lock); return ret; } static void cs40l50_dsp_power_down(void *data) { cs_dsp_power_down(data); } static void cs40l50_dsp_stop(void *data) { cs_dsp_stop(data); } static void cs40l50_dsp_bringup(const struct firmware *bin, void *context) { struct cs40l50 *cs40l50 = context; u32 nwaves; int ret; /* Wavetable is optional; bringup DSP regardless */ cs40l50->bin = bin; ret = cs40l50_reset_dsp(cs40l50); if (ret) { dev_err(cs40l50->dev, "Failed to reset DSP: %d\n", ret); goto err_fw; } ret = regmap_read(cs40l50->regmap, CS40L50_NUM_WAVES, &nwaves); if (ret) goto err_fw; dev_info(cs40l50->dev, "%u RAM effects loaded\n", nwaves); /* Add teardown actions for first-time bringup */ ret = devm_add_action_or_reset(cs40l50->dev, cs40l50_dsp_power_down, &cs40l50->dsp); if (ret) { dev_err(cs40l50->dev, "Failed to add power down action: %d\n", ret); goto err_fw; } ret = devm_add_action_or_reset(cs40l50->dev, cs40l50_dsp_stop, &cs40l50->dsp); if (ret) dev_err(cs40l50->dev, "Failed to add stop action: %d\n", ret); err_fw: release_firmware(cs40l50->bin); release_firmware(cs40l50->fw); } static void cs40l50_request_firmware(const struct firmware *fw, void *context) { struct cs40l50 *cs40l50 = context; int ret; if (!fw) { dev_err(cs40l50->dev, "No firmware file found\n"); return; } cs40l50->fw = fw; ret = request_firmware_nowait(THIS_MODULE, FW_ACTION_UEVENT, CS40L50_WT, cs40l50->dev, GFP_KERNEL, cs40l50, cs40l50_dsp_bringup); if (ret) { dev_err(cs40l50->dev, "Failed to request %s: %d\n", CS40L50_WT, ret); release_firmware(cs40l50->fw); } } struct cs40l50_irq { const char *name; int virq; }; static struct cs40l50_irq cs40l50_irqs[] = { { "DSP", }, { "Global", }, { "Boost UVLO", }, { "Boost current limit", }, { "Boost short", }, { "Boost undervolt", }, { "Overtemp", }, { "Amp short", }, }; static const struct reg_sequence cs40l50_err_rls[] = { { CS40L50_ERR_RLS, CS40L50_GLOBAL_ERR_RLS_SET }, { CS40L50_ERR_RLS, CS40L50_GLOBAL_ERR_RLS_CLEAR }, }; static irqreturn_t cs40l50_hw_err(int irq, void *data) { struct cs40l50 *cs40l50 = data; int ret = 0, i; mutex_lock(&cs40l50->lock); /* Log hardware interrupt and execute error release sequence */ for (i = 1; i < ARRAY_SIZE(cs40l50_irqs); i++) { if (cs40l50_irqs[i].virq == irq) { dev_err(cs40l50->dev, "%s error\n", cs40l50_irqs[i].name); ret = regmap_multi_reg_write(cs40l50->regmap, cs40l50_err_rls, ARRAY_SIZE(cs40l50_err_rls)); break; } } mutex_unlock(&cs40l50->lock); return IRQ_RETVAL(!ret); } static irqreturn_t cs40l50_dsp_queue(int irq, void *data) { struct cs40l50 *cs40l50 = data; u32 rd_ptr, val, wt_ptr; int ret = 0; mutex_lock(&cs40l50->lock); /* Read from DSP queue, log, and update read pointer */ while (!ret) { ret = regmap_read(cs40l50->regmap, CS40L50_DSP_QUEUE_WT, &wt_ptr); if (ret) break; ret = regmap_read(cs40l50->regmap, CS40L50_DSP_QUEUE_RD, &rd_ptr); if (ret) break; /* Check if queue is empty */ if (wt_ptr == rd_ptr) break; ret = regmap_read(cs40l50->regmap, rd_ptr, &val); if (ret) break; dev_dbg(cs40l50->dev, "DSP payload: %#X", val); rd_ptr += sizeof(u32); if (rd_ptr > CS40L50_DSP_QUEUE_END) rd_ptr = CS40L50_DSP_QUEUE_BASE; ret = regmap_write(cs40l50->regmap, CS40L50_DSP_QUEUE_RD, rd_ptr); } mutex_unlock(&cs40l50->lock); return IRQ_RETVAL(!ret); } static int cs40l50_irq_init(struct cs40l50 *cs40l50) { int ret, i, virq; ret = devm_regmap_add_irq_chip(cs40l50->dev, cs40l50->regmap, cs40l50->irq, IRQF_ONESHOT | IRQF_SHARED, 0, &cs40l50_irq_chip, &cs40l50->irq_data); if (ret) { dev_err(cs40l50->dev, "Failed adding IRQ chip\n"); return ret; } for (i = 0; i < ARRAY_SIZE(cs40l50_irqs); i++) { virq = regmap_irq_get_virq(cs40l50->irq_data, i); if (virq < 0) { dev_err(cs40l50->dev, "Failed getting virq for %s\n", cs40l50_irqs[i].name); return virq; } cs40l50_irqs[i].virq = virq; /* Handle DSP and hardware interrupts separately */ ret = devm_request_threaded_irq(cs40l50->dev, virq, NULL, i ? cs40l50_hw_err : cs40l50_dsp_queue, IRQF_ONESHOT | IRQF_SHARED, cs40l50_irqs[i].name, cs40l50); if (ret) { return dev_err_probe(cs40l50->dev, ret, "Failed requesting %s IRQ\n", cs40l50_irqs[i].name); } } return 0; } static int cs40l50_get_model(struct cs40l50 *cs40l50) { int ret; ret = regmap_read(cs40l50->regmap, CS40L50_DEVID, &cs40l50->devid); if (ret) return ret; if (cs40l50->devid != CS40L50_DEVID_A) return -EINVAL; ret = regmap_read(cs40l50->regmap, CS40L50_REVID, &cs40l50->revid); if (ret) return ret; if (cs40l50->revid < CS40L50_REVID_B0) return -EINVAL; dev_dbg(cs40l50->dev, "Cirrus Logic CS40L50 rev. %02X\n", cs40l50->revid); return 0; } static int cs40l50_pm_runtime_setup(struct device *dev) { int ret; pm_runtime_set_autosuspend_delay(dev, CS40L50_AUTOSUSPEND_MS); pm_runtime_use_autosuspend(dev); pm_runtime_get_noresume(dev); ret = pm_runtime_set_active(dev); if (ret) return ret; return devm_pm_runtime_enable(dev); } int cs40l50_probe(struct cs40l50 *cs40l50) { struct device *dev = cs40l50->dev; int ret; mutex_init(&cs40l50->lock); cs40l50->reset_gpio = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_HIGH); if (IS_ERR(cs40l50->reset_gpio)) return dev_err_probe(dev, PTR_ERR(cs40l50->reset_gpio), "Failed getting reset GPIO\n"); ret = devm_regulator_bulk_get_enable(dev, ARRAY_SIZE(cs40l50_supplies), cs40l50_supplies); if (ret) return dev_err_probe(dev, ret, "Failed getting supplies\n"); /* Ensure minimum reset pulse width */ usleep_range(CS40L50_RESET_PULSE_US, CS40L50_RESET_PULSE_US + 100); gpiod_set_value_cansleep(cs40l50->reset_gpio, 0); /* Wait for control port to be ready */ usleep_range(CS40L50_CP_READY_US, CS40L50_CP_READY_US + 100); ret = cs40l50_get_model(cs40l50); if (ret) return dev_err_probe(dev, ret, "Failed to get part number\n"); ret = cs40l50_dsp_init(cs40l50); if (ret) return dev_err_probe(dev, ret, "Failed to initialize DSP\n"); ret = cs40l50_pm_runtime_setup(dev); if (ret) return dev_err_probe(dev, ret, "Failed to initialize runtime PM\n"); ret = cs40l50_irq_init(cs40l50); if (ret) return ret; ret = request_firmware_nowait(THIS_MODULE, FW_ACTION_UEVENT, CS40L50_FW, dev, GFP_KERNEL, cs40l50, cs40l50_request_firmware); if (ret) return dev_err_probe(dev, ret, "Failed to request %s\n", CS40L50_FW); pm_runtime_mark_last_busy(dev); pm_runtime_put_autosuspend(dev); return 0; } EXPORT_SYMBOL_GPL(cs40l50_probe); int cs40l50_remove(struct cs40l50 *cs40l50) { gpiod_set_value_cansleep(cs40l50->reset_gpio, 1); return 0; } EXPORT_SYMBOL_GPL(cs40l50_remove); static int cs40l50_runtime_suspend(struct device *dev) { struct cs40l50 *cs40l50 = dev_get_drvdata(dev); return regmap_write(cs40l50->regmap, CS40L50_DSP_QUEUE, CS40L50_ALLOW_HIBER); } static int cs40l50_runtime_resume(struct device *dev) { struct cs40l50 *cs40l50 = dev_get_drvdata(dev); return cs40l50_dsp_write(dev, cs40l50->regmap, CS40L50_PREVENT_HIBER); } EXPORT_GPL_DEV_PM_OPS(cs40l50_pm_ops) = { RUNTIME_PM_OPS(cs40l50_runtime_suspend, cs40l50_runtime_resume, NULL) }; MODULE_DESCRIPTION("CS40L50 Advanced Haptic Driver"); MODULE_AUTHOR("James Ogletree, Cirrus Logic Inc. "); MODULE_LICENSE("GPL"); MODULE_IMPORT_NS(FW_CS_DSP);