soc-dapm.h 27.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
/*
 * linux/sound/soc-dapm.h -- ALSA SoC Dynamic Audio Power Management
 *
 * Author:		Liam Girdwood
 * Created:		Aug 11th 2005
 * Copyright:	Wolfson Microelectronics. PLC.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */

#ifndef __LINUX_SND_SOC_DAPM_H
#define __LINUX_SND_SOC_DAPM_H

#include <linux/types.h>
#include <sound/control.h>
18
#include <sound/soc-topology.h>
19
#include <sound/asoc.h>
20

21 22
struct device;

23 24 25 26
/* widget has no PM register bit */
#define SND_SOC_NOPM	-1

/*
27
 * SoC dynamic audio power management
28
 *
L
Lucas De Marchi 已提交
29
 * We can have up to 4 power domains
L
Lu Guanqun 已提交
30
 *  1. Codec domain - VREF, VMID
31 32 33 34 35 36 37 38 39 40 41 42 43 44
 *     Usually controlled at codec probe/remove, although can be set
 *     at stream time if power is not needed for sidetone, etc.
 *  2. Platform/Machine domain - physically connected inputs and outputs
 *     Is platform/machine and user action specific, is set in the machine
 *     driver and by userspace e.g when HP are inserted
 *  3. Path domain - Internal codec path mixers
 *     Are automatically set when mixer and mux settings are
 *     changed by the user.
 *  4. Stream domain - DAC's and ADC's.
 *     Enabled when stream playback/capture is started.
 */

/* codec domain */
#define SND_SOC_DAPM_VMID(wname) \
45
{	.id = snd_soc_dapm_vmid, .name = wname, .kcontrol_news = NULL, \
46 47 48
	.num_kcontrols = 0}

/* platform domain */
49 50 51
#define SND_SOC_DAPM_SIGGEN(wname) \
{	.id = snd_soc_dapm_siggen, .name = wname, .kcontrol_news = NULL, \
	.num_kcontrols = 0, .reg = SND_SOC_NOPM }
52
#define SND_SOC_DAPM_INPUT(wname) \
53
{	.id = snd_soc_dapm_input, .name = wname, .kcontrol_news = NULL, \
54
	.num_kcontrols = 0, .reg = SND_SOC_NOPM }
55
#define SND_SOC_DAPM_OUTPUT(wname) \
56
{	.id = snd_soc_dapm_output, .name = wname, .kcontrol_news = NULL, \
57
	.num_kcontrols = 0, .reg = SND_SOC_NOPM }
58
#define SND_SOC_DAPM_MIC(wname, wevent) \
59
{	.id = snd_soc_dapm_mic, .name = wname, .kcontrol_news = NULL, \
60
	.num_kcontrols = 0, .reg = SND_SOC_NOPM, .event = wevent, \
61 62
	.event_flags = SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD}
#define SND_SOC_DAPM_HP(wname, wevent) \
63
{	.id = snd_soc_dapm_hp, .name = wname, .kcontrol_news = NULL, \
64
	.num_kcontrols = 0, .reg = SND_SOC_NOPM, .event = wevent, \
65 66
	.event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD}
#define SND_SOC_DAPM_SPK(wname, wevent) \
67
{	.id = snd_soc_dapm_spk, .name = wname, .kcontrol_news = NULL, \
68
	.num_kcontrols = 0, .reg = SND_SOC_NOPM, .event = wevent, \
69 70
	.event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD}
#define SND_SOC_DAPM_LINE(wname, wevent) \
71
{	.id = snd_soc_dapm_line, .name = wname, .kcontrol_news = NULL, \
72
	.num_kcontrols = 0, .reg = SND_SOC_NOPM, .event = wevent, \
73 74
	.event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD}

75 76 77 78
#define SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert) \
	.reg = wreg, .mask = 1, .shift = wshift, \
	.on_val = winvert ? 0 : 1, .off_val = winvert ? 1 : 0

79 80 81
/* path domain */
#define SND_SOC_DAPM_PGA(wname, wreg, wshift, winvert,\
	 wcontrols, wncontrols) \
82 83 84
{	.id = snd_soc_dapm_pga, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = wncontrols}
85 86
#define SND_SOC_DAPM_OUT_DRV(wname, wreg, wshift, winvert,\
	 wcontrols, wncontrols) \
87 88 89
{	.id = snd_soc_dapm_out_drv, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = wncontrols}
90 91
#define SND_SOC_DAPM_MIXER(wname, wreg, wshift, winvert, \
	 wcontrols, wncontrols)\
92 93 94
{	.id = snd_soc_dapm_mixer, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = wncontrols}
95 96
#define SND_SOC_DAPM_MIXER_NAMED_CTL(wname, wreg, wshift, winvert, \
	 wcontrols, wncontrols)\
97 98 99
{       .id = snd_soc_dapm_mixer_named_ctl, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = wncontrols}
100
#define SND_SOC_DAPM_MICBIAS(wname, wreg, wshift, winvert) \
101 102 103
{	.id = snd_soc_dapm_micbias, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = NULL, .num_kcontrols = 0}
104
#define SND_SOC_DAPM_SWITCH(wname, wreg, wshift, winvert, wcontrols) \
105 106 107
{	.id = snd_soc_dapm_switch, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = 1}
108
#define SND_SOC_DAPM_MUX(wname, wreg, wshift, winvert, wcontrols) \
109 110
{	.id = snd_soc_dapm_mux, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
111
	.kcontrol_news = wcontrols, .num_kcontrols = 1}
112 113 114 115
#define SND_SOC_DAPM_DEMUX(wname, wreg, wshift, winvert, wcontrols) \
{	.id = snd_soc_dapm_demux, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = 1}
116

117 118 119
/* Simplified versions of above macros, assuming wncontrols = ARRAY_SIZE(wcontrols) */
#define SOC_PGA_ARRAY(wname, wreg, wshift, winvert,\
	 wcontrols) \
120 121 122
{	.id = snd_soc_dapm_pga, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = ARRAY_SIZE(wcontrols)}
123 124
#define SOC_MIXER_ARRAY(wname, wreg, wshift, winvert, \
	 wcontrols)\
125 126 127
{	.id = snd_soc_dapm_mixer, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = ARRAY_SIZE(wcontrols)}
128 129
#define SOC_MIXER_NAMED_CTL_ARRAY(wname, wreg, wshift, winvert, \
	 wcontrols)\
130 131 132
{       .id = snd_soc_dapm_mixer_named_ctl, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = ARRAY_SIZE(wcontrols)}
133

134 135 136
/* path domain with event - event handler must return 0 for success */
#define SND_SOC_DAPM_PGA_E(wname, wreg, wshift, winvert, wcontrols, \
	wncontrols, wevent, wflags) \
137 138 139
{	.id = snd_soc_dapm_pga, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = wncontrols, \
140
	.event = wevent, .event_flags = wflags}
141 142
#define SND_SOC_DAPM_OUT_DRV_E(wname, wreg, wshift, winvert, wcontrols, \
	wncontrols, wevent, wflags) \
143 144 145
{	.id = snd_soc_dapm_out_drv, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = wncontrols, \
146
	.event = wevent, .event_flags = wflags}
147 148
#define SND_SOC_DAPM_MIXER_E(wname, wreg, wshift, winvert, wcontrols, \
	wncontrols, wevent, wflags) \
149 150 151
{	.id = snd_soc_dapm_mixer, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = wncontrols, \
152
	.event = wevent, .event_flags = wflags}
153 154
#define SND_SOC_DAPM_MIXER_NAMED_CTL_E(wname, wreg, wshift, winvert, \
	wcontrols, wncontrols, wevent, wflags) \
155 156 157
{       .id = snd_soc_dapm_mixer, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, \
158
	.num_kcontrols = wncontrols, .event = wevent, .event_flags = wflags}
159 160
#define SND_SOC_DAPM_SWITCH_E(wname, wreg, wshift, winvert, wcontrols, \
	wevent, wflags) \
161 162 163
{	.id = snd_soc_dapm_switch, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = 1, \
164 165 166
	.event = wevent, .event_flags = wflags}
#define SND_SOC_DAPM_MUX_E(wname, wreg, wshift, winvert, wcontrols, \
	wevent, wflags) \
167 168 169
{	.id = snd_soc_dapm_mux, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = 1, \
170 171
	.event = wevent, .event_flags = wflags}

172
/* additional sequencing control within an event type */
173 174
#define SND_SOC_DAPM_PGA_S(wname, wsubseq, wreg, wshift, winvert, \
	wevent, wflags) \
175 176 177
{	.id = snd_soc_dapm_pga, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.event = wevent, .event_flags = wflags, \
178
	.subseq = wsubseq}
179 180
#define SND_SOC_DAPM_SUPPLY_S(wname, wsubseq, wreg, wshift, winvert, wevent, \
	wflags)	\
181 182 183
{	.id = snd_soc_dapm_supply, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.event = wevent, .event_flags = wflags, .subseq = wsubseq}
184

185 186 187
/* Simplified versions of above macros, assuming wncontrols = ARRAY_SIZE(wcontrols) */
#define SOC_PGA_E_ARRAY(wname, wreg, wshift, winvert, wcontrols, \
	wevent, wflags) \
188 189 190
{	.id = snd_soc_dapm_pga, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = ARRAY_SIZE(wcontrols), \
191 192 193
	.event = wevent, .event_flags = wflags}
#define SOC_MIXER_E_ARRAY(wname, wreg, wshift, winvert, wcontrols, \
	wevent, wflags) \
194 195 196
{	.id = snd_soc_dapm_mixer, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = ARRAY_SIZE(wcontrols), \
197 198 199
	.event = wevent, .event_flags = wflags}
#define SOC_MIXER_NAMED_CTL_E_ARRAY(wname, wreg, wshift, winvert, \
	wcontrols, wevent, wflags) \
200 201 202 203
{       .id = snd_soc_dapm_mixer, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.kcontrol_news = wcontrols, .num_kcontrols = ARRAY_SIZE(wcontrols), \
	.event = wevent, .event_flags = wflags}
204

205 206
/* events that are pre and post DAPM */
#define SND_SOC_DAPM_PRE(wname, wevent) \
207
{	.id = snd_soc_dapm_pre, .name = wname, .kcontrol_news = NULL, \
208
	.num_kcontrols = 0, .reg = SND_SOC_NOPM, .event = wevent, \
209 210
	.event_flags = SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_PRE_PMD}
#define SND_SOC_DAPM_POST(wname, wevent) \
211
{	.id = snd_soc_dapm_post, .name = wname, .kcontrol_news = NULL, \
212
	.num_kcontrols = 0, .reg = SND_SOC_NOPM, .event = wevent, \
213 214 215
	.event_flags = SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_POST_PMD}

/* stream domain */
216 217
#define SND_SOC_DAPM_AIF_IN(wname, stname, wslot, wreg, wshift, winvert) \
{	.id = snd_soc_dapm_aif_in, .name = wname, .sname = stname, \
218
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), }
219 220 221
#define SND_SOC_DAPM_AIF_IN_E(wname, stname, wslot, wreg, wshift, winvert, \
			      wevent, wflags)				\
{	.id = snd_soc_dapm_aif_in, .name = wname, .sname = stname, \
222
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
223
	.event = wevent, .event_flags = wflags }
224 225
#define SND_SOC_DAPM_AIF_OUT(wname, stname, wslot, wreg, wshift, winvert) \
{	.id = snd_soc_dapm_aif_out, .name = wname, .sname = stname, \
226
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), }
227 228 229
#define SND_SOC_DAPM_AIF_OUT_E(wname, stname, wslot, wreg, wshift, winvert, \
			     wevent, wflags)				\
{	.id = snd_soc_dapm_aif_out, .name = wname, .sname = stname, \
230
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
231
	.event = wevent, .event_flags = wflags }
232
#define SND_SOC_DAPM_DAC(wname, stname, wreg, wshift, winvert) \
233 234
{	.id = snd_soc_dapm_dac, .name = wname, .sname = stname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert) }
235 236
#define SND_SOC_DAPM_DAC_E(wname, stname, wreg, wshift, winvert, \
			   wevent, wflags)				\
237 238
{	.id = snd_soc_dapm_dac, .name = wname, .sname = stname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
239
	.event = wevent, .event_flags = wflags}
240

241
#define SND_SOC_DAPM_ADC(wname, stname, wreg, wshift, winvert) \
242 243
{	.id = snd_soc_dapm_adc, .name = wname, .sname = stname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), }
244 245
#define SND_SOC_DAPM_ADC_E(wname, stname, wreg, wshift, winvert, \
			   wevent, wflags)				\
246 247
{	.id = snd_soc_dapm_adc, .name = wname, .sname = stname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
248
	.event = wevent, .event_flags = wflags}
249 250 251 252
#define SND_SOC_DAPM_CLOCK_SUPPLY(wname) \
{	.id = snd_soc_dapm_clock_supply, .name = wname, \
	.reg = SND_SOC_NOPM, .event = dapm_clock_event, \
	.event_flags = SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD }
253

254
/* generic widgets */
255
#define SND_SOC_DAPM_REG(wid, wname, wreg, wshift, wmask, won_val, woff_val) \
256
{	.id = wid, .name = wname, .kcontrol_news = NULL, .num_kcontrols = 0, \
257 258
	.reg = wreg, .shift = wshift, .mask = wmask, \
	.on_val = won_val, .off_val = woff_val, }
259
#define SND_SOC_DAPM_SUPPLY(wname, wreg, wshift, winvert, wevent, wflags) \
260 261 262
{	.id = snd_soc_dapm_supply, .name = wname, \
	SND_SOC_DAPM_INIT_REG_VAL(wreg, wshift, winvert), \
	.event = wevent, .event_flags = wflags}
263
#define SND_SOC_DAPM_REGULATOR_SUPPLY(wname, wdelay, wflags)	    \
264 265
{	.id = snd_soc_dapm_regulator_supply, .name = wname, \
	.reg = SND_SOC_NOPM, .shift = wdelay, .event = dapm_regulator_event, \
266
	.event_flags = SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD, \
267
	.on_val = wflags}
268

269

270
/* dapm kcontrol types */
P
Philipp Zabel 已提交
271
#define SOC_DAPM_SINGLE(xname, reg, shift, max, invert) \
272 273 274
{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
	.info = snd_soc_info_volsw, \
	.get = snd_soc_dapm_get_volsw, .put = snd_soc_dapm_put_volsw, \
275 276 277 278 279 280
	.private_value = SOC_SINGLE_VALUE(reg, shift, max, invert, 0) }
#define SOC_DAPM_SINGLE_AUTODISABLE(xname, reg, shift, max, invert) \
{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
	.info = snd_soc_info_volsw, \
	.get = snd_soc_dapm_get_volsw, .put = snd_soc_dapm_put_volsw, \
	.private_value = SOC_SINGLE_VALUE(reg, shift, max, invert, 1) }
281 282
#define SOC_DAPM_SINGLE_VIRT(xname, max) \
	SOC_DAPM_SINGLE(xname, SND_SOC_NOPM, 0, max, 0)
P
Philipp Zabel 已提交
283 284 285 286 287 288
#define SOC_DAPM_SINGLE_TLV(xname, reg, shift, max, invert, tlv_array) \
{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
	.info = snd_soc_info_volsw, \
	.access = SNDRV_CTL_ELEM_ACCESS_TLV_READ | SNDRV_CTL_ELEM_ACCESS_READWRITE,\
	.tlv.p = (tlv_array), \
	.get = snd_soc_dapm_get_volsw, .put = snd_soc_dapm_put_volsw, \
289 290 291 292 293 294 295
	.private_value = SOC_SINGLE_VALUE(reg, shift, max, invert, 0) }
#define SOC_DAPM_SINGLE_TLV_AUTODISABLE(xname, reg, shift, max, invert, tlv_array) \
{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
	.info = snd_soc_info_volsw, \
	.access = SNDRV_CTL_ELEM_ACCESS_TLV_READ | SNDRV_CTL_ELEM_ACCESS_READWRITE,\
	.tlv.p = (tlv_array), \
	.get = snd_soc_dapm_get_volsw, .put = snd_soc_dapm_put_volsw, \
296
	.private_value = SOC_SINGLE_VALUE(reg, shift, max, invert, 1) }
297 298
#define SOC_DAPM_SINGLE_TLV_VIRT(xname, max, tlv_array) \
	SOC_DAPM_SINGLE(xname, SND_SOC_NOPM, 0, max, 0, tlv_array)
299 300 301 302 303 304
#define SOC_DAPM_ENUM(xname, xenum) \
{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
	.info = snd_soc_info_enum_double, \
 	.get = snd_soc_dapm_get_enum_double, \
 	.put = snd_soc_dapm_put_enum_double, \
  	.private_value = (unsigned long)&xenum }
305 306 307 308 309 310
#define SOC_DAPM_ENUM_EXT(xname, xenum, xget, xput) \
{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname, \
	.info = snd_soc_info_enum_double, \
	.get = xget, \
	.put = xput, \
	.private_value = (unsigned long)&xenum }
311 312 313 314 315 316
#define SOC_DAPM_PIN_SWITCH(xname) \
{	.iface = SNDRV_CTL_ELEM_IFACE_MIXER, .name = xname " Switch", \
	.info = snd_soc_dapm_info_pin_switch, \
	.get = snd_soc_dapm_get_pin_switch, \
	.put = snd_soc_dapm_put_pin_switch, \
	.private_value = (unsigned long)xname }
317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333

/* dapm stream operations */
#define SND_SOC_DAPM_STREAM_NOP			0x0
#define SND_SOC_DAPM_STREAM_START		0x1
#define SND_SOC_DAPM_STREAM_STOP		0x2
#define SND_SOC_DAPM_STREAM_SUSPEND		0x4
#define SND_SOC_DAPM_STREAM_RESUME		0x8
#define SND_SOC_DAPM_STREAM_PAUSE_PUSH	0x10
#define SND_SOC_DAPM_STREAM_PAUSE_RELEASE	0x20

/* dapm event types */
#define SND_SOC_DAPM_PRE_PMU	0x1 	/* before widget power up */
#define SND_SOC_DAPM_POST_PMU	0x2		/* after widget power up */
#define SND_SOC_DAPM_PRE_PMD	0x4 	/* before widget power down */
#define SND_SOC_DAPM_POST_PMD	0x8		/* after widget power down */
#define SND_SOC_DAPM_PRE_REG	0x10	/* before audio path setup */
#define SND_SOC_DAPM_POST_REG	0x20	/* after audio path setup */
334 335
#define SND_SOC_DAPM_WILL_PMU   0x40    /* called at start of sequence */
#define SND_SOC_DAPM_WILL_PMD   0x80    /* called at start of sequence */
336 337
#define SND_SOC_DAPM_PRE_POST_PMD \
				(SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD)
338 339 340 341 342 343 344

/* convenience event type detection */
#define SND_SOC_DAPM_EVENT_ON(e)	\
	(e & (SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU))
#define SND_SOC_DAPM_EVENT_OFF(e)	\
	(e & (SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD))

345 346 347
/* regulator widget flags */
#define SND_SOC_DAPM_REGULATOR_BYPASS     0x1     /* bypass when disabled */

348 349 350 351
struct snd_soc_dapm_widget;
enum snd_soc_dapm_type;
struct snd_soc_dapm_path;
struct snd_soc_dapm_pin;
352
struct snd_soc_dapm_route;
L
Liam Girdwood 已提交
353
struct snd_soc_dapm_context;
354
struct regulator;
355
struct snd_soc_dapm_widget_list;
356
struct snd_soc_dapm_update;
357

358 359
int dapm_regulator_event(struct snd_soc_dapm_widget *w,
			 struct snd_kcontrol *kcontrol, int event);
360 361
int dapm_clock_event(struct snd_soc_dapm_widget *w,
			 struct snd_kcontrol *kcontrol, int event);
362

363 364 365 366 367 368 369 370 371
/* dapm controls */
int snd_soc_dapm_put_volsw(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *ucontrol);
int snd_soc_dapm_get_volsw(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *ucontrol);
int snd_soc_dapm_get_enum_double(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *ucontrol);
int snd_soc_dapm_put_enum_double(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *ucontrol);
372 373 374 375 376 377
int snd_soc_dapm_info_pin_switch(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_info *uinfo);
int snd_soc_dapm_get_pin_switch(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *uncontrol);
int snd_soc_dapm_put_pin_switch(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *uncontrol);
L
Liam Girdwood 已提交
378
int snd_soc_dapm_new_controls(struct snd_soc_dapm_context *dapm,
379 380
	const struct snd_soc_dapm_widget *widget,
	int num);
381 382 383
int snd_soc_dapm_new_dai_widgets(struct snd_soc_dapm_context *dapm,
				 struct snd_soc_dai *dai);
int snd_soc_dapm_link_dai_widgets(struct snd_soc_card *card);
384
void snd_soc_dapm_connect_dai_link_widgets(struct snd_soc_card *card);
385 386
int snd_soc_dapm_new_pcm(struct snd_soc_card *card,
			 const struct snd_soc_pcm_stream *params,
387
			 unsigned int num_params,
388 389
			 struct snd_soc_dapm_widget *source,
			 struct snd_soc_dapm_widget *sink);
390 391

/* dapm path setup */
392
int snd_soc_dapm_new_widgets(struct snd_soc_card *card);
L
Liam Girdwood 已提交
393 394
void snd_soc_dapm_free(struct snd_soc_dapm_context *dapm);
int snd_soc_dapm_add_routes(struct snd_soc_dapm_context *dapm,
395
			    const struct snd_soc_dapm_route *route, int num);
396 397
int snd_soc_dapm_del_routes(struct snd_soc_dapm_context *dapm,
			    const struct snd_soc_dapm_route *route, int num);
398 399
int snd_soc_dapm_weak_routes(struct snd_soc_dapm_context *dapm,
			     const struct snd_soc_dapm_route *route, int num);
400
void snd_soc_dapm_free_widget(struct snd_soc_dapm_widget *w);
401 402

/* dapm events */
403 404
void snd_soc_dapm_stream_event(struct snd_soc_pcm_runtime *rtd, int stream,
	int event);
405
void snd_soc_dapm_shutdown(struct snd_soc_card *card);
406

407
/* external DAPM widget events */
408
int snd_soc_dapm_mixer_update_power(struct snd_soc_dapm_context *dapm,
409 410
		struct snd_kcontrol *kcontrol, int connect,
		struct snd_soc_dapm_update *update);
411
int snd_soc_dapm_mux_update_power(struct snd_soc_dapm_context *dapm,
412 413
		struct snd_kcontrol *kcontrol, int mux, struct soc_enum *e,
		struct snd_soc_dapm_update *update);
414

415
/* dapm sys fs - used by the core */
416
extern struct attribute *soc_dapm_dev_attrs[];
417 418
void snd_soc_dapm_debugfs_init(struct snd_soc_dapm_context *dapm,
				struct dentry *parent);
419

420
/* dapm audio pin control and status */
L
Liam Girdwood 已提交
421 422
int snd_soc_dapm_enable_pin(struct snd_soc_dapm_context *dapm,
			    const char *pin);
423 424
int snd_soc_dapm_enable_pin_unlocked(struct snd_soc_dapm_context *dapm,
				     const char *pin);
L
Liam Girdwood 已提交
425 426
int snd_soc_dapm_disable_pin(struct snd_soc_dapm_context *dapm,
			     const char *pin);
427 428
int snd_soc_dapm_disable_pin_unlocked(struct snd_soc_dapm_context *dapm,
				      const char *pin);
L
Liam Girdwood 已提交
429
int snd_soc_dapm_nc_pin(struct snd_soc_dapm_context *dapm, const char *pin);
430 431
int snd_soc_dapm_nc_pin_unlocked(struct snd_soc_dapm_context *dapm,
				 const char *pin);
L
Liam Girdwood 已提交
432 433 434
int snd_soc_dapm_get_pin_status(struct snd_soc_dapm_context *dapm,
				const char *pin);
int snd_soc_dapm_sync(struct snd_soc_dapm_context *dapm);
435
int snd_soc_dapm_sync_unlocked(struct snd_soc_dapm_context *dapm);
L
Liam Girdwood 已提交
436
int snd_soc_dapm_force_enable_pin(struct snd_soc_dapm_context *dapm,
437
				  const char *pin);
438 439
int snd_soc_dapm_force_enable_pin_unlocked(struct snd_soc_dapm_context *dapm,
					   const char *pin);
L
Liam Girdwood 已提交
440 441
int snd_soc_dapm_ignore_suspend(struct snd_soc_dapm_context *dapm,
				const char *pin);
442
unsigned int dapm_kcontrol_get_value(const struct snd_kcontrol *kcontrol);
443

444
/* Mostly internal - should not normally be used */
445
void dapm_mark_endpoints_dirty(struct snd_soc_card *card);
446

447 448 449 450
/* dapm path query */
int snd_soc_dapm_dai_get_connected_widgets(struct snd_soc_dai *dai, int stream,
	struct snd_soc_dapm_widget_list **list);

451 452
struct snd_soc_dapm_context *snd_soc_dapm_kcontrol_dapm(
	struct snd_kcontrol *kcontrol);
453

454 455 456
int snd_soc_dapm_force_bias_level(struct snd_soc_dapm_context *dapm,
	enum snd_soc_bias_level level);

457 458 459 460 461
/* dapm widget types */
enum snd_soc_dapm_type {
	snd_soc_dapm_input = 0,		/* input pin */
	snd_soc_dapm_output,		/* output pin */
	snd_soc_dapm_mux,			/* selects 1 analog signal from many inputs */
462
	snd_soc_dapm_demux,			/* connects the input to one of multiple outputs */
463
	snd_soc_dapm_mixer,			/* mixes several analog signals together */
464
	snd_soc_dapm_mixer_named_ctl,		/* mixer with named controls */
465
	snd_soc_dapm_pga,			/* programmable gain/attenuation (volume) */
466
	snd_soc_dapm_out_drv,			/* output driver */
467 468 469 470 471 472 473 474 475 476 477
	snd_soc_dapm_adc,			/* analog to digital converter */
	snd_soc_dapm_dac,			/* digital to analog converter */
	snd_soc_dapm_micbias,		/* microphone bias (power) */
	snd_soc_dapm_mic,			/* microphone */
	snd_soc_dapm_hp,			/* headphones */
	snd_soc_dapm_spk,			/* speaker */
	snd_soc_dapm_line,			/* line input/output */
	snd_soc_dapm_switch,		/* analog switch */
	snd_soc_dapm_vmid,			/* codec bias/vmid - to minimise pops */
	snd_soc_dapm_pre,			/* machine specific pre widget - exec first */
	snd_soc_dapm_post,			/* machine specific post widget - exec last */
478
	snd_soc_dapm_supply,		/* power/clock supply */
479
	snd_soc_dapm_regulator_supply,	/* external regulator */
480
	snd_soc_dapm_clock_supply,	/* external clock */
481 482
	snd_soc_dapm_aif_in,		/* audio interface input */
	snd_soc_dapm_aif_out,		/* audio interface output */
483
	snd_soc_dapm_siggen,		/* signal generator */
484 485
	snd_soc_dapm_dai_in,		/* link to DAI structure */
	snd_soc_dapm_dai_out,
486
	snd_soc_dapm_dai_link,		/* link between two DAI structures */
487
	snd_soc_dapm_kcontrol,		/* Auto-disabled kcontrol */
488 489
};

490
enum snd_soc_dapm_subclass {
491 492
	SND_SOC_DAPM_CLASS_INIT		= 0,
	SND_SOC_DAPM_CLASS_RUNTIME	= 1,
493 494
};

495 496 497 498 499 500 501 502 503 504
/*
 * DAPM audio route definition.
 *
 * Defines an audio route originating at source via control and finishing
 * at sink.
 */
struct snd_soc_dapm_route {
	const char *sink;
	const char *control;
	const char *source;
505 506 507 508

	/* Note: currently only supported for links where source is a supply */
	int (*connected)(struct snd_soc_dapm_widget *source,
			 struct snd_soc_dapm_widget *sink);
509 510
};

511 512
/* dapm audio path between two widgets */
struct snd_soc_dapm_path {
513
	const char *name;
514 515 516 517 518 519 520

	/* source (input) and sink (output) widgets */
	struct snd_soc_dapm_widget *source;
	struct snd_soc_dapm_widget *sink;

	/* status */
	u32 connect:1;	/* source and sink widgets are connected */
M
Mark Brown 已提交
521
	u32 walking:1;  /* path is in the process of being walked */
522
	u32 weak:1;	/* path ignored for power management */
523
	u32 is_supply:1;	/* At least one of the connected widgets is a supply */
524

525 526 527
	int (*connected)(struct snd_soc_dapm_widget *source,
			 struct snd_soc_dapm_widget *sink);

528 529
	struct list_head list_source;
	struct list_head list_sink;
530
	struct list_head list_kcontrol;
531 532 533 534 535 536
	struct list_head list;
};

/* dapm widget */
struct snd_soc_dapm_widget {
	enum snd_soc_dapm_type id;
537 538
	const char *name;		/* widget name */
	const char *sname;	/* stream name */
539
	struct list_head list;
L
Liam Girdwood 已提交
540
	struct snd_soc_dapm_context *dapm;
541

542
	void *priv;				/* widget specific data */
543
	struct regulator *regulator;		/* attached regulator */
544
	const struct snd_soc_pcm_stream *params; /* params for dai links */
545 546
	unsigned int num_params; /* number of params for dai links */
	unsigned int params_select; /* currently selected param for dai link */
547

548
	/* dapm control */
549
	int reg;				/* negative reg = no direct dapm */
550
	unsigned char shift;			/* bits to shift */
551 552 553
	unsigned int mask;			/* non-shifted mask */
	unsigned int on_val;			/* on state value */
	unsigned int off_val;			/* off state value */
554 555 556 557
	unsigned char power:1;			/* block power status */
	unsigned char active:1;			/* active stream on DAC, ADC's */
	unsigned char connected:1;		/* connected codec pin */
	unsigned char new:1;			/* cnew complete */
558
	unsigned char force:1;			/* force state */
559
	unsigned char ignore_suspend:1;         /* kept enabled over suspend */
560 561
	unsigned char new_power:1;		/* power from this run */
	unsigned char power_checked:1;		/* power checked this run */
562 563 564
	unsigned char is_supply:1;		/* Widget is a supply type widget */
	unsigned char is_sink:1;		/* Widget is a sink type widget */
	unsigned char is_source:1;		/* Widget is a source type widget */
565
	int subseq;				/* sort within widget type */
566

567 568
	int (*power_check)(struct snd_soc_dapm_widget *w);

569 570
	/* external events */
	unsigned short event_flags;		/* flags to specify event types */
571
	int (*event)(struct snd_soc_dapm_widget*, struct snd_kcontrol *, int);
572 573 574

	/* kcontrols that relate to this widget */
	int num_kcontrols;
575
	const struct snd_kcontrol_new *kcontrol_news;
576
	struct snd_kcontrol **kcontrols;
577
	struct snd_soc_dobj dobj;
578 579 580 581

	/* widget input and outputs */
	struct list_head sources;
	struct list_head sinks;
582 583

	/* used during DAPM updates */
584
	struct list_head work_list;
585
	struct list_head power_list;
586
	struct list_head dirty;
587 588
	int inputs;
	int outputs;
589 590

	struct clk *clk;
591 592
};

593 594 595 596 597 598 599
struct snd_soc_dapm_update {
	struct snd_kcontrol *kcontrol;
	int reg;
	int mask;
	int val;
};

600 601 602 603
struct snd_soc_dapm_wcache {
	struct snd_soc_dapm_widget *widget;
};

L
Liam Girdwood 已提交
604 605 606 607
/* DAPM context */
struct snd_soc_dapm_context {
	enum snd_soc_bias_level bias_level;
	unsigned int idle_bias_off:1; /* Use BIAS_OFF instead of STANDBY */
608 609
	/* Go to BIAS_OFF in suspend if the DAPM context is idle */
	unsigned int suspend_bias_off:1;
610
	void (*seq_notifier)(struct snd_soc_dapm_context *,
611
			     enum snd_soc_dapm_type, int);
612

L
Liam Girdwood 已提交
613
	struct device *dev; /* from parent - for debug */
614
	struct snd_soc_component *component; /* parent component */
615
	struct snd_soc_card *card; /* parent card */
616 617

	/* used during DAPM updates */
618
	enum snd_soc_bias_level target_bias_level;
619 620
	struct list_head list;

621
	int (*stream_event)(struct snd_soc_dapm_context *dapm, int event);
622 623
	int (*set_bias_level)(struct snd_soc_dapm_context *dapm,
			      enum snd_soc_bias_level level);
624

625 626 627
	struct snd_soc_dapm_wcache path_sink_cache;
	struct snd_soc_dapm_wcache path_source_cache;

L
Liam Girdwood 已提交
628 629 630 631 632
#ifdef CONFIG_DEBUG_FS
	struct dentry *debugfs_dapm;
#endif
};

633 634 635 636 637 638
/* A list of widgets associated with an object, typically a snd_kcontrol */
struct snd_soc_dapm_widget_list {
	int num_widgets;
	struct snd_soc_dapm_widget *widgets[0];
};

639 640 641
struct snd_soc_dapm_stats {
	int power_checks;
	int path_checks;
642
	int neighbour_checks;
643 644
};

645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675
/**
 * snd_soc_dapm_init_bias_level() - Initialize DAPM bias level
 * @dapm: The DAPM context to initialize
 * @level: The DAPM level to initialize to
 *
 * This function only sets the driver internal state of the DAPM level and will
 * not modify the state of the device. Hence it should not be used during normal
 * operation, but only to synchronize the internal state to the device state.
 * E.g. during driver probe to set the DAPM level to the one corresponding with
 * the power-on reset state of the device.
 *
 * To change the DAPM state of the device use snd_soc_dapm_set_bias_level().
 */
static inline void snd_soc_dapm_init_bias_level(
	struct snd_soc_dapm_context *dapm, enum snd_soc_bias_level level)
{
	dapm->bias_level = level;
}

/**
 * snd_soc_dapm_get_bias_level() - Get current DAPM bias level
 * @dapm: The context for which to get the bias level
 *
 * Returns: The current bias level of the passed DAPM context.
 */
static inline enum snd_soc_bias_level snd_soc_dapm_get_bias_level(
	struct snd_soc_dapm_context *dapm)
{
	return dapm->bias_level;
}

676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693
/**
 * snd_soc_dapm_widget_for_each_sink_path - Iterates over all paths leaving a
 *  widget
 * @w: The widget
 * @p: The path iterator variable
 */
#define snd_soc_dapm_widget_for_each_sink_path(w, p) \
	list_for_each_entry(p, &w->sinks, list_source)

/**
 * snd_soc_dapm_widget_for_each_source_path - Iterates over all paths leading to
 *  a widget
 * @w: The widget
 * @p: The path iterator variable
 */
#define snd_soc_dapm_widget_for_each_source_path(w, p) \
	list_for_each_entry(p, &w->sources, list_sink)

694
#endif