STM32+EC20实现4G无线通信

news2025/1/12 20:39:52

EC20是一款集成度非常高的4G无线通信模块,支持多种常见通信频段,能满足几乎所有的M2M(MachinetoMachine)应用需求。模块支持TCP/UDP/FTP等一众网络协议,内置多星座高精度定位GNSS接收机,快速提供准确的经纬度信息,UART接口提供AT命令控制和数据传输。

物联网很多的网关设备因需要会安装在有线网络不易布到线的地方,而有些网关则需要跟着运输工具一起移动,那么就需要产品实现与上位机服务器进行无线通信,而4G的无线通信模块就提供了一种非常便利的实现。本例使用了FreeRTOS作为实时操作系统,嵌入式代码运行在FreeRTOS之上,使用提供系统调用完成了多任务编程。

目录

概念说明

实现原理

嵌入式程序

顶层结构

任务与串口初始化

EC20任务

传感器任务 


概念说明

  • 电平转换:在我们设计的电路中,不同芯片的引脚使用的电压不同,比如常见的1.8V、3.3V、5V等,在两种不同电压芯片引脚之间进行通讯时候,我们需要使得两边的电平都符合自身的需求且能够进行正常的通讯,这就叫电平转换。我们STM32与4G模块连接的控制与通信的引脚就需要做电平转化才能正常通信。
  • AT指令:在物联网中,AT(Attention)命令集可用于控制和调测设备、通信模块入网等。AT 指令以"AT"开始,以"\r "或者"\r\n" 结尾,参数之间使用" ," 隔开,字符串参数使用双引号 "" 包裹,整形参数不适用双引号。
  • FreeRTOS:FreeRTOS是面向微控制器和小型微处理器的实时操作系统 (RTOS),本例在STM32的平台上使用了FreeRTOS作为实时操作系统,以更合理、更有效地利用CPU的资源,简化应用软件的设计,缩短系统开发时间,提供多任务编程环境,并且更好地保证系统的实时性和可靠性。
  • 4G模块:4G模块通常是插针到电路板或将贴片式模块焊到电路板上,并插入相应的SIM卡,通过4G蜂窝网络进行联网。

  • 物联网sim卡:物联网(IoT)的发展及其独特的需求推动了物联网SIM卡的发展,IoT SIM卡是传统SIM卡的变体,旨在存储用户信息并将手机连接到对应运营商的蜂窝网络。
  • APN:接入点名称(APN)用户的模块通过哪种接入方式来访问什么样的网络。一个典型的APN包含的参数有名称、MCC、MNC、接入点、类型。

实现原理

嵌入式程序运行在STM32芯片之上,通过UART与4G模块进行通信,控制协议为AT指令格式。STM32与4G模块之间还有两个引脚:

  • 开/关模块引脚:引脚拉高保持300ms后拉低开机,拉高保持800ms后拉低关机
  • 复位引脚:引脚拉高500ms后拉低复位

下图为实现原理的示意图:


嵌入式程序

顶层结构

本例基于STM32平台,完整工程代码已上传。工程一共涉及三个串口,功能分别如下:

  1. 连接4G模块完成无线数据收发
  2. 负责与传感器等数据采集模块连接
  3. 产品参数配置串口

代码的顶层结构由任务和消息队列与中断构成。程序开始初始化消息队列与串口控制器和相关GPIO之后,注册串口接收函数。接收函数在串口接收中断处调用,将信息发送到消息队列中。最后启动三个任务,来处理三个串口接收队列中的信息。完整的数据流图如下:


任务与串口初始化

下面是除了串口的初始化工作,已增加了中文注释帮助理解,主要包括任务的初始化和启动:

#define START_TASK_PRIO		1			//任务优先级
#define START_STK_SIZE 		64  		//任务堆栈大小
TaskHandle_t StartTask_Handler;			//任务句柄
void start_task(void *pvParameters);	//开始任务声明

#define EC20_TASK_PRIO		10			
#define EC20_STK_SIZE 		128  			
TaskHandle_t ec20_task_handler;			
void ec20_task(void *pvParameters);		

#define STM32_TASK_PRIO		11			
#define STM32_STK_SIZE 		128  			
TaskHandle_t stm32_task_handler;		
void stm32_task(void *pvParameters);	

#define IDLE_TASK_PRIO		8			
#define IDLE_STK_SIZE 		128  	
TaskHandle_t idle_task_handler;			
void idle_task(void *pvParameters);		

#define SET_TASK_PRIO		9		
#define SET_STK_SIZE 		128  		
TaskHandle_t set_task_handler;			
void set_task(void *pvParameters);		
int main(void)
{
	Flash_EnableReadProtection();//开Flash读保护
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//设置系统中断优先级分组4
	//创建开始任务
    xTaskCreate((TaskFunction_t )start_task,           
                (const char*    )"start_task",          
                (uint16_t       )START_STK_SIZE,        
                (void*          )NULL,                  
                (UBaseType_t    )START_TASK_PRIO,       
                (TaskHandle_t*  )&StartTask_Handler);       
    vTaskStartScheduler();       
}

/**********************
@GPIO初始化
**********************/
void gpio_init(void)
{
	GPIO_InitTypeDef  GPIO_InitStructure;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | 
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;		
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	GPIO_ResetBits(GPIOB, GPIO_InitStructure.GPIO_Pin);		
}
/**********************
@启动任务函数
**********************/
void start_task(void *pvParameters)
{
    taskENTER_CRITICAL();           //进入临界区
	gpio_init();	//GPIO初始化
	delay_init();	//延时函数初始化
	global_init();	//全局变量初始化
	msg_init();		//消息队列初始化
	uart3_init(global.baud);//串口3初始化
	uart1_init(115200);//串口1初始化
	uart2_init(global.baud);//串口2初始化
    //创建ec20任务
    xTaskCreate((TaskFunction_t )ec20_task,     	
                (const char*    )"ec20_task",   	
                (uint16_t       )EC20_STK_SIZE, 
                (void*          )NULL,				
                (UBaseType_t    )EC20_TASK_PRIO,	
                (TaskHandle_t*  )&ec20_task_handler);   
      
	//创建传感器任务
				 xTaskCreate((TaskFunction_t )stm32_task,     	
                (const char*    )"stm32_task",   	
                (uint16_t       )STM32_STK_SIZE, 
                (void*          )NULL,				
                (UBaseType_t    )STM32_TASK_PRIO,	
                (TaskHandle_t*  )&stm32_task_handler);
				
//	//创建空闲任务
//				xTaskCreate((TaskFunction_t )idle_task,     	
//                (const char*    )"idle_task",   	
//                (uint16_t       )IDLE_STK_SIZE, 
//                (void*          )NULL,				
//                (UBaseType_t    )IDLE_TASK_PRIO,	
//                (TaskHandle_t*  )&idle_task_handler);   
	//创建参数设置任务
				xTaskCreate((TaskFunction_t )set_task,     	
                (const char*    )"set_task",   	
                (uint16_t       )SET_STK_SIZE, 
                (void*          )NULL,				
                (UBaseType_t    )SET_TASK_PRIO,	
                (TaskHandle_t*  )&set_task_handler);   


    vTaskDelete(StartTask_Handler); //删除开始启动任务
    taskEXIT_CRITICAL();            //退出临界区
}

下面以串口1为例提供相关初始化以及发送和接收中断代码:

#include "global.h"
#include "usart1.h"
#include "msg.h"
#include "global.h"

extern uint8_t callback_ec20_deault(MSG *rx);
static volatile MSG *tx;

static COMMUCATION_RECV_CALLBACK callBack_recv = 0;
//串口接收中断数据处理函数
static COMMUCATION_RECV_CALLBACK default_callback = callback_ec20_deault;
//串口配置
static void uart1_config(u32 baud)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_USART1, ENABLE); 
	//PA9 -> UART1_TX
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;		
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;	
	GPIO_Init(GPIOA, &GPIO_InitStructure);

	//PA10 -> UART1_RX
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
    //USART1 参数配置
	USART_DeInit(USART1);
	USART_InitStructure.USART_BaudRate = baud;						
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;		
	USART_InitStructure.USART_StopBits = USART_StopBits_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(USART1, &USART_InitStructure); 						
	

	USART_ITConfig(USART1, USART_IT_IDLE, ENABLE );	//¿ª¿ÕÏÐÖжÏ
	USART_ITConfig(USART1, USART_IT_ERR, ENABLE);	//¿ª´íÎóÖжÏ

	USART_ClearFlag(USART1, USART_FLAG_TC);
	USART_ClearFlag(USART1, USART_FLAG_RXNE);
	USART_ClearFlag(USART1, USART_FLAG_IDLE);
	USART_ClearFlag(USART1, USART_FLAG_ORE);
	
	USART_Cmd(USART1, ENABLE);  

	//NVIC ÅäÖÃ
	NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;			
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 6;	
	NVIC_InitStructure.NVIC_IRQChannelSubPriority =0;			
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;				
	NVIC_Init(&NVIC_InitStructure);								
}


//dma接收配置 未用
static void dma_recv_config(void)
{	
	DMA_InitTypeDef DMA_InitStructure;				
	
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	
	
	DMA_DeInit(DMA1_Channel5); 
	DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&USART1->DR);	
	DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)msg_recv_ec20[0].buf;	
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;						
	DMA_InitStructure.DMA_BufferSize = global.bufferSize;						
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;		
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;					
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;	
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;			
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;							
	DMA_InitStructure.DMA_Priority = DMA_Priority_High;						
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;							
	DMA_Init(DMA1_Channel5, &DMA_InitStructure);
	
	DMA_ClearFlag(DMA1_FLAG_GL4);       //Çå³ýDMA1ËùÓбêÖ¾

	
	USART_DMACmd(USART1, USART_DMAReq_Rx, ENABLE);	
	DMA_Cmd(DMA1_Channel5, ENABLE);					
		
}

//dma发送配置 未用
static void dma_send_config(void)
{
	DMA_InitTypeDef DMA_InitStructure;
	
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	//DMA1 ʱÖÓ

	DMA_DeInit(DMA1_Channel4);
	DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)(&USART1->DR);
	DMA_InitStructure.DMA_MemoryBaseAddr = (u32)tx->buf;
	DMA_InitStructure.DMA_BufferSize = 0;							
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;				
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;			
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;	
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;					
	DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;			
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;					
	DMA_Init(DMA1_Channel4, &DMA_InitStructure);
	
	DMA_ClearFlag(DMA1_FLAG_GL4);
	

	USART_DMACmd(USART1, USART_DMAReq_Tx, ENABLE);
	DMA_Cmd(DMA1_Channel4, DISABLE);

	DMA_ClearFlag(DMA1_FLAG_GL4);
	DMA_ClearFlag(DMA1_FLAG_HT4);
	DMA_ClearFlag(DMA1_FLAG_TC4);
	DMA_ClearFlag(DMA1_FLAG_TE4);
		
}
//串口1中断处理函数
void USART1_IRQHandler(void)
{
	uint32_t  sr;
	sr = sr;
	
	if(USART_GetITStatus(USART1, USART_IT_TXE) != RESET)//·¢ËÍ¿Õ
	{
		USART_ClearITPendingBit(USART1, USART_IT_TXE);
	}
	
	if(USART_GetITStatus(USART1, USART_IT_TC) != RESET)//·¢ËÍÍê³É
	{
		USART_ClearITPendingBit(USART1, USART_IT_TC);
		USART_ITConfig(USART1, USART_IT_TC, DISABLE);                                  
	}
	
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)//ÊÕµ½
	{	 	
		USART_ClearITPendingBit(USART1, USART_IT_RXNE);//Çå½ÓÊÕÖжϱêÖ¾
	}
	
	if(USART_GetITStatus(USART1, USART_IT_ORE) != RESET)
	{
		sr = USART_ReceiveData(USART1);
		USART_ClearITPendingBit(USART1, USART_IT_ORE);
	}
	
	if(USART_GetITStatus(USART1, USART_IT_IDLE) != RESET)//¿ÕÏÐ
	{
		sr = USART_ReceiveData(USART1);					
		DMA_Cmd(DMA1_Channel5, DISABLE);				
		USART_ClearITPendingBit(USART1, USART_IT_IDLE );
		
		msg_recv_ec20[global.cur_ec20_msg].state = 1;
		msg_recv_ec20[global.cur_ec20_msg].len = global.bufferSize - DMA_GetCurrDataCounter(DMA1_Channel5);	

		callBack_recv((MSG*)&msg_recv_ec20[global.cur_ec20_msg]);//启动回调函数函数处理中断内容
		callBack_recv = default_callback;
	
		DMA1_Channel5->CMAR = (u32)(ec20_rx()->buf);
		DMA1_Channel5->CNDTR = global.bufferSize;				
		DMA_Cmd(DMA1_Channel5, ENABLE);
	}

	if(USART_GetITStatus(USART1 ,USART_IT_PE | USART_IT_FE | USART_IT_NE) != RESET)//´íÎó
	{
		sr = USART_ReceiveData(USART1);
		USART_ClearITPendingBit(USART1, USART_IT_PE | USART_IT_FE | USART_IT_NE);
	}
}
//串口接收DMA中断
void DMA1_Channel5_IRQHandler(void)
{
	if(DMA_GetITStatus(DMA1_IT_TC5) == SET)    
	{
		DMA_ClearITPendingBit(DMA1_IT_TC5);
	}
	if(DMA_GetITStatus(DMA1_IT_TE5) == SET)
	{
		DMA_ClearITPendingBit(DMA1_IT_TE5);
	}
	DMA_ClearITPendingBit(DMA1_IT_TC5);
	DMA_ClearITPendingBit(DMA1_IT_TE5);
	DMA_Cmd(DMA1_Channel5, DISABLE);
	DMA1_Channel5->CNDTR = 100;
	DMA_Cmd(DMA1_Channel5, ENABLE);
}
//串口发送DMA中断
void DMA1_Channel4_IRQHandler(void)
{
	if(DMA_GetITStatus(DMA1_IT_TC4) == SET)    
	{
		DMA_ClearITPendingBit(DMA1_IT_TC4);
	}
	if(DMA_GetITStatus(DMA1_IT_TE4) == SET)
	{
		DMA_ClearITPendingBit(DMA1_IT_TE4);
	}
	if(DMA_GetITStatus(DMA1_IT_GL4) == SET)
	{
		DMA_ClearITPendingBit(DMA1_IT_GL4);
	}
	if(DMA_GetITStatus(DMA1_IT_HT4) == SET)
	{
		DMA_ClearITPendingBit(DMA1_IT_HT4);
	}
	DMA_Cmd(DMA1_Channel4, DISABLE);
}

//串口1初始化
void uart1_init(u32 baud)
{
	uart1_config(baud);
	dma_recv_config();
	dma_send_config();
	callBack_recv = default_callback;//配置串口中断回调函数
}


//串口1发送函数
uint8_t uart1_send(uint8_t *pt_txbuf , uint32_t tx_len, COMMUCATION_RECV_CALLBACK callback)  
{	
	if(pt_txbuf == 0)
		return FALSE;
	
	if(callback != 0)
		callBack_recv = callback;	
	

	DMA_ClearFlag(DMA1_FLAG_TC4);
	DMA_ClearFlag(DMA1_FLAG_TE4);
	
	DMA_Cmd(DMA1_Channel4, DISABLE);	
	DMA1_Channel4->CMAR = (u32)(pt_txbuf);		
	DMA1_Channel4->CNDTR = tx_len;				
	DMA_Cmd(DMA1_Channel4, ENABLE);	
	return TRUE;
}

下面只介绍 EC20任务和传感器任务的实现,剩下一个任务的实现有兴趣的读者可以到我的资源里把工程代码下载下来进行研究,欢迎评论区留言讨论。

EC20任务

下面是EC20模块初始化,建立TCP连接以及接收数据处理整个状态机在一切结果正常的情况下的流程图:

 下面是任务状态机的实现,包括一些模块返回错误的流程跳转,代码中以增加中文注释帮忙理解:

 //AT指令
 const char chk_baud[] = {"AT+IPR?\r\n"};			//查询波特率
 const char set_baud[] = {"AT+IPR=115200\r\n"};		//设置波特率
 const char chk_signal[] = {"AT+CSQ\r\n"};			//查询信号强度
 const char close_cmd_echo[] = {"ATE0\r\n"};	    //关闭数据回显
 const char save_param[] = {"AT&W\r\n"};			//保存设置参数,应答超时300ms
 const char chk_iccid[] = {"AT+QCCID\r\n"};			//查询ccid

 const char chk_pin[] = {"AT+CPIN?\r\n"};			//查询sim卡密码,5s超时
 const char chk_pin_back1[] = {"+CME ERROR: 10"};	//sim卡未插好
 const char chk_pin_back2[] = {"+CPIN: READY"};		//sim卡准备好

 const char chk_sim_reg[] = {"AT+CREG?\r\n"};		//查询sim卡是否注册
 const char chk_net_reg[] = {"AT+CGREG?\r\n"};		//查询net注册
 
 //运营商APN
 const char set_link_ChinaMobile[] = {"AT+QICSGP=1,1,\"CMNET\",\"\",\"\",1\r\n"};//设置接入点(移动)APN,USERNAME,PASSWORD
 const char set_link_UNICOM[] = {"AT+QICSGP=1,1,\"UNINET\",\"\",\"\",1\r\n"};//设置接入点(联通) APN,USERNAME,PASSWORD
 const char set_link_p[] = {"AT+QICSGP=1,1,\"CTLTE\",\"\",\"\",1\r\n"};//设置接入点(电信) APN,USERNAME,PASSWORD
 
 const char chk_link_point[] = {"AT+QICSGP=1\r\n"};	//查询接入点
 const char active_pdp_context[] = {"AT+QIACT=1\r\n"};//激活接入点150s超时
 const char chk_pdp_context[] = {"AT+QIACT?\r\n"};	//查询context,返回当前接入点信息(150s超时)
 const char deactive_pdp_context[] = {"AT+QIDEACT=1\r\n"};//关闭接入点40s超时
//const char creat_tcp[] = {"AT+QIOPEN=1,0,\"TCP\",\"120.55.115.113\",5008,0,2\r\n"};//创建TCP连接(150s超时)
//const char creat_tcp[] = {"AT+QIOPEN=1,0,\"TCP\",\"120.26.204.86\",8001,0,2\r\n"};//创建TCP连接(150s超时)
 
const char creat_tcp_top[] = {"AT+QIOPEN=1,0,\"TCP\",\""};
const char creat_tcp_tail[] = {"\","};//创建TCP连接150s
const char creat_tcp_port[] = {",0,1\r\n"};//服务端口号+传输模式【推送,非透传】
const char creat_tcp_back1[] = {"CONNECT"};//创建TCP成功,透传模式的回应
	
const char close_tcp[] = {"AT+QICLOSE=0\r\n"};		//关闭TCP连接(自己设置超时时间)
const char chk_tcp[] = {"AT+QISTATE=1,0\r\n"};		//查询TCP连接(指定connetcID)
const char chk_data_echo[] = {"AT+QISDE?\r\n"};		//查询数据回显
//const char close_data_echo[] = {"AT+QISDE=0\r\n"};	//关闭数据回显(透传模式不需要)
const char chk_err_code[] = {"AT+QIGETERROR\r\n"};	//查询最近的错误码

const char exit_transpartent[] = {"+++"};		//退出透传模式
const char exit_transpartent2[] = {"++++++++\r\n"};		//命令模式下+++操作

const char change_transparent_mode[] = {"AT+QISWTMD=1,2\r\n"};		//改为透传模式
const char enter_transparent_mode[] = {"ATO\r\n"};		//进入透传模式

const char common_ack[] = {"OK"};	//通用回复"OK"

const char chk_Operator[] = {"AT+COPS?\r\n"};//查是什么卡,移动联通电信
const char Operater1_ack[] = {"CHINA MOBILE"};//中国移动
const char Operater2_ack[] = {"CHN-UNICOM"};//中国联通
const char Operater3_ack[] = {"CHN-CT"};//中国电信
//	const char Operater1_ack[] = {"china mobile"};//中国移动
//	const char Operater2_ack[] = {"chn-unicom"};//中国联通
//	const char Operater3_ack[] = {"chn-ct"};//中国电信

const char chk_signal_quality[] = {"AT+CSQ\r\n"};//查询信号强度
const char chk_signal_ack[] = {"+CQS"};//查询信号强度返回值

	
const char remote_recv[] = {"+QIURC"};//服务器数据返回

const char tcp_direct_ack[] = {"+QIOPEN:"};//tcp连接 direct push 模式返回值


const char send_head[] = {"AT+QISEND=0,"};//通知ec20即将发送数据
const char send_head_ack[] = {">"};
const char sended_data_ack[] = {"SEND OK"};
const char query_sended[] = {"AT+QISEND=0,0\r\n"};//查询发送数据
const char query_sended_ack[] = {"+QISEND:"};
/
//EC20主任务
void ec20_task(void)
{
	u8 rs;

	ec20.fsm = ST_PWR_ON;//初始化状态机
	UPDATE_REBOOT_TIME;	//初始化重启计时
	CLEAR_REBOOT_COUNT;	//初始化重启计数
	
	while(1)
	{
		switch(ec20.fsm)
		{					
			case ST_PWR_ON:		//上电
//				c4g_power_off();
				c4g_power_on();
				ec20.fsm = ST_CLOSE_CMD_ECHO;//跳转关闭回显
			break;
			
			case ST_REBOOT:		//重启
				c4g_power_reset();
				ec20.fsm = ST_CLOSE_CMD_ECHO;//跳转关闭回显
				UPDATE_REBOOT_TIME;
				CLEAR_REBOOT_COUNT;
				global.reboot_time_count = xTaskGetTickCount();//更新全局数据发送时间间隔
			break;
			
			case ST_CLOSE_CMD_ECHO://关闭数据回显
				if(c4g_close_cmd_echo())
				{
					UPDATE_REBOOT_TIME;
					ec20.fsm = ST_CHK_SIM;
				}
				if(JUDGE_REBOOT_TIME(30))//超时则重启
				{
					ec20.fsm = ST_REBOOT;
					printf("EC20 reboot\r\n\r\n");
				}
			break;
				
			case ST_CHK_SIM:	//查看sim卡,物理卡
				rs = c4g_chk_sim();
				if(rs == TRUE)
				{
					ec20.fsm = ST_CHK_SIGNAL;//跳转信号强度
					UPDATE_REBOOT_TIME;
				}else
				if(rs == 2)//未插卡
				{
					printf("no sim\r\n");
					UPDATE_REBOOT_TIME;
				}
				else//重启
				{
					if(JUDGE_REBOOT_TIME(20))//查询时间超过20s
					{
						printf("check sim timeout\r\n\r\n");
						ec20.fsm = ST_REBOOT;
					}
				}
			break;
				
			case ST_CHK_SIGNAL:	//查询信号强度
				rs = c4g_chk_signal();
				if(rs == TRUE)
				{
					ec20.fsm = ST_CHK_SIM_REG;//跳转sim卡注册
					UPDATE_REBOOT_TIME;
				}
				else
				{
					if(JUDGE_REBOOT_TIME(20))//查询时间超过20s
					{
						printf("check signal timeout\r\n\r\n");
						ec20.fsm = ST_REBOOT;
					}
				}
			break;
				
			case ST_CHK_SIM_REG:	//查询sim卡注册
				rs = c4g_sim_reg();
				if(rs == TRUE)
				{
					ec20.fsm = ST_CHK_NET_REG;
					UPDATE_REBOOT_TIME;
				}
				else 
				if(rs == 2)
				{
					if(JUDGE_REBOOT_TIME(90))//查询时间超过90s
					{
						printf("check sim reg timeout\r\n\r\n");
						ec20.fsm = ST_REBOOT;
					}
				}
			break;
				
			case ST_CHK_NET_REG:	//查询网络注册
				rs = c4g_net_reg();
				if(rs == TRUE)
				{
					ec20.fsm = ST_CHK_OPERATER;//跳转查询运营商
					UPDATE_REBOOT_TIME;
				}
				else
				if(rs == 2)
				{
					if(JUDGE_REBOOT_TIME(60))//查询时间超过60s
					{
						printf("check net reg timeout\r\n\r\n");
						ec20.fsm = ST_REBOOT;
					}
				}
			break;
				
			case ST_CHK_OPERATER:	//查询运营商
				rs = c4g_check_Operater();
				if(rs == TRUE)
				{
					ec20.fsm = ST_SET_POINT;//跳转设置接入点
					UPDATE_REBOOT_TIME;
				}
				else
				{
					if(JUDGE_REBOOT_TIME(60))//查询时间超过60s
					{
						printf("Operater check err\r\n\r\n");//没查到运营商
						ec20.fsm = ST_REBOOT;
					}
				}
			break;
				
			case ST_SET_POINT:	//设置接入点,APN
				rs = c4g_set_point();
				if(rs == TRUE)
				{
					ec20.fsm = ST_ACTIVE_POINT;//Ìøת->"¼¤»î½ÓÈëµã"
					UPDATE_REBOOT_TIME;
				}
				else
				{
					if(JUDGE_REBOOT_TIME(60))//²éѯʱ¼ä³¬¹ý60s
					{
						printf("set apn err\r\n\r\n");
						ec20.fsm = ST_REBOOT;
					}
				}
			break;
				
			case ST_CHK_POINT:	//查询接入点
				
			break;
			
			case ST_ACTIVE_POINT:	//激活接入点
				rs = c4g_active_context();
				if(rs == TRUE)
				{
					ec20.fsm = ST_CREAT_TCP;//跳转创建TCP
					UPDATE_REBOOT_TIME;
					CLEAR_REBOOT_COUNT;
				}
				else
				if(rs == 2)//收到数据,但是数据错误
				{
					printf("active PDP Context err 1\r\n\r\n");
					ec20.fsm = ST_DEACTIVE_POINT;//跳转关闭接入点
					UPDATE_REBOOT_TIME;
				}
				else//超过150秒未收到回复
				{
					printf("active PDP Context err 2\r\n\r\n");
					ec20.fsm = ST_REBOOT;//跳转到重启设备
				}
			break;
				
			case ST_DEACTIVE_POINT:	//关闭接入点
				rs = c4g_deactive_context();
				if(rs == TRUE)
				{
					ec20.fsm = ST_CHK_SIM;//跳转查询sim卡
					CLEAR_REBOOT_COUNT;//清除重启计数
					UPDATE_REBOOT_TIME;
				}
				else
				if(rs == 2)//有返回,但是返回数据错误
				{
					if(JUDGE_REBOOT_COUNT(3))//最多查4次,如果还是失败则重启设置
					{
						printf("link point err 1\r\n\r\n");//接入点设置失败
						ec20.fsm = ST_REBOOT;//跳转重启设备
					}
				}
				else//40秒超时无回答
				{
					printf("link point err 2\r\n\r\n");//接入点设置失败
					ec20.fsm = ST_REBOOT;//跳转重启设备
				}
			break;
				
			case ST_CHK_CONTEXT:	//查询接入点文本
				rs = c4g_chk_context();
				if(rs == TRUE){//查询成功,但是未激活
					ec20.fsm = ST_ACTIVE_POINT;
					UPDATE_REBOOT_TIME;
				}else
				if(rs == 3){//已经激活
					ec20.fsm = ST_CHK_TCP;//先查询是否建立了TCP连接,如没有则建立
				}else
				if(rs == 2){//错误信息
					printf("context err 1\r\n\r\n");//查询接入点文本错误
				}
			break;
				
			case ST_CREAT_TCP:	//创建TCP连接
				rs = c4g_creat_tcp();
				if(rs == TRUE)
				{
					ec20.fsm = ST_COMMUCATION;//跳转通信
					global.socket_state = 1;//tcp连接成功
					CLEAR_REBOOT_COUNT;//清除重启计数
					UPDATE_REBOOT_TIME;
				}
				else
				if(rs !=0 && rs != TRUE)//有返回,则返回数据错误
				{
					if(JUDGE_REBOOT_COUNT(5))//五次失败重启
					{
						char tmp[2];
						sprintf(tmp, "%u", rs);
						printf("creat tcp err:");
						printf((const char*)tmp);
						printf("\r\n\r\n");
						ec20.fsm = ST_REBOOT;//跳转重启
					}
				}
				else//创建TCP超时,150s
				{
					printf("creat tcp timout\r\n\r\n");
					ec20.fsm = ST_REBOOT;//跳转重启
				}
			break;
				
			case ST_CHK_TCP:	//查询TCP
				rs = c4g_chk_tcp();
				if(rs == TRUE)//没有建立tcp,需要重新建立
				{
					ec20.fsm = ST_CREAT_TCP;//跳转创建tcp
					UPDATE_REBOOT_TIME;
					CLEAR_REBOOT_COUNT;
				}
				else//********************
				if(rs == 3)//TCP连接已存在
				{
					if( xTaskGetTickCount() - global.reboot_time_count > global.reboot_time_set )//超过系统超时时间
					{
						global.reboot_time_count = xTaskGetTickCount();//更新时间
						ec20.fsm = ST_CLOSE_TCP;//跳转"关闭CTP"
						UPDATE_REBOOT_TIME;
					}
					else
					{
						ec20.fsm = ST_COMMUCATION;//跳转等待接收数据
						UPDATE_REBOOT_TIME;
					}
				}
				else
				if(rs == 4)//正在关闭TCP
				{
					ec20.fsm = ST_CLOSE_TCP;//跳转关闭TCP
					UPDATE_REBOOT_TIME;
				}
				else//超时无人应答
				{
					printf("chk tcp timeout\r\n\r\n");//查询TCP失败
					ec20.fsm = ST_REBOOT;//跳转重启
				}
			break;
				
			case ST_CLOSE_TCP:		//关闭TCP
				rs = c4g_close_tcp();
				if(rs == TRUE)//关闭TCP成功
				{
					ec20.fsm = ST_CREAT_TCP;//跳转创建TCP
					UPDATE_REBOOT_TIME;
					CLEAR_REBOOT_COUNT;
				}
				else
				if(rs == 2)//错误信息
				{
					if( JUDGE_REBOOT_COUNT(5) )//最多查5次
					{
						printf("close tcp err\r\n\r\n");
						ec20.fsm = ST_REBOOT;//跳转重启
					}
				}
				else
				{
					printf("close tcp timeout\r\n\r\n");//关闭TCP超时
					ec20.fsm = ST_REBOOT;//Ì跳转重启
				}
			break;
				
			case ST_COMMUCATION://数据通信
			{
				MSG*cmsg;
				u8 err;
				
				err = getting_queue_ec20_remote((void*)&cmsg, 10*OS_TICKS_PER_SEC);//每10s查询一次
				if(err == pdTRUE)//收到远端数据
				{
					send_to_stm32(cmsg);//发送给传感器串口
					printf("send to user\r\n\r\n");
				}
				else
				{
					if(global.socket_state == 2)
					{
						printf("direct reboot\r\n\r\n");
						ec20.fsm = ST_REBOOT;//跳转重启
					}
					else
					if(global.socket_state == 0)
					{
						printf("transfer to close tcp\r\n\r\n");
						ec20.fsm = ST_CLOSE_TCP;//跳转关闭TCP
						UPDATE_REBOOT_TIME;
					}
				}
			}
			break;
			
			case ST_CHK_DATA_ECHO:	//查询数据回显
				printf("idle \r\n\r\n");
			break;
			case ST_CLOSE_DATA_ECHO://关闭数据回显
				
			break;
			case ST_CHK_ERRCODE:	//查询最近的错误码
				
			break;
			
		}
	}
}

传感器任务 

下面是传感器任务的实现,将传感器通过串口的发过来的数据通过EC20模块发送出去:

void stm32_recv_task(void)
{
	uint8_t rs, err, resend = 0;
	MSG *cmsg;
	
	while(1)
	{
		err = getting_queue_stm32((void*)&cmsg, global.reboot_time_set);//无限等待传感器队列消息
		if(err == pdTRUE)
		{
			global.reboot_time_count = xTaskGetTickCount();
			if(global.socket_state != 1)//tcp未连接
			{
				msg_free(cmsg);//不处理,并释放队列空间
				printf("stm32 recv data no dealwith\r\n");
			}
			else
			{
				//1.将数据发送至ec20
				//2.查询数据是否发送成功,即tcp是有否回应。如果没有发送成功,此条数据重新丢进队列,关闭TCP,重新连接tcp,重新发送此数据
				rs = 0xff; resend = 0;
				do
				{
					rs = send_commucation_head(cmsg->len);
				}while(resend++ < 3 && rs != 1);
				
				if(rs == 1)//发送数据头,通知ec20即将发送数据
				{
					u8 i = 0;
					do{
						uart1_send(cmsg->buf, cmsg->len, 0);//发数据
						rs = sended_result();
						if(rs != 1)//发送失败(EC20本身发送失败,并不是发送远端失败)
						{
							printf("ec20 send err\r\n\r\n");
						}
						else//发送成功
						{
							rs = query_sended_result(cmsg->len);//检查发送到remote是否成功
							if(rs == 0)//超时90s
							{
								rs = 111;
								global.socket_state = 2;//重启
								printf("sended no ack\r\n\r\n");
							}
						}
					}while(i++ < 3 && rs != 1);
					
					if(rs == 1)//发送remote成功
					{
						
					}
					else/发送失败
					{
						printf("send data err\r\n\r\n");
						if(rs != 111)
						{
							global.socket_state = 0;//重新建立tcp
							printf("sended data fail\r\n\r\n");
						}
					}
				}
				else//发送数据头,无'>'回复
				{
					printf("head is no ack\r\n\r\n");
					global.socket_state = 2;//重启
				}
				msg_free(cmsg);//释放消息,这里无论remote是否收到没有
			}
		}
		else//超过最长发送时间间隔
		{
			global.socket_state = 2;//重新建立TCP
			printf("global timeout\r\n\r\n");
		}
	}
}

 十六宿舍 原创作品,转载必须标注原文链接。

©2023 Yang Li. All rights reserved.

欢迎关注 『十六宿舍』,大家喜欢的话,给个👍,更多关于嵌入式相关技术的内容持续更新中。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/436090.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vscode java环境扩展

下载安装jdk: Java Downloads | Oracle 下载安装maven: Maven – Download Apache Maven windows系统选择 Binary zip archive maven目录中的conf文件中有一个settins.xml文件 镜像&#xff08;mirrors&#xff09;二选一 <mirror><id>nexus-aliyun</…

影视动画设计有些SCI期刊推荐? - 易智编译EaseEditing

以下是几本影视动画设计方向的SCI期刊&#xff1a; ACM Transactions on Graphics: 该期刊是ACM&#xff08;Association for Computing Machinery&#xff0c;美国计算机协会&#xff09;下的一个子刊&#xff0c;涵盖了计算机图形学和交互技术等领域&#xff0c;也包括了动…

虚拟化技术 — 硬件辅助的虚拟化技术

目录 文章目录 目录硬件辅助的虚拟化技术概览CPU 虚拟化技术基于二进制翻译的全虚拟化技术&#xff08;Full-Virtualization&#xff09;需要改造 GuestOS 的半虚拟化技术&#xff08;Para-Virtualization&#xff09;Intel VT-x 硬件辅助的虚拟化技术&#xff08;Hardware-ass…

智芯Z20K11x串口printf重定向

智芯Z20K11x串口printf重定向 目录 智芯Z20K11x串口printf重定向前言1 串口代码编写2 Keil勾选MicroLIB库结束语 前言 智芯Z20K11x系列是基于ARM CORTEX M0的中低端微控制器&#xff0c;高达256K P FLASH 128K D FLASH&#xff0c;2路CANFD接口&#xff0c;4路UART/LIN接口&a…

综合能源系统中基于电转气和碳捕集系统的热电联产建模与优化研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

HashMap理解

简单介绍 HashMap是双列集合Map接口下的实现类&#xff0c;jdk1.8之前其数据结构是数组链表、jdk1.8之后是数组链表红黑树&#xff0c;是线程不安全的&#xff0c; 原理图&#xff1a; HashMap放值的过程&#xff1a; HashMap先是根据Hash算法算出key的hashcode码值&#xff0…

Vivado之VIO原理及应用

虚拟输入输出&#xff08;Virtual Input Output,VIO)核是一个可定制的IP核&#xff0c;它可用于实时监视和驱动内部FPGA的信号&#xff0c;如图所示。 可以定制VIO的输入和输出端口的数量与宽度&#xff0c;用于和FPGA设计进行连接。由于VIO核与被监视和驱动的设计同步&#xf…

Linux操作系统原理—内核网络协议栈

前言 本文主要记录 Linux 内核网络协议栈的运行原理 数据报文的封装与分用 封装&#xff1a;当应用程序用 TCP 协议传送数据时&#xff0c;数据首先进入内核网络协议栈中&#xff0c;然后逐一通过 TCP/IP 协议族的每层直到被当作一串比特流送入网络。对于每一层而言&#xff…

SwiftUI 使用 UIPageViewController 翻页后出现空白的原因及解决

问题现象 我们 SwiftUI 开发的 App 需要 UIPageViewController 的翻页功能,这可以非常方便的通过桥接 UIKit 到 SwiftUI 来搞定: 不过,观察上图可以发现 App 翻页显示的并不太对:当用户通过右下角的 UIPageControl 触发翻页时没有问题,但当用户直接手动在 UIPageViewCont…

【毕业设计】基于程序化生成和音频检测的生态仿真与3D内容生成系统----音频检测算法设计

(2条消息) 【开发日志】2022.09.02 ZENO----Audio----Beat detection algorithm----Combine Wav&Mp3_minimp3 和 ffmpeg_EndlessDaydream的博客-CSDN博客https://blog.csdn.net/Angelloveyatou/article/details/126670613 4 音频检测算法设计 4.1 节拍检测算法 4.1.1 节…

JavaSE——数组

这篇文章的面向读者为Java初级程序员&#xff0c;也就是刚刚接触到Java这门语言&#xff0c;里面描述的内容主要是数组相关的内容&#xff0c;讲解了最基础的一些数组扩容思路&#xff0c;数组赋值机制&#xff0c;什么是引用地址&#xff0c;什么是基础数据赋值。 Java该章节数…

JAVA入坑之线程

目录 一、&#xff1a;相关概念 1.1中央处理器(CPU&#xff0c;Central Processing Unit) 1.2程序、进程、线程 1.3线程与任务 二、线程的创建&#xff1a; 2.1继承Thread创建线程&#xff1a; 使用Thread子类创建线程的优缺点 2.2实现Runnable接口创建线程&#xff1a;…

【C++STL精讲】list的使用教程及其模拟实现

文章目录 &#x1f490;专栏导读&#x1f490;文章导读&#x1f337;list是什么&#xff1f;&#x1f337;list如何使用&#xff1f;&#x1f337;list的模拟实现&#x1f33a;定义list类&#x1f33a;构造函数&#x1f33a;push_back&#x1f33a;pop_back &#x1f337;list迭…

利用STM32实现自平衡机器人功能与方法

将机器人整体开源&#xff0c;同时总结一下机器人搭建过程中遇到的坑和未来的改进方向。在分享的文件里包含了结构设计、程序控制、电路设计以及其他模块相关资料供大家参考。 第一&#xff1a;机器人原理分析 首先来看成品图&#xff1a; 如图所示&#xff0c;该机器人根据陀…

宝塔面板设置虚拟内存Swap降低的方法

宝塔面板可以通过设置Swap/虚拟内存的方式来降低内存使用率和负载&#xff0c;使用宝塔面板的Linux工具箱即可设置虚拟内存&#xff0c;新手站长来详细说下宝塔Linux面板设置Swap/虚拟内存的方法&#xff1a; 宝塔面板设置虚拟内存 设置虚拟内存是通过Linux工具箱&#xff0c…

4.0、Java继承与多态 - 抽象类与抽象方法

4.0、Java继承与多态 - 抽象类与抽象方法 先给大家举个例子 -> 创建一个父类 - 图形类&#xff1b;图形类中有一个计算面积的方法 calculateArea()&#xff1b; 创建三个子类 - 正方形、三角形、圆形 类&#xff1b; 由于我们图形类父类中未明确指明是什么图形&#xff0c…

内网域环境搭建学习

建立的关系就是这样&#xff0c;接下来就开始讲解遇到的困难 虚拟机中我们可以克隆来实现域的搭建 可能会出现这样的问题&#xff0c;原因是你的虚拟机没有关闭&#xff0c;所以才会导致这样的原因&#xff0c;解决方法 将虚拟机打开后&#xff0c;电源 -> 关闭客户机&…

MySQL学习笔记第二天

第03章 基本的SELECT语句 1.SQL概述 1.1 SQL背景知识 1946年&#xff0c;世界上第一台电脑诞生&#xff0c;如今&#xff0c;借由这台电脑发展起来的互联网已经自成江湖。在这几十年里&#xff0c;无数的技术、产业在这片江湖里沉浮&#xff0c;有的方兴未艾&#xff0c;有的…

reactxp搭建,start:windows运行不起来

1、官网 reactxp 2、VSCode和Visual Studio2019 安装VSCode Visual Studio 下载地址 先不用勾选工作负荷&#xff0c;直接安装 3、安装nvm 访问下载地址下载安装nvm&#xff1a; 百度云分享 官网直装链接 nvm的github发行界面下载nvm-setup.exe GitCode镜像下载nvm-setup…

ACL配置学习(附练习题)------ensp

从此文了解ACL配置&#xff0c;欢迎学习、指导。 目录 基本ACL配置举例 高级ACL配置举例 ACL配置练习题 定义 访问控制列表ACL&#xff08;Access Control List&#xff09;本质上是一种报文过滤器。 范围: OSI七层模型中的网络层、传输层的信息。 滤芯&#xff1a;五…