18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * ALSA driver for ICEnsemble VT1724 (Envy24HT) 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Lowlevel functions for Pontis MS300 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * Copyright (c) 2004 Takashi Iwai <tiwai@suse.de> 88c2ecf20Sopenharmony_ci */ 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <linux/delay.h> 118c2ecf20Sopenharmony_ci#include <linux/interrupt.h> 128c2ecf20Sopenharmony_ci#include <linux/init.h> 138c2ecf20Sopenharmony_ci#include <linux/slab.h> 148c2ecf20Sopenharmony_ci#include <linux/mutex.h> 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci#include <sound/core.h> 178c2ecf20Sopenharmony_ci#include <sound/info.h> 188c2ecf20Sopenharmony_ci#include <sound/tlv.h> 198c2ecf20Sopenharmony_ci 208c2ecf20Sopenharmony_ci#include "ice1712.h" 218c2ecf20Sopenharmony_ci#include "envy24ht.h" 228c2ecf20Sopenharmony_ci#include "pontis.h" 238c2ecf20Sopenharmony_ci 248c2ecf20Sopenharmony_ci/* I2C addresses */ 258c2ecf20Sopenharmony_ci#define WM_DEV 0x34 268c2ecf20Sopenharmony_ci#define CS_DEV 0x20 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci/* WM8776 registers */ 298c2ecf20Sopenharmony_ci#define WM_HP_ATTEN_L 0x00 /* headphone left attenuation */ 308c2ecf20Sopenharmony_ci#define WM_HP_ATTEN_R 0x01 /* headphone left attenuation */ 318c2ecf20Sopenharmony_ci#define WM_HP_MASTER 0x02 /* headphone master (both channels) */ 328c2ecf20Sopenharmony_ci /* override LLR */ 338c2ecf20Sopenharmony_ci#define WM_DAC_ATTEN_L 0x03 /* digital left attenuation */ 348c2ecf20Sopenharmony_ci#define WM_DAC_ATTEN_R 0x04 358c2ecf20Sopenharmony_ci#define WM_DAC_MASTER 0x05 368c2ecf20Sopenharmony_ci#define WM_PHASE_SWAP 0x06 /* DAC phase swap */ 378c2ecf20Sopenharmony_ci#define WM_DAC_CTRL1 0x07 388c2ecf20Sopenharmony_ci#define WM_DAC_MUTE 0x08 398c2ecf20Sopenharmony_ci#define WM_DAC_CTRL2 0x09 408c2ecf20Sopenharmony_ci#define WM_DAC_INT 0x0a 418c2ecf20Sopenharmony_ci#define WM_ADC_INT 0x0b 428c2ecf20Sopenharmony_ci#define WM_MASTER_CTRL 0x0c 438c2ecf20Sopenharmony_ci#define WM_POWERDOWN 0x0d 448c2ecf20Sopenharmony_ci#define WM_ADC_ATTEN_L 0x0e 458c2ecf20Sopenharmony_ci#define WM_ADC_ATTEN_R 0x0f 468c2ecf20Sopenharmony_ci#define WM_ALC_CTRL1 0x10 478c2ecf20Sopenharmony_ci#define WM_ALC_CTRL2 0x11 488c2ecf20Sopenharmony_ci#define WM_ALC_CTRL3 0x12 498c2ecf20Sopenharmony_ci#define WM_NOISE_GATE 0x13 508c2ecf20Sopenharmony_ci#define WM_LIMITER 0x14 518c2ecf20Sopenharmony_ci#define WM_ADC_MUX 0x15 528c2ecf20Sopenharmony_ci#define WM_OUT_MUX 0x16 538c2ecf20Sopenharmony_ci#define WM_RESET 0x17 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_ci/* 568c2ecf20Sopenharmony_ci * GPIO 578c2ecf20Sopenharmony_ci */ 588c2ecf20Sopenharmony_ci#define PONTIS_CS_CS (1<<4) /* CS */ 598c2ecf20Sopenharmony_ci#define PONTIS_CS_CLK (1<<5) /* CLK */ 608c2ecf20Sopenharmony_ci#define PONTIS_CS_RDATA (1<<6) /* CS8416 -> VT1720 */ 618c2ecf20Sopenharmony_ci#define PONTIS_CS_WDATA (1<<7) /* VT1720 -> CS8416 */ 628c2ecf20Sopenharmony_ci 638c2ecf20Sopenharmony_ci 648c2ecf20Sopenharmony_ci/* 658c2ecf20Sopenharmony_ci * get the current register value of WM codec 668c2ecf20Sopenharmony_ci */ 678c2ecf20Sopenharmony_cistatic unsigned short wm_get(struct snd_ice1712 *ice, int reg) 688c2ecf20Sopenharmony_ci{ 698c2ecf20Sopenharmony_ci reg <<= 1; 708c2ecf20Sopenharmony_ci return ((unsigned short)ice->akm[0].images[reg] << 8) | 718c2ecf20Sopenharmony_ci ice->akm[0].images[reg + 1]; 728c2ecf20Sopenharmony_ci} 738c2ecf20Sopenharmony_ci 748c2ecf20Sopenharmony_ci/* 758c2ecf20Sopenharmony_ci * set the register value of WM codec and remember it 768c2ecf20Sopenharmony_ci */ 778c2ecf20Sopenharmony_cistatic void wm_put_nocache(struct snd_ice1712 *ice, int reg, unsigned short val) 788c2ecf20Sopenharmony_ci{ 798c2ecf20Sopenharmony_ci unsigned short cval; 808c2ecf20Sopenharmony_ci cval = (reg << 9) | val; 818c2ecf20Sopenharmony_ci snd_vt1724_write_i2c(ice, WM_DEV, cval >> 8, cval & 0xff); 828c2ecf20Sopenharmony_ci} 838c2ecf20Sopenharmony_ci 848c2ecf20Sopenharmony_cistatic void wm_put(struct snd_ice1712 *ice, int reg, unsigned short val) 858c2ecf20Sopenharmony_ci{ 868c2ecf20Sopenharmony_ci wm_put_nocache(ice, reg, val); 878c2ecf20Sopenharmony_ci reg <<= 1; 888c2ecf20Sopenharmony_ci ice->akm[0].images[reg] = val >> 8; 898c2ecf20Sopenharmony_ci ice->akm[0].images[reg + 1] = val; 908c2ecf20Sopenharmony_ci} 918c2ecf20Sopenharmony_ci 928c2ecf20Sopenharmony_ci/* 938c2ecf20Sopenharmony_ci * DAC volume attenuation mixer control (-64dB to 0dB) 948c2ecf20Sopenharmony_ci */ 958c2ecf20Sopenharmony_ci 968c2ecf20Sopenharmony_ci#define DAC_0dB 0xff 978c2ecf20Sopenharmony_ci#define DAC_RES 128 988c2ecf20Sopenharmony_ci#define DAC_MIN (DAC_0dB - DAC_RES) 998c2ecf20Sopenharmony_ci 1008c2ecf20Sopenharmony_cistatic int wm_dac_vol_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo) 1018c2ecf20Sopenharmony_ci{ 1028c2ecf20Sopenharmony_ci uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; 1038c2ecf20Sopenharmony_ci uinfo->count = 2; 1048c2ecf20Sopenharmony_ci uinfo->value.integer.min = 0; /* mute */ 1058c2ecf20Sopenharmony_ci uinfo->value.integer.max = DAC_RES; /* 0dB, 0.5dB step */ 1068c2ecf20Sopenharmony_ci return 0; 1078c2ecf20Sopenharmony_ci} 1088c2ecf20Sopenharmony_ci 1098c2ecf20Sopenharmony_cistatic int wm_dac_vol_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 1108c2ecf20Sopenharmony_ci{ 1118c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 1128c2ecf20Sopenharmony_ci unsigned short val; 1138c2ecf20Sopenharmony_ci int i; 1148c2ecf20Sopenharmony_ci 1158c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 1168c2ecf20Sopenharmony_ci for (i = 0; i < 2; i++) { 1178c2ecf20Sopenharmony_ci val = wm_get(ice, WM_DAC_ATTEN_L + i) & 0xff; 1188c2ecf20Sopenharmony_ci val = val > DAC_MIN ? (val - DAC_MIN) : 0; 1198c2ecf20Sopenharmony_ci ucontrol->value.integer.value[i] = val; 1208c2ecf20Sopenharmony_ci } 1218c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 1228c2ecf20Sopenharmony_ci return 0; 1238c2ecf20Sopenharmony_ci} 1248c2ecf20Sopenharmony_ci 1258c2ecf20Sopenharmony_cistatic int wm_dac_vol_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 1268c2ecf20Sopenharmony_ci{ 1278c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 1288c2ecf20Sopenharmony_ci unsigned short oval, nval; 1298c2ecf20Sopenharmony_ci int i, idx, change = 0; 1308c2ecf20Sopenharmony_ci 1318c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 1328c2ecf20Sopenharmony_ci for (i = 0; i < 2; i++) { 1338c2ecf20Sopenharmony_ci nval = ucontrol->value.integer.value[i]; 1348c2ecf20Sopenharmony_ci nval = (nval ? (nval + DAC_MIN) : 0) & 0xff; 1358c2ecf20Sopenharmony_ci idx = WM_DAC_ATTEN_L + i; 1368c2ecf20Sopenharmony_ci oval = wm_get(ice, idx) & 0xff; 1378c2ecf20Sopenharmony_ci if (oval != nval) { 1388c2ecf20Sopenharmony_ci wm_put(ice, idx, nval); 1398c2ecf20Sopenharmony_ci wm_put_nocache(ice, idx, nval | 0x100); 1408c2ecf20Sopenharmony_ci change = 1; 1418c2ecf20Sopenharmony_ci } 1428c2ecf20Sopenharmony_ci } 1438c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 1448c2ecf20Sopenharmony_ci return change; 1458c2ecf20Sopenharmony_ci} 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_ci/* 1488c2ecf20Sopenharmony_ci * ADC gain mixer control (-64dB to 0dB) 1498c2ecf20Sopenharmony_ci */ 1508c2ecf20Sopenharmony_ci 1518c2ecf20Sopenharmony_ci#define ADC_0dB 0xcf 1528c2ecf20Sopenharmony_ci#define ADC_RES 128 1538c2ecf20Sopenharmony_ci#define ADC_MIN (ADC_0dB - ADC_RES) 1548c2ecf20Sopenharmony_ci 1558c2ecf20Sopenharmony_cistatic int wm_adc_vol_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo) 1568c2ecf20Sopenharmony_ci{ 1578c2ecf20Sopenharmony_ci uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; 1588c2ecf20Sopenharmony_ci uinfo->count = 2; 1598c2ecf20Sopenharmony_ci uinfo->value.integer.min = 0; /* mute (-64dB) */ 1608c2ecf20Sopenharmony_ci uinfo->value.integer.max = ADC_RES; /* 0dB, 0.5dB step */ 1618c2ecf20Sopenharmony_ci return 0; 1628c2ecf20Sopenharmony_ci} 1638c2ecf20Sopenharmony_ci 1648c2ecf20Sopenharmony_cistatic int wm_adc_vol_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 1658c2ecf20Sopenharmony_ci{ 1668c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 1678c2ecf20Sopenharmony_ci unsigned short val; 1688c2ecf20Sopenharmony_ci int i; 1698c2ecf20Sopenharmony_ci 1708c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 1718c2ecf20Sopenharmony_ci for (i = 0; i < 2; i++) { 1728c2ecf20Sopenharmony_ci val = wm_get(ice, WM_ADC_ATTEN_L + i) & 0xff; 1738c2ecf20Sopenharmony_ci val = val > ADC_MIN ? (val - ADC_MIN) : 0; 1748c2ecf20Sopenharmony_ci ucontrol->value.integer.value[i] = val; 1758c2ecf20Sopenharmony_ci } 1768c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 1778c2ecf20Sopenharmony_ci return 0; 1788c2ecf20Sopenharmony_ci} 1798c2ecf20Sopenharmony_ci 1808c2ecf20Sopenharmony_cistatic int wm_adc_vol_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 1818c2ecf20Sopenharmony_ci{ 1828c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 1838c2ecf20Sopenharmony_ci unsigned short ovol, nvol; 1848c2ecf20Sopenharmony_ci int i, idx, change = 0; 1858c2ecf20Sopenharmony_ci 1868c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 1878c2ecf20Sopenharmony_ci for (i = 0; i < 2; i++) { 1888c2ecf20Sopenharmony_ci nvol = ucontrol->value.integer.value[i]; 1898c2ecf20Sopenharmony_ci nvol = nvol ? (nvol + ADC_MIN) : 0; 1908c2ecf20Sopenharmony_ci idx = WM_ADC_ATTEN_L + i; 1918c2ecf20Sopenharmony_ci ovol = wm_get(ice, idx) & 0xff; 1928c2ecf20Sopenharmony_ci if (ovol != nvol) { 1938c2ecf20Sopenharmony_ci wm_put(ice, idx, nvol); 1948c2ecf20Sopenharmony_ci change = 1; 1958c2ecf20Sopenharmony_ci } 1968c2ecf20Sopenharmony_ci } 1978c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 1988c2ecf20Sopenharmony_ci return change; 1998c2ecf20Sopenharmony_ci} 2008c2ecf20Sopenharmony_ci 2018c2ecf20Sopenharmony_ci/* 2028c2ecf20Sopenharmony_ci * ADC input mux mixer control 2038c2ecf20Sopenharmony_ci */ 2048c2ecf20Sopenharmony_ci#define wm_adc_mux_info snd_ctl_boolean_mono_info 2058c2ecf20Sopenharmony_ci 2068c2ecf20Sopenharmony_cistatic int wm_adc_mux_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 2078c2ecf20Sopenharmony_ci{ 2088c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 2098c2ecf20Sopenharmony_ci int bit = kcontrol->private_value; 2108c2ecf20Sopenharmony_ci 2118c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 2128c2ecf20Sopenharmony_ci ucontrol->value.integer.value[0] = (wm_get(ice, WM_ADC_MUX) & (1 << bit)) ? 1 : 0; 2138c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 2148c2ecf20Sopenharmony_ci return 0; 2158c2ecf20Sopenharmony_ci} 2168c2ecf20Sopenharmony_ci 2178c2ecf20Sopenharmony_cistatic int wm_adc_mux_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 2188c2ecf20Sopenharmony_ci{ 2198c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 2208c2ecf20Sopenharmony_ci int bit = kcontrol->private_value; 2218c2ecf20Sopenharmony_ci unsigned short oval, nval; 2228c2ecf20Sopenharmony_ci int change; 2238c2ecf20Sopenharmony_ci 2248c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 2258c2ecf20Sopenharmony_ci nval = oval = wm_get(ice, WM_ADC_MUX); 2268c2ecf20Sopenharmony_ci if (ucontrol->value.integer.value[0]) 2278c2ecf20Sopenharmony_ci nval |= (1 << bit); 2288c2ecf20Sopenharmony_ci else 2298c2ecf20Sopenharmony_ci nval &= ~(1 << bit); 2308c2ecf20Sopenharmony_ci change = nval != oval; 2318c2ecf20Sopenharmony_ci if (change) { 2328c2ecf20Sopenharmony_ci wm_put(ice, WM_ADC_MUX, nval); 2338c2ecf20Sopenharmony_ci } 2348c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 2358c2ecf20Sopenharmony_ci return change; 2368c2ecf20Sopenharmony_ci} 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_ci/* 2398c2ecf20Sopenharmony_ci * Analog bypass (In -> Out) 2408c2ecf20Sopenharmony_ci */ 2418c2ecf20Sopenharmony_ci#define wm_bypass_info snd_ctl_boolean_mono_info 2428c2ecf20Sopenharmony_ci 2438c2ecf20Sopenharmony_cistatic int wm_bypass_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 2448c2ecf20Sopenharmony_ci{ 2458c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 2468c2ecf20Sopenharmony_ci 2478c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 2488c2ecf20Sopenharmony_ci ucontrol->value.integer.value[0] = (wm_get(ice, WM_OUT_MUX) & 0x04) ? 1 : 0; 2498c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 2508c2ecf20Sopenharmony_ci return 0; 2518c2ecf20Sopenharmony_ci} 2528c2ecf20Sopenharmony_ci 2538c2ecf20Sopenharmony_cistatic int wm_bypass_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 2548c2ecf20Sopenharmony_ci{ 2558c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 2568c2ecf20Sopenharmony_ci unsigned short val, oval; 2578c2ecf20Sopenharmony_ci int change = 0; 2588c2ecf20Sopenharmony_ci 2598c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 2608c2ecf20Sopenharmony_ci val = oval = wm_get(ice, WM_OUT_MUX); 2618c2ecf20Sopenharmony_ci if (ucontrol->value.integer.value[0]) 2628c2ecf20Sopenharmony_ci val |= 0x04; 2638c2ecf20Sopenharmony_ci else 2648c2ecf20Sopenharmony_ci val &= ~0x04; 2658c2ecf20Sopenharmony_ci if (val != oval) { 2668c2ecf20Sopenharmony_ci wm_put(ice, WM_OUT_MUX, val); 2678c2ecf20Sopenharmony_ci change = 1; 2688c2ecf20Sopenharmony_ci } 2698c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 2708c2ecf20Sopenharmony_ci return change; 2718c2ecf20Sopenharmony_ci} 2728c2ecf20Sopenharmony_ci 2738c2ecf20Sopenharmony_ci/* 2748c2ecf20Sopenharmony_ci * Left/Right swap 2758c2ecf20Sopenharmony_ci */ 2768c2ecf20Sopenharmony_ci#define wm_chswap_info snd_ctl_boolean_mono_info 2778c2ecf20Sopenharmony_ci 2788c2ecf20Sopenharmony_cistatic int wm_chswap_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 2798c2ecf20Sopenharmony_ci{ 2808c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 2818c2ecf20Sopenharmony_ci 2828c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 2838c2ecf20Sopenharmony_ci ucontrol->value.integer.value[0] = (wm_get(ice, WM_DAC_CTRL1) & 0xf0) != 0x90; 2848c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 2858c2ecf20Sopenharmony_ci return 0; 2868c2ecf20Sopenharmony_ci} 2878c2ecf20Sopenharmony_ci 2888c2ecf20Sopenharmony_cistatic int wm_chswap_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 2898c2ecf20Sopenharmony_ci{ 2908c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 2918c2ecf20Sopenharmony_ci unsigned short val, oval; 2928c2ecf20Sopenharmony_ci int change = 0; 2938c2ecf20Sopenharmony_ci 2948c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 2958c2ecf20Sopenharmony_ci oval = wm_get(ice, WM_DAC_CTRL1); 2968c2ecf20Sopenharmony_ci val = oval & 0x0f; 2978c2ecf20Sopenharmony_ci if (ucontrol->value.integer.value[0]) 2988c2ecf20Sopenharmony_ci val |= 0x60; 2998c2ecf20Sopenharmony_ci else 3008c2ecf20Sopenharmony_ci val |= 0x90; 3018c2ecf20Sopenharmony_ci if (val != oval) { 3028c2ecf20Sopenharmony_ci wm_put(ice, WM_DAC_CTRL1, val); 3038c2ecf20Sopenharmony_ci wm_put_nocache(ice, WM_DAC_CTRL1, val); 3048c2ecf20Sopenharmony_ci change = 1; 3058c2ecf20Sopenharmony_ci } 3068c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 3078c2ecf20Sopenharmony_ci return change; 3088c2ecf20Sopenharmony_ci} 3098c2ecf20Sopenharmony_ci 3108c2ecf20Sopenharmony_ci/* 3118c2ecf20Sopenharmony_ci * write data in the SPI mode 3128c2ecf20Sopenharmony_ci */ 3138c2ecf20Sopenharmony_cistatic void set_gpio_bit(struct snd_ice1712 *ice, unsigned int bit, int val) 3148c2ecf20Sopenharmony_ci{ 3158c2ecf20Sopenharmony_ci unsigned int tmp = snd_ice1712_gpio_read(ice); 3168c2ecf20Sopenharmony_ci if (val) 3178c2ecf20Sopenharmony_ci tmp |= bit; 3188c2ecf20Sopenharmony_ci else 3198c2ecf20Sopenharmony_ci tmp &= ~bit; 3208c2ecf20Sopenharmony_ci snd_ice1712_gpio_write(ice, tmp); 3218c2ecf20Sopenharmony_ci} 3228c2ecf20Sopenharmony_ci 3238c2ecf20Sopenharmony_cistatic void spi_send_byte(struct snd_ice1712 *ice, unsigned char data) 3248c2ecf20Sopenharmony_ci{ 3258c2ecf20Sopenharmony_ci int i; 3268c2ecf20Sopenharmony_ci for (i = 0; i < 8; i++) { 3278c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CLK, 0); 3288c2ecf20Sopenharmony_ci udelay(1); 3298c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_WDATA, data & 0x80); 3308c2ecf20Sopenharmony_ci udelay(1); 3318c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CLK, 1); 3328c2ecf20Sopenharmony_ci udelay(1); 3338c2ecf20Sopenharmony_ci data <<= 1; 3348c2ecf20Sopenharmony_ci } 3358c2ecf20Sopenharmony_ci} 3368c2ecf20Sopenharmony_ci 3378c2ecf20Sopenharmony_cistatic unsigned int spi_read_byte(struct snd_ice1712 *ice) 3388c2ecf20Sopenharmony_ci{ 3398c2ecf20Sopenharmony_ci int i; 3408c2ecf20Sopenharmony_ci unsigned int val = 0; 3418c2ecf20Sopenharmony_ci 3428c2ecf20Sopenharmony_ci for (i = 0; i < 8; i++) { 3438c2ecf20Sopenharmony_ci val <<= 1; 3448c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CLK, 0); 3458c2ecf20Sopenharmony_ci udelay(1); 3468c2ecf20Sopenharmony_ci if (snd_ice1712_gpio_read(ice) & PONTIS_CS_RDATA) 3478c2ecf20Sopenharmony_ci val |= 1; 3488c2ecf20Sopenharmony_ci udelay(1); 3498c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CLK, 1); 3508c2ecf20Sopenharmony_ci udelay(1); 3518c2ecf20Sopenharmony_ci } 3528c2ecf20Sopenharmony_ci return val; 3538c2ecf20Sopenharmony_ci} 3548c2ecf20Sopenharmony_ci 3558c2ecf20Sopenharmony_ci 3568c2ecf20Sopenharmony_cistatic void spi_write(struct snd_ice1712 *ice, unsigned int dev, unsigned int reg, unsigned int data) 3578c2ecf20Sopenharmony_ci{ 3588c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_dir(ice, PONTIS_CS_CS|PONTIS_CS_WDATA|PONTIS_CS_CLK); 3598c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_mask(ice, ~(PONTIS_CS_CS|PONTIS_CS_WDATA|PONTIS_CS_CLK)); 3608c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CS, 0); 3618c2ecf20Sopenharmony_ci spi_send_byte(ice, dev & ~1); /* WRITE */ 3628c2ecf20Sopenharmony_ci spi_send_byte(ice, reg); /* MAP */ 3638c2ecf20Sopenharmony_ci spi_send_byte(ice, data); /* DATA */ 3648c2ecf20Sopenharmony_ci /* trigger */ 3658c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CS, 1); 3668c2ecf20Sopenharmony_ci udelay(1); 3678c2ecf20Sopenharmony_ci /* restore */ 3688c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_mask(ice, ice->gpio.write_mask); 3698c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_dir(ice, ice->gpio.direction); 3708c2ecf20Sopenharmony_ci} 3718c2ecf20Sopenharmony_ci 3728c2ecf20Sopenharmony_cistatic unsigned int spi_read(struct snd_ice1712 *ice, unsigned int dev, unsigned int reg) 3738c2ecf20Sopenharmony_ci{ 3748c2ecf20Sopenharmony_ci unsigned int val; 3758c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_dir(ice, PONTIS_CS_CS|PONTIS_CS_WDATA|PONTIS_CS_CLK); 3768c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_mask(ice, ~(PONTIS_CS_CS|PONTIS_CS_WDATA|PONTIS_CS_CLK)); 3778c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CS, 0); 3788c2ecf20Sopenharmony_ci spi_send_byte(ice, dev & ~1); /* WRITE */ 3798c2ecf20Sopenharmony_ci spi_send_byte(ice, reg); /* MAP */ 3808c2ecf20Sopenharmony_ci /* trigger */ 3818c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CS, 1); 3828c2ecf20Sopenharmony_ci udelay(1); 3838c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CS, 0); 3848c2ecf20Sopenharmony_ci spi_send_byte(ice, dev | 1); /* READ */ 3858c2ecf20Sopenharmony_ci val = spi_read_byte(ice); 3868c2ecf20Sopenharmony_ci /* trigger */ 3878c2ecf20Sopenharmony_ci set_gpio_bit(ice, PONTIS_CS_CS, 1); 3888c2ecf20Sopenharmony_ci udelay(1); 3898c2ecf20Sopenharmony_ci /* restore */ 3908c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_mask(ice, ice->gpio.write_mask); 3918c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_dir(ice, ice->gpio.direction); 3928c2ecf20Sopenharmony_ci return val; 3938c2ecf20Sopenharmony_ci} 3948c2ecf20Sopenharmony_ci 3958c2ecf20Sopenharmony_ci 3968c2ecf20Sopenharmony_ci/* 3978c2ecf20Sopenharmony_ci * SPDIF input source 3988c2ecf20Sopenharmony_ci */ 3998c2ecf20Sopenharmony_cistatic int cs_source_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo) 4008c2ecf20Sopenharmony_ci{ 4018c2ecf20Sopenharmony_ci static const char * const texts[] = { 4028c2ecf20Sopenharmony_ci "Coax", /* RXP0 */ 4038c2ecf20Sopenharmony_ci "Optical", /* RXP1 */ 4048c2ecf20Sopenharmony_ci "CD", /* RXP2 */ 4058c2ecf20Sopenharmony_ci }; 4068c2ecf20Sopenharmony_ci return snd_ctl_enum_info(uinfo, 1, 3, texts); 4078c2ecf20Sopenharmony_ci} 4088c2ecf20Sopenharmony_ci 4098c2ecf20Sopenharmony_cistatic int cs_source_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 4108c2ecf20Sopenharmony_ci{ 4118c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 4128c2ecf20Sopenharmony_ci 4138c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 4148c2ecf20Sopenharmony_ci ucontrol->value.enumerated.item[0] = ice->gpio.saved[0]; 4158c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 4168c2ecf20Sopenharmony_ci return 0; 4178c2ecf20Sopenharmony_ci} 4188c2ecf20Sopenharmony_ci 4198c2ecf20Sopenharmony_cistatic int cs_source_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 4208c2ecf20Sopenharmony_ci{ 4218c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 4228c2ecf20Sopenharmony_ci unsigned char val; 4238c2ecf20Sopenharmony_ci int change = 0; 4248c2ecf20Sopenharmony_ci 4258c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 4268c2ecf20Sopenharmony_ci if (ucontrol->value.enumerated.item[0] != ice->gpio.saved[0]) { 4278c2ecf20Sopenharmony_ci ice->gpio.saved[0] = ucontrol->value.enumerated.item[0] & 3; 4288c2ecf20Sopenharmony_ci val = 0x80 | (ice->gpio.saved[0] << 3); 4298c2ecf20Sopenharmony_ci spi_write(ice, CS_DEV, 0x04, val); 4308c2ecf20Sopenharmony_ci change = 1; 4318c2ecf20Sopenharmony_ci } 4328c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 4338c2ecf20Sopenharmony_ci return change; 4348c2ecf20Sopenharmony_ci} 4358c2ecf20Sopenharmony_ci 4368c2ecf20Sopenharmony_ci 4378c2ecf20Sopenharmony_ci/* 4388c2ecf20Sopenharmony_ci * GPIO controls 4398c2ecf20Sopenharmony_ci */ 4408c2ecf20Sopenharmony_cistatic int pontis_gpio_mask_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo) 4418c2ecf20Sopenharmony_ci{ 4428c2ecf20Sopenharmony_ci uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; 4438c2ecf20Sopenharmony_ci uinfo->count = 1; 4448c2ecf20Sopenharmony_ci uinfo->value.integer.min = 0; 4458c2ecf20Sopenharmony_ci uinfo->value.integer.max = 0xffff; /* 16bit */ 4468c2ecf20Sopenharmony_ci return 0; 4478c2ecf20Sopenharmony_ci} 4488c2ecf20Sopenharmony_ci 4498c2ecf20Sopenharmony_cistatic int pontis_gpio_mask_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 4508c2ecf20Sopenharmony_ci{ 4518c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 4528c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 4538c2ecf20Sopenharmony_ci /* 4-7 reserved */ 4548c2ecf20Sopenharmony_ci ucontrol->value.integer.value[0] = (~ice->gpio.write_mask & 0xffff) | 0x00f0; 4558c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 4568c2ecf20Sopenharmony_ci return 0; 4578c2ecf20Sopenharmony_ci} 4588c2ecf20Sopenharmony_ci 4598c2ecf20Sopenharmony_cistatic int pontis_gpio_mask_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 4608c2ecf20Sopenharmony_ci{ 4618c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 4628c2ecf20Sopenharmony_ci unsigned int val; 4638c2ecf20Sopenharmony_ci int changed; 4648c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 4658c2ecf20Sopenharmony_ci /* 4-7 reserved */ 4668c2ecf20Sopenharmony_ci val = (~ucontrol->value.integer.value[0] & 0xffff) | 0x00f0; 4678c2ecf20Sopenharmony_ci changed = val != ice->gpio.write_mask; 4688c2ecf20Sopenharmony_ci ice->gpio.write_mask = val; 4698c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 4708c2ecf20Sopenharmony_ci return changed; 4718c2ecf20Sopenharmony_ci} 4728c2ecf20Sopenharmony_ci 4738c2ecf20Sopenharmony_cistatic int pontis_gpio_dir_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 4748c2ecf20Sopenharmony_ci{ 4758c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 4768c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 4778c2ecf20Sopenharmony_ci /* 4-7 reserved */ 4788c2ecf20Sopenharmony_ci ucontrol->value.integer.value[0] = ice->gpio.direction & 0xff0f; 4798c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 4808c2ecf20Sopenharmony_ci return 0; 4818c2ecf20Sopenharmony_ci} 4828c2ecf20Sopenharmony_ci 4838c2ecf20Sopenharmony_cistatic int pontis_gpio_dir_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 4848c2ecf20Sopenharmony_ci{ 4858c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 4868c2ecf20Sopenharmony_ci unsigned int val; 4878c2ecf20Sopenharmony_ci int changed; 4888c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 4898c2ecf20Sopenharmony_ci /* 4-7 reserved */ 4908c2ecf20Sopenharmony_ci val = ucontrol->value.integer.value[0] & 0xff0f; 4918c2ecf20Sopenharmony_ci changed = (val != ice->gpio.direction); 4928c2ecf20Sopenharmony_ci ice->gpio.direction = val; 4938c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 4948c2ecf20Sopenharmony_ci return changed; 4958c2ecf20Sopenharmony_ci} 4968c2ecf20Sopenharmony_ci 4978c2ecf20Sopenharmony_cistatic int pontis_gpio_data_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 4988c2ecf20Sopenharmony_ci{ 4998c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 5008c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 5018c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_dir(ice, ice->gpio.direction); 5028c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_mask(ice, ice->gpio.write_mask); 5038c2ecf20Sopenharmony_ci ucontrol->value.integer.value[0] = snd_ice1712_gpio_read(ice) & 0xffff; 5048c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 5058c2ecf20Sopenharmony_ci return 0; 5068c2ecf20Sopenharmony_ci} 5078c2ecf20Sopenharmony_ci 5088c2ecf20Sopenharmony_cistatic int pontis_gpio_data_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol) 5098c2ecf20Sopenharmony_ci{ 5108c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol); 5118c2ecf20Sopenharmony_ci unsigned int val, nval; 5128c2ecf20Sopenharmony_ci int changed = 0; 5138c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 5148c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_dir(ice, ice->gpio.direction); 5158c2ecf20Sopenharmony_ci snd_ice1712_gpio_set_mask(ice, ice->gpio.write_mask); 5168c2ecf20Sopenharmony_ci val = snd_ice1712_gpio_read(ice) & 0xffff; 5178c2ecf20Sopenharmony_ci nval = ucontrol->value.integer.value[0] & 0xffff; 5188c2ecf20Sopenharmony_ci if (val != nval) { 5198c2ecf20Sopenharmony_ci snd_ice1712_gpio_write(ice, nval); 5208c2ecf20Sopenharmony_ci changed = 1; 5218c2ecf20Sopenharmony_ci } 5228c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 5238c2ecf20Sopenharmony_ci return changed; 5248c2ecf20Sopenharmony_ci} 5258c2ecf20Sopenharmony_ci 5268c2ecf20Sopenharmony_cistatic const DECLARE_TLV_DB_SCALE(db_scale_volume, -6400, 50, 1); 5278c2ecf20Sopenharmony_ci 5288c2ecf20Sopenharmony_ci/* 5298c2ecf20Sopenharmony_ci * mixers 5308c2ecf20Sopenharmony_ci */ 5318c2ecf20Sopenharmony_ci 5328c2ecf20Sopenharmony_cistatic const struct snd_kcontrol_new pontis_controls[] = { 5338c2ecf20Sopenharmony_ci { 5348c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 5358c2ecf20Sopenharmony_ci .access = (SNDRV_CTL_ELEM_ACCESS_READWRITE | 5368c2ecf20Sopenharmony_ci SNDRV_CTL_ELEM_ACCESS_TLV_READ), 5378c2ecf20Sopenharmony_ci .name = "PCM Playback Volume", 5388c2ecf20Sopenharmony_ci .info = wm_dac_vol_info, 5398c2ecf20Sopenharmony_ci .get = wm_dac_vol_get, 5408c2ecf20Sopenharmony_ci .put = wm_dac_vol_put, 5418c2ecf20Sopenharmony_ci .tlv = { .p = db_scale_volume }, 5428c2ecf20Sopenharmony_ci }, 5438c2ecf20Sopenharmony_ci { 5448c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 5458c2ecf20Sopenharmony_ci .access = (SNDRV_CTL_ELEM_ACCESS_READWRITE | 5468c2ecf20Sopenharmony_ci SNDRV_CTL_ELEM_ACCESS_TLV_READ), 5478c2ecf20Sopenharmony_ci .name = "Capture Volume", 5488c2ecf20Sopenharmony_ci .info = wm_adc_vol_info, 5498c2ecf20Sopenharmony_ci .get = wm_adc_vol_get, 5508c2ecf20Sopenharmony_ci .put = wm_adc_vol_put, 5518c2ecf20Sopenharmony_ci .tlv = { .p = db_scale_volume }, 5528c2ecf20Sopenharmony_ci }, 5538c2ecf20Sopenharmony_ci { 5548c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 5558c2ecf20Sopenharmony_ci .name = "CD Capture Switch", 5568c2ecf20Sopenharmony_ci .info = wm_adc_mux_info, 5578c2ecf20Sopenharmony_ci .get = wm_adc_mux_get, 5588c2ecf20Sopenharmony_ci .put = wm_adc_mux_put, 5598c2ecf20Sopenharmony_ci .private_value = 0, 5608c2ecf20Sopenharmony_ci }, 5618c2ecf20Sopenharmony_ci { 5628c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 5638c2ecf20Sopenharmony_ci .name = "Line Capture Switch", 5648c2ecf20Sopenharmony_ci .info = wm_adc_mux_info, 5658c2ecf20Sopenharmony_ci .get = wm_adc_mux_get, 5668c2ecf20Sopenharmony_ci .put = wm_adc_mux_put, 5678c2ecf20Sopenharmony_ci .private_value = 1, 5688c2ecf20Sopenharmony_ci }, 5698c2ecf20Sopenharmony_ci { 5708c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 5718c2ecf20Sopenharmony_ci .name = "Analog Bypass Switch", 5728c2ecf20Sopenharmony_ci .info = wm_bypass_info, 5738c2ecf20Sopenharmony_ci .get = wm_bypass_get, 5748c2ecf20Sopenharmony_ci .put = wm_bypass_put, 5758c2ecf20Sopenharmony_ci }, 5768c2ecf20Sopenharmony_ci { 5778c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 5788c2ecf20Sopenharmony_ci .name = "Swap Output Channels", 5798c2ecf20Sopenharmony_ci .info = wm_chswap_info, 5808c2ecf20Sopenharmony_ci .get = wm_chswap_get, 5818c2ecf20Sopenharmony_ci .put = wm_chswap_put, 5828c2ecf20Sopenharmony_ci }, 5838c2ecf20Sopenharmony_ci { 5848c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 5858c2ecf20Sopenharmony_ci .name = "IEC958 Input Source", 5868c2ecf20Sopenharmony_ci .info = cs_source_info, 5878c2ecf20Sopenharmony_ci .get = cs_source_get, 5888c2ecf20Sopenharmony_ci .put = cs_source_put, 5898c2ecf20Sopenharmony_ci }, 5908c2ecf20Sopenharmony_ci /* FIXME: which interface? */ 5918c2ecf20Sopenharmony_ci { 5928c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_CARD, 5938c2ecf20Sopenharmony_ci .name = "GPIO Mask", 5948c2ecf20Sopenharmony_ci .info = pontis_gpio_mask_info, 5958c2ecf20Sopenharmony_ci .get = pontis_gpio_mask_get, 5968c2ecf20Sopenharmony_ci .put = pontis_gpio_mask_put, 5978c2ecf20Sopenharmony_ci }, 5988c2ecf20Sopenharmony_ci { 5998c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_CARD, 6008c2ecf20Sopenharmony_ci .name = "GPIO Direction", 6018c2ecf20Sopenharmony_ci .info = pontis_gpio_mask_info, 6028c2ecf20Sopenharmony_ci .get = pontis_gpio_dir_get, 6038c2ecf20Sopenharmony_ci .put = pontis_gpio_dir_put, 6048c2ecf20Sopenharmony_ci }, 6058c2ecf20Sopenharmony_ci { 6068c2ecf20Sopenharmony_ci .iface = SNDRV_CTL_ELEM_IFACE_CARD, 6078c2ecf20Sopenharmony_ci .name = "GPIO Data", 6088c2ecf20Sopenharmony_ci .info = pontis_gpio_mask_info, 6098c2ecf20Sopenharmony_ci .get = pontis_gpio_data_get, 6108c2ecf20Sopenharmony_ci .put = pontis_gpio_data_put, 6118c2ecf20Sopenharmony_ci }, 6128c2ecf20Sopenharmony_ci}; 6138c2ecf20Sopenharmony_ci 6148c2ecf20Sopenharmony_ci 6158c2ecf20Sopenharmony_ci/* 6168c2ecf20Sopenharmony_ci * WM codec registers 6178c2ecf20Sopenharmony_ci */ 6188c2ecf20Sopenharmony_cistatic void wm_proc_regs_write(struct snd_info_entry *entry, struct snd_info_buffer *buffer) 6198c2ecf20Sopenharmony_ci{ 6208c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = entry->private_data; 6218c2ecf20Sopenharmony_ci char line[64]; 6228c2ecf20Sopenharmony_ci unsigned int reg, val; 6238c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 6248c2ecf20Sopenharmony_ci while (!snd_info_get_line(buffer, line, sizeof(line))) { 6258c2ecf20Sopenharmony_ci if (sscanf(line, "%x %x", ®, &val) != 2) 6268c2ecf20Sopenharmony_ci continue; 6278c2ecf20Sopenharmony_ci if (reg <= 0x17 && val <= 0xffff) 6288c2ecf20Sopenharmony_ci wm_put(ice, reg, val); 6298c2ecf20Sopenharmony_ci } 6308c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 6318c2ecf20Sopenharmony_ci} 6328c2ecf20Sopenharmony_ci 6338c2ecf20Sopenharmony_cistatic void wm_proc_regs_read(struct snd_info_entry *entry, struct snd_info_buffer *buffer) 6348c2ecf20Sopenharmony_ci{ 6358c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = entry->private_data; 6368c2ecf20Sopenharmony_ci int reg, val; 6378c2ecf20Sopenharmony_ci 6388c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 6398c2ecf20Sopenharmony_ci for (reg = 0; reg <= 0x17; reg++) { 6408c2ecf20Sopenharmony_ci val = wm_get(ice, reg); 6418c2ecf20Sopenharmony_ci snd_iprintf(buffer, "%02x = %04x\n", reg, val); 6428c2ecf20Sopenharmony_ci } 6438c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 6448c2ecf20Sopenharmony_ci} 6458c2ecf20Sopenharmony_ci 6468c2ecf20Sopenharmony_cistatic void wm_proc_init(struct snd_ice1712 *ice) 6478c2ecf20Sopenharmony_ci{ 6488c2ecf20Sopenharmony_ci snd_card_rw_proc_new(ice->card, "wm_codec", ice, wm_proc_regs_read, 6498c2ecf20Sopenharmony_ci wm_proc_regs_write); 6508c2ecf20Sopenharmony_ci} 6518c2ecf20Sopenharmony_ci 6528c2ecf20Sopenharmony_cistatic void cs_proc_regs_read(struct snd_info_entry *entry, struct snd_info_buffer *buffer) 6538c2ecf20Sopenharmony_ci{ 6548c2ecf20Sopenharmony_ci struct snd_ice1712 *ice = entry->private_data; 6558c2ecf20Sopenharmony_ci int reg, val; 6568c2ecf20Sopenharmony_ci 6578c2ecf20Sopenharmony_ci mutex_lock(&ice->gpio_mutex); 6588c2ecf20Sopenharmony_ci for (reg = 0; reg <= 0x26; reg++) { 6598c2ecf20Sopenharmony_ci val = spi_read(ice, CS_DEV, reg); 6608c2ecf20Sopenharmony_ci snd_iprintf(buffer, "%02x = %02x\n", reg, val); 6618c2ecf20Sopenharmony_ci } 6628c2ecf20Sopenharmony_ci val = spi_read(ice, CS_DEV, 0x7f); 6638c2ecf20Sopenharmony_ci snd_iprintf(buffer, "%02x = %02x\n", 0x7f, val); 6648c2ecf20Sopenharmony_ci mutex_unlock(&ice->gpio_mutex); 6658c2ecf20Sopenharmony_ci} 6668c2ecf20Sopenharmony_ci 6678c2ecf20Sopenharmony_cistatic void cs_proc_init(struct snd_ice1712 *ice) 6688c2ecf20Sopenharmony_ci{ 6698c2ecf20Sopenharmony_ci snd_card_ro_proc_new(ice->card, "cs_codec", ice, cs_proc_regs_read); 6708c2ecf20Sopenharmony_ci} 6718c2ecf20Sopenharmony_ci 6728c2ecf20Sopenharmony_ci 6738c2ecf20Sopenharmony_cistatic int pontis_add_controls(struct snd_ice1712 *ice) 6748c2ecf20Sopenharmony_ci{ 6758c2ecf20Sopenharmony_ci unsigned int i; 6768c2ecf20Sopenharmony_ci int err; 6778c2ecf20Sopenharmony_ci 6788c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(pontis_controls); i++) { 6798c2ecf20Sopenharmony_ci err = snd_ctl_add(ice->card, snd_ctl_new1(&pontis_controls[i], ice)); 6808c2ecf20Sopenharmony_ci if (err < 0) 6818c2ecf20Sopenharmony_ci return err; 6828c2ecf20Sopenharmony_ci } 6838c2ecf20Sopenharmony_ci 6848c2ecf20Sopenharmony_ci wm_proc_init(ice); 6858c2ecf20Sopenharmony_ci cs_proc_init(ice); 6868c2ecf20Sopenharmony_ci 6878c2ecf20Sopenharmony_ci return 0; 6888c2ecf20Sopenharmony_ci} 6898c2ecf20Sopenharmony_ci 6908c2ecf20Sopenharmony_ci 6918c2ecf20Sopenharmony_ci/* 6928c2ecf20Sopenharmony_ci * initialize the chip 6938c2ecf20Sopenharmony_ci */ 6948c2ecf20Sopenharmony_cistatic int pontis_init(struct snd_ice1712 *ice) 6958c2ecf20Sopenharmony_ci{ 6968c2ecf20Sopenharmony_ci static const unsigned short wm_inits[] = { 6978c2ecf20Sopenharmony_ci /* These come first to reduce init pop noise */ 6988c2ecf20Sopenharmony_ci WM_ADC_MUX, 0x00c0, /* ADC mute */ 6998c2ecf20Sopenharmony_ci WM_DAC_MUTE, 0x0001, /* DAC softmute */ 7008c2ecf20Sopenharmony_ci WM_DAC_CTRL1, 0x0000, /* DAC mute */ 7018c2ecf20Sopenharmony_ci 7028c2ecf20Sopenharmony_ci WM_POWERDOWN, 0x0008, /* All power-up except HP */ 7038c2ecf20Sopenharmony_ci WM_RESET, 0x0000, /* reset */ 7048c2ecf20Sopenharmony_ci }; 7058c2ecf20Sopenharmony_ci static const unsigned short wm_inits2[] = { 7068c2ecf20Sopenharmony_ci WM_MASTER_CTRL, 0x0022, /* 256fs, slave mode */ 7078c2ecf20Sopenharmony_ci WM_DAC_INT, 0x0022, /* I2S, normal polarity, 24bit */ 7088c2ecf20Sopenharmony_ci WM_ADC_INT, 0x0022, /* I2S, normal polarity, 24bit */ 7098c2ecf20Sopenharmony_ci WM_DAC_CTRL1, 0x0090, /* DAC L/R */ 7108c2ecf20Sopenharmony_ci WM_OUT_MUX, 0x0001, /* OUT DAC */ 7118c2ecf20Sopenharmony_ci WM_HP_ATTEN_L, 0x0179, /* HP 0dB */ 7128c2ecf20Sopenharmony_ci WM_HP_ATTEN_R, 0x0179, /* HP 0dB */ 7138c2ecf20Sopenharmony_ci WM_DAC_ATTEN_L, 0x0000, /* DAC 0dB */ 7148c2ecf20Sopenharmony_ci WM_DAC_ATTEN_L, 0x0100, /* DAC 0dB */ 7158c2ecf20Sopenharmony_ci WM_DAC_ATTEN_R, 0x0000, /* DAC 0dB */ 7168c2ecf20Sopenharmony_ci WM_DAC_ATTEN_R, 0x0100, /* DAC 0dB */ 7178c2ecf20Sopenharmony_ci /* WM_DAC_MASTER, 0x0100, */ /* DAC master muted */ 7188c2ecf20Sopenharmony_ci WM_PHASE_SWAP, 0x0000, /* phase normal */ 7198c2ecf20Sopenharmony_ci WM_DAC_CTRL2, 0x0000, /* no deemphasis, no ZFLG */ 7208c2ecf20Sopenharmony_ci WM_ADC_ATTEN_L, 0x0000, /* ADC muted */ 7218c2ecf20Sopenharmony_ci WM_ADC_ATTEN_R, 0x0000, /* ADC muted */ 7228c2ecf20Sopenharmony_ci#if 0 7238c2ecf20Sopenharmony_ci WM_ALC_CTRL1, 0x007b, /* */ 7248c2ecf20Sopenharmony_ci WM_ALC_CTRL2, 0x0000, /* */ 7258c2ecf20Sopenharmony_ci WM_ALC_CTRL3, 0x0000, /* */ 7268c2ecf20Sopenharmony_ci WM_NOISE_GATE, 0x0000, /* */ 7278c2ecf20Sopenharmony_ci#endif 7288c2ecf20Sopenharmony_ci WM_DAC_MUTE, 0x0000, /* DAC unmute */ 7298c2ecf20Sopenharmony_ci WM_ADC_MUX, 0x0003, /* ADC unmute, both CD/Line On */ 7308c2ecf20Sopenharmony_ci }; 7318c2ecf20Sopenharmony_ci static const unsigned char cs_inits[] = { 7328c2ecf20Sopenharmony_ci 0x04, 0x80, /* RUN, RXP0 */ 7338c2ecf20Sopenharmony_ci 0x05, 0x05, /* slave, 24bit */ 7348c2ecf20Sopenharmony_ci 0x01, 0x00, 7358c2ecf20Sopenharmony_ci 0x02, 0x00, 7368c2ecf20Sopenharmony_ci 0x03, 0x00, 7378c2ecf20Sopenharmony_ci }; 7388c2ecf20Sopenharmony_ci unsigned int i; 7398c2ecf20Sopenharmony_ci 7408c2ecf20Sopenharmony_ci ice->vt1720 = 1; 7418c2ecf20Sopenharmony_ci ice->num_total_dacs = 2; 7428c2ecf20Sopenharmony_ci ice->num_total_adcs = 2; 7438c2ecf20Sopenharmony_ci 7448c2ecf20Sopenharmony_ci /* to remember the register values */ 7458c2ecf20Sopenharmony_ci ice->akm = kzalloc(sizeof(struct snd_akm4xxx), GFP_KERNEL); 7468c2ecf20Sopenharmony_ci if (! ice->akm) 7478c2ecf20Sopenharmony_ci return -ENOMEM; 7488c2ecf20Sopenharmony_ci ice->akm_codecs = 1; 7498c2ecf20Sopenharmony_ci 7508c2ecf20Sopenharmony_ci /* HACK - use this as the SPDIF source. 7518c2ecf20Sopenharmony_ci * don't call snd_ice1712_gpio_get/put(), otherwise it's overwritten 7528c2ecf20Sopenharmony_ci */ 7538c2ecf20Sopenharmony_ci ice->gpio.saved[0] = 0; 7548c2ecf20Sopenharmony_ci 7558c2ecf20Sopenharmony_ci /* initialize WM8776 codec */ 7568c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(wm_inits); i += 2) 7578c2ecf20Sopenharmony_ci wm_put(ice, wm_inits[i], wm_inits[i+1]); 7588c2ecf20Sopenharmony_ci schedule_timeout_uninterruptible(1); 7598c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(wm_inits2); i += 2) 7608c2ecf20Sopenharmony_ci wm_put(ice, wm_inits2[i], wm_inits2[i+1]); 7618c2ecf20Sopenharmony_ci 7628c2ecf20Sopenharmony_ci /* initialize CS8416 codec */ 7638c2ecf20Sopenharmony_ci /* assert PRST#; MT05 bit 7 */ 7648c2ecf20Sopenharmony_ci outb(inb(ICEMT1724(ice, AC97_CMD)) | 0x80, ICEMT1724(ice, AC97_CMD)); 7658c2ecf20Sopenharmony_ci mdelay(5); 7668c2ecf20Sopenharmony_ci /* deassert PRST# */ 7678c2ecf20Sopenharmony_ci outb(inb(ICEMT1724(ice, AC97_CMD)) & ~0x80, ICEMT1724(ice, AC97_CMD)); 7688c2ecf20Sopenharmony_ci 7698c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(cs_inits); i += 2) 7708c2ecf20Sopenharmony_ci spi_write(ice, CS_DEV, cs_inits[i], cs_inits[i+1]); 7718c2ecf20Sopenharmony_ci 7728c2ecf20Sopenharmony_ci return 0; 7738c2ecf20Sopenharmony_ci} 7748c2ecf20Sopenharmony_ci 7758c2ecf20Sopenharmony_ci 7768c2ecf20Sopenharmony_ci/* 7778c2ecf20Sopenharmony_ci * Pontis boards don't provide the EEPROM data at all. 7788c2ecf20Sopenharmony_ci * hence the driver needs to sets up it properly. 7798c2ecf20Sopenharmony_ci */ 7808c2ecf20Sopenharmony_ci 7818c2ecf20Sopenharmony_cistatic const unsigned char pontis_eeprom[] = { 7828c2ecf20Sopenharmony_ci [ICE_EEP2_SYSCONF] = 0x08, /* clock 256, mpu401, spdif-in/ADC, 1DAC */ 7838c2ecf20Sopenharmony_ci [ICE_EEP2_ACLINK] = 0x80, /* I2S */ 7848c2ecf20Sopenharmony_ci [ICE_EEP2_I2S] = 0xf8, /* vol, 96k, 24bit, 192k */ 7858c2ecf20Sopenharmony_ci [ICE_EEP2_SPDIF] = 0xc3, /* out-en, out-int, spdif-in */ 7868c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_DIR] = 0x07, 7878c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_DIR1] = 0x00, 7888c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_DIR2] = 0x00, /* ignored */ 7898c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_MASK] = 0x0f, /* 4-7 reserved for CS8416 */ 7908c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_MASK1] = 0xff, 7918c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_MASK2] = 0x00, /* ignored */ 7928c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_STATE] = 0x06, /* 0-low, 1-high, 2-high */ 7938c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_STATE1] = 0x00, 7948c2ecf20Sopenharmony_ci [ICE_EEP2_GPIO_STATE2] = 0x00, /* ignored */ 7958c2ecf20Sopenharmony_ci}; 7968c2ecf20Sopenharmony_ci 7978c2ecf20Sopenharmony_ci/* entry point */ 7988c2ecf20Sopenharmony_cistruct snd_ice1712_card_info snd_vt1720_pontis_cards[] = { 7998c2ecf20Sopenharmony_ci { 8008c2ecf20Sopenharmony_ci .subvendor = VT1720_SUBDEVICE_PONTIS_MS300, 8018c2ecf20Sopenharmony_ci .name = "Pontis MS300", 8028c2ecf20Sopenharmony_ci .model = "ms300", 8038c2ecf20Sopenharmony_ci .chip_init = pontis_init, 8048c2ecf20Sopenharmony_ci .build_controls = pontis_add_controls, 8058c2ecf20Sopenharmony_ci .eeprom_size = sizeof(pontis_eeprom), 8068c2ecf20Sopenharmony_ci .eeprom_data = pontis_eeprom, 8078c2ecf20Sopenharmony_ci }, 8088c2ecf20Sopenharmony_ci { } /* terminator */ 8098c2ecf20Sopenharmony_ci}; 810