mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	Clean up the existing export namespace code along the same lines of
commit 33def8498f ("treewide: Convert macro and uses of __section(foo)
to __section("foo")") and for the same reason, it is not desired for the
namespace argument to be a macro expansion itself.
Scripted using
  git grep -l -e MODULE_IMPORT_NS -e EXPORT_SYMBOL_NS | while read file;
  do
    awk -i inplace '
      /^#define EXPORT_SYMBOL_NS/ {
        gsub(/__stringify\(ns\)/, "ns");
        print;
        next;
      }
      /^#define MODULE_IMPORT_NS/ {
        gsub(/__stringify\(ns\)/, "ns");
        print;
        next;
      }
      /MODULE_IMPORT_NS/ {
        $0 = gensub(/MODULE_IMPORT_NS\(([^)]*)\)/, "MODULE_IMPORT_NS(\"\\1\")", "g");
      }
      /EXPORT_SYMBOL_NS/ {
        if ($0 ~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+),/) {
  	if ($0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/ &&
  	    $0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(\)/ &&
  	    $0 !~ /^my/) {
  	  getline line;
  	  gsub(/[[:space:]]*\\$/, "");
  	  gsub(/[[:space:]]/, "", line);
  	  $0 = $0 " " line;
  	}
  	$0 = gensub(/(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/,
  		    "\\1(\\2, \"\\3\")", "g");
        }
      }
      { print }' $file;
  done
Requested-by: Masahiro Yamada <masahiroy@kernel.org>
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Link: https://mail.google.com/mail/u/2/#inbox/FMfcgzQXKWgMmjdFwwdsfgxzKpVHWPlc
Acked-by: Greg KH <gregkh@linuxfoundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
	
			
		
			
				
	
	
		
			570 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			570 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// 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 <james.ogletree@cirrus.com>
 | 
						|
 */
 | 
						|
 | 
						|
#include <linux/firmware/cirrus/cs_dsp.h>
 | 
						|
#include <linux/firmware/cirrus/wmfw.h>
 | 
						|
#include <linux/mfd/core.h>
 | 
						|
#include <linux/mfd/cs40l50.h>
 | 
						|
#include <linux/pm_runtime.h>
 | 
						|
#include <linux/regulator/consumer.h>
 | 
						|
 | 
						|
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. <james.ogletree@cirrus.com>");
 | 
						|
MODULE_LICENSE("GPL");
 | 
						|
MODULE_IMPORT_NS("FW_CS_DSP");
 |