CurieNano 直接播放声音

前一段拿到了 DFRobot的CurieNano控制板。除了控制器,还有两个徽章以及多功能便携工具卡:

220537naqkmwaaihumzhak

可以看到这次的主角 Curie Nano 非常小巧,但是功能和全尺寸的 Arduino 101 相比毫不逊色。接下来我就使用这块小板子完成直接播放声音的功能,之前我在【参考1】中介绍过。 唯一的问题是直接寄存器仿佛古文一般面目可憎,又好比英文小说,每一个字母都认识,但是放在一起变成了天书。 仔细捋了一下原理,重新编写了一次。
从原理出发,声音播放就是在特定的频率下(通常大于8000Hz),将数字量化后的信号按照模拟的方式输出。而对于Arduino 来说,最简单的模拟输出就是用PWM。参考1的代码用了一个定时器,8000Hz触发一次,这样整体的代码非常复杂。我们修改为循环,输出之后做足够的延时(1/8000=125us),实际上的输出也是8000Hz,但是整体变得直观和容易理解多了。此外,默认的 PWM输出频率过小(关于 Curie的PWM我会另外介绍),这会导致analogWrite输出的PWM还没有来得及输出完整的一个周期下面的数据就过来。因此,需要升高PWM的频率。对于101 来说,使用analogWriteFrequency(Pin,Frequency);即可指定输出的 PWM 频率(意想不到的简单)。
首先,我们要大概估计一下空间,编写下面这样的简单代码:

void setup() {
analogWriteFrequency(OUTPIN, 64000);
}
void loop() {
}

 

剩余空间差不多有 155648-48016=107632 Bytes,大概可以存放107632/8000=13.4秒的声音(实际上还有额外库函数的开销,实际写代码根据编译结果删除了很多数据)。
220537ll4xtkt883x45pxn

硬件连接上非常简单:Uno 的地接在喇叭负极标志处上,然后随便选一个 支持PWM的引脚作为输出,这里我选择的是Pin9接在喇叭的正极标记处。

220538gm9mnu44d8f4440p

播放的内容是罗大佑的《恋曲1990》,首先用工具将音频转化为 8000Hz,8Bits,然后用【参考4】的工具转化数据为 C的格式,放置在代码中即可。
不完整的程序如下:

#define OUTPIN 9

const unsigned char sample[] PROGMEM =
{
        0x80, 0x80, 0x84, 0x85, 0x85, 0x83, 0x83, 0x81, 0x82, 0x83, 0x85, 0x82, 0x80, 0x7F, 0x7F, 0x80, 0x83, 0x81, 0x81, 0x80, 0x7F, 0x85, 0x87, 0x87, 0x86, 0x84, 0x84, 0x87, 0x84, 0x82, 0x81, 0x80,
        0x82, 0x87, 0x88, 0x88, 0x89, 0x84, 0x80, 0x7F, 0x80, 0x81, 0x82, 0x82, 0x83, 0x82, 0x83, 0x84, 0x83, 0x82, 0x80, 0x7F, 0x7D, 0x7A, 0x78, 0x75, 0x74, 0x72, 0x72, 0x74, 0x75, 0x73, 0x70, 0x70,
//很多很多数据
        0x7C, 0x76, 0x76, 0x7C, 0x76, 0x72, 0x77, 0x79, 0x77, 0x7B, 0x7F, 0x84, 0x87, 0x85, 0x7F, 0x85, 0x88, 0x86, 0x82, 0x81, 0x83, 0x82, 0x80, 0x7F, 0x82, 0x80, 0x7A, 0x74, 0x73, 0x74, 0x77, 0x7A,
        0x84, 0x88, 0x87, 0x83, 0x85, 0x8B, 0x90, 0x91, 0x8F, 0x8A, 0x8A, 0x88, 0x88, 0x82, 0x7F, 0x7D, 0x74, 0x6D, 0x6A, 0x67, 0x6B, 0x6E, 0x74, 0x75, 0x74, 0x6F, 0x70, 0x73, 0x77, 0x7B, 0x7B, 0x7B,
        }
        void setup()
        {
                analogWriteFrequency(OUTPIN, 64000);
        }

        void loop()
        {
                for (unsigned int i = 0; i < sizeof(sample); i++)
                {
                        analogWrite(OUTPIN, pgm_read_byte(&sample));
                        delayMicroseconds(125);
                }
        }

 

整个程序源代码500K。

220538s16xtttm8ltsusqu

完整代码下载:

1990s

针对这种代码的调试上有一些建议:

1. 最好使用歌曲作为素材,理由是单纯的音乐通常无法分清楚当前的播放是否正常。当然,如果你的喇叭质量不好,使用<<月亮之上>>这样的也可以掩盖破音(这也是为什么山寨机最喜欢放这首曲子的原因);
2. 为了安全起见,建议在喇叭上串联一个电阻,防止电流超过40ma;
3. 因为数据被定义在 Flash中,所以一定要使用pgm_read_byte(&sample)这种形式读取出来,切忌直接使用sample.这个问题让我头晕了很久.
4.
有了这样的方法, 你可以尝试在自己的作品中加入声音的功能。 后面我还会尝试使用板载SPI芯片来扩展存储更多的音频。

参考:
1. https://www.lab-z.com/arduinosound/使用Arduino直接发声
2. http://www.diy-robots.com/?p=852Arduino系列教程之 – PWM的秘密(下)
3. http://www.diy-robots.com/?p=814Arduino系列教程之 – PWM的秘密(上)
4. http://www.arduino.cn/thread-46025-1-1.html一个wave 转h 文件的工具

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注