mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-12-14 06:24:53 +08:00
d8a766a16e
Now we create the standard HD-audio bus (/sys/bus/hdaudio), and bind the codec driver with the codec device over there. This is the first step of the whole transition so that the changes to each codec driver are kept as minimal as possible. Each codec driver needs to register hda_codec_driver struct containing the currently existing preset via the new helper macro module_hda_codec_driver(). The old hda_codec_preset_list is replaced with this infrastructure. The generic parsers (for HDMI and other) are also included in the preset with the special IDs to bind uniquely. In HD-audio core side, the device binding code is split to hda_bind.c. It provides the snd_hda_bus_type implementation to match the codec driver with the given codec vendor ID. It also manages the module auto-loading by itself like before: when the matching isn't found, it tries to probe the corresponding codec modules, and finally falls back to the generic drivers. (The special ID mentioned above is set at this stage.) The only visible change to outside is that the hdaudio sysfs entry now appears in /sys/bus/devices, not as a sound class device. More works to move the suspend/resume and remove ops will be (hopefully) done in later patches. Signed-off-by: Takashi Iwai <tiwai@suse.de>
145 lines
3.6 KiB
C
145 lines
3.6 KiB
C
/*
|
|
* Universal Interface for Intel High Definition Audio Codec
|
|
*
|
|
* HD audio interface patch for C-Media CMI9880
|
|
*
|
|
* Copyright (c) 2004 Takashi Iwai <tiwai@suse.de>
|
|
*
|
|
*
|
|
* This driver 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 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This driver 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include <linux/init.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/module.h>
|
|
#include <sound/core.h>
|
|
#include "hda_codec.h"
|
|
#include "hda_local.h"
|
|
#include "hda_auto_parser.h"
|
|
#include "hda_jack.h"
|
|
#include "hda_generic.h"
|
|
|
|
struct cmi_spec {
|
|
struct hda_gen_spec gen;
|
|
};
|
|
|
|
/*
|
|
* stuff for auto-parser
|
|
*/
|
|
static const struct hda_codec_ops cmi_auto_patch_ops = {
|
|
.build_controls = snd_hda_gen_build_controls,
|
|
.build_pcms = snd_hda_gen_build_pcms,
|
|
.init = snd_hda_gen_init,
|
|
.free = snd_hda_gen_free,
|
|
.unsol_event = snd_hda_jack_unsol_event,
|
|
};
|
|
|
|
static int patch_cmi9880(struct hda_codec *codec)
|
|
{
|
|
struct cmi_spec *spec;
|
|
struct auto_pin_cfg *cfg;
|
|
int err;
|
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
|
if (spec == NULL)
|
|
return -ENOMEM;
|
|
|
|
codec->spec = spec;
|
|
cfg = &spec->gen.autocfg;
|
|
snd_hda_gen_spec_init(&spec->gen);
|
|
|
|
err = snd_hda_parse_pin_defcfg(codec, cfg, NULL, 0);
|
|
if (err < 0)
|
|
goto error;
|
|
err = snd_hda_gen_parse_auto_config(codec, cfg);
|
|
if (err < 0)
|
|
goto error;
|
|
|
|
codec->patch_ops = cmi_auto_patch_ops;
|
|
return 0;
|
|
|
|
error:
|
|
snd_hda_gen_free(codec);
|
|
return err;
|
|
}
|
|
|
|
static int patch_cmi8888(struct hda_codec *codec)
|
|
{
|
|
struct cmi_spec *spec;
|
|
struct auto_pin_cfg *cfg;
|
|
int err;
|
|
|
|
spec = kzalloc(sizeof(*spec), GFP_KERNEL);
|
|
if (!spec)
|
|
return -ENOMEM;
|
|
|
|
codec->spec = spec;
|
|
cfg = &spec->gen.autocfg;
|
|
snd_hda_gen_spec_init(&spec->gen);
|
|
|
|
/* mask NID 0x10 from the playback volume selection;
|
|
* it's a headphone boost volume handled manually below
|
|
*/
|
|
spec->gen.out_vol_mask = (1ULL << 0x10);
|
|
|
|
err = snd_hda_parse_pin_defcfg(codec, cfg, NULL, 0);
|
|
if (err < 0)
|
|
goto error;
|
|
err = snd_hda_gen_parse_auto_config(codec, cfg);
|
|
if (err < 0)
|
|
goto error;
|
|
|
|
if (get_defcfg_device(snd_hda_codec_get_pincfg(codec, 0x10)) ==
|
|
AC_JACK_HP_OUT) {
|
|
static const struct snd_kcontrol_new amp_kctl =
|
|
HDA_CODEC_VOLUME("Headphone Amp Playback Volume",
|
|
0x10, 0, HDA_OUTPUT);
|
|
if (!snd_hda_gen_add_kctl(&spec->gen, NULL, &_kctl)) {
|
|
err = -ENOMEM;
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
codec->patch_ops = cmi_auto_patch_ops;
|
|
return 0;
|
|
|
|
error:
|
|
snd_hda_gen_free(codec);
|
|
return err;
|
|
}
|
|
|
|
/*
|
|
* patch entries
|
|
*/
|
|
static const struct hda_codec_preset snd_hda_preset_cmedia[] = {
|
|
{ .id = 0x13f68888, .name = "CMI8888", .patch = patch_cmi8888 },
|
|
{ .id = 0x13f69880, .name = "CMI9880", .patch = patch_cmi9880 },
|
|
{ .id = 0x434d4980, .name = "CMI9880", .patch = patch_cmi9880 },
|
|
{} /* terminator */
|
|
};
|
|
|
|
MODULE_ALIAS("snd-hda-codec-id:13f68888");
|
|
MODULE_ALIAS("snd-hda-codec-id:13f69880");
|
|
MODULE_ALIAS("snd-hda-codec-id:434d4980");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
MODULE_DESCRIPTION("C-Media HD-audio codec");
|
|
|
|
static struct hda_codec_driver cmedia_driver = {
|
|
.preset = snd_hda_preset_cmedia,
|
|
};
|
|
|
|
module_hda_codec_driver(cmedia_driver);
|