當(dāng)前位置:首頁 > 公眾號精選 > 嵌入式云IOT技術(shù)圈
[導(dǎo)讀]上節(jié),我們了解了小熊派上的ESP8266模塊,這節(jié),我們實(shí)現(xiàn)一個程序,讓手機(jī)發(fā)指令來控制開發(fā)板上LED燈的亮滅吧,上節(jié)的文章鏈接如下: 基于小熊派WIFI-ESP8266實(shí)踐(上) 1、了解硬件 編寫程序之前先來看看ESP8266硬件模塊的接口電路原理圖: 以下是ESP8266模塊

上節(jié),我們了解了小熊派上的ESP8266模塊,這節(jié),我們實(shí)現(xiàn)一個程序,讓手機(jī)發(fā)指令來控制開發(fā)板上LED燈的亮滅吧,上節(jié)的文章鏈接如下:

基于小熊派WIFI-ESP8266實(shí)踐(上)

1、了解硬件

編寫程序之前先來看看ESP8266硬件模塊的接口電路原理圖:

以下是ESP8266模塊對應(yīng)底板的硬件連接:

LPUART是什么鬼?我們具體來看看STM32L431RCTx這款芯片關(guān)于LPUART的描述吧,該介紹位于STM32L431 Datasheet的第48頁:

文檔的意思大概是,這是一個低功耗的UART,可以以更低的時鐘頻率實(shí)現(xiàn)高波特率的通信,同時支持從停止模式喚醒且喚醒事件是可編程的,還有就是具有極低的功耗,如果是更高速度的時鐘還可以更高的波特率進(jìn)行數(shù)據(jù)傳輸。


既然是這樣,我們就把它當(dāng)普通串口使用就行啦!其余的功能后面用到了再去詳細(xì)了解!


在軟件編程之前,我們先來了解下與ESP8266通信相關(guān)的注意事項(xiàng),打開開發(fā)板ESP8266相關(guān)的規(guī)格書,簡要瀏覽一下,我們可以看到以下的描述:

2、STM32CubeMX配置

這里我們直接之前利用上次編寫光強(qiáng)那個工程就可以了,鏈接如下:

基于小熊派光強(qiáng)傳感器BH1750狀態(tài)機(jī)驅(qū)動項(xiàng)目再度升級(帶上位機(jī)曲線顯示)

在此基礎(chǔ)上添加ESP8266的串口,所以在STM32CubeMx對應(yīng)的LPUART1的配置如下,其余參數(shù)默認(rèn)即可,其余的關(guān)于ESP8266的上電,硬件復(fù)位這些管腳都不需要配置,因?yàn)橛布o我們做好了,我們專注于與ESP8266通信就可以了。

由于軟件接收的AT指令回復(fù)有可能是不定長數(shù)據(jù),且可能存在多個\r\n的情況,所以這里我們使用DMA來做接收會更簡單一些,一般用環(huán)形緩沖實(shí)現(xiàn)也可以,但是STM32有這么優(yōu)秀的DMA功能,我當(dāng)然用!

由于AT指令是一個處理收發(fā)的過程,所以我們還需要將接收中斷配置上:

3、軟件編程

有了ESP8266,能做的事情很多,比如,讓我們來設(shè)計幾個簡單的控制指令:

指令 功能
LEDON 打開底板上的LED燈
LEDOFF 關(guān)閉底板上的LED燈
LEDBLINK 讓底板上的LED燈閃爍
PLOTDISPLAY 顯示上位機(jī)曲線(打印數(shù)據(jù))
PLOTCLOSEDISPLAY 關(guān)閉上位機(jī)曲線顯示(取消打印數(shù)據(jù))

指令下發(fā)代碼實(shí)現(xiàn)框架如下:

/*wifi接收指令處理*/
static void Wifi_Recv_Cmd_Process(void)
{
static int cmd_index = 0 ;
char display_buf[20] = {0};
char *cmd[] = {"LEDON", "LEDOFF", "LEDBLINK", "PLOTDISPLAY", "PLOTCLOSEDISPLAY"};

if(strstr((char *)esp8266_info.rx_buffer, cmd[cmd_index]) != NULL)
{
HAL_UART_DMAStop(&hlpuart1);

switch(cmd_index)
{
case 0:
printf("接收到開燈指令\n");
printf("接收到客戶端發(fā)來的指令:%s\n", esp8266_info.rx_buffer);
HAL_GPIO_WritePin(GPIOC, BOARD_LED_Pin, GPIO_PIN_SET);
break ;

case 1:
printf("接收到關(guān)燈指令\n");
printf("接收到客戶端發(fā)來的指令:%s\n", esp8266_info.rx_buffer);
LED_BLINK_controld(0);
HAL_GPIO_WritePin(GPIOC, BOARD_LED_Pin, GPIO_PIN_RESET);
break ;

case 2:
printf("接收到閃燈指令\n");
printf("接收到客戶端發(fā)來的指令:%s\n", esp8266_info.rx_buffer);
LED_BLINK_controld(1);
break ;

case 3:
printf("接收到顯示曲線指令\n");
printf("接收到客戶端發(fā)來的指令:%s\n", esp8266_info.rx_buffer);
plot_display_controld(1);
break ;

case 4:
printf("接收到關(guān)閉顯示曲線指令\n");
printf("接收到客戶端發(fā)來的指令:%s\n", esp8266_info.rx_buffer);
plot_display_controld(0);
break ;
}

LCD_Fill(0, 146, 240, 146 + 24, BLACK);
sprintf(display_buf, "CMD:%s", cmd[cmd_index]);
LCD_ShowString(0, 146, 240, 24, 24, (char *)display_buf); //顯示字符串,字體大小16*16
memset(esp8266_info.rx_buffer, 0, RX_BUFF_SIZE);
HAL_UART_Receive_DMA(&hlpuart1, esp8266_info.rx_buffer, RX_BUFF_SIZE);
}

++cmd_index ;

if(5 == cmd_index)
cmd_index = 0 ;
}

最終看到的效果:這里把數(shù)據(jù)顯示做了相應(yīng)的調(diào)整,我使用了手機(jī)的一個TCP/UDP測試工具,連接了ESP8266,然后下發(fā)指令,就像下面這樣:

當(dāng)匹配到數(shù)據(jù)中含有對應(yīng)指令的時候,則執(zhí)行具體的操作,并將指令顯示到LCD上。

那么要實(shí)現(xiàn)這樣,就必須把ESP8266作為服務(wù)器,手機(jī)作為客戶端,客戶端連接服務(wù)器后,向服務(wù)器發(fā)送指令,我們來看看esp8266.h的實(shí)現(xiàn):

#ifndef __ESP8266_H
#define __ESP8266_H
#include "main.h"


/*發(fā)送數(shù)據(jù)最大長度*/
#define TX_BUFF_SIZE 50
/*接收數(shù)據(jù)最大長度*/
#define RX_BUFF_SIZE 150

/*ESP8266作為熱點(diǎn)時的名稱*/
#define WIFI_HOT_SPOT_SSID "BearPi_ESP8266"
/*ESP8266作為熱點(diǎn)時的密碼*/
#define WIFI_HOT_SPOT_PASSWORD "12345678"
/*AP PORT*/
#define AP_PORT 8080

typedef struct
{
/*wifi ap運(yùn)行狀態(tài)機(jī)*/
uint8_t wifi_apr_status ;
/*AT指令發(fā)送緩存*/
uint8_t tx_buffer[TX_BUFF_SIZE];
/*接收緩存*/
uint8_t rx_buffer[RX_BUFF_SIZE];
/*發(fā)送標(biāo)志*/
uint8_t tx_flag ;
/*multi_timer定時器句柄*/
Timer wifi_timer ;
/*定時器計數(shù)值*/
uint16_t wifi_timer_count ;
/*wifi完成標(biāo)志*/
uint8_t wifi_completed_flag ;
/*定時回調(diào)*/
void (*wifi_timeout_cb)(void);
} wifi_ap_info ;



/*測試WIFI*/
#define WIFI_AT_TEST "AT\r\n"
/*設(shè)置或關(guān)閉回顯*/
#define WIFI_ATE_SET "ATE%d\r\n"
/*設(shè)置WIFI模式*/
#define WIFI_AT_SET_MODE "AT+CWMODE=%d\r\n"
/*創(chuàng)建WIFI熱點(diǎn)*/
#define WIFI_AT_SAP "AT+CWSAP=\"%s\",\"%s\",%d,%d\r\n"
/*配置多連接模式*/
#define WIFI_AT_MULTPLE "AT+CIPMUX=%d\r\n"
/*開啟服務(wù)器模式*/
#define WIFI_OPEN_SMODE "AT+CIPSERVER=%d,%d\r\n"
/*設(shè)置與服務(wù)器的主動斷開時間*/
#define WIFI_SET_STO "AT+CIPSTO=%d\r\n"
/*查看WIFI作為服務(wù)器時的地址*/
#define WIFI_VIEW_ADDR "AT+CIFSR\r\n"


/*每個狀態(tài)機(jī)執(zhí)行的超時查詢時間*/
#define WIFI_TEST_TIMEOUT 1000
#define WIFI_SET_ATE_TIMEOUT 200
#define WIFI_SET_MODE_TIMEOUT 200
#define WIFI_BUILD_AP_INFO_TIMEOUT 4000
#define WIFI_CONFIG_MULTPLE_CONNECT_TIMEOUT 200
#define WIFI_OPEN_SERVER_MODE_TIMEOUT 1000
#define WIFI_VIEW_IPADDR_TIMEOUT 1000

/*每個狀態(tài)機(jī)對應(yīng)的序號*/
enum
{
ITEM_WIFI_TEST = 0,
ITEM_WIFI_SATE,
ITEM_WIFI_SMODE,
ITEM_WIFI_BUIAP,
ITEM_WIFI_CMULT,
ITEM_WIFI_OSERV,
ITEM_WIFI_STIMO,
ITEM_WIFI_VADDR,
ITEM_WIFI_GDATA,
ITEM_WIFI_ERROR = 99
};

/*ESP8266作為AP模式進(jìn)行初始化*/
void Init_ESP8266_AP_Mode(void);
/*Wifi作為服務(wù)器時的服務(wù)*/
void ESP8266_AP_Mode_Setting(void);
/*wifi發(fā)送命令*/
void wifi_send_cmd(const char *format, ...);

#endif //__ESP8266_h

這里我們再次運(yùn)用了multi_timer,可見我多么喜歡它!

由于代碼較多,我們只挑核心部分出來講解就可以了,其它留給讀者自行實(shí)踐。

1、Init_ESP8266_AP_Mode函數(shù)實(shí)現(xiàn)

/*ESP8266作為AP模式進(jìn)行初始化*/
void Init_ESP8266_AP_Mode(void)
{
esp8266_info.tx_flag = 1 ;
esp8266_info.wifi_apr_status = ITEM_WIFI_TEST ;
esp8266_info.wifi_timer_count = 0 ;
esp8266_info.wifi_completed_flag = 1 ;
esp8266_info.wifi_timeout_cb = wifi_timeout_callback ;
/*開啟1ms軟件定時器*/
timer_init(&esp8266_info.wifi_timer, esp8266_info.wifi_timeout_cb, 1, 1);
timer_start(&esp8266_info.wifi_timer);
}

這里對結(jié)構(gòu)體參數(shù)進(jìn)行了初始化,在這里用multi_timer開啟一個1ms的軟件定時器,定時時基由系統(tǒng)時鐘產(chǎn)生,一次中斷為1ms,主要是用來產(chǎn)生延時的,發(fā)送完AT指令給ESP8266后,一般要延時一段時間,再去查串口緩存區(qū)是否有ESP8266的回復(fù)數(shù)據(jù),定時器回調(diào)函數(shù)如下:

static void wifi_timeout_callback(void)
{
if(0 == esp8266_info.wifi_completed_flag)
++esp8266_info.wifi_timer_count ;
}

當(dāng)esp8266_info.wifi_completed_flag標(biāo)志為0時,esp8266_info.wifi_timer_count變量自加產(chǎn)生對應(yīng)的延時,當(dāng)esp8266_info.wifi_completed_flag為1時,回調(diào)函數(shù)不會做任何操作,根據(jù)這樣的想法,我們簡單實(shí)現(xiàn)發(fā)送WIFI測試指令AT\r\n

/*測試*/
void WIFI_Test(void)
{
uint8_t ret = 0 ;
static uint8_t err_count = 0 ;

/*當(dāng)前為發(fā)送狀態(tài)*/
if(1 == esp8266_info.tx_flag)
{
/*復(fù)位參數(shù)*/
Reset_Wifi_Para();
/*發(fā)送測試指令*/
wifi_send_cmd(WIFI_AT_TEST);
/*將發(fā)送狀態(tài)設(shè)置為0,即為接收狀態(tài)*/
esp8266_info.tx_flag = 0 ;
/*清空定時計數(shù)器*/
esp8266_info.wifi_timer_count = 0 ;
/*開啟定時計數(shù)標(biāo)志*/
esp8266_info.wifi_completed_flag = 0 ;
}
/*當(dāng)前為接收狀態(tài)*/
else
{
/*判斷定時計數(shù)到WIFI_TEST_TIMEOUT==>1000ms了沒有?*/
if(WIFI_TEST_TIMEOUT == esp8266_info.wifi_timer_count)
{
/*關(guān)閉定時計數(shù)標(biāo)志*/
esp8266_info.wifi_completed_flag = 1 ;
/*清空定時計數(shù)器*/
esp8266_info.wifi_timer_count = 0 ;
/*檢查DMA接收緩存中是否包含OK子串*/
ret = AT_Check_Answer("OK");

/*失敗,錯誤超過3次,返回出錯狀態(tài)*/
if(ret != 0)
{
esp8266_info.tx_flag = 1 ;
++err_count;

if(err_count > 3)
{
err_count = 0 ;
esp8266_info.wifi_apr_status = ITEM_WIFI_ERROR ;
printf("WIFI初始化失敗\n");
}
}
else
{
esp8266_info.tx_flag = 1 ;
/*將狀態(tài)標(biāo)記為下一個指令的處理流程*/
esp8266_info.wifi_apr_status = ITEM_WIFI_SATE ;
printf("wifi測試成功! 回復(fù)%s\n", esp8266_info.rx_buffer);
}
}
}
}

復(fù)位參數(shù)的實(shí)現(xiàn)邏輯:

/*復(fù)位wifi收發(fā)參數(shù)*/
void Reset_Wifi_Para(void)
{
/*停止DMA接收*/
HAL_UART_DMAStop(&hlpuart1);
/*清除發(fā)送接收緩存*/
memset(esp8266_info.tx_buffer, 0, TX_BUFF_SIZE);
memset(esp8266_info.rx_buffer, 0, RX_BUFF_SIZE);
/*開啟DMA接收*/
HAL_UART_Receive_DMA(&hlpuart1, esp8266_info.rx_buffer, RX_BUFF_SIZE);
}

雖然這段測試AT的代碼相比很多例程看起來都長,但是它沒有硬延時!沒有硬延時!沒有硬延時!重要的事情說三遍,這里就體現(xiàn)狀態(tài)機(jī)結(jié)合定時器超時處理的好處了,那么其他狀態(tài)也是類似的實(shí)現(xiàn)方法,大家可以下載工程源碼去自己看,最后用這么一個函數(shù)來綜合體現(xiàn)狀態(tài)機(jī)的切換:

2、ESP8266_AP_Mode_Setting函數(shù)實(shí)現(xiàn)

/*ESP8266作為AP模式進(jìn)行設(shè)置*/
void ESP8266_AP_Mode_Setting(void)
{
static uint8_t error_flag = 0 ;

switch(esp8266_info.wifi_apr_status)
{
case ITEM_WIFI_TEST:
WIFI_Test();
break ;

case ITEM_WIFI_SATE:
WIFI_SET_ATE(1);
break ;

case ITEM_WIFI_SMODE:
WIFI_SET_MODE(2);
break ;

case ITEM_WIFI_BUIAP:
WIFI_BUILD_AP_INFO(WIFI_HOT_SPOT_SSID, WIFI_HOT_SPOT_PASSWORD, 1, 4);
break ;

case ITEM_WIFI_CMULT:
WIFI_CONFIG_MULTPLE_CONNECT(1);
break ;

case ITEM_WIFI_OSERV:
WIFI_OPEN_SERVER_MODE(1, AP_PORT);
break ;

case ITEM_WIFI_STIMO:
WIFI_CONFIG_SERVER_TIMEOUT(0);
break ;

case ITEM_WIFI_VADDR:
WIFI_VIEW_IPADDR();
break ;

case ITEM_WIFI_GDATA:
Wifi_Recv_Cmd_Process();
break ;

default:
if(0 == error_flag)
{
error_flag = 1 ;

if(ITEM_WIFI_ERROR == esp8266_info.wifi_apr_status)
{
printf("WIFI出錯\n");
}
}

break ;
}
}

最后,我們只要在主函數(shù)中循環(huán)調(diào)用這段代碼就行了:

while (1)
{
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
Light_Sensor_Service(); /*光強(qiáng)傳感器處理和現(xiàn)實(shí)*/
ESP8266_AP_Mode_Setting(); /*ESP8266 AP模式下的狀態(tài)機(jī)*/
LED_Blink_Service(); /*LED閃爍燈服務(wù)*/
timer_loop(); /*multi_timer循環(huán)代用*/
}

3、wifi_send_cmd函數(shù)實(shí)現(xiàn)

wifi發(fā)送指令的實(shí)現(xiàn):

/*wifi發(fā)送命令*/
void wifi_send_cmd(const char *format, ...)
{
va_list args;
uint32_t length;
va_start(args, format);
length = vsnprintf((char *)esp8266_info.tx_buffer, sizeof(esp8266_info.tx_buffer), (char *)format, args);
va_end(args);
HAL_UART_Transmit(&hlpuart1, (uint8_t *)esp8266_info.tx_buffer, length, HAL_MAX_DELAY);
}

發(fā)送很簡單,就把這個函數(shù)當(dāng)做printf函數(shù)來用就可以了,這是一個可變參函數(shù),參數(shù)個數(shù)可以根據(jù)format做動態(tài)調(diào)整。

大致框架講解完畢了,接下來看下效果:

發(fā)送LED燈閃爍指令:

發(fā)送曲線顯示指令(代碼默認(rèn)將曲線顯示用標(biāo)志位做了屏蔽,這里只要看到串口有一連串?dāng)?shù)據(jù)即可):

例程下載

鏈接:https://pan.baidu.com/s/1P8yjbuljvcqZute1ToGjVQ
提取碼:ni46
復(fù)制這段內(nèi)容后打開百度網(wǎng)盤手機(jī)App,操作更方便哦

公眾號粉絲福利時刻

這里我給大家申請到了福利,本公眾號讀者購買小熊派開發(fā)板可享受9折優(yōu)惠,有需要購買小熊派的朋友,淘寶搜索即可,跟客戶說你是公眾號:嵌入式云IOT技術(shù)圈 的粉絲,立享9折優(yōu)惠!

往期精彩

網(wǎng)紅物聯(lián)網(wǎng)開發(fā)板小熊派使用評測

基于小熊派WIFI-ESP8266實(shí)踐(上)

超輕量級網(wǎng)紅軟件定時器multi_timer(51+stm32雙平臺實(shí)戰(zhàn))

基于小熊派光強(qiáng)傳感器BH1750實(shí)踐(multi_timer+狀態(tài)機(jī)工程應(yīng)用)

基于小熊派光強(qiáng)傳感器BH1750狀態(tài)機(jī)驅(qū)動項(xiàng)目升級(帶LCD屏顯示)

基于小熊派光強(qiáng)傳感器BH1750狀態(tài)機(jī)驅(qū)動項(xiàng)目再度升級(帶上位機(jī)曲線顯示)

若覺得本次分享的文章對您有幫助,隨手點(diǎn)[在看]并轉(zhuǎn)發(fā)分享,也是對我的支持。

免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點(diǎn),不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉