整理东西整理出了几块RDA5851S模块

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_i++

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_单片机_02

本着物尽其用的原则,我查了一下相关资料,发现是在车载上用的,还可以进行AT控制

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_ico_03

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_单片机_04

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_android 车载蓝牙不显示歌曲信息_05

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_单片机_06

看到这里我觉得还好,因为这种可以AT控制的蓝牙模块还是比较常见的,

本着试一试的心思,接电源线,想到接功放和喇叭太费事儿,直接连接了一个耳机口,插上耳机,通电

蓝牙名称还是很好找的  CAR KIT ,点击蓝牙连接,手机放歌,耳机里传出了声音,一切正常,除了没有蓝牙连接提示音,断开连接提示音。

接着用USB转串口接到蓝牙的RX TX上,数据显示乱码,切换波特率,再来。。。

终于,波特率57600情况下,显示正常了,哎等等,怎么还有乱码?

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_android 车载蓝牙不显示歌曲信息_07

换歌再来,还是乱码

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_android 车载蓝牙不显示歌曲信息_08

灵机一动,放个英文歌

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_单片机_09

发现了新大陆

MI1后面就是歌名

MI2后面就是歌手

那么问题来了,汉字歌名怎么不行呢?

看看16进制到底发了什么

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_i++_10

找个转换工具转一下

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_i++_11

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_蓝牙_12

喜出望外,说明汉字歌曲是可以的,但为什么这里可以显示,串口助手就不行?

换一个支持改编码的串口助手

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_android 车载蓝牙不显示歌曲信息_13

原来如此,模块发送出来是UTF-8编码的数据,英文数字因为编码一样可以正常显示,而汉字编码则为乱码。找到原因就好办了。

因为单片机显示汉字是GBK编码的字库,而现在是UTF-8编码,UTF-8可以转GBK吗?查查查起来。。。

UTF-8不能直接转GBK,但是UTF-8可以转成Unicode,而Unicode经过一个大表可以对应到GBK。

代码实现

/************************************************************************
* 函数名称: StrProcess_UTF8toGBK
* 函数功能: 将网络上 utf8 的字符串转换为 GBK 字符串 使得兼容 原子LCD屏驱动程序
* 函数输入: void input: c_utf8 原utf8 字符串 同时 新的 gbk字符串会将其覆盖
* length 你想要设置的 中间缓存块的大小
* 函数输出: void output:字符总数 (一个中文算两个字节)
* 作者 :
* 文件依存:
* #include “ff.h”
* #include “malloc.h”
* #include “stm32f10x.h”
************************************************************************/
int16_t StrProcess_UTF8toGBK(u8 *c_utf8, u16 length)
{
    char low;
    char high;
    char middle;

    // ff_uni2oem(,FF_CODE_PAGE);
    /* !< 首先 将 utf8 转换成标准 Unicode 然后查表将 Unicode 转化为GBK */
    u16 outputSize = 0; //记录转换后的gbk字符串长度
    u8 *pInput = c_utf8;
    u8 *c_gbk = mymalloc(0, length); /* !< 申请内存,也可以外部传入一个 数据缓存空间 */
    u8 *pOutput = c_gbk;
    u16 *uni = (u16 *)c_gbk;
    u16 gbk;
    /* !< 以下中间代码来自于 CSDN @bladeandmaster88 公开的源码 */
    while (*pInput)
    {
        if (*pInput > 0x00 && *pInput <= 0x7F) //处理单字节UTF8字符(英文字母、数字)
        {
            *pOutput = *pInput;
            pOutput += 1;
            *pOutput = 0; //小端法表示,在高地址填补0
        }
        else if (((*pInput) & 0xE0) == 0xC0) //处理双字节UTF8字符
        {
            high = *pInput;
            pInput += 1;
            low = *pInput;
            if ((low & 0xC0) != 0x80) //检查是否为合法的UTF8字符表示
            {
                return -1; //如果不是则报错
            }

            *pOutput = (high << 6) + (low & 0x3F);
            pOutput++;
            *pOutput = (high >> 2) & 0x07;
        }
        else if (((*pInput) & 0xF0) == 0xE0) //处理三字节UTF8字符
        {
            high = *pInput;
            pInput++;
            middle = *pInput;
            pInput++;
            low = *pInput;
            if (((middle & 0xC0) != 0x80) || ((low & 0xC0) != 0x80))
            {
                return -1;
            }
            *pOutput = (middle << 6) + (low & 0x3F);//取出middle的低两位与low的低6位,组合成unicode字符的低8位
            pOutput++;
            *pOutput = (high << 4) + ((middle >> 2) & 0x0F); //取出high的低四位与middle的中间四位,组合成unicode字符的高8位
        }
        else //对于其他字节数的UTF8字符不进行处理
        {
            return -1;
        }
        pInput++;//处理下一个utf8字符
        pOutput++;
    }
    //unicode字符串后面,有两个\0
    *pOutput = 0;
    pOutput++;
    *pOutput = 0;
    /* !< 感谢 @bladeandmaster88 的开源支持 */
    pInput = c_utf8;
    while(*uni != 0)

    {
        gbk = ff_convert((WCHAR) * uni, 0);
        //    gbk = ff_uni2oem(*uni,FF_CODE_PAGE);        /* !< Unicode 向 GBK 转换函数  */
        uni++;
        if(gbk & 0xff00)
        {
            *pInput = ((gbk & 0xff00) >> 8);
            pInput++;
            *pInput = (gbk & 0x00ff);
            pInput++;
            outputSize += 2;
        }
        else
        {
            *pInput = (gbk & 0x00ff);
            pInput++;
            outputSize++;
        }


    }
    *pInput = '\0';         /* !< 加上结束符号 */
    myfree(0, c_gbk);       /* !< 释放内存 */
    return outputSize;
}

蓝牙数据解析

void Data_Analysis()
{
    char *ptr;
    uint8_t lenth = 0;
    uint8_t i = 0;

    ptr = strstr((char *)STR_BUF, "IB");		//已连接
    if(ptr != NULL)
    {
        Connect_Flag = 1;
    }
    else
    {
        ptr = strstr((char *)STR_BUF, "IA");		//断开连接
        if(ptr != NULL)
        {
            Connect_Flag = 0;
        }
    }
    ptr = strstr((char *)STR_BUF, "MB");		//开始播放
    if(ptr != NULL)
    {
        Play_Flag = 1;
    }
    else
    {
        ptr = strstr((char *)STR_BUF, "MA");		//停止播放
        if(ptr != NULL)
        {
            Play_Flag = 0;
        }
    }
    ptr = strstr((char *)STR_BUF, "MI1");		//歌名
    if(ptr != NULL)
    {
        Connect_Flag = 1;
        Play_Flag = 1;
        ptr = ptr + 3;
        lenth = 127;
        i = 0;
        memset(STR_Name , 0 , 127);
        while ((*ptr != 0x0D) && (lenth --))
        {
            STR_Name[i++] = *(ptr++);
        }
    }
    ptr = strstr((char *)STR_BUF, "MI2");		//歌手
    if(ptr != NULL)
    {
        ptr = ptr + 3;
        lenth = 127;
        i = 0;
        memset(STR_Singer , 0 , 127);
        while ((*ptr != 0x0D) && (lenth --))
        {
            STR_Singer[i++] = *(ptr++);
        }
    }
}

接下来就是STM32显示了

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_ico_14

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_ico_15

接着又画了一下PCB,毕竟总是连着线也不太好,带有充电电路,小集成功放,MOS管控制蓝牙电源,状态指示灯等(毕竟这个蓝牙没连接提示音)。

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_android 车载蓝牙不显示歌曲信息_16

焊接调试好没问题。

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_i++_17

android 车载蓝牙不显示歌曲信息 车上蓝牙不显示歌名_ico_18

自动居中显示,如果超长则滚动显示

void Dis_Play()
{
    uint8_t STR_Name_len;
    uint8_t STR_Singer_len;
    int8_t STR_Name_can;
    int8_t STR_Singer_can;
    uint8_t STR_Name_symbol = 0;
    uint8_t STR_Singer_symbol = 0;
    int8_t i , j ;

    if(Play_Flag)
    {
        Lcd_P16x16Ch(88, 0, (uint8_t *)"播放", 0);
        Connect_Flag = 1;
    }
    else
    {
        if(Connect_Flag)
        {
            Lcd_P16x16Ch(88, 0, (uint8_t *)"暂停", 0);
        }

    }
    if(Connect_Flag)
    {
        Lcd_P16x16Ch(8, 0, (uint8_t *)"已连接", 0);
				LED2_ON;
    }
    else
    {
        Lcd_P16x16Ch(8, 0, (uint8_t *)"未连接", 0);
				LED2_OFF;
    }

    Lcd_fill(0, 4, 128, 4, 0);


    if(STR_Name_lenth > 16)	//歌名超长滚动处理
    {
        STR_Name_can = STR_Name_lenth - STR_Name_num ;

        if(STR_Name_can < 0)
        {
            STR_Name_num = 0;
            STR_Name_can = STR_Name_lenth - STR_Name_num ;
        }

        if(STR_Name_can > 16)
        {

            for(i = 0 ; i < 16 ; i ++)
            {
                if(STR_Name [i + STR_Name_num] <= 0x80)
                {
                    STR_Name_symbol ++;
                }
                Dis_STR_Name[i] = STR_Name [i + STR_Name_num] ;
            }

        }
        else
        {
            for(i = 0 ; i < STR_Name_can ; i ++)
            {
                Dis_STR_Name[i] = STR_Name [i + STR_Name_num] ;
            }
            Dis_STR_Name[i ++] = 0x20 ;
            Dis_STR_Name[i ++] = 0x20 ;
            STR_Name_can = STR_Name_can + 2 ;
            j = 0 ;
            for(; i < 16 ; i++)
            {
                Dis_STR_Name[i] = STR_Name [j++] ;
            }

            for(i = 0; i < 16 ; i++)
            {
                if(Dis_STR_Name[i] <= 0x80)
                {
                    STR_Name_symbol ++;
                }
            }

        }


        if(STR_Name_symbol % 2)
        {
            Dis_STR_Name[15] = 0;
        }
        if(Dis_STR_Name[0] <= 0x80)
        {
            STR_Name_num += 1;
        }
        else
        {
            STR_Name_num += 2;
        }

        STR_Name_len = 0;
    }
    else		//歌名自动居中处理
    {
        memset(Dis_STR_Name , 0 , 16);
        for(i = 0 ; i < STR_Name_lenth ; i ++)
        {
            Dis_STR_Name[i] = STR_Name [i] ;
        }
        STR_Name_len = (128 - STR_Name_lenth * 8) / 2;
    }



    if(STR_Singer_lenth > 16)	//歌手超长滚动处理
    {
        STR_Singer_can = STR_Singer_lenth - STR_Singer_num ;

        if(STR_Singer_can < 0)
        {
            STR_Singer_num = 0;
            STR_Singer_can = STR_Singer_lenth - STR_Singer_num ;
        }

        if(STR_Singer_can > 16)
        {

            for(i = 0 ; i < 16 ; i ++)
            {
                if(STR_Singer [i + STR_Singer_num] <= 0x80)
                {
                    STR_Singer_symbol ++;
                }
                Dis_STR_Singer[i] = STR_Singer [i + STR_Singer_num] ;
            }

        }
        else
        {
            for(i = 0 ; i < STR_Singer_can ; i ++)
            {
                Dis_STR_Singer[i] = STR_Singer [i + STR_Singer_num] ;
            }
            Dis_STR_Singer[i ++] = 0x20 ;
            Dis_STR_Singer[i ++] = 0x20 ;
            STR_Singer_can = STR_Singer_can + 2 ;
            j = 0 ;
            for(; i < 16 ; i++)
            {
                Dis_STR_Singer[i] = STR_Singer [j++] ;
            }

            for(i = 0; i < 16 ; i++)
            {
                if(Dis_STR_Singer[i] <= 0x80)
                {
                    STR_Singer_symbol ++;
                }
            }

        }


        if(STR_Singer_symbol % 2)
        {
            Dis_STR_Singer[15] = 0;
        }
        if(Dis_STR_Singer[0] <= 0x80)
        {
            STR_Singer_num += 1;
        }
        else
        {
            STR_Singer_num += 2;
        }

        STR_Singer_len = 0;
    }
    else	//歌手自动居中处理
    {
        memset(Dis_STR_Singer , 0 , 16);
        for(i = 0 ; i < STR_Singer_lenth ; i ++)
        {
            Dis_STR_Singer[i] = STR_Singer [i] ;
        }
        STR_Singer_len = (128 - STR_Singer_lenth * 8) / 2;
    }

    Lcd_P16x16Ch(STR_Name_len, 4, Dis_STR_Name, 0);		//歌名显示
    Lcd_P16x16Ch(STR_Singer_len, 6, Dis_STR_Singer, 0);		//歌手显示

}

 

大功告成,试了一下,几乎所有音乐软件都可以(主流的那几个都试了,包括自带的音乐播放器,还有一些游戏背景音乐也可以)可以同步显示歌名歌手,再也不用为不知道现在放的什么歌而苦恼了,也不用打开手机查看了。

以下就是全部下载资料

 

后续:因为汉字显示是Unicode通过一个大表可以对应到GBK,而这个表只有基本汉字(意思就是没有一些生僻字和部分特殊符号),即使这样,程序也达到了190K,不过实际使用中影响不大,我切了几十首歌才有一个因为特殊符号乱码的(可能我找的歌也不到位)。