⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 emumixer.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
/* *  Copyright (c) by Jaroslav Kysela <perex@suse.cz>, *                   Takashi Iwai <tiwai@suse.de> *                   Creative Labs, Inc. *  Routines for control of EMU10K1 chips / mixer routines *  Multichannel PCM support Copyright (c) Lee Revell <rlrevell@joe-job.com> * *  BUGS: *    -- * *  TODO: *    -- * *   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 of the License, or *   (at your option) any later version. * *   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, write to the Free Software *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA * */#include <sound/driver.h>#include <linux/time.h>#include <linux/init.h>#include <sound/core.h>#include <sound/emu10k1.h>#define AC97_ID_STAC9758	0x83847658static int snd_emu10k1_spdif_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo){	uinfo->type = SNDRV_CTL_ELEM_TYPE_IEC958;	uinfo->count = 1;	return 0;}static int snd_emu10k1_spdif_get(snd_kcontrol_t * kcontrol,                                 snd_ctl_elem_value_t * ucontrol){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);	unsigned long flags;	spin_lock_irqsave(&emu->reg_lock, flags);	ucontrol->value.iec958.status[0] = (emu->spdif_bits[idx] >> 0) & 0xff;	ucontrol->value.iec958.status[1] = (emu->spdif_bits[idx] >> 8) & 0xff;	ucontrol->value.iec958.status[2] = (emu->spdif_bits[idx] >> 16) & 0xff;	ucontrol->value.iec958.status[3] = (emu->spdif_bits[idx] >> 24) & 0xff;	spin_unlock_irqrestore(&emu->reg_lock, flags);	return 0;}static int snd_emu10k1_spdif_get_mask(snd_kcontrol_t * kcontrol,				      snd_ctl_elem_value_t * ucontrol){	ucontrol->value.iec958.status[0] = 0xff;	ucontrol->value.iec958.status[1] = 0xff;	ucontrol->value.iec958.status[2] = 0xff;	ucontrol->value.iec958.status[3] = 0xff;	return 0;}#if 0static int snd_audigy_spdif_output_rate_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo){	static char *texts[] = {"44100", "48000", "96000"};	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;	uinfo->count = 1;	uinfo->value.enumerated.items = 3;	if (uinfo->value.enumerated.item >= uinfo->value.enumerated.items)		uinfo->value.enumerated.item = uinfo->value.enumerated.items - 1;	strcpy(uinfo->value.enumerated.name, texts[uinfo->value.enumerated.item]);	return 0;}static int snd_audigy_spdif_output_rate_get(snd_kcontrol_t * kcontrol,                                 snd_ctl_elem_value_t * ucontrol){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	unsigned int tmp;	unsigned long flags;		spin_lock_irqsave(&emu->reg_lock, flags);	tmp = snd_emu10k1_ptr_read(emu, A_SPDIF_SAMPLERATE, 0);	switch (tmp & A_SPDIF_RATE_MASK) {	case A_SPDIF_44100:		ucontrol->value.enumerated.item[0] = 0;		break;	case A_SPDIF_48000:		ucontrol->value.enumerated.item[0] = 1;		break;	case A_SPDIF_96000:		ucontrol->value.enumerated.item[0] = 2;		break;	default:		ucontrol->value.enumerated.item[0] = 1;	}	spin_unlock_irqrestore(&emu->reg_lock, flags);	return 0;}static int snd_audigy_spdif_output_rate_put(snd_kcontrol_t * kcontrol,                                 snd_ctl_elem_value_t * ucontrol){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	int change;	unsigned int reg, val, tmp;	unsigned long flags;	switch(ucontrol->value.enumerated.item[0]) {	case 0:		val = A_SPDIF_44100;		break;	case 1:		val = A_SPDIF_48000;		break;	case 2:		val = A_SPDIF_96000;		break;	default:		val = A_SPDIF_48000;		break;	}		spin_lock_irqsave(&emu->reg_lock, flags);	reg = snd_emu10k1_ptr_read(emu, A_SPDIF_SAMPLERATE, 0);	tmp = reg & ~A_SPDIF_RATE_MASK;	tmp |= val;	if ((change = (tmp != reg)))		snd_emu10k1_ptr_write(emu, A_SPDIF_SAMPLERATE, 0, tmp);	spin_unlock_irqrestore(&emu->reg_lock, flags);	return change;}static snd_kcontrol_new_t snd_audigy_spdif_output_rate ={	.access =	SNDRV_CTL_ELEM_ACCESS_READWRITE,	.iface =        SNDRV_CTL_ELEM_IFACE_MIXER,	.name =         "Audigy SPDIF Output Sample Rate",	.count =	1,	.info =         snd_audigy_spdif_output_rate_info,	.get =          snd_audigy_spdif_output_rate_get,	.put =          snd_audigy_spdif_output_rate_put};#endifstatic int snd_emu10k1_spdif_put(snd_kcontrol_t * kcontrol,                                 snd_ctl_elem_value_t * ucontrol){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	unsigned int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);	int change;	unsigned int val;	unsigned long flags;	val = (ucontrol->value.iec958.status[0] << 0) |	      (ucontrol->value.iec958.status[1] << 8) |	      (ucontrol->value.iec958.status[2] << 16) |	      (ucontrol->value.iec958.status[3] << 24);	spin_lock_irqsave(&emu->reg_lock, flags);	change = val != emu->spdif_bits[idx];	if (change) {		snd_emu10k1_ptr_write(emu, SPCS0 + idx, 0, val);		emu->spdif_bits[idx] = val;	}	spin_unlock_irqrestore(&emu->reg_lock, flags);	return change;}static snd_kcontrol_new_t snd_emu10k1_spdif_mask_control ={	.access =	SNDRV_CTL_ELEM_ACCESS_READ,	.iface =        SNDRV_CTL_ELEM_IFACE_PCM,	.name =         SNDRV_CTL_NAME_IEC958("",PLAYBACK,MASK),	.count =	4,	.info =         snd_emu10k1_spdif_info,	.get =          snd_emu10k1_spdif_get_mask};static snd_kcontrol_new_t snd_emu10k1_spdif_control ={	.iface =	SNDRV_CTL_ELEM_IFACE_PCM,	.name =         SNDRV_CTL_NAME_IEC958("",PLAYBACK,DEFAULT),	.count =	4,	.info =         snd_emu10k1_spdif_info,	.get =          snd_emu10k1_spdif_get,	.put =          snd_emu10k1_spdif_put};static void update_emu10k1_fxrt(emu10k1_t *emu, int voice, unsigned char *route){	if (emu->audigy) {		snd_emu10k1_ptr_write(emu, A_FXRT1, voice,				      snd_emu10k1_compose_audigy_fxrt1(route));		snd_emu10k1_ptr_write(emu, A_FXRT2, voice,				      snd_emu10k1_compose_audigy_fxrt2(route));	} else {		snd_emu10k1_ptr_write(emu, FXRT, voice,				      snd_emu10k1_compose_send_routing(route));	}}static void update_emu10k1_send_volume(emu10k1_t *emu, int voice, unsigned char *volume){	snd_emu10k1_ptr_write(emu, PTRX_FXSENDAMOUNT_A, voice, volume[0]);	snd_emu10k1_ptr_write(emu, PTRX_FXSENDAMOUNT_B, voice, volume[1]);	snd_emu10k1_ptr_write(emu, PSST_FXSENDAMOUNT_C, voice, volume[2]);	snd_emu10k1_ptr_write(emu, DSL_FXSENDAMOUNT_D, voice, volume[3]);	if (emu->audigy) {		unsigned int val = ((unsigned int)volume[4] << 24) |			((unsigned int)volume[5] << 16) |			((unsigned int)volume[6] << 8) |			(unsigned int)volume[7];		snd_emu10k1_ptr_write(emu, A_SENDAMOUNTS, voice, val);	}}/* PCM stream controls */static int snd_emu10k1_send_routing_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;	uinfo->count = emu->audigy ? 3*8 : 3*4;	uinfo->value.integer.min = 0;	uinfo->value.integer.max = emu->audigy ? 0x3f : 0x0f;	return 0;}static int snd_emu10k1_send_routing_get(snd_kcontrol_t * kcontrol,                                        snd_ctl_elem_value_t * ucontrol){	unsigned long flags;	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	emu10k1_pcm_mixer_t *mix = &emu->pcm_mixer[snd_ctl_get_ioffidx(kcontrol, &ucontrol->id)];	int voice, idx;	int num_efx = emu->audigy ? 8 : 4;	int mask = emu->audigy ? 0x3f : 0x0f;	spin_lock_irqsave(&emu->reg_lock, flags);	for (voice = 0; voice < 3; voice++)		for (idx = 0; idx < num_efx; idx++)			ucontrol->value.integer.value[(voice * num_efx) + idx] = 				mix->send_routing[voice][idx] & mask;	spin_unlock_irqrestore(&emu->reg_lock, flags);	return 0;}static int snd_emu10k1_send_routing_put(snd_kcontrol_t * kcontrol,                                        snd_ctl_elem_value_t * ucontrol){	unsigned long flags;	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	emu10k1_pcm_mixer_t *mix = &emu->pcm_mixer[snd_ctl_get_ioffidx(kcontrol, &ucontrol->id)];	int change = 0, voice, idx, val;	int num_efx = emu->audigy ? 8 : 4;	int mask = emu->audigy ? 0x3f : 0x0f;	spin_lock_irqsave(&emu->reg_lock, flags);	for (voice = 0; voice < 3; voice++)		for (idx = 0; idx < num_efx; idx++) {			val = ucontrol->value.integer.value[(voice * num_efx) + idx] & mask;			if (mix->send_routing[voice][idx] != val) {				mix->send_routing[voice][idx] = val;				change = 1;			}		}		if (change && mix->epcm) {		if (mix->epcm->voices[0] && mix->epcm->voices[1]) {			update_emu10k1_fxrt(emu, mix->epcm->voices[0]->number,					    &mix->send_routing[1][0]);			update_emu10k1_fxrt(emu, mix->epcm->voices[1]->number,					    &mix->send_routing[2][0]);		} else if (mix->epcm->voices[0]) {			update_emu10k1_fxrt(emu, mix->epcm->voices[0]->number,					    &mix->send_routing[0][0]);		}	}	spin_unlock_irqrestore(&emu->reg_lock, flags);	return change;}static snd_kcontrol_new_t snd_emu10k1_send_routing_control ={	.access =	SNDRV_CTL_ELEM_ACCESS_READWRITE | SNDRV_CTL_ELEM_ACCESS_INACTIVE,	.iface =        SNDRV_CTL_ELEM_IFACE_PCM,	.name =         "EMU10K1 PCM Send Routing",	.count =	32,	.info =         snd_emu10k1_send_routing_info,	.get =          snd_emu10k1_send_routing_get,	.put =          snd_emu10k1_send_routing_put};static int snd_emu10k1_send_volume_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;	uinfo->count = emu->audigy ? 3*8 : 3*4;	uinfo->value.integer.min = 0;	uinfo->value.integer.max = 255;	return 0;}static int snd_emu10k1_send_volume_get(snd_kcontrol_t * kcontrol,                                       snd_ctl_elem_value_t * ucontrol){	unsigned long flags;	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	emu10k1_pcm_mixer_t *mix = &emu->pcm_mixer[snd_ctl_get_ioffidx(kcontrol, &ucontrol->id)];	int idx;	int num_efx = emu->audigy ? 8 : 4;	spin_lock_irqsave(&emu->reg_lock, flags);	for (idx = 0; idx < 3*num_efx; idx++)		ucontrol->value.integer.value[idx] = mix->send_volume[idx/num_efx][idx%num_efx];	spin_unlock_irqrestore(&emu->reg_lock, flags);	return 0;}static int snd_emu10k1_send_volume_put(snd_kcontrol_t * kcontrol,                                       snd_ctl_elem_value_t * ucontrol){	unsigned long flags;	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	emu10k1_pcm_mixer_t *mix = &emu->pcm_mixer[snd_ctl_get_ioffidx(kcontrol, &ucontrol->id)];	int change = 0, idx, val;	int num_efx = emu->audigy ? 8 : 4;	spin_lock_irqsave(&emu->reg_lock, flags);	for (idx = 0; idx < 3*num_efx; idx++) {		val = ucontrol->value.integer.value[idx] & 255;		if (mix->send_volume[idx/num_efx][idx%num_efx] != val) {			mix->send_volume[idx/num_efx][idx%num_efx] = val;			change = 1;		}	}	if (change && mix->epcm) {		if (mix->epcm->voices[0] && mix->epcm->voices[1]) {			update_emu10k1_send_volume(emu, mix->epcm->voices[0]->number,						   &mix->send_volume[1][0]);			update_emu10k1_send_volume(emu, mix->epcm->voices[1]->number,						   &mix->send_volume[2][0]);		} else if (mix->epcm->voices[0]) {			update_emu10k1_send_volume(emu, mix->epcm->voices[0]->number,						   &mix->send_volume[0][0]);		}	}	spin_unlock_irqrestore(&emu->reg_lock, flags);	return change;}static snd_kcontrol_new_t snd_emu10k1_send_volume_control ={	.access =	SNDRV_CTL_ELEM_ACCESS_READWRITE | SNDRV_CTL_ELEM_ACCESS_INACTIVE,	.iface =        SNDRV_CTL_ELEM_IFACE_PCM,	.name =         "EMU10K1 PCM Send Volume",	.count =	32,	.info =         snd_emu10k1_send_volume_info,	.get =          snd_emu10k1_send_volume_get,	.put =          snd_emu10k1_send_volume_put};static int snd_emu10k1_attn_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo){	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;	uinfo->count = 3;	uinfo->value.integer.min = 0;	uinfo->value.integer.max = 0xffff;	return 0;}static int snd_emu10k1_attn_get(snd_kcontrol_t * kcontrol,                                snd_ctl_elem_value_t * ucontrol){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	emu10k1_pcm_mixer_t *mix = &emu->pcm_mixer[snd_ctl_get_ioffidx(kcontrol, &ucontrol->id)];	unsigned long flags;	int idx;	spin_lock_irqsave(&emu->reg_lock, flags);	for (idx = 0; idx < 3; idx++)		ucontrol->value.integer.value[idx] = mix->attn[idx];	spin_unlock_irqrestore(&emu->reg_lock, flags);	return 0;}static int snd_emu10k1_attn_put(snd_kcontrol_t * kcontrol,				snd_ctl_elem_value_t * ucontrol){	unsigned long flags;	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	emu10k1_pcm_mixer_t *mix = &emu->pcm_mixer[snd_ctl_get_ioffidx(kcontrol, &ucontrol->id)];	int change = 0, idx, val;	spin_lock_irqsave(&emu->reg_lock, flags);	for (idx = 0; idx < 3; idx++) {		val = ucontrol->value.integer.value[idx] & 0xffff;		if (mix->attn[idx] != val) {			mix->attn[idx] = val;			change = 1;		}	}	if (change && mix->epcm) {		if (mix->epcm->voices[0] && mix->epcm->voices[1]) {			snd_emu10k1_ptr_write(emu, VTFT_VOLUMETARGET, mix->epcm->voices[0]->number, mix->attn[1]);			snd_emu10k1_ptr_write(emu, VTFT_VOLUMETARGET, mix->epcm->voices[1]->number, mix->attn[2]);		} else if (mix->epcm->voices[0]) {			snd_emu10k1_ptr_write(emu, VTFT_VOLUMETARGET, mix->epcm->voices[0]->number, mix->attn[0]);		}	}	spin_unlock_irqrestore(&emu->reg_lock, flags);	return change;}static snd_kcontrol_new_t snd_emu10k1_attn_control ={	.access =	SNDRV_CTL_ELEM_ACCESS_READWRITE | SNDRV_CTL_ELEM_ACCESS_INACTIVE,	.iface =        SNDRV_CTL_ELEM_IFACE_PCM,	.name =         "EMU10K1 PCM Volume",	.count =	32,	.info =         snd_emu10k1_attn_info,	.get =          snd_emu10k1_attn_get,	.put =          snd_emu10k1_attn_put};/* Mutichannel PCM stream controls */static int snd_emu10k1_efx_send_routing_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo){	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;	uinfo->count = emu->audigy ? 8 : 4;	uinfo->value.integer.min = 0;	uinfo->value.integer.max = emu->audigy ? 0x3f : 0x0f;	return 0;}static int snd_emu10k1_efx_send_routing_get(snd_kcontrol_t * kcontrol,                                        snd_ctl_elem_value_t * ucontrol){	unsigned long flags;	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	emu10k1_pcm_mixer_t *mix = &emu->efx_pcm_mixer[snd_ctl_get_ioffidx(kcontrol, &ucontrol->id)];	int idx;	int num_efx = emu->audigy ? 8 : 4;	int mask = emu->audigy ? 0x3f : 0x0f;	spin_lock_irqsave(&emu->reg_lock, flags);	for (idx = 0; idx < num_efx; idx++)		ucontrol->value.integer.value[idx] = 			mix->send_routing[0][idx] & mask;	spin_unlock_irqrestore(&emu->reg_lock, flags);	return 0;}static int snd_emu10k1_efx_send_routing_put(snd_kcontrol_t * kcontrol,                                        snd_ctl_elem_value_t * ucontrol){	unsigned long flags;	emu10k1_t *emu = snd_kcontrol_chip(kcontrol);	int ch = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);	emu10k1_pcm_mixer_t *mix = &emu->efx_pcm_mixer[ch];	int change = 0, idx, val;	int num_efx = emu->audigy ? 8 : 4;	int mask = emu->audigy ? 0x3f : 0x0f;	spin_lock_irqsave(&emu->reg_lock, flags);	for (idx = 0; idx < num_efx; idx++) {		val = ucontrol->value.integer.value[idx] & mask;		if (mix->send_routing[0][idx] != val) {			mix->send_routing[0][idx] = val;			change = 1;		}	}		if (change && mix->epcm) {		if (mix->epcm->voices[ch]) {			update_emu10k1_fxrt(emu, mix->epcm->voices[ch]->number,

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -