博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
SylixOS音频驱动移植
阅读量:6226 次
发布时间:2019-06-21

本文共 3842 字,大约阅读时间需要 12 分钟。

hot3.png

1. 适用范围

       本文档为实现Nuc970平台音频驱动的方法总结,以此提供一些SylixOS音频驱动移植方法的参考。

2. 原理概述

2.1 Codec编解码芯片

        声音信号分为模拟信号和数字信号,Codec编解码芯片主要功能就是实现模拟信号与数字信号的互相转换。

        本文调试的Codec型号为NAU8822L,其结构如图 2-1所示。

图 2-1 NAU8822L编解码芯片

        其中主要使用到的是以下三个部分:

  • Mixer    混音器设备,它的作用是将多个信号组合或者叠加在一起。

         由输入混音器(input mixer)和输出混音器(output mixer)组成。

  • ADC    模拟信号转换数字信号单元。

         由左路ADC和右路ADC组成。

  • DAC    数字信号转换模拟信号单元。 

         由左路DAC和右路DAC组成。

2.2 采样率、量化位数、声道

  • 采样率        每秒从连续信号中提取并组成离散信号的采样个数。
  • 量化位数    模拟量转换成数字量之后的数据位数。
  • 声道            声音录制时的音源数量或回放时相应的扬声器数量。

2.3 I2S总线

        I2S(Inter-IC Sound Bus)是飞利浦公司为数字音频设备之间的音频数据传输而制定的一种总线标准。I2S有3个主要信号,如表 2-1所示。 

表 2-1 i2s信号类型

信号 名称 功能
BCLK 位时钟

数字音频的每一位数据,BCLK都有1个脉冲。

频率 = 2 ×采样频率×量化位数

LRCK 左右声道切换时钟

用于切换左右声道的数据,频率等于采样频率。

为"1"表示正在传输的是左声道的数据,

为"0"表示正在传输的是右声道的数据。

SDATA 串行数据 用二进制补码表示的音频数据

2.4 OSS统一音频接口

2.4.1 声卡模型

        SylixOS中使用的是OSS简化后的声卡模型,将声卡抽象为了以下两种设备:

  • /dev/dsp    实现录音和放音功能。

        向该设备写数据即意味着激活声卡上的D/A转换器进行放音,

        向该设备读数据则意味着激活声卡上的A/D转换器进行录音。

  • /dev/mixer    实现混音器调节功能。

         对该设备调用ioctl,即可控制声卡上的Mixer功能进行调节。

2.4.2 IO控制

        /dev/dsp和/dev/mixer分别提供了ioctl的接口用于对其功能进行配置和操作。

       开发时,应该根据OSS的框架以及实际的Codec情况在驱动中实现这些命令相关的功能。

  • /dev/dsp提供了如表 2-2所示的ioctl命令。

表 2-2 dsp相关ioctl参数

命令
说明
SNDCTL_DSP_SETTRIGGER 用于启动播放和录音功能
SNDCTL_DSP_GETOSPACE
SNDCTL_DSP_GETISPACE
用于获得输出/输入buffer相关的信息
SNDCTL_DSP_GETCAPS 用于获得Codec相关的属性参数
SNDCTL_DSP_SETFRAGMENT 用于设置Codec和buffer相关的参数
…… ……
  • /dev/mixer提供了如表 2-3所示的ioctl命令

表 2-3 mixer相关ioctl参数

命令 说明
SOUND_MIXER_VOLUME 用于调节主输出音量
SOUND_MIXER_PCM 用于调节Codec(ADC)输出音量
SOUND_MIXER_MIC 用于调节Mic输入音量
…… ……

3. 技术实现

3.1 驱动框架

        在Nuc970的音频驱动代码中采用了如图 3-1所示的框架。

  • Device层         用于实现抽象的dsp和mixer设备,提供统一的API接口。
  • AudioLib层      用于实现Codec和I2S的相关逻辑,供Device层调用,Codec和I2S之间应同步配置。
  • Hardware层    用于实现具体的Codec和I2S的寄存器访问,和具体硬件相关。

图 3-1音频驱动框架

3.2 调试方法

        当喇叭还未有声音产生或录音文件没有数据时,需要借助示波器进行调试。

        总体思路是在Codec的输入端和输出端分别测量数字信号和模拟信号,用于确定问题点所在位置。

        数字信号下的音频波形即为i2s输出的波形,大致形状如图 3-2所示。

图 3-2 i2s波形

        数字信号下的模拟波形大致形状如图 3-3所示。

图 3-3音频模拟信号波形

3.2.1 播放音乐调试

    当调试播放功能时,可以参考如图 3-4所示流程,确定问题点所在位置,以此明确需要修改的内容。

图 3-4播放音乐功能调试流程

3.2.2 录制声音调试

    当调试录音功能时,可以参考如图 3-5所示流程,确定问题点所在位置,以此明确需要修改的内容。

图 3-5录制声音功能调试流程

3.3 DMA缓冲机制

        Nuc970的I2S可以将DMA分块,对于Play和Record使用的DMA都可以按图 3-6所示各自等分成2至8块。

图 3-6 DMA扫描

        I2S的DMA本身处于实时刷新状态。以播放为例,DMA不停地从内存搬运数据送往Codec。如果DMA正在读取区域内的数据仍未更新,Codec转换出来的模拟音频信号就会产生杂音。因此,必须保证DMA读取对应内存区域前,其中的数据已经被更新。

        Nuc970可以在某分块区域被DMA读取完后产生中断,软件通过读取寄存器可以得知是哪个分块已经被读取完成,此时就可以更新这块刚刚被读取完的分块中的数据,以保证下次DMA读取此分区时其中的数据已经为有效数据,如图 3-7所示。

图 3-7扫描分块结束后产生中断

3.4 播放音乐代码实现

3.4.1 应用代码实现

程序清单 3-1播放音乐应用代码实现

/* * 以只写方式打开dsp设备 */iDspFd = open("/dev/dsp", O_WRONLY, 0666);if (iDspFd < 0) {    printf("failed to open /dev/dsp device!\n");    break;} /* * 设置量化位数 */iSampleFmt = wavHeader.sPCMBitWidth == 16 ? AFMT_S16_LE : AFMT_S8;stRet = ioctl(iDspFd, SNDCTL_DSP_SETFMT, &iSampleFmt);if (stRet < 0) {    printf("failed to set sample format!\n");    close(iDspFd);    break;} /* * 设置声道数 */iChannels = wavHeader.sChannel;stRet = ioctl(iDspFd, SNDCTL_DSP_CHANNELS, &iChannels);if (stRet < 0) {    printf("failed to set channels!\n");    close(iDspFd);    break;} /* * 设置采样率 */iSampleRate = wavHeader.lSampleRate;stRet = ioctl(iDspFd, SNDCTL_DSP_SPEED, &iSampleRate);if (stRet < 0) {    printf("failed to set sample rate!\n");    close(iDspFd);    break;} /* * 打开wav音频文件 */iFileFd = open(pcFileName, O_RDONLY, 0666);if (iFileFd  < 0) {    printf("failed to open test audio file %s!\n", pcFileName);    close(iDspFd);    break;} read(iFileFd, pcBuffer, 0x2E); /* * 循环读取wav音频文件,写入dsp进行播放 */while((stLen = read(iFileFd, pcBuffer, __OSS_PLAY_BUFFER_LEN)) > 0) {    pcPtr = pcBuffer;     while (stLen > 0) {        stRet = write(iDspFd, pcPtr, stLen);        if (stRet < 0) {            break;        }        pcPtr += stRet;        stLen -= stRet;    }}

3.4.2 驱动代码实现

        Nuc970采用了如图 3-8所示的音频驱动代码结构,此代码结构与图 3-1所示驱动框架相对应。

图 3-8代码结构

        Audio驱动实现的基本流程如图 3-9所示,开始播放声音时,驱动需要先对Codec进行配置,然后开启DMA传输,此时DMA就在实时读取内存。

        当DMA其中一个区域读取完成后,就会产生中断,此时将用户层写入的新数据填充到DMA中即可。

        当音频文件播放结束时,停止DMA传输即可。

图 3-9音频驱动流程

4. 总结

        本文章结合Nuc970说明了SylixOS中音频驱动移植的方法。

转载于:https://my.oschina.net/wPCBsKzQ/blog/884220

你可能感兴趣的文章
基于BIGINT溢出错误的SQL注入
查看>>
Burp Suite使用介绍(二)
查看>>
Struts2 Tomcat class.classLoader.resources.dirContext.docBase赋值造成的DoS及远程代码执行利用!...
查看>>
当失控的预装行为以非正当手段伸向行货机时_北京鼎开预装刷机数据统计apk(rom固化版)分析...
查看>>
最近招聘的一些思考
查看>>
PHP 单元测试
查看>>
魔幻特效,慢放世界,nova 3带你玩转抖音新技能
查看>>
声明式调用---Feign
查看>>
有效的沟通,如忍者的最后一击!
查看>>
从零开始搭建一个简单的基于webpack的vue开发环境
查看>>
【功能盘点】升级后的媒体处理MPS有哪些能力?
查看>>
聊聊redis的slowlog与latency monitor
查看>>
【iOS 印象】Swift 中值类型与引用类型指北
查看>>
vim-galore 中文翻译
查看>>
云数据库Memcache版使用教程
查看>>
重构了一下小博客
查看>>
redux简单实现与分析
查看>>
iOS12的捷径你玩过了吗
查看>>
iOS实现UITableViewCell点击展开文本内容
查看>>
基本特效:饿了么丝滑无缝过度搜索栏的实现
查看>>