OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区-51CTO.COM

OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放

奶盖
发布于 2022-3-28 17:31
浏览
0收藏

在LYEVK-38613861开发板的学习过程中,发现缺少基于鸿蒙2.0的相关文档,同时缺少基于2.0系统 IoT接口的应用示例。在学习PWM相关接口的过程中,了解到PWM接口驱动蜂鸣器可以实现类似音乐播放的效果,个人觉得是个不错的思路,就有了本次的学习开发之旅。

环境准备

1、开发环境、编译环境搭建,参考官方文档,此处不在赘述。参考链接如下:

2、OpenHarmony 2.0 Canary源码 源码获取,参考:

3、LYEVK-3861 IoT物联网开发板套件

开发调试

2.1 相关基础知识介绍

PWM输出方波的IOT接口

鸿蒙系统IoT硬件子系统提供了一些外设相关的接口,目录位于:

base/iot_hardware/peripheral/interfaces/kits

PWM相关接口,接口头文件为iot_pwm.h,其中开始输出方波的接口为:

unsigned int IoTPwmStart(unsigned int port, unsigned short duty, unsigned int freq);

接口参数介绍:

freq:IoTPwmStart接口中freq参数是分频倍数,PWM实际输出的方波频率等于 PWM时钟源频率 除以 分频倍数,即 f = Fcs / freq 其中,Fcs是PWM时钟源频率; duty:IoTPwmStart接口的duty参数可以控制输出方波的占空比,占空比是指PWM输出的方波波形的高电平时间占整个方波周期的比例,具体占空比值取值为1到99,例如想要输出占空比 50%的方波信号,那么duty填的值就要是50。

音符-频率对应关系OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区 这个表中有一个规律——音高升高一个八度,频率升高一倍。

hi_u32 hi_pwm_set_clock(hi_pwm_clk_source clk_type);

160M时钟源条件下,输出方波的最低频率是:160M/65535=2441.44...,这个频率略高,在上面的表格中没有找到音名。通过调用hi_pwm_set_clock接口,可以修改时钟源,将时钟源设置为晶体时钟且时钟频率为40MHz,40M/65535= 610.3...,这样就能够输出E5及以上的所有音符。

2.2 曲谱转换

由于个人比较喜欢《蜜雪冰城》,我选择了《蜜雪冰城主题曲》的曲谱作为素材,简谱如下:OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区 简谱转换

每个音符都需要有节拍,在外面的代码里体现为停顿时间,不同音符的不同停顿时间,可以实现简单的音乐起伏。

常见的节拍简谱对应:OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区

通过简谱和以上表格的对应,就可以将我们现有的简谱,转换成为可以被程序识别的”程序谱子“。

2.3 编写代码

接口初始化

修改device/hisilicon/hispark_pegasus/sdk_liteos/build/config/usr_config.mk配置文件,打开PWM编译支持,如已打开,可略过:

# CONFIG_UART_DMA_SUPPORT is not set
CONFIG_PWM_SUPPORT=y
# CONFIG_PWM_HOLD_AFTER_REBOOT is not set

修改device/hisilicon/hispark_pegasus/sdk_liteos/app/wifiiot_app/init/app_io_init.c的如下代码

#ifdef CONFIG_PWM_SUPPORT 
    /* PWM 0/2/3/4/5 配置同理 */
    //hi_io_set_func(HI_IO_NAME_GPIO_8, HI_IO_FUNC_GPIO_8_PWM1_OUT);
    //GPIO引脚复用
    hi_io_set_func(HI_IO_NAME_GPIO_8, HI_IO_FUNC_GPIO_8_GPIO);  //button
    hi_io_set_pull(HI_IO_NAME_GPIO_8, HI_IO_PULL_UP);
    hi_io_set_func(HI_IO_NAME_GPIO_9,HI_IO_FUNC_GPIO_9_PWM0_OUT);// PWM
#endif

备注:2.0把GPIO的引脚复用从应用层移到了板级,对IO的功能做了更加细分的处理,之前一直以为2.0移除了相关的接口实现,最后才发现2.0做了功能上的优化。

实例代码

按键驱动OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区

说明:按键主要是为了开启音乐的播放。

程序”曲谱

《蜜雪冰城主题曲》的“程序”曲谱,定义如下:OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区以上的曲谱,看起来比较直观,也比较容易理解,其中的一些宏定义定义如下:OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区 有了以上的枚举,可以自己直接谱曲,甚至不用拘泥于现有的曲子,当然放出来的具体效果如何,就因人而异了。

音符频率

以下是IoTPwmStart接口的入口参数freq的定义,和前述枚举是一一对应的:OpenHarmony 基于LYEVK-3861开发板 IOT套件的音乐播放-鸿蒙开发者社区

音乐处理

/* 音乐处理*/
static void *BeeperMusicTask(const char *arg)
{
    (void)arg;
    int status = 0;
    printf("BeeperMusicTask start!\r\n");

    hi_pwm_set_clock(PWM_CLK_XTAL); // 设置时钟源为晶体时钟(40MHz,默认时钟源160MHz)

    while (1)
    {
        usleep(M_INTERVAL_TIME_US);
        /*第一次点击按键播放,播放完以后按键才能继续生效*/
        if (music == 1)
        {
            for (size_t i = 0; i < sizeof(g_interval) / sizeof(g_interval[0]); i++)
            {  
                uint32 tune = g_interval[i].tuneNotes; // 音符
                uint16 freqDivisor = g_tuneFreqs[tune];
                uint32 tuneInterval = g_interval[i].interval * (TICKS_DELAY); // 音符时间
                IoTPwmStart(IOT_PWM_PORT0, PWM_DUTY, freqDivisor);
                usleep(tuneInterval);
                IoTPwmStop(IOT_PWM_PORT0);
                music = 0;
            }
        }
    }

    return NULL;
}

功能展示

​ 以上都完成之后,就可以编译、烧录,并测试最后的完成效果。

​ 说明:

1、本程序需要Hi3861开发板配合交通灯板实现完整的演示。

​ 2、烧录完成之后,RESET重启开发板之后,音乐不会播放。

​ 3、重启开发板之后,点击交通灯板开发播放音乐;播放过程中,按键失去效果,待播放完成之后,再次点击按键,音乐继续播放。

已于2022-5-5 14:16:49修改
1
收藏
回复
举报
回复
    相关推荐