162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Universal Interface for Intel High Definition Audio Codec 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * HD audio interface patch for Silicon Labs 3054/5 modem codec 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Copyright (c) 2005 Sasha Khapyorsky <sashak@alsa-project.org> 862306a36Sopenharmony_ci * Takashi Iwai <tiwai@suse.de> 962306a36Sopenharmony_ci */ 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include <linux/init.h> 1262306a36Sopenharmony_ci#include <linux/delay.h> 1362306a36Sopenharmony_ci#include <linux/slab.h> 1462306a36Sopenharmony_ci#include <linux/module.h> 1562306a36Sopenharmony_ci#include <sound/core.h> 1662306a36Sopenharmony_ci#include <sound/hda_codec.h> 1762306a36Sopenharmony_ci#include "hda_local.h" 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci/* si3054 verbs */ 2062306a36Sopenharmony_ci#define SI3054_VERB_READ_NODE 0x900 2162306a36Sopenharmony_ci#define SI3054_VERB_WRITE_NODE 0x100 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci/* si3054 nodes (registers) */ 2462306a36Sopenharmony_ci#define SI3054_EXTENDED_MID 2 2562306a36Sopenharmony_ci#define SI3054_LINE_RATE 3 2662306a36Sopenharmony_ci#define SI3054_LINE_LEVEL 4 2762306a36Sopenharmony_ci#define SI3054_GPIO_CFG 5 2862306a36Sopenharmony_ci#define SI3054_GPIO_POLARITY 6 2962306a36Sopenharmony_ci#define SI3054_GPIO_STICKY 7 3062306a36Sopenharmony_ci#define SI3054_GPIO_WAKEUP 8 3162306a36Sopenharmony_ci#define SI3054_GPIO_STATUS 9 3262306a36Sopenharmony_ci#define SI3054_GPIO_CONTROL 10 3362306a36Sopenharmony_ci#define SI3054_MISC_AFE 11 3462306a36Sopenharmony_ci#define SI3054_CHIPID 12 3562306a36Sopenharmony_ci#define SI3054_LINE_CFG1 13 3662306a36Sopenharmony_ci#define SI3054_LINE_STATUS 14 3762306a36Sopenharmony_ci#define SI3054_DC_TERMINATION 15 3862306a36Sopenharmony_ci#define SI3054_LINE_CONFIG 16 3962306a36Sopenharmony_ci#define SI3054_CALLPROG_ATT 17 4062306a36Sopenharmony_ci#define SI3054_SQ_CONTROL 18 4162306a36Sopenharmony_ci#define SI3054_MISC_CONTROL 19 4262306a36Sopenharmony_ci#define SI3054_RING_CTRL1 20 4362306a36Sopenharmony_ci#define SI3054_RING_CTRL2 21 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_ci/* extended MID */ 4662306a36Sopenharmony_ci#define SI3054_MEI_READY 0xf 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci/* line level */ 4962306a36Sopenharmony_ci#define SI3054_ATAG_MASK 0x00f0 5062306a36Sopenharmony_ci#define SI3054_DTAG_MASK 0xf000 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci/* GPIO bits */ 5362306a36Sopenharmony_ci#define SI3054_GPIO_OH 0x0001 5462306a36Sopenharmony_ci#define SI3054_GPIO_CID 0x0002 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci/* chipid and revisions */ 5762306a36Sopenharmony_ci#define SI3054_CHIPID_CODEC_REV_MASK 0x000f 5862306a36Sopenharmony_ci#define SI3054_CHIPID_DAA_REV_MASK 0x00f0 5962306a36Sopenharmony_ci#define SI3054_CHIPID_INTERNATIONAL 0x0100 6062306a36Sopenharmony_ci#define SI3054_CHIPID_DAA_ID 0x0f00 6162306a36Sopenharmony_ci#define SI3054_CHIPID_CODEC_ID (1<<12) 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci/* si3054 codec registers (nodes) access macros */ 6462306a36Sopenharmony_ci#define GET_REG(codec,reg) (snd_hda_codec_read(codec,reg,0,SI3054_VERB_READ_NODE,0)) 6562306a36Sopenharmony_ci#define SET_REG(codec,reg,val) (snd_hda_codec_write(codec,reg,0,SI3054_VERB_WRITE_NODE,val)) 6662306a36Sopenharmony_ci#define SET_REG_CACHE(codec,reg,val) \ 6762306a36Sopenharmony_ci snd_hda_codec_write_cache(codec,reg,0,SI3054_VERB_WRITE_NODE,val) 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_cistruct si3054_spec { 7162306a36Sopenharmony_ci unsigned international; 7262306a36Sopenharmony_ci}; 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci/* 7662306a36Sopenharmony_ci * Modem mixer 7762306a36Sopenharmony_ci */ 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci#define PRIVATE_VALUE(reg,mask) ((reg<<16)|(mask&0xffff)) 8062306a36Sopenharmony_ci#define PRIVATE_REG(val) ((val>>16)&0xffff) 8162306a36Sopenharmony_ci#define PRIVATE_MASK(val) (val&0xffff) 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci#define si3054_switch_info snd_ctl_boolean_mono_info 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_cistatic int si3054_switch_get(struct snd_kcontrol *kcontrol, 8662306a36Sopenharmony_ci struct snd_ctl_elem_value *uvalue) 8762306a36Sopenharmony_ci{ 8862306a36Sopenharmony_ci struct hda_codec *codec = snd_kcontrol_chip(kcontrol); 8962306a36Sopenharmony_ci u16 reg = PRIVATE_REG(kcontrol->private_value); 9062306a36Sopenharmony_ci u16 mask = PRIVATE_MASK(kcontrol->private_value); 9162306a36Sopenharmony_ci uvalue->value.integer.value[0] = (GET_REG(codec, reg)) & mask ? 1 : 0 ; 9262306a36Sopenharmony_ci return 0; 9362306a36Sopenharmony_ci} 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_cistatic int si3054_switch_put(struct snd_kcontrol *kcontrol, 9662306a36Sopenharmony_ci struct snd_ctl_elem_value *uvalue) 9762306a36Sopenharmony_ci{ 9862306a36Sopenharmony_ci struct hda_codec *codec = snd_kcontrol_chip(kcontrol); 9962306a36Sopenharmony_ci u16 reg = PRIVATE_REG(kcontrol->private_value); 10062306a36Sopenharmony_ci u16 mask = PRIVATE_MASK(kcontrol->private_value); 10162306a36Sopenharmony_ci if (uvalue->value.integer.value[0]) 10262306a36Sopenharmony_ci SET_REG_CACHE(codec, reg, (GET_REG(codec, reg)) | mask); 10362306a36Sopenharmony_ci else 10462306a36Sopenharmony_ci SET_REG_CACHE(codec, reg, (GET_REG(codec, reg)) & ~mask); 10562306a36Sopenharmony_ci return 0; 10662306a36Sopenharmony_ci} 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_ci#define SI3054_KCONTROL(kname,reg,mask) { \ 10962306a36Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, \ 11062306a36Sopenharmony_ci .name = kname, \ 11162306a36Sopenharmony_ci .subdevice = HDA_SUBDEV_NID_FLAG | reg, \ 11262306a36Sopenharmony_ci .info = si3054_switch_info, \ 11362306a36Sopenharmony_ci .get = si3054_switch_get, \ 11462306a36Sopenharmony_ci .put = si3054_switch_put, \ 11562306a36Sopenharmony_ci .private_value = PRIVATE_VALUE(reg,mask), \ 11662306a36Sopenharmony_ci} 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_cistatic const struct snd_kcontrol_new si3054_modem_mixer[] = { 12062306a36Sopenharmony_ci SI3054_KCONTROL("Off-hook Switch", SI3054_GPIO_CONTROL, SI3054_GPIO_OH), 12162306a36Sopenharmony_ci SI3054_KCONTROL("Caller ID Switch", SI3054_GPIO_CONTROL, SI3054_GPIO_CID), 12262306a36Sopenharmony_ci {} 12362306a36Sopenharmony_ci}; 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_cistatic int si3054_build_controls(struct hda_codec *codec) 12662306a36Sopenharmony_ci{ 12762306a36Sopenharmony_ci return snd_hda_add_new_ctls(codec, si3054_modem_mixer); 12862306a36Sopenharmony_ci} 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_ci 13162306a36Sopenharmony_ci/* 13262306a36Sopenharmony_ci * PCM callbacks 13362306a36Sopenharmony_ci */ 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_cistatic int si3054_pcm_prepare(struct hda_pcm_stream *hinfo, 13662306a36Sopenharmony_ci struct hda_codec *codec, 13762306a36Sopenharmony_ci unsigned int stream_tag, 13862306a36Sopenharmony_ci unsigned int format, 13962306a36Sopenharmony_ci struct snd_pcm_substream *substream) 14062306a36Sopenharmony_ci{ 14162306a36Sopenharmony_ci u16 val; 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci SET_REG(codec, SI3054_LINE_RATE, substream->runtime->rate); 14462306a36Sopenharmony_ci val = GET_REG(codec, SI3054_LINE_LEVEL); 14562306a36Sopenharmony_ci val &= 0xff << (8 * (substream->stream != SNDRV_PCM_STREAM_PLAYBACK)); 14662306a36Sopenharmony_ci val |= ((stream_tag & 0xf) << 4) << (8 * (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)); 14762306a36Sopenharmony_ci SET_REG(codec, SI3054_LINE_LEVEL, val); 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci snd_hda_codec_setup_stream(codec, hinfo->nid, 15062306a36Sopenharmony_ci stream_tag, 0, format); 15162306a36Sopenharmony_ci return 0; 15262306a36Sopenharmony_ci} 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_cistatic int si3054_pcm_open(struct hda_pcm_stream *hinfo, 15562306a36Sopenharmony_ci struct hda_codec *codec, 15662306a36Sopenharmony_ci struct snd_pcm_substream *substream) 15762306a36Sopenharmony_ci{ 15862306a36Sopenharmony_ci static const unsigned int rates[] = { 8000, 9600, 16000 }; 15962306a36Sopenharmony_ci static const struct snd_pcm_hw_constraint_list hw_constraints_rates = { 16062306a36Sopenharmony_ci .count = ARRAY_SIZE(rates), 16162306a36Sopenharmony_ci .list = rates, 16262306a36Sopenharmony_ci .mask = 0, 16362306a36Sopenharmony_ci }; 16462306a36Sopenharmony_ci substream->runtime->hw.period_bytes_min = 80; 16562306a36Sopenharmony_ci return snd_pcm_hw_constraint_list(substream->runtime, 0, 16662306a36Sopenharmony_ci SNDRV_PCM_HW_PARAM_RATE, &hw_constraints_rates); 16762306a36Sopenharmony_ci} 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_cistatic const struct hda_pcm_stream si3054_pcm = { 17162306a36Sopenharmony_ci .substreams = 1, 17262306a36Sopenharmony_ci .channels_min = 1, 17362306a36Sopenharmony_ci .channels_max = 1, 17462306a36Sopenharmony_ci .nid = 0x1, 17562306a36Sopenharmony_ci .rates = SNDRV_PCM_RATE_8000|SNDRV_PCM_RATE_16000|SNDRV_PCM_RATE_KNOT, 17662306a36Sopenharmony_ci .formats = SNDRV_PCM_FMTBIT_S16_LE, 17762306a36Sopenharmony_ci .maxbps = 16, 17862306a36Sopenharmony_ci .ops = { 17962306a36Sopenharmony_ci .open = si3054_pcm_open, 18062306a36Sopenharmony_ci .prepare = si3054_pcm_prepare, 18162306a36Sopenharmony_ci }, 18262306a36Sopenharmony_ci}; 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_ci 18562306a36Sopenharmony_cistatic int si3054_build_pcms(struct hda_codec *codec) 18662306a36Sopenharmony_ci{ 18762306a36Sopenharmony_ci struct hda_pcm *info; 18862306a36Sopenharmony_ci 18962306a36Sopenharmony_ci info = snd_hda_codec_pcm_new(codec, "Si3054 Modem"); 19062306a36Sopenharmony_ci if (!info) 19162306a36Sopenharmony_ci return -ENOMEM; 19262306a36Sopenharmony_ci info->stream[SNDRV_PCM_STREAM_PLAYBACK] = si3054_pcm; 19362306a36Sopenharmony_ci info->stream[SNDRV_PCM_STREAM_CAPTURE] = si3054_pcm; 19462306a36Sopenharmony_ci info->stream[SNDRV_PCM_STREAM_PLAYBACK].nid = codec->core.mfg; 19562306a36Sopenharmony_ci info->stream[SNDRV_PCM_STREAM_CAPTURE].nid = codec->core.mfg; 19662306a36Sopenharmony_ci info->pcm_type = HDA_PCM_TYPE_MODEM; 19762306a36Sopenharmony_ci return 0; 19862306a36Sopenharmony_ci} 19962306a36Sopenharmony_ci 20062306a36Sopenharmony_ci 20162306a36Sopenharmony_ci/* 20262306a36Sopenharmony_ci * Init part 20362306a36Sopenharmony_ci */ 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_cistatic int si3054_init(struct hda_codec *codec) 20662306a36Sopenharmony_ci{ 20762306a36Sopenharmony_ci struct si3054_spec *spec = codec->spec; 20862306a36Sopenharmony_ci unsigned wait_count; 20962306a36Sopenharmony_ci u16 val; 21062306a36Sopenharmony_ci 21162306a36Sopenharmony_ci if (snd_hdac_regmap_add_vendor_verb(&codec->core, 21262306a36Sopenharmony_ci SI3054_VERB_WRITE_NODE)) 21362306a36Sopenharmony_ci return -ENOMEM; 21462306a36Sopenharmony_ci 21562306a36Sopenharmony_ci snd_hda_codec_write(codec, AC_NODE_ROOT, 0, AC_VERB_SET_CODEC_RESET, 0); 21662306a36Sopenharmony_ci snd_hda_codec_write(codec, codec->core.mfg, 0, AC_VERB_SET_STREAM_FORMAT, 0); 21762306a36Sopenharmony_ci SET_REG(codec, SI3054_LINE_RATE, 9600); 21862306a36Sopenharmony_ci SET_REG(codec, SI3054_LINE_LEVEL, SI3054_DTAG_MASK|SI3054_ATAG_MASK); 21962306a36Sopenharmony_ci SET_REG(codec, SI3054_EXTENDED_MID, 0); 22062306a36Sopenharmony_ci 22162306a36Sopenharmony_ci wait_count = 10; 22262306a36Sopenharmony_ci do { 22362306a36Sopenharmony_ci msleep(2); 22462306a36Sopenharmony_ci val = GET_REG(codec, SI3054_EXTENDED_MID); 22562306a36Sopenharmony_ci } while ((val & SI3054_MEI_READY) != SI3054_MEI_READY && wait_count--); 22662306a36Sopenharmony_ci 22762306a36Sopenharmony_ci if((val&SI3054_MEI_READY) != SI3054_MEI_READY) { 22862306a36Sopenharmony_ci codec_err(codec, "si3054: cannot initialize. EXT MID = %04x\n", val); 22962306a36Sopenharmony_ci /* let's pray that this is no fatal error */ 23062306a36Sopenharmony_ci /* return -EACCES; */ 23162306a36Sopenharmony_ci } 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_ci SET_REG(codec, SI3054_GPIO_POLARITY, 0xffff); 23462306a36Sopenharmony_ci SET_REG(codec, SI3054_GPIO_CFG, 0x0); 23562306a36Sopenharmony_ci SET_REG(codec, SI3054_MISC_AFE, 0); 23662306a36Sopenharmony_ci SET_REG(codec, SI3054_LINE_CFG1,0x200); 23762306a36Sopenharmony_ci 23862306a36Sopenharmony_ci if((GET_REG(codec,SI3054_LINE_STATUS) & (1<<6)) == 0) { 23962306a36Sopenharmony_ci codec_dbg(codec, 24062306a36Sopenharmony_ci "Link Frame Detect(FDT) is not ready (line status: %04x)\n", 24162306a36Sopenharmony_ci GET_REG(codec,SI3054_LINE_STATUS)); 24262306a36Sopenharmony_ci } 24362306a36Sopenharmony_ci 24462306a36Sopenharmony_ci spec->international = GET_REG(codec, SI3054_CHIPID) & SI3054_CHIPID_INTERNATIONAL; 24562306a36Sopenharmony_ci 24662306a36Sopenharmony_ci return 0; 24762306a36Sopenharmony_ci} 24862306a36Sopenharmony_ci 24962306a36Sopenharmony_cistatic void si3054_free(struct hda_codec *codec) 25062306a36Sopenharmony_ci{ 25162306a36Sopenharmony_ci kfree(codec->spec); 25262306a36Sopenharmony_ci} 25362306a36Sopenharmony_ci 25462306a36Sopenharmony_ci 25562306a36Sopenharmony_ci/* 25662306a36Sopenharmony_ci */ 25762306a36Sopenharmony_ci 25862306a36Sopenharmony_cistatic const struct hda_codec_ops si3054_patch_ops = { 25962306a36Sopenharmony_ci .build_controls = si3054_build_controls, 26062306a36Sopenharmony_ci .build_pcms = si3054_build_pcms, 26162306a36Sopenharmony_ci .init = si3054_init, 26262306a36Sopenharmony_ci .free = si3054_free, 26362306a36Sopenharmony_ci}; 26462306a36Sopenharmony_ci 26562306a36Sopenharmony_cistatic int patch_si3054(struct hda_codec *codec) 26662306a36Sopenharmony_ci{ 26762306a36Sopenharmony_ci struct si3054_spec *spec = kzalloc(sizeof(*spec), GFP_KERNEL); 26862306a36Sopenharmony_ci if (spec == NULL) 26962306a36Sopenharmony_ci return -ENOMEM; 27062306a36Sopenharmony_ci codec->spec = spec; 27162306a36Sopenharmony_ci codec->patch_ops = si3054_patch_ops; 27262306a36Sopenharmony_ci return 0; 27362306a36Sopenharmony_ci} 27462306a36Sopenharmony_ci 27562306a36Sopenharmony_ci/* 27662306a36Sopenharmony_ci * patch entries 27762306a36Sopenharmony_ci */ 27862306a36Sopenharmony_cistatic const struct hda_device_id snd_hda_id_si3054[] = { 27962306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x163c3055, "Si3054", patch_si3054), 28062306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x163c3155, "Si3054", patch_si3054), 28162306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x11c13026, "Si3054", patch_si3054), 28262306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x11c13055, "Si3054", patch_si3054), 28362306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x11c13155, "Si3054", patch_si3054), 28462306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x10573055, "Si3054", patch_si3054), 28562306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x10573057, "Si3054", patch_si3054), 28662306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x10573155, "Si3054", patch_si3054), 28762306a36Sopenharmony_ci /* VIA HDA on Clevo m540 */ 28862306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x11063288, "Si3054", patch_si3054), 28962306a36Sopenharmony_ci /* Asus A8J Modem (SM56) */ 29062306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x15433155, "Si3054", patch_si3054), 29162306a36Sopenharmony_ci /* LG LW20 modem */ 29262306a36Sopenharmony_ci HDA_CODEC_ENTRY(0x18540018, "Si3054", patch_si3054), 29362306a36Sopenharmony_ci {} 29462306a36Sopenharmony_ci}; 29562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(hdaudio, snd_hda_id_si3054); 29662306a36Sopenharmony_ci 29762306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 29862306a36Sopenharmony_ciMODULE_DESCRIPTION("Si3054 HD-audio modem codec"); 29962306a36Sopenharmony_ci 30062306a36Sopenharmony_cistatic struct hda_codec_driver si3054_driver = { 30162306a36Sopenharmony_ci .id = snd_hda_id_si3054, 30262306a36Sopenharmony_ci}; 30362306a36Sopenharmony_ci 30462306a36Sopenharmony_cimodule_hda_codec_driver(si3054_driver); 305