162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * LM4857 AMP driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright 2007 Wolfson Microelectronics PLC. 662306a36Sopenharmony_ci * Author: Graeme Gregory 762306a36Sopenharmony_ci * graeme.gregory@wolfsonmicro.com 862306a36Sopenharmony_ci * Copyright 2011 Lars-Peter Clausen <lars@metafoo.de> 962306a36Sopenharmony_ci */ 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include <linux/init.h> 1262306a36Sopenharmony_ci#include <linux/module.h> 1362306a36Sopenharmony_ci#include <linux/i2c.h> 1462306a36Sopenharmony_ci#include <linux/regmap.h> 1562306a36Sopenharmony_ci#include <linux/slab.h> 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci#include <sound/core.h> 1862306a36Sopenharmony_ci#include <sound/soc.h> 1962306a36Sopenharmony_ci#include <sound/tlv.h> 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_cistatic const struct reg_default lm4857_default_regs[] = { 2262306a36Sopenharmony_ci { 0x0, 0x00 }, 2362306a36Sopenharmony_ci { 0x1, 0x00 }, 2462306a36Sopenharmony_ci { 0x2, 0x00 }, 2562306a36Sopenharmony_ci { 0x3, 0x00 }, 2662306a36Sopenharmony_ci}; 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_ci/* The register offsets in the cache array */ 2962306a36Sopenharmony_ci#define LM4857_MVOL 0 3062306a36Sopenharmony_ci#define LM4857_LVOL 1 3162306a36Sopenharmony_ci#define LM4857_RVOL 2 3262306a36Sopenharmony_ci#define LM4857_CTRL 3 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci/* the shifts required to set these bits */ 3562306a36Sopenharmony_ci#define LM4857_3D 5 3662306a36Sopenharmony_ci#define LM4857_WAKEUP 5 3762306a36Sopenharmony_ci#define LM4857_EPGAIN 4 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_cistatic const unsigned int lm4857_mode_values[] = { 4062306a36Sopenharmony_ci 0, 4162306a36Sopenharmony_ci 6, 4262306a36Sopenharmony_ci 7, 4362306a36Sopenharmony_ci 8, 4462306a36Sopenharmony_ci 9, 4562306a36Sopenharmony_ci}; 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_cistatic const char * const lm4857_mode_texts[] = { 4862306a36Sopenharmony_ci "Off", 4962306a36Sopenharmony_ci "Earpiece", 5062306a36Sopenharmony_ci "Loudspeaker", 5162306a36Sopenharmony_ci "Loudspeaker + Headphone", 5262306a36Sopenharmony_ci "Headphone", 5362306a36Sopenharmony_ci}; 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_cistatic SOC_VALUE_ENUM_SINGLE_AUTODISABLE_DECL(lm4857_mode_enum, 5662306a36Sopenharmony_ci LM4857_CTRL, 0, 0xf, lm4857_mode_texts, lm4857_mode_values); 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_cistatic const struct snd_kcontrol_new lm4857_mode_ctrl = 5962306a36Sopenharmony_ci SOC_DAPM_ENUM("Mode", lm4857_mode_enum); 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_cistatic const struct snd_soc_dapm_widget lm4857_dapm_widgets[] = { 6262306a36Sopenharmony_ci SND_SOC_DAPM_INPUT("IN"), 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_ci SND_SOC_DAPM_DEMUX("Mode", SND_SOC_NOPM, 0, 0, &lm4857_mode_ctrl), 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("LS"), 6762306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("HP"), 6862306a36Sopenharmony_ci SND_SOC_DAPM_OUTPUT("EP"), 6962306a36Sopenharmony_ci}; 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_cistatic const DECLARE_TLV_DB_SCALE(stereo_tlv, -4050, 150, 0); 7262306a36Sopenharmony_cistatic const DECLARE_TLV_DB_SCALE(mono_tlv, -3450, 150, 0); 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_cistatic const struct snd_kcontrol_new lm4857_controls[] = { 7562306a36Sopenharmony_ci SOC_SINGLE_TLV("Left Playback Volume", LM4857_LVOL, 0, 31, 0, 7662306a36Sopenharmony_ci stereo_tlv), 7762306a36Sopenharmony_ci SOC_SINGLE_TLV("Right Playback Volume", LM4857_RVOL, 0, 31, 0, 7862306a36Sopenharmony_ci stereo_tlv), 7962306a36Sopenharmony_ci SOC_SINGLE_TLV("Mono Playback Volume", LM4857_MVOL, 0, 31, 0, 8062306a36Sopenharmony_ci mono_tlv), 8162306a36Sopenharmony_ci SOC_SINGLE("Spk 3D Playback Switch", LM4857_LVOL, LM4857_3D, 1, 0), 8262306a36Sopenharmony_ci SOC_SINGLE("HP 3D Playback Switch", LM4857_RVOL, LM4857_3D, 1, 0), 8362306a36Sopenharmony_ci SOC_SINGLE("Fast Wakeup Playback Switch", LM4857_CTRL, 8462306a36Sopenharmony_ci LM4857_WAKEUP, 1, 0), 8562306a36Sopenharmony_ci SOC_SINGLE("Earpiece 6dB Playback Switch", LM4857_CTRL, 8662306a36Sopenharmony_ci LM4857_EPGAIN, 1, 0), 8762306a36Sopenharmony_ci}; 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_cistatic const struct snd_soc_dapm_route lm4857_routes[] = { 9062306a36Sopenharmony_ci { "Mode", NULL, "IN" }, 9162306a36Sopenharmony_ci { "LS", "Loudspeaker", "Mode" }, 9262306a36Sopenharmony_ci { "LS", "Loudspeaker + Headphone", "Mode" }, 9362306a36Sopenharmony_ci { "HP", "Headphone", "Mode" }, 9462306a36Sopenharmony_ci { "HP", "Loudspeaker + Headphone", "Mode" }, 9562306a36Sopenharmony_ci { "EP", "Earpiece", "Mode" }, 9662306a36Sopenharmony_ci}; 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_cistatic const struct snd_soc_component_driver lm4857_component_driver = { 9962306a36Sopenharmony_ci .controls = lm4857_controls, 10062306a36Sopenharmony_ci .num_controls = ARRAY_SIZE(lm4857_controls), 10162306a36Sopenharmony_ci .dapm_widgets = lm4857_dapm_widgets, 10262306a36Sopenharmony_ci .num_dapm_widgets = ARRAY_SIZE(lm4857_dapm_widgets), 10362306a36Sopenharmony_ci .dapm_routes = lm4857_routes, 10462306a36Sopenharmony_ci .num_dapm_routes = ARRAY_SIZE(lm4857_routes), 10562306a36Sopenharmony_ci}; 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_cistatic const struct regmap_config lm4857_regmap_config = { 10862306a36Sopenharmony_ci .val_bits = 6, 10962306a36Sopenharmony_ci .reg_bits = 2, 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci .max_register = LM4857_CTRL, 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_ci .cache_type = REGCACHE_FLAT, 11462306a36Sopenharmony_ci .reg_defaults = lm4857_default_regs, 11562306a36Sopenharmony_ci .num_reg_defaults = ARRAY_SIZE(lm4857_default_regs), 11662306a36Sopenharmony_ci}; 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_cistatic int lm4857_i2c_probe(struct i2c_client *i2c) 11962306a36Sopenharmony_ci{ 12062306a36Sopenharmony_ci struct regmap *regmap; 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_ci regmap = devm_regmap_init_i2c(i2c, &lm4857_regmap_config); 12362306a36Sopenharmony_ci if (IS_ERR(regmap)) 12462306a36Sopenharmony_ci return PTR_ERR(regmap); 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ci return devm_snd_soc_register_component(&i2c->dev, 12762306a36Sopenharmony_ci &lm4857_component_driver, NULL, 0); 12862306a36Sopenharmony_ci} 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_cistatic const struct i2c_device_id lm4857_i2c_id[] = { 13162306a36Sopenharmony_ci { "lm4857", 0 }, 13262306a36Sopenharmony_ci { } 13362306a36Sopenharmony_ci}; 13462306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, lm4857_i2c_id); 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_cistatic struct i2c_driver lm4857_i2c_driver = { 13762306a36Sopenharmony_ci .driver = { 13862306a36Sopenharmony_ci .name = "lm4857", 13962306a36Sopenharmony_ci }, 14062306a36Sopenharmony_ci .probe = lm4857_i2c_probe, 14162306a36Sopenharmony_ci .id_table = lm4857_i2c_id, 14262306a36Sopenharmony_ci}; 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_cimodule_i2c_driver(lm4857_i2c_driver); 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ciMODULE_AUTHOR("Lars-Peter Clausen <lars@metafoo.de>"); 14762306a36Sopenharmony_ciMODULE_DESCRIPTION("LM4857 amplifier driver"); 14862306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 149