如何設計基于STM8的ADC0832采集及藍牙通信系統?
最近在淘寶逛的時候發現了一款單片機,STM8。相比之前一直使用的也是8位的AVR相比,感覺STM8更為強大,芯片特點如下:
本文引用地址:http://www.104case.com/article/201709/364892.htm內核:具有3級流水線的哈佛結構、擴展指令集
程序存儲器:8K字節Flash;RAM:1K字節
數據存儲器:640 字節真正的數據EEPROM;可達30萬次擦寫
更重要的一點就是STM8系列若使用庫編程的話,可以方便的不同芯片的程序移植。甚至可以方便的移植到STM32上面,大大減輕了更新硬件的重寫程序的工作量。
ADC0832 為8位分辨率A/D轉換芯片,其最高分辨可達256級,可以適應一般的模擬量轉換要求。其內部電源輸入與參考電壓的復用,使得芯片的模擬電壓輸入在0~5V之間。芯片轉換時間僅為32μS,據有雙數據輸出可作為數據校驗,以減少數據誤差,轉換速度快且穩定性能強。獨立的芯片使能輸入,使多器件掛接和處理器控制變的更加方便。通過DI 數據輸入端,可以輕易的實現通道功能的選擇。(簡述和圖片均來之百度百科)

本文適合STM8控制ADC0832,程序是使用庫編程,編譯工具IAR。其實STM8也自帶ADC轉換模塊了......
本程序還包括藍牙串口通信,方便將得到數據從串口輸出,我是編寫了安卓上位機的app,方便在安卓上面顯示圖像。
程序還是用了定時器TIM4,確保每次采樣的間隔大致相等,對之后的數據處理提供了基礎。
先介紹核心mian.c文件,主要功能是初始化串口UART1,定時器TIMER4,還有一個發送16進制的函數。其中發送完數據再發送一個字符’U’作為一個數據的結束(你也可以自己定義)。這里說說為什么要選用16進制,而不是10進制,STM8速度有限,為了減少單指令操作,程序用了移位操作,這樣可得到16進制每位數值,在發送到安卓上位機,上位機運算速度快,再轉化成10進制,這樣可以資源合理分配。
main.c程序:
#include "stm8s.h"
#include "stm8s_it.h"
uint8_t HexTable[]={'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
uint8_t i=0;
//串口UART1初始化
void Init_UART(void)
{
//默認初始化
UART1_DeInit();
//設置波特率9600 8位數據 1位停止位 無校驗 外部時鐘不可用 模式接收發送
UART1_Init((u32)9600, UART1_WORDLENGTH_8D, UART1_STOPBITS_1, UART1_PARITY_NO, UART1_SYNCMODE_CLOCK_DISABLE, UART1_MODE_TXRX_ENABLE);
//設置接收寄存器溢出中斷
UART1_ITConfig(UART1_IT_RXNE_OR, ENABLE);
}
//定時器TIM4初始化
void Init_Timer4(void)
{
//1ms中斷一次
TIM4_TimeBaseInit(TIM4_PRESCALER_128, 124);
/* Clear TIM4 update flag */
TIM4_ClearFlag(TIM4_FLAG_UPDATE);
/* Enable update interrupt */
TIM4_ITConfig(TIM4_IT_UPDATE, ENABLE);
TIM4_Cmd(ENABLE);
}
//發送字節
void Send(uint8_t dat)
{
//檢查并等待發送寄存器是否為空
while(( UART1_GetFlagStatus(UART1_FLAG_TXE)==RESET));
//發送字節
UART1_SendData8(dat);
}
//發送16位16進制
void UART1_mysend16hex(u16 dat)
{
Send(HexTable[(dat>>12)&0x0f]);
Send(HexTable[(dat>>8)&0x0f]);
Send(HexTable[(dat>>4)&0x0f]);
Send(HexTable[(dat)&0x0f]);
}
//發送8位16進制
void UART1_mysend8hex(uint8_t dat)
{
Send(HexTable[(dat>>4)&0x0f]);
Send(HexTable[(dat)&0x0f]);
Send('U');
}
void main()
{
//初始化
Init_UART();
Init_Timer4();
//中斷開啟
enableInterrupts();
while(1)
{
}
}
//這個必須加上 不然會報錯 估計是庫的要求
#ifdef USE_FULL_ASSERT
void assert_failed(u8* file, u32 line)
{
while (1)
{
}
}
#endif
接下來說說中斷函數表stm8s_it.c
其中只要選用兩個中斷函數就可以了:
INTERRUPT_HANDLER(UART1_RX_IRQHandler, 18) 接收寄存器溢出中斷
里面添加安卓上位機發送過來的數據的處理程序,我這里寫的是ADC0832通道選擇的判斷。
INTERRUPT_HANDLER(TIM4_UPD_OVF_IRQHandler, 23) 定時器4計數器溢出中斷
里面添加初始化ADC0832和ADC0832數據讀取并UART1發送到安卓上位機。
stm8s_it.c程序:
#include "stm8s_it.h"
#include "ADC0832.h"
extern uint8_t i;
uint8_t channel=1 ;
//接收寄存器溢出中斷
INTERRUPT_HANDLER(UART1_RX_IRQHandler, 18)
{
/* In order to detect unexpected events during development,
it is recommended to set a breakpoint on the following instruction.
*/
//下面是我做的安卓上位機發送過來的數據判斷,這里可以改成自己想要的程序
uint8_t tempData;
tempData = UART1_ReceiveData8();
if(tempData=='A')
{
channel = 0;
}
if(tempData=='Z')
{
channel = 1;
}
//清除UART1中斷標識符
UART1_ClearITPendingBit(UART1_IT_RXNE);
}
//定時器4計數器溢出中斷
評論