亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? hda_codec.c

?? linux 內核源代碼
?? C
?? 第 1 頁 / 共 5 頁
字號:
/* * Universal Interface for Intel High Definition Audio Codec * * 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 <sound/driver.h>#include <linux/init.h>#include <linux/delay.h>#include <linux/slab.h>#include <linux/pci.h>#include <linux/mutex.h>#include <sound/core.h>#include "hda_codec.h"#include <sound/asoundef.h>#include <sound/tlv.h>#include <sound/initval.h>#include "hda_local.h"#include <sound/hda_hwdep.h>#ifdef CONFIG_SND_HDA_POWER_SAVE/* define this option here to hide as static */static int power_save = CONFIG_SND_HDA_POWER_SAVE_DEFAULT;module_param(power_save, int, 0644);MODULE_PARM_DESC(power_save, "Automatic power-saving timeout "		 "(in second, 0 = disable).");#endif/* * vendor / preset table */struct hda_vendor_id {	unsigned int id;	const char *name;};/* codec vendor labels */static struct hda_vendor_id hda_vendor_ids[] = {	{ 0x10ec, "Realtek" },	{ 0x1057, "Motorola" },	{ 0x1106, "VIA" },	{ 0x11d4, "Analog Devices" },	{ 0x13f6, "C-Media" },	{ 0x14f1, "Conexant" },	{ 0x434d, "C-Media" },	{ 0x8384, "SigmaTel" },	{} /* terminator */};/* codec presets */#include "hda_patch.h"#ifdef CONFIG_SND_HDA_POWER_SAVEstatic void hda_power_work(struct work_struct *work);static void hda_keep_power_on(struct hda_codec *codec);#elsestatic inline void hda_keep_power_on(struct hda_codec *codec) {}#endif/** * snd_hda_codec_read - send a command and get the response * @codec: the HDA codec * @nid: NID to send the command * @direct: direct flag * @verb: the verb to send * @parm: the parameter for the verb * * Send a single command and read the corresponding response. * * Returns the obtained response value, or -1 for an error. */unsigned int snd_hda_codec_read(struct hda_codec *codec, hda_nid_t nid,				int direct,				unsigned int verb, unsigned int parm){	unsigned int res;	snd_hda_power_up(codec);	mutex_lock(&codec->bus->cmd_mutex);	if (!codec->bus->ops.command(codec, nid, direct, verb, parm))		res = codec->bus->ops.get_response(codec);	else		res = (unsigned int)-1;	mutex_unlock(&codec->bus->cmd_mutex);	snd_hda_power_down(codec);	return res;}/** * snd_hda_codec_write - send a single command without waiting for response * @codec: the HDA codec * @nid: NID to send the command * @direct: direct flag * @verb: the verb to send * @parm: the parameter for the verb * * Send a single command without waiting for response. * * Returns 0 if successful, or a negative error code. */int snd_hda_codec_write(struct hda_codec *codec, hda_nid_t nid, int direct,			 unsigned int verb, unsigned int parm){	int err;	snd_hda_power_up(codec);	mutex_lock(&codec->bus->cmd_mutex);	err = codec->bus->ops.command(codec, nid, direct, verb, parm);	mutex_unlock(&codec->bus->cmd_mutex);	snd_hda_power_down(codec);	return err;}/** * snd_hda_sequence_write - sequence writes * @codec: the HDA codec * @seq: VERB array to send * * Send the commands sequentially from the given array. * The array must be terminated with NID=0. */void snd_hda_sequence_write(struct hda_codec *codec, const struct hda_verb *seq){	for (; seq->nid; seq++)		snd_hda_codec_write(codec, seq->nid, 0, seq->verb, seq->param);}/** * snd_hda_get_sub_nodes - get the range of sub nodes * @codec: the HDA codec * @nid: NID to parse * @start_id: the pointer to store the start NID * * Parse the NID and store the start NID of its sub-nodes. * Returns the number of sub-nodes. */int snd_hda_get_sub_nodes(struct hda_codec *codec, hda_nid_t nid,			  hda_nid_t *start_id){	unsigned int parm;	parm = snd_hda_param_read(codec, nid, AC_PAR_NODE_COUNT);	if (parm == -1)		return 0;	*start_id = (parm >> 16) & 0x7fff;	return (int)(parm & 0x7fff);}/** * snd_hda_get_connections - get connection list * @codec: the HDA codec * @nid: NID to parse * @conn_list: connection list array * @max_conns: max. number of connections to store * * Parses the connection list of the given widget and stores the list * of NIDs. * * Returns the number of connections, or a negative error code. */int snd_hda_get_connections(struct hda_codec *codec, hda_nid_t nid,			    hda_nid_t *conn_list, int max_conns){	unsigned int parm;	int i, conn_len, conns;	unsigned int shift, num_elems, mask;	hda_nid_t prev_nid;	snd_assert(conn_list && max_conns > 0, return -EINVAL);	parm = snd_hda_param_read(codec, nid, AC_PAR_CONNLIST_LEN);	if (parm & AC_CLIST_LONG) {		/* long form */		shift = 16;		num_elems = 2;	} else {		/* short form */		shift = 8;		num_elems = 4;	}	conn_len = parm & AC_CLIST_LENGTH;	mask = (1 << (shift-1)) - 1;	if (!conn_len)		return 0; /* no connection */	if (conn_len == 1) {		/* single connection */		parm = snd_hda_codec_read(codec, nid, 0,					  AC_VERB_GET_CONNECT_LIST, 0);		conn_list[0] = parm & mask;		return 1;	}	/* multi connection */	conns = 0;	prev_nid = 0;	for (i = 0; i < conn_len; i++) {		int range_val;		hda_nid_t val, n;		if (i % num_elems == 0)			parm = snd_hda_codec_read(codec, nid, 0,						  AC_VERB_GET_CONNECT_LIST, i);		range_val = !!(parm & (1 << (shift-1))); /* ranges */		val = parm & mask;		parm >>= shift;		if (range_val) {			/* ranges between the previous and this one */			if (!prev_nid || prev_nid >= val) {				snd_printk(KERN_WARNING "hda_codec: "					   "invalid dep_range_val %x:%x\n",					   prev_nid, val);				continue;			}			for (n = prev_nid + 1; n <= val; n++) {				if (conns >= max_conns) {					snd_printk(KERN_ERR						   "Too many connections\n");					return -EINVAL;				}				conn_list[conns++] = n;			}		} else {			if (conns >= max_conns) {				snd_printk(KERN_ERR "Too many connections\n");				return -EINVAL;			}			conn_list[conns++] = val;		}		prev_nid = val;	}	return conns;}/** * snd_hda_queue_unsol_event - add an unsolicited event to queue * @bus: the BUS * @res: unsolicited event (lower 32bit of RIRB entry) * @res_ex: codec addr and flags (upper 32bit or RIRB entry) * * Adds the given event to the queue.  The events are processed in * the workqueue asynchronously.  Call this function in the interrupt * hanlder when RIRB receives an unsolicited event. * * Returns 0 if successful, or a negative error code. */int snd_hda_queue_unsol_event(struct hda_bus *bus, u32 res, u32 res_ex){	struct hda_bus_unsolicited *unsol;	unsigned int wp;	unsol = bus->unsol;	if (!unsol)		return 0;	wp = (unsol->wp + 1) % HDA_UNSOL_QUEUE_SIZE;	unsol->wp = wp;	wp <<= 1;	unsol->queue[wp] = res;	unsol->queue[wp + 1] = res_ex;	schedule_work(&unsol->work);	return 0;}/* * process queueud unsolicited events */static void process_unsol_events(struct work_struct *work){	struct hda_bus_unsolicited *unsol =		container_of(work, struct hda_bus_unsolicited, work);	struct hda_bus *bus = unsol->bus;	struct hda_codec *codec;	unsigned int rp, caddr, res;	while (unsol->rp != unsol->wp) {		rp = (unsol->rp + 1) % HDA_UNSOL_QUEUE_SIZE;		unsol->rp = rp;		rp <<= 1;		res = unsol->queue[rp];		caddr = unsol->queue[rp + 1];		if (!(caddr & (1 << 4))) /* no unsolicited event? */			continue;		codec = bus->caddr_tbl[caddr & 0x0f];		if (codec && codec->patch_ops.unsol_event)			codec->patch_ops.unsol_event(codec, res);	}}/* * initialize unsolicited queue */static int __devinit init_unsol_queue(struct hda_bus *bus){	struct hda_bus_unsolicited *unsol;	if (bus->unsol) /* already initialized */		return 0;	unsol = kzalloc(sizeof(*unsol), GFP_KERNEL);	if (!unsol) {		snd_printk(KERN_ERR "hda_codec: "			   "can't allocate unsolicited queue\n");		return -ENOMEM;	}	INIT_WORK(&unsol->work, process_unsol_events);	unsol->bus = bus;	bus->unsol = unsol;	return 0;}/* * destructor */static void snd_hda_codec_free(struct hda_codec *codec);static int snd_hda_bus_free(struct hda_bus *bus){	struct hda_codec *codec, *n;	if (!bus)		return 0;	if (bus->unsol) {		flush_scheduled_work();		kfree(bus->unsol);	}	list_for_each_entry_safe(codec, n, &bus->codec_list, list) {		snd_hda_codec_free(codec);	}	if (bus->ops.private_free)		bus->ops.private_free(bus);	kfree(bus);	return 0;}static int snd_hda_bus_dev_free(struct snd_device *device){	struct hda_bus *bus = device->device_data;	return snd_hda_bus_free(bus);}/** * snd_hda_bus_new - create a HDA bus * @card: the card entry * @temp: the template for hda_bus information * @busp: the pointer to store the created bus instance * * Returns 0 if successful, or a negative error code. */int __devinit snd_hda_bus_new(struct snd_card *card,			      const struct hda_bus_template *temp,			      struct hda_bus **busp){	struct hda_bus *bus;	int err;	static struct snd_device_ops dev_ops = {		.dev_free = snd_hda_bus_dev_free,	};	snd_assert(temp, return -EINVAL);	snd_assert(temp->ops.command && temp->ops.get_response, return -EINVAL);	if (busp)		*busp = NULL;	bus = kzalloc(sizeof(*bus), GFP_KERNEL);	if (bus == NULL) {		snd_printk(KERN_ERR "can't allocate struct hda_bus\n");		return -ENOMEM;	}	bus->card = card;	bus->private_data = temp->private_data;	bus->pci = temp->pci;	bus->modelname = temp->modelname;	bus->ops = temp->ops;	mutex_init(&bus->cmd_mutex);	INIT_LIST_HEAD(&bus->codec_list);	err = snd_device_new(card, SNDRV_DEV_BUS, bus, &dev_ops);	if (err < 0) {		snd_hda_bus_free(bus);		return err;	}	if (busp)		*busp = bus;	return 0;}#ifdef CONFIG_SND_HDA_GENERIC#define is_generic_config(codec) \	(codec->bus->modelname && !strcmp(codec->bus->modelname, "generic"))#else#define is_generic_config(codec)	0#endif/* * find a matching codec preset */static const struct hda_codec_preset __devinit *find_codec_preset(struct hda_codec *codec){	const struct hda_codec_preset **tbl, *preset;	if (is_generic_config(codec))		return NULL; /* use the generic parser */	for (tbl = hda_preset_tables; *tbl; tbl++) {		for (preset = *tbl; preset->id; preset++) {			u32 mask = preset->mask;			if (!mask)				mask = ~0;			if (preset->id == (codec->vendor_id & mask) &&			    (!preset->rev ||			     preset->rev == codec->revision_id))				return preset;		}	}	return NULL;}/* * snd_hda_get_codec_name - store the codec name */void snd_hda_get_codec_name(struct hda_codec *codec,			    char *name, int namelen){	const struct hda_vendor_id *c;	const char *vendor = NULL;	u16 vendor_id = codec->vendor_id >> 16;	char tmp[16];	for (c = hda_vendor_ids; c->id; c++) {		if (c->id == vendor_id) {			vendor = c->name;			break;		}	}	if (!vendor) {		sprintf(tmp, "Generic %04x", vendor_id);		vendor = tmp;	}	if (codec->preset && codec->preset->name)		snprintf(name, namelen, "%s %s", vendor, codec->preset->name);	else		snprintf(name, namelen, "%s ID %x", vendor,			 codec->vendor_id & 0xffff);}/* * look for an AFG and MFG nodes */static void __devinit setup_fg_nodes(struct hda_codec *codec){	int i, total_nodes;	hda_nid_t nid;	total_nodes = snd_hda_get_sub_nodes(codec, AC_NODE_ROOT, &nid);	for (i = 0; i < total_nodes; i++, nid++) {		unsigned int func;		func = snd_hda_param_read(codec, nid, AC_PAR_FUNCTION_TYPE);		switch (func & 0xff) {		case AC_GRP_AUDIO_FUNCTION:			codec->afg = nid;			break;		case AC_GRP_MODEM_FUNCTION:			codec->mfg = nid;			break;		default:			break;		}	}}/*

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
日韩国产欧美三级| 欧美日韩一区二区三区高清| 欧美日韩精品欧美日韩精品一综合 | 欧美亚洲综合在线| 亚洲男人的天堂在线aⅴ视频| 懂色av一区二区三区免费看| 精品成a人在线观看| 麻豆成人av在线| 欧美一区二区久久| 国内成人精品2018免费看| 欧美成人三级在线| 国产精品亚洲一区二区三区妖精| 日韩一区二区免费在线电影| 美女任你摸久久| 精品人伦一区二区色婷婷| 国产综合久久久久影院| 国产欧美一区二区精品性| av在线一区二区| 亚洲超碰精品一区二区| 日韩三级中文字幕| 成人av在线一区二区三区| 亚洲国产精品久久久久秋霞影院| 欧美视频中文一区二区三区在线观看| 亚洲成人av电影| 精品成人a区在线观看| 一本色道久久综合精品竹菊| 日韩黄色免费电影| 欧美激情中文字幕| 欧美久久久久久久久久| 成人天堂资源www在线| 一区二区三区不卡视频| 国产欧美日韩视频在线观看| 91久久精品一区二区三| 国产麻豆精品95视频| 亚洲图片一区二区| 欧美国产日韩在线观看| 日韩欧美在线网站| 欧洲视频一区二区| 色综合久久久网| 成人永久看片免费视频天堂| 亚洲综合在线第一页| 国产精品少妇自拍| 久久久久久久久蜜桃| 欧美日韩国产色站一区二区三区| 成人免费看片app下载| 国内成+人亚洲+欧美+综合在线| 丝袜美腿亚洲综合| 亚洲综合久久久久| 一区二区三区色| 亚洲免费资源在线播放| 国产精品三级电影| 国产色综合一区| 日本一区二区三区久久久久久久久不 | 亚洲私人影院在线观看| 欧美极品少妇xxxxⅹ高跟鞋| 久久老女人爱爱| 国产三级久久久| 国产精品视频在线看| 国产精品乱人伦中文| 亚洲视频综合在线| 有码一区二区三区| 日韩精品电影一区亚洲| 视频一区免费在线观看| 麻豆视频一区二区| 国产福利一区二区三区| jlzzjlzz亚洲日本少妇| 日本福利一区二区| 日韩一级视频免费观看在线| 欧美va亚洲va国产综合| 亚洲品质自拍视频网站| 亚洲国产sm捆绑调教视频| 另类小说图片综合网| 成人激情免费视频| 欧美日本免费一区二区三区| 日韩午夜中文字幕| 亚洲欧美一区二区三区久本道91| 亚洲成人免费看| 成人免费黄色大片| 欧美精品久久久久久久多人混战 | 欧美二区乱c少妇| 国产精品的网站| 国产真实精品久久二三区| 色哟哟国产精品| 国产日产欧美一区| 天堂在线亚洲视频| 在线观看一区不卡| 中文字幕乱码亚洲精品一区| 日本aⅴ精品一区二区三区| 成人免费视频播放| 国产欧美综合在线观看第十页| 国产一区二区三区精品欧美日韩一区二区三区 | 国产成人综合自拍| 911国产精品| 亚洲欧美另类图片小说| 蜜臀91精品一区二区三区| 在线看国产一区| 亚洲视频免费看| 成人激情电影免费在线观看| 日韩精品一区二区在线观看| 亚洲成a人v欧美综合天堂下载 | 久久久三级国产网站| 久久精品人人做人人爽97| 免费看日韩精品| 在线免费精品视频| 一区二区三区高清| 欧美性色综合网| 亚洲成人福利片| 日韩一区二区三区免费观看| 天堂资源在线中文精品| 6080亚洲精品一区二区| 日本不卡一二三区黄网| 欧美精品一区在线观看| 国产成人精品一区二区三区四区 | 最新国产成人在线观看| 成人国产精品免费观看视频| 国产精品成人免费| 日本国产一区二区| 久久99国产精品久久| 国产天堂亚洲国产碰碰| 欧美三级午夜理伦三级中视频| 亚洲成av人综合在线观看| 精品国产一二三| 91免费精品国自产拍在线不卡| 中文字幕一区二区三区在线不卡 | 丁香网亚洲国际| 亚洲综合网站在线观看| 日韩欧美一级片| 91丨porny丨蝌蚪视频| 日韩中文字幕区一区有砖一区| 日韩免费观看高清完整版| 成人一区二区三区视频| 亚洲电影一区二区| 久久精品亚洲一区二区三区浴池 | 日韩视频国产视频| 99国产欧美久久久精品| 久久se这里有精品| 午夜亚洲福利老司机| 亚洲国产精华液网站w| 欧美一区二区三区电影| 91黄色激情网站| 99视频有精品| 豆国产96在线|亚洲| 毛片av一区二区三区| 免费看黄色91| 日本强好片久久久久久aaa| 一区二区三区在线看| 中文字幕中文字幕中文字幕亚洲无线| 日韩午夜在线影院| 日韩视频免费观看高清在线视频| 欧美日韩国产在线播放网站| 日本道免费精品一区二区三区| 国产精品一区二区三区网站| 日本不卡视频在线| 理论片日本一区| 国产成人自拍网| 成人精品一区二区三区四区| 国产精品中文字幕欧美| 国产99精品国产| 99这里只有久久精品视频| 91美女福利视频| 欧美日韩精品欧美日韩精品| 91超碰这里只有精品国产| 日韩欧美在线网站| 国产精品网站在线| 亚洲欧美另类小说| 日韩精品三区四区| 国产福利91精品一区二区三区| 国产成人av福利| 在线一区二区三区四区五区| 欧美欧美欧美欧美首页| 国产三级精品三级| 亚洲一区二区三区四区在线观看| 日本欧洲一区二区| 91丝袜国产在线播放| 日韩一区二区在线观看视频播放| 国产三级一区二区| 亚洲国产一区二区三区| 国产精品亚洲午夜一区二区三区| 欧美中文字幕亚洲一区二区va在线| 337p亚洲精品色噜噜| 中文字幕一区日韩精品欧美| 日本在线不卡一区| 欧美日韩视频在线第一区| 国产清纯在线一区二区www| 日本亚洲欧美天堂免费| 97精品久久久久中文字幕| 精品久久久久久久久久久久久久久久久| 国产欧美精品一区二区三区四区| 婷婷夜色潮精品综合在线| 成人高清免费在线播放| 久久精品欧美日韩| 极品销魂美女一区二区三区| 欧美综合色免费| 亚洲成人免费观看| 欧美三级资源在线| 一区二区三区高清| 欧美日韩在线免费视频| 亚洲一区在线观看免费 | 亚洲午夜影视影院在线观看| 91在线视频官网|