Merge remote-tracking branch 'kraxel/audio.1' into staging
# By Bandan Das (3) and Gerd Hoffmann (1) # Via Gerd Hoffmann * kraxel/audio.1: audio: remove CONFIG_MIXEMU configure option hda-codec: make mixemu selectable at runtime hda-codec: refactor common definitions into a header file audio maintainers update Message-id: 1380011943-15083-1-git-send-email-kraxel@redhat.com
This commit is contained in:
commit
d7f0efcb22
@ -638,6 +638,7 @@ Subsystems
|
||||
----------
|
||||
Audio
|
||||
M: Vassili Karpov (malc) <av1474@comtv.ru>
|
||||
M: Gerd Hoffmann <kraxel@redhat.com>
|
||||
S: Maintained
|
||||
F: audio/
|
||||
F: hw/audio/
|
||||
|
@ -348,7 +348,6 @@ void mixeng_clear (struct st_sample *buf, int len)
|
||||
|
||||
void mixeng_volume (struct st_sample *buf, int len, struct mixeng_volume *vol)
|
||||
{
|
||||
#ifdef CONFIG_MIXEMU
|
||||
if (vol->mute) {
|
||||
mixeng_clear (buf, len);
|
||||
return;
|
||||
@ -364,9 +363,4 @@ void mixeng_volume (struct st_sample *buf, int len, struct mixeng_volume *vol)
|
||||
#endif
|
||||
buf += 1;
|
||||
}
|
||||
#else
|
||||
(void) buf;
|
||||
(void) len;
|
||||
(void) vol;
|
||||
#endif
|
||||
}
|
||||
|
8
configure
vendored
8
configure
vendored
@ -215,7 +215,6 @@ linux_user="no"
|
||||
bsd_user="no"
|
||||
guest_base="yes"
|
||||
uname_release=""
|
||||
mixemu="no"
|
||||
aix="no"
|
||||
blobs="yes"
|
||||
pkgversion=""
|
||||
@ -869,8 +868,6 @@ for opt do
|
||||
;;
|
||||
--enable-fdt) fdt="yes"
|
||||
;;
|
||||
--enable-mixemu) mixemu="yes"
|
||||
;;
|
||||
--disable-linux-aio) linux_aio="no"
|
||||
;;
|
||||
--enable-linux-aio) linux_aio="yes"
|
||||
@ -1103,7 +1100,6 @@ echo " (affects only QEMU, not qemu-img)"
|
||||
echo " --block-drv-ro-whitelist=L"
|
||||
echo " set block driver read-only whitelist"
|
||||
echo " (affects only QEMU, not qemu-img)"
|
||||
echo " --enable-mixemu enable mixer emulation"
|
||||
echo " --disable-xen disable xen backend driver support"
|
||||
echo " --enable-xen enable xen backend driver support"
|
||||
echo " --disable-xen-pci-passthrough"
|
||||
@ -3694,7 +3690,6 @@ echo "mingw32 support $mingw32"
|
||||
echo "Audio drivers $audio_drv_list"
|
||||
echo "Block whitelist (rw) $block_drv_rw_whitelist"
|
||||
echo "Block whitelist (ro) $block_drv_ro_whitelist"
|
||||
echo "Mixer emulation $mixemu"
|
||||
echo "VirtFS support $virtfs"
|
||||
echo "VNC support $vnc"
|
||||
if test "$vnc" = "yes" ; then
|
||||
@ -3881,9 +3876,6 @@ if test "$audio_win_int" = "yes" ; then
|
||||
fi
|
||||
echo "CONFIG_BDRV_RW_WHITELIST=$block_drv_rw_whitelist" >> $config_host_mak
|
||||
echo "CONFIG_BDRV_RO_WHITELIST=$block_drv_ro_whitelist" >> $config_host_mak
|
||||
if test "$mixemu" = "yes" ; then
|
||||
echo "CONFIG_MIXEMU=y" >> $config_host_mak
|
||||
fi
|
||||
if test "$vnc" = "yes" ; then
|
||||
echo "CONFIG_VNC=y" >> $config_host_mak
|
||||
fi
|
||||
|
456
hw/audio/hda-codec-common.h
Normal file
456
hw/audio/hda-codec-common.h
Normal file
@ -0,0 +1,456 @@
|
||||
/*
|
||||
* 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
|
@ -118,428 +118,12 @@ static void hda_codec_parse_fmt(uint32_t format, struct audsettings *as)
|
||||
#define QEMU_HDA_AMP_NONE (0)
|
||||
#define QEMU_HDA_AMP_STEPS 0x4a
|
||||
|
||||
#ifdef CONFIG_MIXEMU
|
||||
# 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
|
||||
#define PARAM mixemu
|
||||
#define HDA_MIXER
|
||||
#include "hda-codec-common.h"
|
||||
|
||||
/* common: audio output widget */
|
||||
static const desc_param common_params_audio_dac[] = {
|
||||
{
|
||||
.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 common_params_audio_adc[] = {
|
||||
{
|
||||
.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 common_params_audio_lineout[] = {
|
||||
{
|
||||
.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 common_params_audio_linein[] = {
|
||||
{
|
||||
.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 output_params_root[] = {
|
||||
{
|
||||
.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 output_params_audio_func[] = {
|
||||
{
|
||||
.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 output_nodes[] = {
|
||||
{
|
||||
.nid = AC_NODE_ROOT,
|
||||
.name = "root",
|
||||
.params = output_params_root,
|
||||
.nparams = ARRAY_SIZE(output_params_root),
|
||||
},{
|
||||
.nid = 1,
|
||||
.name = "func",
|
||||
.params = output_params_audio_func,
|
||||
.nparams = ARRAY_SIZE(output_params_audio_func),
|
||||
},{
|
||||
.nid = 2,
|
||||
.name = "dac",
|
||||
.params = common_params_audio_dac,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_dac),
|
||||
.stindex = 0,
|
||||
},{
|
||||
.nid = 3,
|
||||
.name = "out",
|
||||
.params = common_params_audio_lineout,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_lineout),
|
||||
.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 output = {
|
||||
.name = "output",
|
||||
.iid = QEMU_HDA_ID_OUTPUT,
|
||||
.nodes = output_nodes,
|
||||
.nnodes = ARRAY_SIZE(output_nodes),
|
||||
};
|
||||
|
||||
/* duplex: root node */
|
||||
static const desc_param duplex_params_root[] = {
|
||||
{
|
||||
.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 duplex_params_audio_func[] = {
|
||||
{
|
||||
.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 duplex_nodes[] = {
|
||||
{
|
||||
.nid = AC_NODE_ROOT,
|
||||
.name = "root",
|
||||
.params = duplex_params_root,
|
||||
.nparams = ARRAY_SIZE(duplex_params_root),
|
||||
},{
|
||||
.nid = 1,
|
||||
.name = "func",
|
||||
.params = duplex_params_audio_func,
|
||||
.nparams = ARRAY_SIZE(duplex_params_audio_func),
|
||||
},{
|
||||
.nid = 2,
|
||||
.name = "dac",
|
||||
.params = common_params_audio_dac,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_dac),
|
||||
.stindex = 0,
|
||||
},{
|
||||
.nid = 3,
|
||||
.name = "out",
|
||||
.params = common_params_audio_lineout,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_lineout),
|
||||
.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 = common_params_audio_adc,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_adc),
|
||||
.stindex = 1,
|
||||
.conn = (uint32_t[]) { 5 },
|
||||
},{
|
||||
.nid = 5,
|
||||
.name = "in",
|
||||
.params = common_params_audio_linein,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_linein),
|
||||
.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 duplex = {
|
||||
.name = "duplex",
|
||||
.iid = QEMU_HDA_ID_DUPLEX,
|
||||
.nodes = duplex_nodes,
|
||||
.nnodes = ARRAY_SIZE(duplex_nodes),
|
||||
};
|
||||
|
||||
/* micro: root node */
|
||||
static const desc_param micro_params_root[] = {
|
||||
{
|
||||
.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 micro_params_audio_func[] = {
|
||||
{
|
||||
.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 micro_nodes[] = {
|
||||
{
|
||||
.nid = AC_NODE_ROOT,
|
||||
.name = "root",
|
||||
.params = micro_params_root,
|
||||
.nparams = ARRAY_SIZE(micro_params_root),
|
||||
},{
|
||||
.nid = 1,
|
||||
.name = "func",
|
||||
.params = micro_params_audio_func,
|
||||
.nparams = ARRAY_SIZE(micro_params_audio_func),
|
||||
},{
|
||||
.nid = 2,
|
||||
.name = "dac",
|
||||
.params = common_params_audio_dac,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_dac),
|
||||
.stindex = 0,
|
||||
},{
|
||||
.nid = 3,
|
||||
.name = "out",
|
||||
.params = common_params_audio_lineout,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_lineout),
|
||||
.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 = common_params_audio_adc,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_adc),
|
||||
.stindex = 1,
|
||||
.conn = (uint32_t[]) { 5 },
|
||||
},{
|
||||
.nid = 5,
|
||||
.name = "in",
|
||||
.params = common_params_audio_linein,
|
||||
.nparams = ARRAY_SIZE(common_params_audio_linein),
|
||||
.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 micro = {
|
||||
.name = "micro",
|
||||
.iid = QEMU_HDA_ID_MICRO,
|
||||
.nodes = micro_nodes,
|
||||
.nnodes = ARRAY_SIZE(micro_nodes),
|
||||
};
|
||||
#define PARAM nomixemu
|
||||
#include "hda-codec-common.h"
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
|
||||
@ -585,6 +169,7 @@ struct HDAAudioState {
|
||||
|
||||
/* properties */
|
||||
uint32_t debug;
|
||||
bool mixer;
|
||||
};
|
||||
|
||||
static void hda_audio_input_cb(void *opaque, int avail)
|
||||
@ -1006,23 +591,42 @@ static const VMStateDescription vmstate_hda_audio = {
|
||||
};
|
||||
|
||||
static Property hda_audio_properties[] = {
|
||||
DEFINE_PROP_UINT32("debug", HDAAudioState, debug, 0),
|
||||
DEFINE_PROP_UINT32("debug", HDAAudioState, debug, 0),
|
||||
DEFINE_PROP_BOOL("mixer", HDAAudioState, mixer, true),
|
||||
DEFINE_PROP_END_OF_LIST(),
|
||||
};
|
||||
|
||||
static int hda_audio_init_output(HDACodecDevice *hda)
|
||||
{
|
||||
return hda_audio_init(hda, &output);
|
||||
HDAAudioState *a = DO_UPCAST(HDAAudioState, hda, hda);
|
||||
|
||||
if (!a->mixer) {
|
||||
return hda_audio_init(hda, &output_nomixemu);
|
||||
} else {
|
||||
return hda_audio_init(hda, &output_mixemu);
|
||||
}
|
||||
}
|
||||
|
||||
static int hda_audio_init_duplex(HDACodecDevice *hda)
|
||||
{
|
||||
return hda_audio_init(hda, &duplex);
|
||||
HDAAudioState *a = DO_UPCAST(HDAAudioState, hda, hda);
|
||||
|
||||
if (!a->mixer) {
|
||||
return hda_audio_init(hda, &duplex_nomixemu);
|
||||
} else {
|
||||
return hda_audio_init(hda, &duplex_mixemu);
|
||||
}
|
||||
}
|
||||
|
||||
static int hda_audio_init_micro(HDACodecDevice *hda)
|
||||
{
|
||||
return hda_audio_init(hda, µ);
|
||||
HDAAudioState *a = DO_UPCAST(HDAAudioState, hda, hda);
|
||||
|
||||
if (!a->mixer) {
|
||||
return hda_audio_init(hda, µ_nomixemu);
|
||||
} else {
|
||||
return hda_audio_init(hda, µ_mixemu);
|
||||
}
|
||||
}
|
||||
|
||||
static void hda_audio_output_class_init(ObjectClass *klass, void *data)
|
||||
|
Loading…
Reference in New Issue
Block a user