当前位置:首页 > 单片机 > 单片机
[导读]/*** 有待学习 *****/ Linux 下的IIS 音频驱动程序主要都在/kernel/drivers/sound/s3c2410-uda1341.c 文件中。 在音频驱动程序中有2个比较重要的结构体:typedef struct {int size; /* buffer size */char *start; /

/*** 有待学习 *****/

Linux 下的IIS 音频驱动程序主要都在/kernel/drivers/sound/s3c2410-uda1341.c 文件中。

在音频驱动程序中有2个比较重要的结构体:

typedef struct {

int size; /* buffer size */

char *start; /* point to actual buffer */(内存虚拟地址起始地址)

dma_addr_t dma_addr; /* physical buffer address */(内存物理地址起始地址)

struct semaphore sem; /* down before touching the buffer */

int master; /* owner for buffer allocation, contain size when true */(内存大小)

} audio_buf_t;

typedef struct {

audio_buf_t *buffers; /* pointer to audio buffer structures */

audio_buf_t *buf; /* current buffer used by read/write */

u_int buf_idx; /* index for the pointer above */

u_int fragsize; /* fragment i.e. buffer size */(音频缓冲区片大小)

u_int nbfrags; /* nbr of fragments */(音频缓冲区片数量)

dmach_t dma_ch; /* DMA channel (channel2 for audio) */

} 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。

音频驱动的file_operations 结构定义如下:

static struct file_operations smdk2410_audio_fops = {

llseek: smdk2410_audio_llseek,

write: smdk2410_audio_write,

read: smdk2410_audio_read,

poll: smdk2410_audio_poll,

ioctl: smdk2410_audio_ioctl,

open: smdk2410_audio_open,

release: smdk2410_audio_release

};

static struct file_operations smdk2410_mixer_fops = {

ioctl: smdk2410_mixer_ioctl,

open: smdk2410_mixer_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 文件中:

/* For spinlocks etc */

#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 文件中:

/*

* Save the current interrupt enable state & disable IRQs

*/

#define __save_flags_cli(x)

do {

unsigned long temp;

__asm__ __volatile__(

" mov %0, pc @ save_flags_clin"

" orr %1, %0, #0x08000000n"

" and %0, %0, #0x0c000000n"

" teqp %1, #0n"

: "=r" (x), "=r" (temp)

:

: "memory");

} while (0)

最后用ARM 汇编指令实现了保存IRQ 和FIQ 的中断使能状态,并禁止IRQ 中断。

/*

* restore saved IRQ & FIQ state

*/

#define __restore_flags(x)

do {

unsigned long temp;

__asm__ __volatile__(

" mov %0, pc @ restore_flagsn"

" bic %0, %0, #0x0c000000n"

" orr %0, %0, %1n"

" teqp %0, #0n"

: "=&r" (temp)

: "r" (x)

: "memory");

} while (0)

最后用ARM 汇编指令实现了恢复IRQ 和FIQ 的中断使能状态。

/* GPB 4: L3CLOCK, OUTPUT */

set_gpio_ctrl(GPIO_L3CLOCK);

/* GPB 3: L3DATA, OUTPUT */

set_gpio_ctrl(GPIO_L3DATA);

/* GPB 2: L3MODE, OUTPUT */

set_gpio_ctrl(GPIO_L3MODE);

/* GPE 3: I2SSDI */

set_gpio_ctrl(GPIO_E3 | GPIO_PULLUP_EN | GPIO_MODE_I2SSDI);

/* GPE 0: I2SLRCK */

set_gpio_ctrl(GPIO_E0 | GPIO_PULLUP_EN | GPIO_MODE_I2SSDI);

/* GPE 1: I2SSCLK */

set_gpio_ctrl(GPIO_E1 | GPIO_PULLUP_EN | GPIO_MODE_I2SSCLK);

/* GPE 2: CDCLK */

set_gpio_ctrl(GPIO_E2 | GPIO_PULLUP_EN | GPIO_MODE_CDCLK);

/* GPE 4: I2SSDO */

set_gpio_ctrl(GPIO_E4 | GPIO_PULLUP_EN | GPIO_MODE_I2SSDO);

接下来马上设置与UDA1341 芯片相关GPIO 引脚。这里首先将GPB4,GPB3,GPB2 这3个GPIO 引脚设置为输出模式,参考原理图后,得知这3个引脚分别连接UDA1341 芯片的L3CLOCK,L3DATA,L3MODE 这3个引脚,作为这3个信号的输入。

在/kernel/drivers/sound/s3c2410-uda1341.c 文件中:

#define GPIO_L3CLOCK (GPIO_MODE_OUT | GPIO_PULLUP_DIS | GPIO_B4)

#define GPIO_L3DATA (GPIO_MODE_OUT | GPIO_PULLUP_DIS | GPIO_B3)

#define GPIO_L3MODE (GPIO_MODE_OUT | GPIO_PULLUP_DIS | GPIO_B2)

然后继续设置与IIS 控制器输出信号相关GPIO 引脚。将GPE0~GPE4 这5个引脚设置为IIS 接口的信号模式。需要通过配置GPECON 寄存器来设定该端口管脚的输出模式,对应位如下:

[9:8] [7:6] [5:4] [3:2] [1:0]

GPE4 GPE3 GPE2 GPE1 GPE0

参考S3C2410 芯片datasheet 的I/O口章节,都要设为10(二进制)。

local_irq_restore(flags);

设置完GPIO 口的工作模式,就可以前面已经分析过的local_irq_restore 宏函数来恢复IRQ 和FIQ 的中断使能状态。

init_uda1341();

这里调用了init_uda1341 函数来初始化UDA1341 芯片,该函数会在后面说明。

output_stream.dma_ch = DMA_CH2;

if (audio_init_dma(&output_stream, "UDA1341 out")) {

audio_clear_dma(&output_stream);

printk( KERN_WARNING AUDIO_NAME_VERBOSE

": unable to get DMA channelsn" );

return -EBUSY;

}

input_stream.dma_ch = DMA_CH1;

if (audio_init_dma(&input_stream, "UDA1341 in")) {

audio_clear_dma(&input_stream);

printk( KERN_WARNING AUDIO_NAME_VERBOSE

": unable to get DMA channelsn" );

return -EBUSY;

}

在全局变量中定义了,两个audio_stream_t 结构的变量,分别是output_stream 和input_stream,一个作为输出音频缓冲区,一个作为输入音频缓冲区。

将输出音频缓冲区的DMA 通道设为通道2,输入音频缓冲区的DMA 通道设为通道1。

在/kernel/include/asm-arm/arch-s3c2410/dma.h 文件中:

#define DMA_CH0 0

#define DMA_CH1 1

#define DMA_CH2 2

#define DMA_CH3 3

通过查阅S3C2410 芯片datasheet 中的DMA 章节,知道该芯片共有4个DMA 通道,DMA 控制器的每个通道可以从4个DMA 源中选择一个DMA 请求源。其中,通道1具有IIS 输入源,而通道2具有IIS 输出和输入源。所以要以全双工模式进行音频数据传输的话,只有将输出音频缓冲区的设为DMA 通道2,输入音频缓冲区设为DMA 通道1。

接着调用2次audio_init_dma 函数来分别对输出和输入音频缓冲区的DMA 通道进行初始化设置。该函数比较简单,定义如下:

static int __init audio_init_dma(audio_stream_t * s, char *desc)

{

if(s->dma_ch == DMA_CH2)

return s3c2410_request_dma("I2SSDO", s->dma_ch, audio_dmaout_done_callback, NULL);

else if(s->dma_ch == DMA_CH1)

return s3c2410_request_dma("I2SSDI", s->dma_ch, NULL ,audio_dmain_done_callback);

else

return 1;

}

这个函数其实就是对DMA 的通道号进行判断,然后调用了s3c2410_request_dma 函数来向内核申请一个DMA 通道。

在/kernel/arch/arm/mach-s3c2410/dma.c 文件中:

int s3c2410_request_dma(const char *device_id, dmach_t channel,

dma_callback_t write_cb, dma_callback_t read_cb)

在该函数中会分配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,

device_id, (void *)dma);

即申请了一个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 =

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭