在音频驱动程序中有2个比较重要的结构体:
typedef struct {
int size;
char *start;
dma_addr_t dma_addr; (内存物理地址起始地址)
struct semaphore sem;
int master;
} audio_buf_t;
typedef struct {
audio_buf_t *buffers;
audio_buf_t *buf;
u_int buf_idx;
u_int fragsize;
u_int nbfrags;
dmach_t dma_ch;
} audio_stream_t;
这是一个管理多缓冲区的结构体,结构体audio_stream_t 为音频流数据组成了一个环形缓冲区。(audio_buf_t *buffers 同触摸屏驱动中struct TS_DEV 结构中的TS_RET buf[MAX_TS_BUF] 意义一样,都为环形缓冲区)用audio_buf_t 来管理一段内存,在用audio_stream_t 来管理N 个audio_buf_t。
static struct file_operations smdk2410_audio_fops = {
llseek:
write:
read:
poll:
ioctl:
open:
release: smdk2410_audio_release
};
static struct file_operations smdk2410_mixer_fops = {
ioctl:
open:
release: smdk2410_mixer_release
};
这里定义了两种类型设备的file_operations 结构,前者是DSP 设备,后者是混频器设备。
------------------------------------------------------------------------
int __init s3c2410_uda1341_init(void)
该函数首先会初始化I/O 和UDA1341 芯片,然后申请2个DMA 通道用于音频传输。
local_irq_save(flags);
调用该宏函数来保存IRQ 中断使能状态,并禁止IRQ 中断。
在/kernel/include/asm-arm/system.h 文件中:
#define local_irq_save(x) __save_flags_cli(x)
#define local_irq_restore(x) __restore_flags(x)
在/kernel/include/asm-arm/proc-armo/system.h 文件中:
#define __save_flags_cli(x)
do {
" mov %0, pc
" orr %1, %0, #0x08000000\n"
" and %0, %0, #0x0c000000\n"
" teqp %1, #0\n"
} while (0)
最后用ARM 汇编指令实现了保存IRQ 和FIQ 的中断使能状态,并禁止IRQ 中断。
#define __restore_flags(x)
do {
" mov %0, pc
" bic %0, %0, #0x0c000000\n"
" orr %0, %0, %1\n"
" teqp %0, #0\n"
} while (0)
最后用ARM 汇编指令实现了恢复IRQ 和FIQ 的中断使能状态。
set_gpio_ctrl(GPIO_L3CLOCK);
set_gpio_ctrl(GPIO_L3DATA);
set_gpio_ctrl(GPIO_L3MODE);
set_gpio_ctrl(GPIO_E3 | GPIO_PULLUP_EN | GPIO_MODE_I2SSDI);
set_gpio_ctrl(GPIO_E0 | GPIO_PULLUP_EN | GPIO_MODE_I2SSDI);
set_gpio_ctrl(GPIO_E1 | GPIO_PULLUP_EN | GPIO_MODE_I2SSCLK);
set_gpio_ctrl(GPIO_E2 | GPIO_PULLUP_EN | GPIO_MODE_CDCLK);
set_gpio_ctrl(GPIO_E4 | GPIO_PULLUP_EN | GPIO_MODE_I2SSDO);
在/kernel/drivers/sound/s3c2410-uda1341.c 文件中:
#define GPIO_L3CLOCK
#define GPIO_L3DATA
#define GPIO_L3MODE
[9:8] [7:6] [5:4] [3:2] [1:0]
GPE4
参考S3C2410 芯片datasheet 的I/O口章节,都要设为10(二进制)。
local_irq_restore(flags);
init_uda1341();
output_stream.dma_ch = DMA_CH2;
if (audio_init_dma(&output_stream, "UDA1341 out")) {
}
input_stream.dma_ch = DMA_CH1;
在/kernel/include/asm-arm/arch-s3c2410/dma.h 文件中:
#define DMA_CH0
#define DMA_CH1
#define DMA_CH2
#define DMA_CH3
通过查阅S3C2410 芯片datasheet 中的DMA 章节,知道该芯片共有4个DMA 通道,DMA 控制器的每个通道可以从4个DMA 源中选择一个DMA 请求源。其中,通道1具有IIS 输入源,而通道2具有IIS 输出和输入源。所以要以全双工模式进行音频数据传输的话,只有将输出音频缓冲区的设为DMA 通道2,输入音频缓冲区设为DMA 通道1。
static int __init audio_init_dma(audio_stream_t * s, char *desc)
{
if(s->dma_ch == DMA_CH2)
else if(s->dma_ch == DMA_CH1)
else
}
在/kernel/arch/arm/mach-s3c2410/dma.c 文件中:
int s3c2410_request_dma(const char *device_id, dmach_t channel,
在该函数中会分配DMA 通道,并申请DMA 中断,即当DMA 传输结束时,会响应中断请求,调用回调函数。这里的参数中,device_id 为设备id 号,用字符串来表示;channel 为DMA 通道号,将前面定义的通道号1,2传入;write_cb 和read_cb 分别指向DMA 发送和读取结束时调用的函数,即DMA 传输结束时调用的回调函数。
在该函数中有:
err = request_irq(dma->irq, dma_irq_handler, 0 * SA_INTERRUPT,
即申请了一个DMA 的中断号,中断处理子程序为dma_irq_handler 函数,然后:
dma->write.callback = write_cb;
dma->read.callback = read_cb;
将读写DMA 中断的两个回调函数指针传入。
在/kernel/arch/arm/mach-s3c2410/dma.c 文件中:
static void dma_irq_handler(int irq, void *dev_id, struct pt_regs *regs)
{
s3c2410_dma_t *dma = (s3c2410_dma_t *)dev_id;
DPRINTK(__FUNCTION__"\n");
s3c2410_dma_done(dma);
}
在中断处理子程序中,调用了s3c2410_dma_done 函数,该函数定义如下:
static inline void s3c2410_dma_done(s3c2410_dma_t *dma)
{
dma_buf_t *buf = dma->curr;
dma_callback_t callback;
if (buf->write) callback = dma->write.callback;
else callback = dma->read.callback;
#ifdef HOOK_LOST_INT
stop_dma_timer();
#endif
DPRINTK("IRQ: b=%#x st=%ld\n", (int)buf->id, (long)dma->regs->DSTAT);
if (callback)
kfree(buf);
dma->active = 0;
process_dma(dma);
}
最后在s3c2410_dma_done 函数中,通过callback 函数指针调用了DMA 发送和读取的回调函数。
在/kernel/arch/arm/mach-s3c2410/dma.c 文件中:
void s3c2410_free_dma(dmach_t channel)
该函数中释放了已申请的DMA 通道,并调用了free_irq 函数来释放已分配的DMA 发送和读取结束的中断号。
audio_dev_dsp = register_sound_dsp(&smdk2410_audio_fops, -1);
audio_dev_mixer = register_sound_mixer(&smdk2410_mixer_fops, -1);
在/kernel/drivers/sound/sound_core.c 文件中:
int register_sound_dsp(struct file_operations *fops, int dev)
int register_sound_mixer(struct file_operations *fops, int dev)
这两个函数的参数一样,fops 为传给内核的file_operations 结构中的接口函数,dev 为分配的设备序号,设为-1 表示由内核自动分配一个空闲的序号。
static void in 接着来看uda1341_l3_data 函数:
local_irq_save(flags);
write_gpio_bit(GPIO_L3MODE, 1);
udelay(1);
write_gpio_bit(GPIO_L3MODE, 0);
udelay(1);
write_gpio_bit(GPIO_L3MODE, 1);
for (i = 0; i < 8; i++) {
}
write_gpio_bit(GPIO_L3MODE, 1);
write_gpio_bit(GPIO_L3MODE, 0);
udelay(1);
write_gpio_bit(GPIO_L3MODE, 1);
local_irq_restore(flags);
------------------------------------------------------------------------
void __exit s3c2410_uda1341_exit(void)
unregister_sound_dsp(audio_dev_dsp);
unregister_sound_mixer(audio_dev_mixer);
在/kernel/drivers/sound/sound_core.c 文件中:
void unregister_sound_dsp(int unit)
void unregister_sound_mixer(int unit)
这两个函数的参数一样,为刚才调用注册函数时返回的内核所分配的设备序号。
audio_clear_dma(&output_stream);
audio_clear_dma(&input_stream);
------------------------------------------------------------------------
static int smdk2410_audio_open(struct inode *inode, struct file *file)
if ((file->f_flags & O_ACCMODE) == O_RDONLY) {
} else if ((file->f_flags & O_ACCMODE) == O_WRONLY) {
} else if ((file->f_flags & O_ACCMODE) == O_RDWR) {
} else
if (cold) {
}
MOD_INC_USE_COUNT;
------------------------------------------------------------------------
static void init_s3c2410_iis_bus_tx(void)
IISPSR = (IISPSR_A(iispsr_value(S_CLOCK_FREQ, 44100))
IISPSR_A(iispsr_value(S_CLOCK_FREQ, 44100)) = IISPSR_A(iispsr_value(384, 44100)) = (一个0~31 之间的值)<<5 预分频控制器A,用于内部时钟块
IISPSR_B(iispsr_value(S_CLOCK_FREQ, 44100))) = (一个0~31 之间的值)<<0 预分频控制器B,用于外部时钟块
uda1341_volume = 62 - ((DEF_VOLUME * 61) / 100);
uda1341_boost = 0;
uda_sampling = DATA2_DEEMP_NONE;
uda_sampling &= ~(DATA2_MUTE);
local_irq_save(flags);
write_gpio_bit(GPIO_L3MODE, 1);
write_gpio_bit(GPIO_L3CLOCK, 1);
local_irq_restore(flags);
/
uda1341_l3_data(EXTADDR(EXT2));
uda1341_l3_data(EXTDATA(EXT2_MIC_GAIN(0x6)) | EXT2_MIXMODE_CH1);
【其实这里的“uda1341_l3_data(uda_sampling); ”,这句话应该是不正确的,不是准备再将音量调到最大。应该改为:
uda_l3_data(DATA2 | uda_sampling);
即用10000000(二进制)参数设置静音关闭和高低音模式为flat 模式(高低音增益都为0dB)等。】
------------------------------------------------------------------------
static void uda1341_l3_address(u8 data)
local_irq_save(flags);
write_gpio_bit(GPIO_L3MODE, 0);
write_gpio_bit(GPIO_L3DATA, 0);
write_gpio_bit(GPIO_L3CLOCK, 1);
udelay(1);
在/kernel/include/asm-arm/delay.h 文件中:
extern void udelay(unsigned long usecs);
在/kernel/include/linux/delay.h 文件中:
#ifdef notdef
#define mdelay(n) (\
{unsigned long msec=(n); while (msec--) udelay(1000);})
#else
#define mdelay(n) (\
(__builtin_constant_p(n) && (n)<=MAX_UDELAY_MS) ? udelay((n)*1000) : \
({unsigned long msec=(n); while (msec--) udelay(1000);}))
#endif
在/kernel/arch/arm/lib/delay.S 文件中:
ENTRY(udelay)
最后用ARM 汇编指令实现了微秒级的短暂延时。
for (i = 0; i < 8; i++) {
}
write_gpio_bit(GPIO_L3MODE, 1);
udelay(1);
local_irq_restore(flags);