LPT26x-HSF-4MB-Hilink_14.2..../application/samples/wifi/sta_sample/sta_sample.c

279 lines
10 KiB
C
Raw Permalink Normal View History

2025-05-13 22:00:58 +08:00
/**
* Copyright (c) HiSilicon (Shanghai) Technologies Co., Ltd. 2022-2023. All rights reserved.
*
* Description: Application core main function for standard \n
*
* History: \n
* 2022-07-27, Create file. \n
*/
#include "lwip/netifapi.h"
#include "wifi_hotspot.h"
#include "wifi_hotspot_config.h"
#include "td_base.h"
#include "td_type.h"
#include "stdlib.h"
#include "uart.h"
#include "cmsis_os2.h"
#include "app_init.h"
#include "soc_osal.h"
#define WIFI_IFNAME_MAX_SIZE 16
#define WIFI_MAX_SSID_LEN 33
#define WIFI_SCAN_AP_LIMIT 64
#define WIFI_MAC_LEN 6
#define WIFI_STA_SAMPLE_LOG "[WIFI_STA_SAMPLE]"
#define WIFI_NOT_AVALLIABLE 0
#define WIFI_AVALIABE 1
#define WIFI_GET_IP_MAX_COUNT 300
#define WIFI_TASK_PRIO (osPriority_t)(13)
#define WIFI_TASK_DURATION_MS 2000
#define WIFI_TASK_STACK_SIZE 0x1000
static td_void wifi_scan_state_changed(td_s32 state, td_s32 size);
static td_void wifi_connection_changed(td_s32 state, const wifi_linked_info_stru *info, td_s32 reason_code);
wifi_event_stru wifi_event_cb = {
.wifi_event_connection_changed = wifi_connection_changed,
.wifi_event_scan_state_changed = wifi_scan_state_changed,
};
enum {
WIFI_STA_SAMPLE_INIT = 0, /* 0:初始态 */
WIFI_STA_SAMPLE_SCANING, /* 1:扫描中 */
WIFI_STA_SAMPLE_SCAN_DONE, /* 2:扫描完成 */
WIFI_STA_SAMPLE_FOUND_TARGET, /* 3:匹配到目标AP */
WIFI_STA_SAMPLE_CONNECTING, /* 4:连接中 */
WIFI_STA_SAMPLE_CONNECT_DONE, /* 5:关联成功 */
WIFI_STA_SAMPLE_GET_IP, /* 6:获取IP */
} wifi_state_enum;
static td_u8 g_wifi_state = WIFI_STA_SAMPLE_INIT;
/*****************************************************************************
STA
*****************************************************************************/
static td_void wifi_scan_state_changed(td_s32 state, td_s32 size)
{
UNUSED(state);
UNUSED(size);
PRINT("%s::Scan done!.\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_SCAN_DONE;
return;
}
/*****************************************************************************
STA
*****************************************************************************/
static td_void wifi_connection_changed(td_s32 state, const wifi_linked_info_stru *info, td_s32 reason_code)
{
UNUSED(info);
UNUSED(reason_code);
if (state == WIFI_NOT_AVALLIABLE) {
PRINT("%s::Connect fail!. try agin !\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_INIT;
} else {
PRINT("%s::Connect succ!.\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_CONNECT_DONE;
}
}
/*****************************************************************************
STA AP
*****************************************************************************/
td_s32 example_get_match_network(wifi_sta_config_stru *expected_bss)
{
td_s32 ret;
td_u32 num = 64; /* 64:扫描到的Wi-Fi网络数量 */
td_char expected_ssid[] = "my_softAP";
td_char key[] = "my_password"; /* 待连接的网络接入密码 */
td_bool find_ap = TD_FALSE;
td_u8 bss_index;
/* 获取扫描结果 */
td_u32 scan_len = sizeof(wifi_scan_info_stru) * WIFI_SCAN_AP_LIMIT;
wifi_scan_info_stru *result = osal_kmalloc(scan_len, OSAL_GFP_ATOMIC);
if (result == TD_NULL) {
return -1;
}
memset_s(result, scan_len, 0, scan_len);
ret = wifi_sta_get_scan_info(result, &num);
if (ret != 0) {
osal_kfree(result);
return -1;
}
/* 筛选扫描到的Wi-Fi网络选择待连接的网络 */
for (bss_index = 0; bss_index < num; bss_index ++) {
if (strlen(expected_ssid) == strlen(result[bss_index].ssid)) {
if (memcmp(expected_ssid, result[bss_index].ssid, strlen(expected_ssid)) == 0) {
find_ap = TD_TRUE;
break;
}
}
}
/* 未找到待连接AP,可以继续尝试扫描或者退出 */
if (find_ap == TD_FALSE) {
osal_kfree(result);
return -1;
}
/* 找到网络后复制网络信息和接入密码 */
if (memcpy_s(expected_bss->ssid, WIFI_MAX_SSID_LEN, expected_ssid, strlen(expected_ssid)) != 0) {
osal_kfree(result);
return -1;
}
if (memcpy_s(expected_bss->bssid, WIFI_MAC_LEN, result[bss_index].bssid, WIFI_MAC_LEN) != 0) {
osal_kfree(result);
return -1;
}
expected_bss->security_type = result[bss_index].security_type;
if (memcpy_s(expected_bss->pre_shared_key, WIFI_MAX_SSID_LEN, key, strlen(key)) != 0) {
osal_kfree(result);
return -1;
}
expected_bss->ip_type = 1; /* 1IP类型为动态DHCP获取 */
osal_kfree(result);
return 0;
}
/*****************************************************************************
STA
*****************************************************************************/
td_bool example_check_connect_status(td_void)
{
td_u8 index;
wifi_linked_info_stru wifi_status;
/* 获取网络连接状态共查询5次每次间隔500ms */
for (index = 0; index < 5; index ++) {
(void)osDelay(50); /* 50: 延时500ms */
memset_s(&wifi_status, sizeof(wifi_linked_info_stru), 0, sizeof(wifi_linked_info_stru));
if (wifi_sta_get_ap_info(&wifi_status) != 0) {
continue;
}
if (wifi_status.conn_state == 1) {
return 0; /* 连接成功退出循环 */
}
}
return -1;
}
/*****************************************************************************
STA DHCP状态查询
*****************************************************************************/
td_bool example_check_dhcp_status(struct netif *netif_p, td_u32 *wait_count)
{
if ((ip_addr_isany(&(netif_p->ip_addr)) == 0) && (*wait_count <= WIFI_GET_IP_MAX_COUNT)) {
/* DHCP成功 */
PRINT("%s::STA DHCP success.\r\n", WIFI_STA_SAMPLE_LOG);
return 0;
}
if (*wait_count > WIFI_GET_IP_MAX_COUNT) {
PRINT("%s::STA DHCP timeout, try again !.\r\n", WIFI_STA_SAMPLE_LOG);
*wait_count = 0;
g_wifi_state = WIFI_STA_SAMPLE_INIT;
}
return -1;
}
td_s32 example_sta_function(td_void)
{
td_char ifname[WIFI_IFNAME_MAX_SIZE + 1] = "wlan0"; /* 创建的STA接口名 */
wifi_sta_config_stru expected_bss = {0}; /* 连接请求信息 */
struct netif *netif_p = TD_NULL;
td_u32 wait_count = 0;
/* 创建STA接口 */
if (wifi_sta_enable() != 0) {
return -1;
}
PRINT("%s::STA enable succ.\r\n", WIFI_STA_SAMPLE_LOG);
do {
(void)osDelay(1); /* 1: 等待10ms后判断状态 */
if (g_wifi_state == WIFI_STA_SAMPLE_INIT) {
PRINT("%s::Scan start!\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_SCANING;
/* 启动STA扫描 */
if (wifi_sta_scan() != 0) {
g_wifi_state = WIFI_STA_SAMPLE_INIT;
continue;
}
} else if (g_wifi_state == WIFI_STA_SAMPLE_SCAN_DONE) {
/* 获取待连接的网络 */
if (example_get_match_network(&expected_bss) != 0) {
PRINT("%s::Do not find AP, try again !\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_INIT;
continue;
}
g_wifi_state = WIFI_STA_SAMPLE_FOUND_TARGET;
} else if (g_wifi_state == WIFI_STA_SAMPLE_FOUND_TARGET) {
PRINT("%s::Connect start.\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_CONNECTING;
/* 启动连接 */
if (wifi_sta_connect(&expected_bss) != 0) {
g_wifi_state = WIFI_STA_SAMPLE_INIT;
continue;
}
} else if (g_wifi_state == WIFI_STA_SAMPLE_CONNECT_DONE) {
PRINT("%s::DHCP start.\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_GET_IP;
netif_p = netifapi_netif_find(ifname);
if (netif_p == TD_NULL || netifapi_dhcp_start(netif_p) != 0) {
PRINT("%s::find netif or start DHCP fail, try again !\r\n", WIFI_STA_SAMPLE_LOG);
g_wifi_state = WIFI_STA_SAMPLE_INIT;
continue;
}
} else if (g_wifi_state == WIFI_STA_SAMPLE_GET_IP) {
if (example_check_dhcp_status(netif_p, &wait_count) == 0) {
break;
}
wait_count++;
}
} while (1);
return 0;
}
int sta_sample_init(void *param)
{
param = param;
/* 注册事件回调 */
if (wifi_register_event_cb(&wifi_event_cb) != 0) {
PRINT("%s::wifi_event_cb register fail.\r\n", WIFI_STA_SAMPLE_LOG);
return -1;
}
PRINT("%s::wifi_event_cb register succ.\r\n", WIFI_STA_SAMPLE_LOG);
/* 等待wifi初始化完成 */
while (wifi_is_wifi_inited() == 0) {
(void)osDelay(10); /* 1: 等待100ms后判断状态 */
}
PRINT("%s::wifi init succ.\r\n", WIFI_STA_SAMPLE_LOG);
if (example_sta_function() != 0) {
PRINT("%s::example_sta_function fail.\r\n", WIFI_STA_SAMPLE_LOG);
return -1;
}
return 0;
}
static void sta_sample_entry(void)
{
osThreadAttr_t attr;
attr.name = "sta_sample_task";
attr.attr_bits = 0U;
attr.cb_mem = NULL;
attr.cb_size = 0U;
attr.stack_mem = NULL;
attr.stack_size = WIFI_TASK_STACK_SIZE;
attr.priority = WIFI_TASK_PRIO;
if (osThreadNew((osThreadFunc_t)sta_sample_init, NULL, &attr) == NULL) {
PRINT("%s::Create sta_sample_task fail.\r\n", WIFI_STA_SAMPLE_LOG);
}
PRINT("%s::Create sta_sample_task succ.\r\n", WIFI_STA_SAMPLE_LOG);
}
/* Run the sta_sample_task. */
app_run(sta_sample_entry);