一、模块来源
模块实物展示:
二、规格参数
输入电压:2.8V-5.5V
IO电压:3.3V(模块引脚输出电压)
额定电流:500uA~10mA
控制方式:串口
以上信息见厂家资料文件
三、移植过程
3.1查看资料
此芯片最大的优势在于能够灵活的更换 SPI-flash 内的语音内容,省去了传统语音芯片需要安装上位机更换语音的麻烦,SPI FLASH 直接模拟成 U 盘,跟拷贝 U 盘一样,非常方便。使得产品研发和生产变得便捷简单。一线串口控制模式、RX232 串口控制可选,为研发提供更多的选择性。
USB 数据线连接芯片,插上电脑,电脑会自动识别,如下图所示:出厂内部自带了10首语音。
注意语音内容不可太长,防止内存不够。
电脑自动识别
U盘内部
该模块分有单独IO控制、一线串行控制、两线串口控制。
单独IO口控制
如果需要更新语音,把要放置的声音拷贝过去按照相应格式命名即可。按 5 位数字进行命名,即 00001、00002......00007。
一线串行控制
两线串口控制
波特率为 9600,数据位:8 ,停止位 1 位,检验位 N
3.2引脚选择
模块接线图
3.3移植至工程
移植步骤中的导入.c和.h文件与【CW32模块使用】DHT11温湿度传感器相同,只是将.c和.h文件更改为bsp_jq8900.c与bsp_jq8900.h。这里不再过多讲述,移植完成后面修改相关代码。
在文件bsp_jq8900.c中,编写如下代码。
/*
* Change Logs:
* Date Author Notes
* 2024-06-25 LCKFB-LP first version
*/
#include "bsp_jq8900.h"
#include "stdio.h"
#include "string.h"
unsigned char JQ8900_RX_BUFF[JQ8900_RX_LEN_MAX];
unsigned char JQ8900_RX_FLAG = 0;
unsigned char JQ8900_RX_LEN = 0;
/************************************************************
* 函数名称:JQ8900_USART_Init
* 函数说明:连接JQ8900的初始化
* 型 参:bund=串口波特率
* 返 回 值:无
* 备 注:无
*************************************************************/
void JQ8900_USART_Init(unsigned int bund)
{
GPIO_InitTypeDef GPIO_InitStruct; // GPIO初始化结构体
BSP_JQ8900_GPIO_RCC_ENABLE(); // 使能GPIO时钟
BSP_JQ8900_UART_RCC_ENABLE(); // 使能UART时钟
GPIO_InitStruct.Pins = BSP_JQ8900_TX_PIN; // GPIO引脚
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // 推挽输出
GPIO_InitStruct.Speed = GPIO_SPEED_HIGH; // 输出速度高
GPIO_Init(BSP_JQ8900_GPIO_PORT, &GPIO_InitStruct); // 初始化
GPIO_InitStruct.Pins = BSP_JQ8900_RX_PIN; // GPIO引脚
GPIO_InitStruct.Mode = GPIO_MODE_INPUT_PULLUP; // 上拉输入
GPIO_Init(BSP_JQ8900_GPIO_PORT, &GPIO_InitStruct); // 初始化
BSP_JQ8900_AF_UART_TX(); // UART_TX复用
BSP_JQ8900_AF_UART_RX(); // UART_RX复用
// 配置UART
USART_InitTypeDef USART_InitStructure;
USART_InitStructure.USART_BaudRate = bund; // 波特率
USART_InitStructure.USART_Over = USART_Over_16; // 配置USART的过采样率。
USART_InitStructure.USART_Source = USART_Source_PCLK; // 设置时钟源
USART_InitStructure.USART_UclkFreq = 64000000; //设置USART时钟频率(和主频一致即可)
USART_InitStructure.USART_StartBit = USART_StartBit_FE; //RXD下降沿开始
USART_InitStructure.USART_StopBits = USART_StopBits_1; // 停止位1
USART_InitStructure.USART_Parity = USART_Parity_No ; // 不使用校验
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; // 不使用流控
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; // 收发模式
USART_Init(BSP_JQ8900_UART, &USART_InitStructure); // 初始化串口2
// 优先级,无优先级分组
NVIC_SetPriority(BSP_JQ8900_IRQ, 0);
// UARTx中断使能
NVIC_EnableIRQ(BSP_JQ8900_IRQ);
// 使能UARTx RC中断
USART_ITConfig(BSP_JQ8900_UART, USART_IT_RC, ENABLE);
}
/******************************************************************
* 函 数 名 称:JQ8900_USART_Send_Bit
* 函 数 说 明:向JQ8900模块发送单个字符
* 函 数 形 参:ch=字符
* 函 数 返 回:无
* 作 者:LC
* 备 注:无
******************************************************************/
void JQ8900_USART_Send_Bit(unsigned char ch)
{
USART_SendData(BSP_JQ8900_UART, (uint8_t)ch);
while( RESET == USART_GetFlagStatus(BSP_JQ8900_UART, USART_FLAG_TXE) ){} // 等待发送数据缓冲区标志置位
}
/******************************************************************
* 函 数 名 称:JQ8900_USART_send_String
* 函 数 说 明:向JQ8900模块发送字符串
* 函 数 形 参:str=发送的字符串
* 函 数 返 回:无
* 作 者:LC
* 备 注:无
******************************************************************/
void JQ8900_USART_send_String(unsigned char *str, unsigned int len)
{
while( len-- )
{
JQ8900_USART_Send_Bit(*str++);
}
}
//清除串口接收的数据
/******************************************************************
* 函 数 名 称:Clear_JQ8900_RX_BUFF
* 函 数 说 明:清除JQ8900发过来的数据
* 函 数 形 参:无
* 函 数 返 回:无
* 作 者:LC
* 备 注:无
******************************************************************/
void Clear_JQ8900_RX_BUFF(void)
{
unsigned char i = JQ8900_RX_LEN_MAX-1;
while(i)
{
JQ8900_RX_BUFF[i--] = 0;
}
JQ8900_RX_LEN = 0;
JQ8900_RX_FLAG = 0;
}
/******************************************************************
* 函 数 名 称:JQ8900_Init
* 函 数 说 明:JQ8900模块初始化
* 函 数 形 参:无
* 函 数 返 回:无
* 作 者:LC
* 备 注:JQ8900的默认波特率是9600
******************************************************************/
void JQ8900_Init(void)
{
JQ8900_USART_Init(9600);//默认波特率为9600
GPIO_InitTypeDef GPIO_InitStruct; // GPIO初始化结构体
BSP_JQ8900_GPIO_RCC_ENABLE(); // 使能GPIO时钟
GPIO_InitStruct.Pins = GPIO_JQ8900_APP; // GPIO引脚
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // 推挽输出
GPIO_InitStruct.Speed = GPIO_SPEED_HIGH; // 输出速度高
GPIO_Init(BSP_JQ8900_GPIO_PORT, &GPIO_InitStruct); // 初始化
SET_JQ8900_APP(1);
}
/**********************************************************
* 函 数 名 称:SendData
* 函 数 功 能:一线串行通信控制
* 传 入 参 数:addr=发送的指令
* 函 数 返 回:无
* 作 者:LC
* 备 注:无
**********************************************************/
void SendData ( unsigned char addr )
{
unsigned char i;
SET_JQ8900_APP(1); /*开始拉高 */
delay_us ( 500 );
SET_JQ8900_APP(0);/*开始引导码*/
delay_ms ( 4 );/*此处延时最少要大于2ms,官方建议4MS */
for ( i = 0; i < 8; i++ ) /*总共8位数据,从低位开始 */
{
//数据总是从1开始
SET_JQ8900_APP(1);
if ( addr & 0x01 )
{
delay_us ( 1300 );/*3:1表示数据位1,每个位用两个脉冲表示 */
SET_JQ8900_APP(0);
delay_us ( 500 );
}
else
{
delay_us ( 500 );/*1:3表示数据位0 ,每个位用两个脉冲表示 */
SET_JQ8900_APP(0);
delay_us ( 1300 );
}
addr >>= 1;
}
SET_JQ8900_APP(1);
delay_ms(10);//两个字节之间延时建议在10ms以上
}
/******************************************************************
* 函 数 名 称:JQ8900_USART_IRQHandler
* 函 数 说 明:连接JQ8900的串口中断服务函数
* 函 数 形 参:无
* 函 数 返 回:无
* 作 者:LC
* 备 注:无
******************************************************************/
void JQ8900_USART_IRQHandler(void)
{
if(USART_GetITStatus(BSP_JQ8900_UART,USART_IT_RC) != RESET) // 接收缓冲区不为空
{
//接收数据
JQ8900_RX_BUFF[ JQ8900_RX_LEN ] = USART_ReceiveData(BSP_JQ8900_UART);
#if DEBUG
//测试,查看接收到了什么数据
printf("%c", JQ8900_RX_BUFF[ JQ8900_RX_LEN ]);
#endif
//接收长度限制
JQ8900_RX_LEN = ( JQ8900_RX_LEN + 1 ) % JQ8900_RX_LEN_MAX;
JQ8900_RX_BUFF[JQ8900_RX_LEN] = '\0'; //字符串结尾补 '\0'
JQ8900_RX_FLAG = SET; // 接收完成
USART_ClearITPendingBit(BSP_JQ8900_UART, USART_IT_RC); // 清除中断标志位
}
}
在文件bsp_jq8900.h中,编写如下代码。
/*
* Change Logs:
* Date Author Notes
* 2024-06-25 LCKFB-LP first version
*/
#ifndef _BSP_JQ8900_H_
#define _BSP_JQ8900_H_
#include "string.h"
#include "board.h"
//是否开启串口调试,查看WIFI回显数据
#define DEBUG 1
#define JQ8900_RX_LEN_MAX 250 //串口接收最大长度
/**************************** 串口配置 ****************************/
#define BSP_JQ8900_GPIO_RCC_ENABLE() __RCC_GPIOA_CLK_ENABLE() // 串口TX的端口时钟
#define BSP_JQ8900_UART_RCC_ENABLE() __RCC_UART2_CLK_ENABLE() // 串口2的时钟
#define BSP_JQ8900_AF_UART_TX() PA02_AFx_UART2TXD()
#define BSP_JQ8900_AF_UART_RX() PA03_AFx_UART2RXD()
#define BSP_JQ8900_GPIO_PORT CW_GPIOA // GPIO的端口
#define BSP_JQ8900_TX_PIN GPIO_PIN_2 // 串口TX的引脚
#define BSP_JQ8900_RX_PIN GPIO_PIN_3 // 串口RX的引脚
#define BSP_JQ8900_UART CW_UART2 // 串口2
#define BSP_JQ8900_IRQ UART2_IRQn // 串口2中断
#define BSP_JQ8900_IRQHandler UART2_IRQHandler // 串口2中断服务函数
#define GPIO_JQ8900_APP GPIO_PIN_1
#define SET_JQ8900_APP(x) GPIO_WritePin(BSP_JQ8900_GPIO_PORT, GPIO_JQ8900_APP, x?GPIO_Pin_SET:GPIO_Pin_RESET);
void JQ8900_Init(void);
void JQ8900_USART_send_String(unsigned char *str, unsigned int len);
void SendData ( unsigned char addr );
void JQ8900_USART_Send_Bit(unsigned char ch);
#endif
四、移植验证
在自己工程中的main主函数中,编写如下。
/*
* Change Logs:
* Date Author Notes
* 2024-06-25 LCKFB-LP first version
*/
#include "board.h"
#include "stdio.h"
#include "bsp_uart.h"
#include "bsp_jq8900.h"
int32_t main(void)
{
//下一曲(没有指定曲目则从1开始一直播报到最后)
uint8_t send_buff[4] = {0xAA,0x06,0x00,0xB0};
board_init();
uart1_init(115200U);
JQ8900_Init();
printf("Start...\r\n");
while(1)
{
//两线串口测试
JQ8900_USART_send_String(send_buff,4);
delay_ms(3000);
}
}
移植现象:
移植现象:持续播报模块内部的全部语音。
模块移植成功案例代码:
链接:https://pan.baidu.com/s/1XcBxcE_BNWIzfEHu9G5c1g?pwd=LCKF
提取码:LCKF