| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * HD audio interface patch for Creative X-Fi CA0110-IBG chip | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2008 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/pci.h>
 | 
					
						
							| 
									
										
										
										
											2011-07-15 12:38:28 -04:00
										 |  |  | #include <linux/module.h>
 | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | #include <sound/core.h>
 | 
					
						
							|  |  |  | #include "hda_codec.h"
 | 
					
						
							|  |  |  | #include "hda_local.h"
 | 
					
						
							| 
									
										
										
										
											2012-05-07 17:42:31 +02:00
										 |  |  | #include "hda_auto_parser.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | #include "hda_jack.h"
 | 
					
						
							|  |  |  | #include "hda_generic.h"
 | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-02 11:36:00 +02:00
										 |  |  | static const struct hda_codec_ops ca0110_patch_ops = { | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | 	.build_controls = snd_hda_gen_build_controls, | 
					
						
							|  |  |  | 	.build_pcms = snd_hda_gen_build_pcms, | 
					
						
							|  |  |  | 	.init = snd_hda_gen_init, | 
					
						
							|  |  |  | 	.free = snd_hda_gen_free, | 
					
						
							| 
									
										
										
										
											2013-01-18 07:51:17 +01:00
										 |  |  | 	.unsol_event = snd_hda_jack_unsol_event, | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int ca0110_parse_auto_config(struct hda_codec *codec) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | 	struct hda_gen_spec *spec = codec->spec; | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 	int err; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | 	err = snd_hda_parse_pin_defcfg(codec, &spec->autocfg, NULL, 0); | 
					
						
							|  |  |  | 	if (err < 0) | 
					
						
							|  |  |  | 		return err; | 
					
						
							|  |  |  | 	err = snd_hda_gen_parse_auto_config(codec, &spec->autocfg); | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 	if (err < 0) | 
					
						
							|  |  |  | 		return err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-01 08:34:32 +02:00
										 |  |  | static int patch_ca0110(struct hda_codec *codec) | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | 	struct hda_gen_spec *spec; | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 	int err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	spec = kzalloc(sizeof(*spec), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!spec) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | 	snd_hda_gen_spec_init(spec); | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 	codec->spec = spec; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | 	spec->multi_cap_vol = 1; | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 	codec->bus->needs_damn_long_delay = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = ca0110_parse_auto_config(codec); | 
					
						
							|  |  |  | 	if (err < 0) | 
					
						
							|  |  |  | 		goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	codec->patch_ops = ca0110_patch_ops; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  error: | 
					
						
							| 
									
										
										
										
											2012-12-19 18:04:37 +01:00
										 |  |  | 	snd_hda_gen_free(codec); | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 	return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * patch entries | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2011-05-02 11:36:00 +02:00
										 |  |  | static const struct hda_codec_preset snd_hda_preset_ca0110[] = { | 
					
						
							| 
									
										
										
										
											2009-04-16 08:53:34 +02:00
										 |  |  | 	{ .id = 0x1102000a, .name = "CA0110-IBG", .patch = patch_ca0110 }, | 
					
						
							|  |  |  | 	{ .id = 0x1102000b, .name = "CA0110-IBG", .patch = patch_ca0110 }, | 
					
						
							|  |  |  | 	{ .id = 0x1102000d, .name = "SB0880 X-Fi", .patch = patch_ca0110 }, | 
					
						
							|  |  |  | 	{} /* terminator */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MODULE_ALIAS("snd-hda-codec-id:1102000a"); | 
					
						
							|  |  |  | MODULE_ALIAS("snd-hda-codec-id:1102000b"); | 
					
						
							|  |  |  | MODULE_ALIAS("snd-hda-codec-id:1102000d"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); | 
					
						
							|  |  |  | MODULE_DESCRIPTION("Creative CA0110-IBG HD-audio codec"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct hda_codec_preset_list ca0110_list = { | 
					
						
							|  |  |  | 	.preset = snd_hda_preset_ca0110, | 
					
						
							|  |  |  | 	.owner = THIS_MODULE, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int __init patch_ca0110_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return snd_hda_add_codec_preset(&ca0110_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void __exit patch_ca0110_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	snd_hda_delete_codec_preset(&ca0110_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module_init(patch_ca0110_init) | 
					
						
							|  |  |  | module_exit(patch_ca0110_exit) |