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

📄 p16v.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 3 页
字号:
		table_base[i*2]=runtime->dma_addr+(i*period_size_bytes);		table_base[(i*2)+1]=period_size_bytes<<16;	} 	snd_emu10k1_ptr20_write(emu, PLAYBACK_LIST_ADDR, channel, emu->p16v_buffer.addr+(8*16*channel));	snd_emu10k1_ptr20_write(emu, PLAYBACK_LIST_SIZE, channel, (runtime->periods - 1) << 19);	snd_emu10k1_ptr20_write(emu, PLAYBACK_LIST_PTR, channel, 0);	snd_emu10k1_ptr20_write(emu, PLAYBACK_DMA_ADDR, channel, runtime->dma_addr);	//snd_emu10k1_ptr20_write(emu, PLAYBACK_PERIOD_SIZE, channel, frames_to_bytes(runtime, runtime->period_size)<<16); // buffer size in bytes	snd_emu10k1_ptr20_write(emu, PLAYBACK_PERIOD_SIZE, channel, 0); // buffer size in bytes	snd_emu10k1_ptr20_write(emu, PLAYBACK_POINTER, channel, 0);	snd_emu10k1_ptr20_write(emu, 0x07, channel, 0x0);	snd_emu10k1_ptr20_write(emu, 0x08, channel, 0);	return 0;}/* prepare capture callback */static int snd_p16v_pcm_prepare_capture(snd_pcm_substream_t *substream){	emu10k1_t *emu = snd_pcm_substream_chip(substream);	snd_pcm_runtime_t *runtime = substream->runtime;	int channel = substream->pcm->device - emu->p16v_device_offset;	u32 tmp;	//printk("prepare capture:channel_number=%d, rate=%d, format=0x%x, channels=%d, buffer_size=%ld, period_size=%ld, frames_to_bytes=%d\n",channel, runtime->rate, runtime->format, runtime->channels, runtime->buffer_size, runtime->period_size,  frames_to_bytes(runtime, 1));	tmp = snd_emu10k1_ptr_read(emu, A_SPDIF_SAMPLERATE, channel);        switch (runtime->rate) {	case 44100:	  snd_emu10k1_ptr_write(emu, A_SPDIF_SAMPLERATE, channel, (tmp & ~0x0e00) | 0x0800);	  break;	case 96000:	  snd_emu10k1_ptr_write(emu, A_SPDIF_SAMPLERATE, channel, (tmp & ~0x0e00) | 0x0400);	  break;	case 192000:	  snd_emu10k1_ptr_write(emu, A_SPDIF_SAMPLERATE, channel, (tmp & ~0x0e00) | 0x0200);	  break;	case 48000:	default:	  snd_emu10k1_ptr_write(emu, A_SPDIF_SAMPLERATE, channel, (tmp & ~0x0e00) | 0x0000);	  break;	}	/* FIXME: Check emu->buffer.size before actually writing to it. */	snd_emu10k1_ptr20_write(emu, 0x13, channel, 0);	snd_emu10k1_ptr20_write(emu, CAPTURE_DMA_ADDR, channel, runtime->dma_addr);	snd_emu10k1_ptr20_write(emu, CAPTURE_BUFFER_SIZE, channel, frames_to_bytes(runtime, runtime->buffer_size)<<16); // buffer size in bytes	snd_emu10k1_ptr20_write(emu, CAPTURE_POINTER, channel, 0);	//snd_emu10k1_ptr20_write(emu, CAPTURE_SOURCE, 0x0, 0x333300e4); /* Select MIC or Line in */	//snd_emu10k1_ptr20_write(emu, EXTENDED_INT_MASK, 0, snd_emu10k1_ptr20_read(emu, EXTENDED_INT_MASK, 0) | (0x110000<<channel));	return 0;}static void snd_p16v_intr_enable(emu10k1_t *emu, unsigned int intrenb){	unsigned long flags;	unsigned int enable;	spin_lock_irqsave(&emu->emu_lock, flags);	enable = inl(emu->port + INTE2) | intrenb;	outl(enable, emu->port + INTE2);	spin_unlock_irqrestore(&emu->emu_lock, flags);}static void snd_p16v_intr_disable(emu10k1_t *emu, unsigned int intrenb){	unsigned long flags;	unsigned int disable;	spin_lock_irqsave(&emu->emu_lock, flags);	disable = inl(emu->port + INTE2) & (~intrenb);	outl(disable, emu->port + INTE2);	spin_unlock_irqrestore(&emu->emu_lock, flags);}/* trigger_playback callback */static int snd_p16v_pcm_trigger_playback(snd_pcm_substream_t *substream,				    int cmd){	emu10k1_t *emu = snd_pcm_substream_chip(substream);	snd_pcm_runtime_t *runtime;	emu10k1_pcm_t *epcm;	int channel;	int result = 0;	struct list_head *pos;        snd_pcm_substream_t *s;	u32 basic = 0;	u32 inte = 0;	int running=0;	switch (cmd) {	case SNDRV_PCM_TRIGGER_START:		running=1;		break;	case SNDRV_PCM_TRIGGER_STOP:	default:		running=0;		break;	}        snd_pcm_group_for_each(pos, substream) {                s = snd_pcm_group_substream_entry(pos);		runtime = s->runtime;		epcm = runtime->private_data;		channel = substream->pcm->device-emu->p16v_device_offset;		//snd_printk("p16v channel=%d\n",channel);		epcm->running = running;		basic |= (0x1<<channel);		inte |= (INTE2_PLAYBACK_CH_0_LOOP<<channel);                snd_pcm_trigger_done(s, substream);        }	//snd_printk("basic=0x%x, inte=0x%x\n",basic, inte);	switch (cmd) {	case SNDRV_PCM_TRIGGER_START:		snd_p16v_intr_enable(emu, inte);		snd_emu10k1_ptr20_write(emu, BASIC_INTERRUPT, 0, snd_emu10k1_ptr20_read(emu, BASIC_INTERRUPT, 0)| (basic));		break;	case SNDRV_PCM_TRIGGER_STOP:		snd_emu10k1_ptr20_write(emu, BASIC_INTERRUPT, 0, snd_emu10k1_ptr20_read(emu, BASIC_INTERRUPT, 0) & ~(basic));		snd_p16v_intr_disable(emu, inte);		break;	default:		result = -EINVAL;		break;	}	return result;}/* trigger_capture callback */static int snd_p16v_pcm_trigger_capture(snd_pcm_substream_t *substream,                                   int cmd){	emu10k1_t *emu = snd_pcm_substream_chip(substream);	snd_pcm_runtime_t *runtime = substream->runtime;	emu10k1_pcm_t *epcm = runtime->private_data;	int channel = 0;	int result = 0;	u32 inte = INTE2_CAPTURE_CH_0_LOOP | INTE2_CAPTURE_CH_0_HALF_LOOP;	switch (cmd) {	case SNDRV_PCM_TRIGGER_START:		snd_p16v_intr_enable(emu, inte);		snd_emu10k1_ptr20_write(emu, BASIC_INTERRUPT, 0, snd_emu10k1_ptr20_read(emu, BASIC_INTERRUPT, 0)|(0x100<<channel));		epcm->running = 1;		break;	case SNDRV_PCM_TRIGGER_STOP:		snd_emu10k1_ptr20_write(emu, BASIC_INTERRUPT, 0, snd_emu10k1_ptr20_read(emu, BASIC_INTERRUPT, 0) & ~(0x100<<channel));		snd_p16v_intr_disable(emu, inte);		//snd_emu10k1_ptr20_write(emu, EXTENDED_INT_MASK, 0, snd_emu10k1_ptr20_read(emu, EXTENDED_INT_MASK, 0) & ~(0x110000<<channel));		epcm->running = 0;		break;	default:		result = -EINVAL;		break;	}	return result;}/* pointer_playback callback */static snd_pcm_uframes_tsnd_p16v_pcm_pointer_playback(snd_pcm_substream_t *substream){	emu10k1_t *emu = snd_pcm_substream_chip(substream);	snd_pcm_runtime_t *runtime = substream->runtime;	emu10k1_pcm_t *epcm = runtime->private_data;	snd_pcm_uframes_t ptr, ptr1, ptr2,ptr3,ptr4 = 0;	int channel = substream->pcm->device - emu->p16v_device_offset;	if (!epcm->running)		return 0;	ptr3 = snd_emu10k1_ptr20_read(emu, PLAYBACK_LIST_PTR, channel);	ptr1 = snd_emu10k1_ptr20_read(emu, PLAYBACK_POINTER, channel);	ptr4 = snd_emu10k1_ptr20_read(emu, PLAYBACK_LIST_PTR, channel);	if (ptr3 != ptr4) ptr1 = snd_emu10k1_ptr20_read(emu, PLAYBACK_POINTER, channel);	ptr2 = bytes_to_frames(runtime, ptr1);	ptr2+= (ptr4 >> 3) * runtime->period_size;	ptr=ptr2;        if (ptr >= runtime->buffer_size)		ptr -= runtime->buffer_size;	return ptr;}/* pointer_capture callback */static snd_pcm_uframes_tsnd_p16v_pcm_pointer_capture(snd_pcm_substream_t *substream){	emu10k1_t *emu = snd_pcm_substream_chip(substream);	snd_pcm_runtime_t *runtime = substream->runtime;	emu10k1_pcm_t *epcm = runtime->private_data;	snd_pcm_uframes_t ptr, ptr1, ptr2 = 0;	int channel = 0;	if (!epcm->running)		return 0;	ptr1 = snd_emu10k1_ptr20_read(emu, CAPTURE_POINTER, channel);	ptr2 = bytes_to_frames(runtime, ptr1);	ptr=ptr2;	if (ptr >= runtime->buffer_size) {		ptr -= runtime->buffer_size;		printk(KERN_WARNING "buffer capture limited!\n");	}	//printk("ptr1 = 0x%lx, ptr2=0x%lx, ptr=0x%lx, buffer_size = 0x%x, period_size = 0x%x, bits=%d, rate=%d\n", ptr1, ptr2, ptr, (int)runtime->buffer_size, (int)runtime->period_size, (int)runtime->frame_bits, (int)runtime->rate);	return ptr;}/* operators */static snd_pcm_ops_t snd_p16v_playback_front_ops = {	.open =        snd_p16v_pcm_open_playback_front,	.close =       snd_p16v_pcm_close_playback,	.ioctl =       snd_pcm_lib_ioctl,	.hw_params =   snd_p16v_pcm_hw_params_playback,	.hw_free =     snd_p16v_pcm_hw_free_playback,	.prepare =     snd_p16v_pcm_prepare_playback,	.trigger =     snd_p16v_pcm_trigger_playback,	.pointer =     snd_p16v_pcm_pointer_playback,};static snd_pcm_ops_t snd_p16v_capture_ops = {	.open =        snd_p16v_pcm_open_capture,	.close =       snd_p16v_pcm_close_capture,	.ioctl =       snd_pcm_lib_ioctl,	.hw_params =   snd_p16v_pcm_hw_params_capture,	.hw_free =     snd_p16v_pcm_hw_free_capture,	.prepare =     snd_p16v_pcm_prepare_capture,	.trigger =     snd_p16v_pcm_trigger_capture,	.pointer =     snd_p16v_pcm_pointer_capture,};int snd_p16v_free(emu10k1_t *chip){	// release the data	if (chip->p16v_buffer.area) {		snd_dma_free_pages(&chip->p16v_buffer);		//snd_printk("period lables free: %p\n", &chip->p16v_buffer);	}	return 0;}static void snd_p16v_pcm_free(snd_pcm_t *pcm){	emu10k1_t *emu = pcm->private_data;	//snd_printk("snd_p16v_pcm_free pcm: called\n");	snd_pcm_lib_preallocate_free_for_all(pcm);	emu->pcm = NULL;}int snd_p16v_pcm(emu10k1_t *emu, int device, snd_pcm_t **rpcm){	snd_pcm_t *pcm;	snd_pcm_substream_t *substream;	int err;        int capture=1;  	//snd_printk("snd_p16v_pcm called. device=%d\n", device);	emu->p16v_device_offset = device;	if (rpcm)		*rpcm = NULL;	if ((err = snd_pcm_new(emu->card, "p16v", device, 1, capture, &pcm)) < 0)		return err;  	pcm->private_data = emu;	pcm->private_free = snd_p16v_pcm_free;	// Single playback 8 channel device.	// Single capture 2 channel device.	snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_p16v_playback_front_ops);	snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &snd_p16v_capture_ops);	pcm->info_flags = 0;	pcm->dev_subclass = SNDRV_PCM_SUBCLASS_GENERIC_MIX;	strcpy(pcm->name, "p16v");	emu->pcm = pcm;	for(substream = pcm->streams[SNDRV_PCM_STREAM_PLAYBACK].substream; 	    substream; 	    substream = substream->next) {		if ((err = snd_pcm_lib_preallocate_pages(substream, 							 SNDRV_DMA_TYPE_DEV, 							 snd_dma_pci_data(emu->pci), 							 ((65536 - 64) * 8), ((65536 - 64) * 8))) < 0) 			return err;		//snd_printk("preallocate playback substream: err=%d\n", err);	}	for (substream = pcm->streams[SNDRV_PCM_STREAM_CAPTURE].substream; 	      substream; 	      substream = substream->next) { 		if ((err = snd_pcm_lib_preallocate_pages(substream, 	                                           SNDRV_DMA_TYPE_DEV, 	                                           snd_dma_pci_data(emu->pci), 	                                           65536 - 64, 65536 - 64)) < 0)			return err;		//snd_printk("preallocate capture substream: err=%d\n", err);	}  	if (rpcm)		*rpcm = pcm;  	return 0;}static int snd_p16v_volume_info(snd_kcontrol_t *kcontrol, snd_ctl_elem_info_t * uinfo){        uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;        uinfo->count = 2;        uinfo->value.integer.min = 0;        uinfo->value.integer.max = 255;        return 0;}static int snd_p16v_volume_get(snd_kcontrol_t * kcontrol,                                       snd_ctl_elem_value_t * ucontrol, int reg, int high_low){        emu10k1_t *emu = snd_kcontrol_chip(kcontrol);        u32 value;        value = snd_emu10k1_ptr20_read(emu, reg, high_low);	if (high_low == 1) {        	ucontrol->value.integer.value[0] = 0xff - ((value >> 24) & 0xff); /* Left */        	ucontrol->value.integer.value[1] = 0xff - ((value >> 16) & 0xff); /* Right */	} else {        	ucontrol->value.integer.value[0] = 0xff - ((value >> 8) & 0xff); /* Left */        	ucontrol->value.integer.value[1] = 0xff - ((value >> 0) & 0xff); /* Right */	}        return 0;}static int snd_p16v_volume_get_spdif_front(snd_kcontrol_t * kcontrol,                                       snd_ctl_elem_value_t * ucontrol){	int high_low = 0;	int reg = PLAYBACK_VOLUME_MIXER7;        return snd_p16v_volume_get(kcontrol, ucontrol, reg, high_low);}static int snd_p16v_volume_get_spdif_center_lfe(snd_kcontrol_t * kcontrol,                                       snd_ctl_elem_value_t * ucontrol){	int high_low = 1;	int reg = PLAYBACK_VOLUME_MIXER7;        return snd_p16v_volume_get(kcontrol, ucontrol, reg, high_low);}static int snd_p16v_volume_get_spdif_unknown(snd_kcontrol_t * kcontrol,                                       snd_ctl_elem_value_t * ucontrol){

⌨️ 快捷键说明

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