一致术语
1. JZ 君正4760b mips
2. Audio buff 便是整个list_head+ list_node *4 + 整个音频区
3. APP 应用程序
全体运用框图
Buff分配流程
首要在probe JZ mixer设备的时分进行初始化DMA及缓冲buff. 在函数init_jz_i2s中调用audio_init_endpoint进行实始化。
fragsize= JZCODEC_RW_BUFFER_SIZE * PAGE_SIZE;
fragstotal= JZCODEC_RW_BUFFER_TOTAL;
audio_init_endpoint(&out_endpoint,fragsize, fragstotal);
audio_init_endpoint(&in_endpoint,fragsize, fragstotal);
fragsize= JZCODEC_RW_BUFFER_SIZE * PAGE_SIZE;fragstotal= JZCODEC_RW_BUFFER_TOTAL;audio_init_endpoint(&out_endpoint,fragsize, fragstotal);audio_init_endpoint(&in_endpoint,fragsize, fragstotal);
进入audio_init_endpoint函数,在这儿先表一下in_endpoint和out_ endpoint结构体, 初始化为0.
static audio_pipe out_endpoint = {
.mem = 0,
.savenode = 0,
.fragsize = 0,
.fragstotal = 0,
.trans_state = 0,
};
staTIc audio_pipe in_endpoint= {
.mem = 0,
.savenode = 0,
.fragsize = 0,
.fragstotal = 0,
.trans_state = 0,
};
staTIc audio_pipe out_endpoint = {.mem = 0,.savenode = 0,.fragsize = 0,.fragstotal = 0,.trans_state = 0,};staTIc audio_pipe in_endpoint= {.mem = 0,.savenode = 0,.fragsize = 0,.fragstotal = 0,.trans_state = 0,};
void audio_init_endpoint(audio_pipe*endpoint, unsigned int pagesize,
unsigned int count)
{
audio_resizemem_endpoint(endpoint,pagesize, count);
spin_lock_init(&endpoint->lock);
init_waitqueue_head(&endpoint->q_full);
endpoint->avialable_couter= 0;
endpoint->filter= NULL;
if(endpoint == &in_endpoint) {
init_audio_audiodma(endpoint,CODEC_RMODE);
//INIT_WORK(&endpoint->work, audio_in_endpoint_work);
endpoint->handle= handle_in_endpoint_work;
}
if(endpoint == &out_endpoint) {
init_audio_audiodma(endpoint,CODEC_WMODE);
//INIT_WORK(&endpoint->work, audio_out_endpoint_work);
endpoint->handle= handle_out_endpoint_work;
}
}
int audio_resizemem_endpoint(audio_pipe*endpoint, unsigned int pagesize,
unsigned int count)
{
intret = init_audio_node(&endpoint->mem, pagesize, count,(int*)&endpoint->
fragmem_start);
if(ret) {
endpoint->fragsize= pagesize;
endpoint->fragstotal= count;
endpoint->memsize= ret;
}
returnret;
}
void audio_init_endpoint(audio_pipe*endpoint, unsigned int pagesize,unsigned int count){audio_resizemem_endpoint(endpoint,pagesize, count);spin_lock_init(&endpoint->lock);init_waitqueue_head(&endpoint->q_full);endpoint->avialable_couter= 0;endpoint->filter= NULL;if(endpoint == &in_endpoint) {init_audio_audiodma(endpoint,CODEC_RMODE);//INIT_WORK(&endpoint->work, audio_in_endpoint_work);endpoint->handle= handle_in_endpoint_work;}if(endpoint == &out_endpoint) {init_audio_audiodma(endpoint,CODEC_WMODE);//INIT_WORK(&endpoint->work, audio_out_endpoint_work);endpoint->handle= handle_out_endpoint_work;}}int audio_resizemem_endpoint(audio_pipe*endpoint, unsigned int pagesize,unsigned int count){intret = init_audio_node(&endpoint->mem, pagesize, count,(int*)&endpoint->fragmem_start);if(ret) {endpoint->fragsize= pagesize;endpoint->fragstotal= count;endpoint->memsize= ret;}returnret;}
咱们进入audio_resizemem_endpoint剖析内存的分配, 其间调用init_audio_node进行本质的
分配, 先介绍一下这儿的变量含义,了解了这些含义,剖析起来就称心如意了,刻不容缓
了。
JZCODEC_RW_BUFFER_SIZE // 1 * 4096
JZCODEC_RW_BUFFER_TOTAL; // 4个buff
unsignedint fact; //分配物理页的order
audio_node *pbuff; //代表1个audio node
audio_head *phead; //代表整个audiobuff 链表
unsignedint *mem; //分配得到整个audiobuff的虚拟地址
structlist_head *audio_wfree; //freebuff 链表
structlist_head *audio_wuse; //usebuff 链表
int memsize; //链表头+节点+audiobuff占用总空间
int datasize; //audiobuff占用的总空间
int headlistsize; //链表头+4个节点占用总空间
JZCODEC_RW_BUFFER_SIZE // 1 * 4096JZCODEC_RW_BUFFER_TOTAL; // 4个buffunsignedint fact; //分配物理页的orderaudio_node *pbuff; //代表1个audio nodeaudio_head *phead; //代表整个audiobuff 链表unsignedint *mem; //分配得到整个audiobuff的虚拟地址structlist_head *audio_wfree; //freebuff 链表structlist_head *audio_wuse; //usebuff 链表int memsize; //链表头+节点+audiobuff占用总空间int datasize; //audiobuff占用的总空间int headlistsize; //链表头+4个节点占用总空间
//Alloc memory first, to avail fail
datasize = ALIGN_PAGE_SIZE(pagesize * count);
headlistsize = ALIGN_PAGE_SIZE(count *sizeof(audio_node) + sizeof(audio_head));
memsize = headlistsize + datasize;
fact = get_order(memsize);
mem= (unsigned int *)__get_free_pages(GFP_KERNEL | GFP_DMA, fact);
//Alloc memory first, to avail faildatasize = ALIGN_PAGE_SIZE(pagesize * count);headlistsize = ALIGN_PAGE_SIZE(count *sizeof(audio_node) + sizeof(audio_head));memsize = headlistsize + datasize;fact = get_order(memsize);mem= (unsigned int *)__get_free_pages(GFP_KERNEL | GFP_DMA, fact);
首要取得整个audiobuff的巨细 datasize, 再取得list_head + list_node * 4的巨细
headlistsize, 然后核算总巨细memsize, 用__get_free_pages分配接连的物理地址空间,这儿
留意一下,回来的是物理地址。
//Free old buffer
if(*memory) {
phead = (audio_head *)*memory;
fact = phead->fact;
free_pages((unsignedlong)*memory, fact);
*memory = NULL;
}
*memory= mem;
//Free old bufferif(*memory) {phead = (audio_head *)*memory;fact = phead->fact;free_pages((unsignedlong)*memory, fact);*memory = NULL;}*memory= mem;
这儿查看一下是不是有分配audiobuff, 假如有就释放掉,再将新分配的地址赋值给memory.
在mixer的ioctl中SNDCTL_DSP_SETFRAGMENT也会重新分配巨细的。
phead = (audio_head *)*memory;
phead->fact = fact;
phead->listsize = headlistsize;
phead->datasize = datasize;
audio_wuse = &(phead->use);
audio_wfree = &(phead->free);
INIT_LIST_HEAD(audio_wuse);
INIT_LIST_HEAD(audio_wfree);
phead = (audio_head *)*memory;phead->fact = fact;phead->listsize = headlistsize;phead->datasize = datasize;audio_wuse = &(phead->use);audio_wfree = &(phead->free);INIT_LIST_HEAD(audio_wuse);INIT_LIST_HEAD(audio_wfree);
接下来便是给listhead赋值,然后初始化audio_wuse和audio_wfree链表。
pbuff= (audio_node *)((unsigned int)*memory + sizeof(audio_head));
*fragmem_start= (int)((unsigned int)*memory + headlistsize);
for(i = 0; i < count; i++) {
pbuff->pBuf = (unsigned int)*memory + headlistsize +pagesize * i;
pbuff->phyaddr = (unsigned int)virt_to_phys((void*)pbuff->pBuf);
pbuff->start = 0;
pbuff->end = 0;
DEBUG
pbuff->pBufID = i;
DPRINT_Q(“audio_notebuffer[%d] = %x\n”, i, (unsigned int)pbuff->pBuf);
list_add(&pbuff->list,audio_wfree);
pbuff++;
}
pbuff= (audio_node *)((unsigned int)*memory + sizeof(audio_head));*fragmem_start= (int)((unsigned int)*memory + headlistsize);for(i = 0; i < count; i++) {pbuff->pBuf = (unsigned int)*memory + headlistsize +pagesize * i;pbuff->phyaddr = (unsigned int)virt_to_phys((void*)pbuff->pBuf);pbuff->start = 0;pbuff->end = 0;#ifdef Q_DEBUGpbuff->pBufID = i;#endifDPRINT_Q(“audio_notebuffer[%d] = %x\n”, i, (unsigned int)pbuff->pBuf);list_add(&pbuff->list,audio_wfree);pbuff++;}
在这儿将audio_wfree 链表悉数初始化,留意此刻wfree链表中有4个结点,wuse中
没有结点。
if (ret) {
endpoint->fragsize =pagesize;
endpoint->fragstotal =count;
endpoint->memsize = ret;
}
if (ret) {endpoint->fragsize =pagesize;endpoint->fragstotal =count;endpoint->memsize = ret;}
接下来回来到audio_resizemem_endpoint, 将endpoint结构赋值。
Audio buff分配好后,再回来到audio_init_endpoint对DMA进行初始化, 这儿依据传
入的是play或许record进行初始化,大致上是相同的,这儿只剖析 replay的初始化
if ((ch =jz_request_aic_dma(DMA_ID_I2S_TX,”audio dac“, jz_i2s_dma_irq,IRQF_DISABLED,
endpoint)) < 0) {
printk(KERN_ERR “%s:can't reqeust DMA DAC channel.\n”, __FUNCTION__);
return -1;
}
chan->io = i;
chan->dev_id = dev_id;
chan->dev_str = dev_str;
chan->fifo_addr = CPHYSADDR(AIC_DR);
switch (dev_id) {
case DMA_ID_AIC_TX:
chan->mode = DMA_AIC_TX_CMD_UNPACK | DMA_MODE_WRITE;
chan->source = DMAC_DRSR_RS_AICOUT;
break;
case DMA_ID_AIC_RX:
chan->mode = DMA_32BIT_RX_CMD | DMA_MODE_READ;
chan->source = DMAC_DRSR_RS_AICIN;
break;
default:
printk(“JZ AIC: %s:%d,need fix !!!\n”, __FUNCTION__, __LINE__);
BUG_ON(1);
}
if ((ch =jz_request_aic_dma(DMA_ID_I2S_TX,”audio dac”, jz_i2s_dma_irq,IRQF_DISABLED,endpoint)) < 0) {printk(KERN_ERR "%s:can't reqeust DMA DAC channel.\n", __FUNCTION__);return -1;}chan->io = i;chan->dev_id = dev_id;chan->dev_str = dev_str;chan->fifo_addr = CPHYSADDR(AIC_DR);switch (dev_id) {case DMA_ID_AIC_TX:chan->mode = DMA_AIC_TX_CMD_UNPACK | DMA_MODE_WRITE;chan->source = DMAC_DRSR_RS_AICOUT;break;case DMA_ID_AIC_RX:chan->mode = DMA_32BIT_RX_CMD | DMA_MODE_READ;chan->source = DMAC_DRSR_RS_AICIN;break;default:printk(“JZ AIC: %s:%d,need fix !!!\n”, __FUNCTION__, __LINE__);BUG_ON(1);}
调用jz_request_aic_dma函数注册DMA中止,填写chan的形式与地址填充,最终发动
AIC DMA的时钟,再回来到init_audio_replaydma中,初始化寄存器,DMA就初始化OK了。
至此整个AUDIO BUFF就初始化完成了。
播映流程剖析
先上图来说话:
首要进入jz_audio_write函数, 里面有两个分支,依据形式进行区别的,第1种是运用
mmap方法传输的,因为咱们要剖析audio buff, 就不重视mmap方法了,各位如有想了解
的话,能够发与我评论, dyronchina@gmail.com, 下边就直接进入audiobuff 方法的
jz_audio_write_data函数剖析。
while(count >= pout_endpoint->fragsize) {
bat_cnt= endpoint_put_userdata(pout_endpoint,
&(buffer[usecount]),
pout_endpoint->fragsize);
//Prepare data success.
if(bat_cnt > 0) {
usecount+= bat_cnt;
count-= bat_cnt;
DPRINT(“bat_cnt= %d\n”, bat_cnt);
}
//Perhaps non node is avialable.
elseif (bat_cnt == 0) {
DPRINT(“bat_cnt== 0\n”);
break;
}
//Error occured.
else{
//break and handle prepared data.
if(usecount > 0) {
DPRINT(“bat_cnt< 0, usecount > 0\n”);
break;
}
//Has not prepared any data and return error when prepared data.
else{
DPRINT(“bat_cnt< 0, usecount == 0\n");
returnbat_cnt;
}
}
}
while(count >= pout_endpoint->fragsize) {bat_cnt= endpoint_put_userdata(pout_endpoint,&(buffer[usecount]),pout_endpoint->fragsize);//Prepare data success.if(bat_cnt > 0) {usecount+= bat_cnt;count-= bat_cnt;DPRINT(“bat_cnt= %d\n”, bat_cnt);}//Perhaps non node is avialable.elseif (bat_cnt == 0) {DPRINT(“bat_cnt== 0\n”);break;}//Error occured.else{//break and handle prepared data.if(usecount > 0) {DPRINT(“bat_cnt< 0, usecount > 0\n”);break;}//Has not prepared any data and return error when prepared data.else{DPRINT(“bat_cnt< 0, usecount == 0\n");returnbat_cnt;}}}
首要查看传入音频块的巨细,有必要大于fragsize(4096),先来全体剖析这个函数,然后
再进入内部剖析细节,bat_cnt 依据回来值进行处理,回来0标明没有wfree节点,就直接
跳出,进行播映录音了。
DPRINT(“<<<);
node= endpoint_get_outnode(endpoint);
if(!node)
return0;
if(copy_from_user((void *)node->pBuf, buffer, count)) {
printk(“JZI2S: copy_from_user failed !\n”);
return-EFAULT;
}
LEAVE();
returnendpoint_post_outnode(endpoint,node,count);
DPRINT(“<<
用endpoint_get_outnode取得一个wfreenode,取得一个后就从wfree list上删除去当
前节点, 播映完成后就回来结点,这样就形成了一个环形缓冲区。
最终回来到jz_audio_write_data, 最终假如当时while 循环完毕后还有剩下的数据,就
录音流程剖析
因为播映和录音基本上是相同的,故在此不做剖析了
重新做一次播映。