457 lines
14 KiB
C
457 lines
14 KiB
C
|
/*
|
||
|
* Common code to disable/enable mixer emulation at run time
|
||
|
*
|
||
|
* Copyright (C) 2013 Red Hat, Inc.
|
||
|
*
|
||
|
* Written by Bandan Das <bsd@redhat.com>
|
||
|
* with important bits picked up from hda-codec.c
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU General Public License as
|
||
|
* published by the Free Software Foundation; either version 2 or
|
||
|
* (at your option) version 3 of the License.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* HDA codec descriptions
|
||
|
*/
|
||
|
|
||
|
#ifdef HDA_MIXER
|
||
|
#define QEMU_HDA_ID_OUTPUT ((QEMU_HDA_ID_VENDOR << 16) | 0x12)
|
||
|
#define QEMU_HDA_ID_DUPLEX ((QEMU_HDA_ID_VENDOR << 16) | 0x22)
|
||
|
#define QEMU_HDA_ID_MICRO ((QEMU_HDA_ID_VENDOR << 16) | 0x32)
|
||
|
#define QEMU_HDA_AMP_CAPS \
|
||
|
(AC_AMPCAP_MUTE | \
|
||
|
(QEMU_HDA_AMP_STEPS << AC_AMPCAP_OFFSET_SHIFT) | \
|
||
|
(QEMU_HDA_AMP_STEPS << AC_AMPCAP_NUM_STEPS_SHIFT) | \
|
||
|
(3 << AC_AMPCAP_STEP_SIZE_SHIFT))
|
||
|
#else
|
||
|
#define QEMU_HDA_ID_OUTPUT ((QEMU_HDA_ID_VENDOR << 16) | 0x11)
|
||
|
#define QEMU_HDA_ID_DUPLEX ((QEMU_HDA_ID_VENDOR << 16) | 0x21)
|
||
|
#define QEMU_HDA_ID_MICRO ((QEMU_HDA_ID_VENDOR << 16) | 0x31)
|
||
|
#define QEMU_HDA_AMP_CAPS QEMU_HDA_AMP_NONE
|
||
|
#endif
|
||
|
|
||
|
|
||
|
/* common: audio output widget */
|
||
|
static const desc_param glue(common_params_audio_dac_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_AUDIO_WIDGET_CAP,
|
||
|
.val = ((AC_WID_AUD_OUT << AC_WCAP_TYPE_SHIFT) |
|
||
|
AC_WCAP_FORMAT_OVRD |
|
||
|
AC_WCAP_AMP_OVRD |
|
||
|
AC_WCAP_OUT_AMP |
|
||
|
AC_WCAP_STEREO),
|
||
|
},{
|
||
|
.id = AC_PAR_PCM,
|
||
|
.val = QEMU_HDA_PCM_FORMATS,
|
||
|
},{
|
||
|
.id = AC_PAR_STREAM,
|
||
|
.val = AC_SUPFMT_PCM,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_IN_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_OUT_CAP,
|
||
|
.val = QEMU_HDA_AMP_CAPS,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* common: audio input widget */
|
||
|
static const desc_param glue(common_params_audio_adc_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_AUDIO_WIDGET_CAP,
|
||
|
.val = ((AC_WID_AUD_IN << AC_WCAP_TYPE_SHIFT) |
|
||
|
AC_WCAP_CONN_LIST |
|
||
|
AC_WCAP_FORMAT_OVRD |
|
||
|
AC_WCAP_AMP_OVRD |
|
||
|
AC_WCAP_IN_AMP |
|
||
|
AC_WCAP_STEREO),
|
||
|
},{
|
||
|
.id = AC_PAR_CONNLIST_LEN,
|
||
|
.val = 1,
|
||
|
},{
|
||
|
.id = AC_PAR_PCM,
|
||
|
.val = QEMU_HDA_PCM_FORMATS,
|
||
|
},{
|
||
|
.id = AC_PAR_STREAM,
|
||
|
.val = AC_SUPFMT_PCM,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_IN_CAP,
|
||
|
.val = QEMU_HDA_AMP_CAPS,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_OUT_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* common: pin widget (line-out) */
|
||
|
static const desc_param glue(common_params_audio_lineout_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_AUDIO_WIDGET_CAP,
|
||
|
.val = ((AC_WID_PIN << AC_WCAP_TYPE_SHIFT) |
|
||
|
AC_WCAP_CONN_LIST |
|
||
|
AC_WCAP_STEREO),
|
||
|
},{
|
||
|
.id = AC_PAR_PIN_CAP,
|
||
|
.val = AC_PINCAP_OUT,
|
||
|
},{
|
||
|
.id = AC_PAR_CONNLIST_LEN,
|
||
|
.val = 1,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_IN_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_OUT_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* common: pin widget (line-in) */
|
||
|
static const desc_param glue(common_params_audio_linein_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_AUDIO_WIDGET_CAP,
|
||
|
.val = ((AC_WID_PIN << AC_WCAP_TYPE_SHIFT) |
|
||
|
AC_WCAP_STEREO),
|
||
|
},{
|
||
|
.id = AC_PAR_PIN_CAP,
|
||
|
.val = AC_PINCAP_IN,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_IN_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_OUT_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* output: root node */
|
||
|
static const desc_param glue(output_params_root_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_VENDOR_ID,
|
||
|
.val = QEMU_HDA_ID_OUTPUT,
|
||
|
},{
|
||
|
.id = AC_PAR_SUBSYSTEM_ID,
|
||
|
.val = QEMU_HDA_ID_OUTPUT,
|
||
|
},{
|
||
|
.id = AC_PAR_REV_ID,
|
||
|
.val = 0x00100101,
|
||
|
},{
|
||
|
.id = AC_PAR_NODE_COUNT,
|
||
|
.val = 0x00010001,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* output: audio function */
|
||
|
static const desc_param glue(output_params_audio_func_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_FUNCTION_TYPE,
|
||
|
.val = AC_GRP_AUDIO_FUNCTION,
|
||
|
},{
|
||
|
.id = AC_PAR_SUBSYSTEM_ID,
|
||
|
.val = QEMU_HDA_ID_OUTPUT,
|
||
|
},{
|
||
|
.id = AC_PAR_NODE_COUNT,
|
||
|
.val = 0x00020002,
|
||
|
},{
|
||
|
.id = AC_PAR_PCM,
|
||
|
.val = QEMU_HDA_PCM_FORMATS,
|
||
|
},{
|
||
|
.id = AC_PAR_STREAM,
|
||
|
.val = AC_SUPFMT_PCM,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_IN_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_OUT_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_GPIO_CAP,
|
||
|
.val = 0,
|
||
|
},{
|
||
|
.id = AC_PAR_AUDIO_FG_CAP,
|
||
|
.val = 0x00000808,
|
||
|
},{
|
||
|
.id = AC_PAR_POWER_STATE,
|
||
|
.val = 0,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* output: nodes */
|
||
|
static const desc_node glue(output_nodes_, PARAM)[] = {
|
||
|
{
|
||
|
.nid = AC_NODE_ROOT,
|
||
|
.name = "root",
|
||
|
.params = glue(output_params_root_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(output_params_root_, PARAM)),
|
||
|
},{
|
||
|
.nid = 1,
|
||
|
.name = "func",
|
||
|
.params = glue(output_params_audio_func_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(output_params_audio_func_, PARAM)),
|
||
|
},{
|
||
|
.nid = 2,
|
||
|
.name = "dac",
|
||
|
.params = glue(common_params_audio_dac_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_dac_, PARAM)),
|
||
|
.stindex = 0,
|
||
|
},{
|
||
|
.nid = 3,
|
||
|
.name = "out",
|
||
|
.params = glue(common_params_audio_lineout_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_lineout_, PARAM)),
|
||
|
.config = ((AC_JACK_PORT_COMPLEX << AC_DEFCFG_PORT_CONN_SHIFT) |
|
||
|
(AC_JACK_LINE_OUT << AC_DEFCFG_DEVICE_SHIFT) |
|
||
|
(AC_JACK_CONN_UNKNOWN << AC_DEFCFG_CONN_TYPE_SHIFT) |
|
||
|
(AC_JACK_COLOR_GREEN << AC_DEFCFG_COLOR_SHIFT) |
|
||
|
0x10),
|
||
|
.pinctl = AC_PINCTL_OUT_EN,
|
||
|
.conn = (uint32_t[]) { 2 },
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/* output: codec */
|
||
|
static const desc_codec glue(output_, PARAM) = {
|
||
|
.name = "output",
|
||
|
.iid = QEMU_HDA_ID_OUTPUT,
|
||
|
.nodes = glue(output_nodes_, PARAM),
|
||
|
.nnodes = ARRAY_SIZE(glue(output_nodes_, PARAM)),
|
||
|
};
|
||
|
|
||
|
/* duplex: root node */
|
||
|
static const desc_param glue(duplex_params_root_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_VENDOR_ID,
|
||
|
.val = QEMU_HDA_ID_DUPLEX,
|
||
|
},{
|
||
|
.id = AC_PAR_SUBSYSTEM_ID,
|
||
|
.val = QEMU_HDA_ID_DUPLEX,
|
||
|
},{
|
||
|
.id = AC_PAR_REV_ID,
|
||
|
.val = 0x00100101,
|
||
|
},{
|
||
|
.id = AC_PAR_NODE_COUNT,
|
||
|
.val = 0x00010001,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* duplex: audio function */
|
||
|
static const desc_param glue(duplex_params_audio_func_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_FUNCTION_TYPE,
|
||
|
.val = AC_GRP_AUDIO_FUNCTION,
|
||
|
},{
|
||
|
.id = AC_PAR_SUBSYSTEM_ID,
|
||
|
.val = QEMU_HDA_ID_DUPLEX,
|
||
|
},{
|
||
|
.id = AC_PAR_NODE_COUNT,
|
||
|
.val = 0x00020004,
|
||
|
},{
|
||
|
.id = AC_PAR_PCM,
|
||
|
.val = QEMU_HDA_PCM_FORMATS,
|
||
|
},{
|
||
|
.id = AC_PAR_STREAM,
|
||
|
.val = AC_SUPFMT_PCM,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_IN_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_OUT_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_GPIO_CAP,
|
||
|
.val = 0,
|
||
|
},{
|
||
|
.id = AC_PAR_AUDIO_FG_CAP,
|
||
|
.val = 0x00000808,
|
||
|
},{
|
||
|
.id = AC_PAR_POWER_STATE,
|
||
|
.val = 0,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* duplex: nodes */
|
||
|
static const desc_node glue(duplex_nodes_, PARAM)[] = {
|
||
|
{
|
||
|
.nid = AC_NODE_ROOT,
|
||
|
.name = "root",
|
||
|
.params = glue(duplex_params_root_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(duplex_params_root_, PARAM)),
|
||
|
},{
|
||
|
.nid = 1,
|
||
|
.name = "func",
|
||
|
.params = glue(duplex_params_audio_func_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(duplex_params_audio_func_, PARAM)),
|
||
|
},{
|
||
|
.nid = 2,
|
||
|
.name = "dac",
|
||
|
.params = glue(common_params_audio_dac_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_dac_, PARAM)),
|
||
|
.stindex = 0,
|
||
|
},{
|
||
|
.nid = 3,
|
||
|
.name = "out",
|
||
|
.params = glue(common_params_audio_lineout_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_lineout_, PARAM)),
|
||
|
.config = ((AC_JACK_PORT_COMPLEX << AC_DEFCFG_PORT_CONN_SHIFT) |
|
||
|
(AC_JACK_LINE_OUT << AC_DEFCFG_DEVICE_SHIFT) |
|
||
|
(AC_JACK_CONN_UNKNOWN << AC_DEFCFG_CONN_TYPE_SHIFT) |
|
||
|
(AC_JACK_COLOR_GREEN << AC_DEFCFG_COLOR_SHIFT) |
|
||
|
0x10),
|
||
|
.pinctl = AC_PINCTL_OUT_EN,
|
||
|
.conn = (uint32_t[]) { 2 },
|
||
|
},{
|
||
|
.nid = 4,
|
||
|
.name = "adc",
|
||
|
.params = glue(common_params_audio_adc_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_adc_, PARAM)),
|
||
|
.stindex = 1,
|
||
|
.conn = (uint32_t[]) { 5 },
|
||
|
},{
|
||
|
.nid = 5,
|
||
|
.name = "in",
|
||
|
.params = glue(common_params_audio_linein_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_linein_, PARAM)),
|
||
|
.config = ((AC_JACK_PORT_COMPLEX << AC_DEFCFG_PORT_CONN_SHIFT) |
|
||
|
(AC_JACK_LINE_IN << AC_DEFCFG_DEVICE_SHIFT) |
|
||
|
(AC_JACK_CONN_UNKNOWN << AC_DEFCFG_CONN_TYPE_SHIFT) |
|
||
|
(AC_JACK_COLOR_RED << AC_DEFCFG_COLOR_SHIFT) |
|
||
|
0x20),
|
||
|
.pinctl = AC_PINCTL_IN_EN,
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/* duplex: codec */
|
||
|
static const desc_codec glue(duplex_, PARAM) = {
|
||
|
.name = "duplex",
|
||
|
.iid = QEMU_HDA_ID_DUPLEX,
|
||
|
.nodes = glue(duplex_nodes_, PARAM),
|
||
|
.nnodes = ARRAY_SIZE(glue(duplex_nodes_, PARAM)),
|
||
|
};
|
||
|
|
||
|
/* micro: root node */
|
||
|
static const desc_param glue(micro_params_root_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_VENDOR_ID,
|
||
|
.val = QEMU_HDA_ID_MICRO,
|
||
|
},{
|
||
|
.id = AC_PAR_SUBSYSTEM_ID,
|
||
|
.val = QEMU_HDA_ID_MICRO,
|
||
|
},{
|
||
|
.id = AC_PAR_REV_ID,
|
||
|
.val = 0x00100101,
|
||
|
},{
|
||
|
.id = AC_PAR_NODE_COUNT,
|
||
|
.val = 0x00010001,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* micro: audio function */
|
||
|
static const desc_param glue(micro_params_audio_func_, PARAM)[] = {
|
||
|
{
|
||
|
.id = AC_PAR_FUNCTION_TYPE,
|
||
|
.val = AC_GRP_AUDIO_FUNCTION,
|
||
|
},{
|
||
|
.id = AC_PAR_SUBSYSTEM_ID,
|
||
|
.val = QEMU_HDA_ID_MICRO,
|
||
|
},{
|
||
|
.id = AC_PAR_NODE_COUNT,
|
||
|
.val = 0x00020004,
|
||
|
},{
|
||
|
.id = AC_PAR_PCM,
|
||
|
.val = QEMU_HDA_PCM_FORMATS,
|
||
|
},{
|
||
|
.id = AC_PAR_STREAM,
|
||
|
.val = AC_SUPFMT_PCM,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_IN_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_AMP_OUT_CAP,
|
||
|
.val = QEMU_HDA_AMP_NONE,
|
||
|
},{
|
||
|
.id = AC_PAR_GPIO_CAP,
|
||
|
.val = 0,
|
||
|
},{
|
||
|
.id = AC_PAR_AUDIO_FG_CAP,
|
||
|
.val = 0x00000808,
|
||
|
},{
|
||
|
.id = AC_PAR_POWER_STATE,
|
||
|
.val = 0,
|
||
|
},
|
||
|
};
|
||
|
|
||
|
/* micro: nodes */
|
||
|
static const desc_node glue(micro_nodes_, PARAM)[] = {
|
||
|
{
|
||
|
.nid = AC_NODE_ROOT,
|
||
|
.name = "root",
|
||
|
.params = glue(micro_params_root_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(micro_params_root_, PARAM)),
|
||
|
},{
|
||
|
.nid = 1,
|
||
|
.name = "func",
|
||
|
.params = glue(micro_params_audio_func_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(micro_params_audio_func_, PARAM)),
|
||
|
},{
|
||
|
.nid = 2,
|
||
|
.name = "dac",
|
||
|
.params = glue(common_params_audio_dac_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_dac_, PARAM)),
|
||
|
.stindex = 0,
|
||
|
},{
|
||
|
.nid = 3,
|
||
|
.name = "out",
|
||
|
.params = glue(common_params_audio_lineout_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_lineout_, PARAM)),
|
||
|
.config = ((AC_JACK_PORT_COMPLEX << AC_DEFCFG_PORT_CONN_SHIFT) |
|
||
|
(AC_JACK_SPEAKER << AC_DEFCFG_DEVICE_SHIFT) |
|
||
|
(AC_JACK_CONN_UNKNOWN << AC_DEFCFG_CONN_TYPE_SHIFT) |
|
||
|
(AC_JACK_COLOR_GREEN << AC_DEFCFG_COLOR_SHIFT) |
|
||
|
0x10),
|
||
|
.pinctl = AC_PINCTL_OUT_EN,
|
||
|
.conn = (uint32_t[]) { 2 },
|
||
|
},{
|
||
|
.nid = 4,
|
||
|
.name = "adc",
|
||
|
.params = glue(common_params_audio_adc_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_adc_, PARAM)),
|
||
|
.stindex = 1,
|
||
|
.conn = (uint32_t[]) { 5 },
|
||
|
},{
|
||
|
.nid = 5,
|
||
|
.name = "in",
|
||
|
.params = glue(common_params_audio_linein_, PARAM),
|
||
|
.nparams = ARRAY_SIZE(glue(common_params_audio_linein_, PARAM)),
|
||
|
.config = ((AC_JACK_PORT_COMPLEX << AC_DEFCFG_PORT_CONN_SHIFT) |
|
||
|
(AC_JACK_MIC_IN << AC_DEFCFG_DEVICE_SHIFT) |
|
||
|
(AC_JACK_CONN_UNKNOWN << AC_DEFCFG_CONN_TYPE_SHIFT) |
|
||
|
(AC_JACK_COLOR_RED << AC_DEFCFG_COLOR_SHIFT) |
|
||
|
0x20),
|
||
|
.pinctl = AC_PINCTL_IN_EN,
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/* micro: codec */
|
||
|
static const desc_codec glue(micro_, PARAM) = {
|
||
|
.name = "micro",
|
||
|
.iid = QEMU_HDA_ID_MICRO,
|
||
|
.nodes = glue(micro_nodes_, PARAM),
|
||
|
.nnodes = ARRAY_SIZE(glue(micro_nodes_, PARAM)),
|
||
|
};
|
||
|
|
||
|
#undef PARAM
|
||
|
#undef HDA_MIXER
|
||
|
#undef QEMU_HDA_ID_OUTPUT
|
||
|
#undef QEMU_HDA_ID_DUPLEX
|
||
|
#undef QEMU_HDA_ID_MICRO
|
||
|
#undef QEMU_HDA_AMP_CAPS
|