在stm32中要實現數據通訊,首先要設置相關的寄存器,這里不做相關的介紹,直接說代碼相關的能內容及相關函數對應的用法。
直接上代碼。
1.串口通訊代碼
usart.h
#ifndef __USART_H
#define __USART_H
#include "stdio.h"
#include "sys.h"
#define USART_REC_LEN 200 //定義最大接收字節數 200
#define EN_USART3_RX 1 //使能(1)/禁止(0)串口1接收
extern u8 USART_RX_BUF[USART_REC_LEN]; //接收緩沖,最大USART_REC_LEN個字節.末字節為換行符
extern u16 USART_RX_STA; //接收狀態標記
//如果想串口中斷接收,請不要注釋以下宏定義
void uart_init(u32 bound);
#endif
對應的usart.c代碼
#include "sys.h"
#include "usart.h"
//加入以下代碼,支持printf函數,而不需要選擇use MicroLIB
#if 1
#pragma import(__use_no_semihosting)
//標準庫需要的支持函數
struct __FILE
{
int handle;
};
FILE __stdout;
//定義_sys_exit()以避免使用半主機模式
void _sys_exit(int x)
{
x = x;
}
//重定義fputc函數
int fputc(int ch, FILE *f)
{
while((USART3->SR&0X40)==0);//循環發送,直到發送完畢
USART3->DR = (u8) ch;
return ch;
}
#endif
/*使用microLib的方法*/
/*
int fputc(int ch, FILE *f)
{
USART_SendData(USART1, (uint8_t) ch);
while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET) {}
return ch;
}
int GetKey (void) {
while (!(USART1->SR & USART_FLAG_RXNE));
return ((int)(USART1->DR & 0x1FF));
}
*/
#if EN_USART3_RX //如果使能了接收
//串口1中斷服務程序
//注意,讀取USARTx->SR能避免莫名其妙的錯誤
u8 USART_RX_BUF[USART_REC_LEN]; //接收緩沖,最大USART_REC_LEN個字節.
//接收狀態
//bit15, 接收完成標志
//bit14, 接收到0x0d
//bit13~0, 接收到的有效字節數目
u16 USART_RX_STA=0; //接收狀態標記
void uart_init(u32 bound)
{
//GPIO端口設置
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3,ENABLE);//使能USART1,GPIOA時鐘
//USART3_TX GPIOB.10
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //PB.10
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //復用推挽輸出
GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOB.10
//USART3_RX GPIOB.11初始化
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;//PA10
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空輸入
GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOB.11
//Usart3 NVIC 配置
NVIC_InitStructure.NVIC_IRQChannel = USART3_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//搶占優先級3
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子優先級3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //根據指定的參數初始化VIC寄存器
//USART 初始化設置
USART_InitStructure.USART_BaudRate = bound;//串口波特率
USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字長為8位數據格式
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(USART3, &USART_InitStructure); //初始化串口3
USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);//開啟串口接受中斷
USART_Cmd(USART3, ENABLE); //使能串口3
}
void USART3_IRQHandler(void) //串口3中斷服務程序
{
u8 Res;
#if SYSTEM_SUPPORT_OS //如果SYSTEM_SUPPORT_OS為真,則需要支持OS.
OSIntEnter();
#endif
if(USART_GetITStatus(USART3, USART_IT_RXNE) != RESET) //接收中斷(接收到的數據必須是0x0d 0x0a結尾)
{
Res = USART_ReceiveData(USART3); //讀取接收到的數據
printf("get data %c \r\n",Res);
if((USART_RX_STA&0x8000)==0)//接收未完成
{
if(USART_RX_STA&0x4000)//接收到了0x0d
{
if(Res!=0x0a)USART_RX_STA=0;//接收錯誤,重新開始
else USART_RX_STA|=0x8000; //接收完成了
}
else //還沒收到0X0D
{
if(Res==0x0d)USART_RX_STA|=0x4000;
else
{
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA++;
if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收數據錯誤,重新開始接收
}
}
}
}
#if SYSTEM_SUPPORT_OS //如果SYSTEM_SUPPORT_OS為真,則需要支持OS.
OSIntExit();
#endif
}
#endif
通過上述的函數,只需要在main函數設置相關的打印函數即可打印相關的內容,這里不做相關的陳述,后面主函數會進行相關的設置。
2.進行相關的adc函數配置
adc.h函數代碼如下:
#ifndef __ADC_H
#define __ADC_H
#include "sys.h"
void Adc_Init(void);
u16 Get_Adc(u8 ch);
u16 Get_Adc_Average(u8 ch,u8 times);
#endif
對應的adc.c函數代碼:
#include "adc.h"
#include "delay.h"
//////////////////////////////////////////////////////////////////////////////////
//adc配置函數
//////////////////////////////////////////////////////////////////////////////////
//初始化ADC
//我們默認將開啟通道0~3
void Adc_Init(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1 , ENABLE ); //使能ADC1通道時鐘
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //設置ADC分頻因子6 72M/6=12,ADC最大時間不能超過14M
//PC0 作為模擬通道輸入引腳
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入引腳
GPIO_Init(GPIOC, &GPIO_InitStructure);
ADC_DeInit(ADC1); //復位ADC1
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在獨立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模數轉換工作在單通道模式
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模數轉換工作在單次轉換模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //轉換由軟件而不是外部觸發啟動
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC數據右對齊
ADC_InitStructure.ADC_NbrOfChannel = 1; //順序進行規則轉換的ADC通道的數目
ADC_Init(ADC1, &ADC_InitStructure); //根據ADC_InitStruct中指定的參數初始化外設ADCx的寄存器
ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1
ADC_ResetCalibration(ADC1); //使能復位校準
while(ADC_GetResetCalibrationStatus(ADC1)); //等待復位校準結束
ADC_StartCalibration(ADC1); //開啟AD校準
while(ADC_GetCalibrationStatus(ADC1)); //等待校準結束
// ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的軟件轉換啟動功能
}
//獲得ADC值
//ch:通道值 0~3
u16 Get_Adc(u8 ch)
{
//設置指定ADC的規則組通道,一個序列,采樣時間
ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采樣時間為239.5周期
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的軟件轉換啟動功能
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待轉換結束
return ADC_GetConversionValue(ADC1); //返回最近一次ADC1規則組的轉換結果
}
u16 Get_Adc_Average(u8 ch,u8 times)
{
u32 temp_val=0;
u8 t;
for(t=0;t;t++)>
3.主函數內容
#include "delay.h"
#include "common.h"
#include "usart.h"
#include "can.h"
#include "string.h"
#define adcx adc;
int main(void)
{
int real_100_vol;
u8 Res,res,data;
u16 adcx;
u8 i=0,t=0;
u8 cnt=11;
delay_init(); //延時函數初始化
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//設置中斷優先級分組為組2:2位搶占優先級,2位響應優先級
uart_init(115200); //串口初始化為115200
Adc_Init(); //adc初始化
CAN_Mode_Init(CAN_SJW_1tq,CAN_BS2_8tq,CAN_BS1_9tq,8,CAN_Mode_LoopBack);//CAN初始化環回模式,波特率500Kbps ,設置8為500k,設置4為250k
printf("APP start OK!\r\n");
while(1)
{
char canbuf_8;
int a = 165;
adc=Get_Adc_Average(ADC_Channel_10,50);
vol_led=(float)adc*(3.3/4096);
// adcx=vol_led;
// vol_led-=adcx;
// vol_led*=1000;
printf("AD值:%d\r\n 電壓值:%f V",adc,vol_led); //打印ad值和電壓值
H文件與C文件進行配置。
即可通過串口讀取到相關的數據AD值和電壓值。
審核編輯:湯梓紅
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。
舉報投訴
-
寄存器
+關注
關注
31文章
5336瀏覽量
120232 -
數據
+關注
關注
8文章
7003瀏覽量
88943 -
串口
+關注
關注
14文章
1551瀏覽量
76431
發布評論請先 登錄
相關推薦
維愛普 數據通訊設備穩定性:濾波器的核心作用
在當今這個信息爆炸的時代,數據通訊設備已經成為我們日常生活中不可或缺的一部分。無論是智能手機、電腦,還是網絡通信基站、數據中心,這些設備都在不斷地傳輸、處理著海量的數據。然而,在
安信可智能無線數據通訊Ra-09-DTU
Ra-09-DTU 是由安信可科技開發的一款智能無線數據通訊DTU,采用Ra-09 LoRa 模組,利用 LoRa 無線技術可用于超長距離擴頻通信。其芯片 STM32WLE5CCU6 是通用
求助,關于ADS125H02數據通訊格式及CRC校驗問題求解
of the initial data.
第1條中,將所有數據“左移擴展為40bit,最右邊用1補齊”,與傳統的crc計算方法不一致,傳統的計算方法是在數據末尾補8個0。是手冊筆誤還是本就與正常的crc計算方法
發表于 12-03 06:54
EtherCAT轉Modbus TCP總線協議轉換網關
遠創智控YC-ECT-TCP型網關實現EtherCAT網絡與Modbus TCP網絡之間的數據通訊,即將Modbus TCP設備轉換為EtherCAT設備。
EtherCAT轉Modbus TCP協議網關(JM-ECT-TCP)
JM-ECT-TCP網關實現EtherCAT網絡與Modbus TCP網絡之間的數據通訊,即將Modbus TCP設備轉換為EtherCAT設備。
STM32F103 USB通訊出現數據丟失的原因?如何解決?
用ST32F103的USB 做數據通訊,有大量的數據通過電腦發送到設備,少量數據回傳到電腦,中斷優先級設為2 ,還有幾個定時器的也是中斷,優先級比USB通訊的高,發現在大量連續發送
發表于 04-29 07:28
淺談光伏電站數據通訊管理機的設計與應用
淺談光伏電站數據通訊管理機的設計與應用 張穎姣 江蘇安科瑞電器制造有限公司江蘇江陰214405 摘要 :設計了一種分布式光伏電站數據通訊管理機,包括硬件系統和軟件系統.硬件系統基于ARM嵌入式處理器
PLC之間數據通訊,PLC與智能儀表、設備之間WIFI無線通訊
各種PLC之間雙向通訊 不同品牌不同型號的PLC,實現多對多的N:N通訊;同時支持以太網、串口連接;無需編寫程序,簡單配置主從站點的起始地址、數據長度即可;各站點多線程方式獨立運行,
發表于 01-07 11:22
虹科技術 | 保障數據傳輸穩定性:BabyLIN產品的CRC算法實現
CRC校驗(循環冗余校驗)是數據通訊中最常采用的校驗方式。CAN協議中,總線通信節點也常采用CRC算法對各種總線傳輸的數據進行校驗。
PLC通過智能網關,與OPCUA服務端通訊,實現標簽數據讀寫
通過‘功能’->‘數據上報與平臺對接’,選擇‘專用協議數據通訊’,進入以下頁面配置發送數據(OPCUA發送到PLC的數據)、接收數據
發表于 01-02 16:41
評論