新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > 如何設(shè)計基于STM8的ADC0832采集及藍牙通信系統(tǒng)?

如何設(shè)計基于STM8的ADC0832采集及藍牙通信系統(tǒng)?

作者: 時間:2017-09-27 來源:網(wǎng)絡(luò) 收藏

  最近在淘寶逛的時候發(fā)現(xiàn)了一款單片機,。相比之前一直使用的也是8位的AVR相比,感覺更為強大,芯片特點如下:

本文引用地址:http://m.butianyuan.cn/article/201709/364892.htm

  內(nèi)核:具有3級流水線的哈佛結(jié)構(gòu)、擴展指令集

  程序存儲器:8K字節(jié)Flash;RAM:1K字節(jié)

  數(shù)據(jù)存儲器:640 字節(jié)真正的數(shù)據(jù)EEPROM;可達30萬次擦寫

  更重要的一點就是系列若使用庫編程的話,可以方便的不同芯片的程序移植。甚至可以方便的移植到STM32上面,大大減輕了更新硬件的重寫程序的工作量。

   為8位分辨率A/D轉(zhuǎn)換芯片,其最高分辨可達256級,可以適應(yīng)一般的模擬量轉(zhuǎn)換要求。其內(nèi)部電源輸入與參考電壓的復(fù)用,使得芯片的模擬電壓輸入在0~5V之間。芯片轉(zhuǎn)換時間僅為32μS,據(jù)有雙數(shù)據(jù)輸出可作為數(shù)據(jù)校驗,以減少數(shù)據(jù)誤差,轉(zhuǎn)換速度快且穩(wěn)定性能強。獨立的芯片使能輸入,使多器件掛接和處理器控制變的更加方便。通過DI 數(shù)據(jù)輸入端,可以輕易的實現(xiàn)通道功能的選擇。(簡述和圖片均來之百度百科)

    

1.jpg

 

  本文適合STM8控制,程序是使用庫編程,編譯工具IAR。其實STM8也自帶ADC轉(zhuǎn)換模塊了......

  本程序還包括藍牙串口通信,方便將得到數(shù)據(jù)從串口輸出,我是編寫了安卓上位機的app,方便在安卓上面顯示圖像。

  程序還是用了定時器TIM4,確保每次采樣的間隔大致相等,對之后的數(shù)據(jù)處理提供了基礎(chǔ)。

  先介紹核心mian.c文件,主要功能是初始化串口UART1,定時器TIMER4,還有一個發(fā)送16進制的函數(shù)。其中發(fā)送完數(shù)據(jù)再發(fā)送一個字符’U’作為一個數(shù)據(jù)的結(jié)束(你也可以自己定義)。這里說說為什么要選用16進制,而不是10進制,STM8速度有限,為了減少單指令操作,程序用了移位操作,這樣可得到16進制每位數(shù)值,在發(fā)送到安卓上位機,上位機運算速度快,再轉(zhuǎn)化成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();

  //設(shè)置波特率9600 8位數(shù)據(jù) 1位停止位 無校驗 外部時鐘不可用 模式接收發(fā)送

  UART1_Init((u32)9600, UART1_WORDLENGTH_8D, UART1_STOPBITS_1, UART1_PARITY_NO, UART1_SYNCMODE_CLOCK_DISABLE, UART1_MODE_TXRX_ENABLE);

  //設(shè)置接收寄存器溢出中斷

  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);

  }

  //發(fā)送字節(jié)

  void Send(uint8_t dat)

  {

  //檢查并等待發(fā)送寄存器是否為空

  while(( UART1_GetFlagStatus(UART1_FLAG_TXE)==RESET));

  //發(fā)送字節(jié)

  UART1_SendData8(dat);

  }

  //發(fā)送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]);

  }

  //發(fā)送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

  接下來說說中斷函數(shù)表stm8s_it.c

  其中只要選用兩個中斷函數(shù)就可以了:

  INTERRUPT_HANDLER(UART1_RX_IRQHandler, 18) 接收寄存器溢出中斷

  里面添加安卓上位機發(fā)送過來的數(shù)據(jù)的處理程序,我這里寫的是通道選擇的判斷。

  INTERRUPT_HANDLER(TIM4_UPD_OVF_IRQHandler, 23) 定時器4計數(shù)器溢出中斷

  里面添加初始化ADC0832和ADC0832數(shù)據(jù)讀取并UART1發(fā)送到安卓上位機。

  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.

  */

  //下面是我做的安卓上位機發(fā)送過來的數(shù)據(jù)判斷,這里可以改成自己想要的程序

  uint8_t tempData;

  tempData = UART1_ReceiveData8();

  if(tempData=='A')

  {

  channel = 0;

  }

  if(tempData=='Z')

  {

  channel = 1;

  }

  //清除UART1中斷標(biāo)識符

  UART1_ClearITPendingBit(UART1_IT_RXNE);

  }

  //定時器4計數(shù)器溢出中斷



關(guān)鍵詞: STM8 ADC0832

評論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉