《電子技術(shù)應(yīng)用》
您所在的位置:首頁 > 嵌入式技術(shù) > 設(shè)計(jì)應(yīng)用 > 單片機(jī)開發(fā)中的一些實(shí)用技巧(上)
單片機(jī)開發(fā)中的一些實(shí)用技巧(上)
摘要: 很多朋友正在學(xué)習(xí)單片機(jī)開發(fā)技術(shù),但開發(fā)中免不了要碰到這樣、那樣的問題,有些問題可能無礙大局,但有一些問題卻直接影響到產(chǎn)品的成本、體積、性能。這里介紹筆者的幾個(gè)技巧,希望對(duì)大家的工作有幫助。
關(guān)鍵詞: 51單片機(jī) 單片機(jī)
Abstract:
Key words :

  很多朋友正在學(xué)習(xí)單片機(jī)開發(fā)技術(shù),但開發(fā)中免不了要碰到這樣、那樣的問題,有些問題可能無礙大局,但有一些問題卻直接影響到產(chǎn)品的成本、體積、性能。這里介紹筆者的幾個(gè)技巧,希望對(duì)大家的工作有幫助。

  一.C語言中嵌入?yún)R編語言

  單片機(jī)開發(fā)中,通常我們使用C語言編寫主程序,這樣可以充分借助C語言工具提供的運(yùn)算庫函數(shù)及強(qiáng)大的數(shù)據(jù)處理能力。但C語言的可控性不及匯編語言,在有些對(duì)時(shí)序要求嚴(yán)格的處理上,我們還需用靈活性更強(qiáng)的匯編語言來編寫。上海AVR單片機(jī)培訓(xùn)這樣就產(chǎn)生了C語言和匯編語言混合編程的問題,一般分成三種方式:1.匯編語言調(diào)用C語言函數(shù);2. C語言調(diào)用匯編語言;3. C語言中嵌入?yún)R編語言。這里我們主要介紹第3種,即C語言中嵌入?yún)R編語言。

  下面的一段程序是主程序調(diào)用精確的205μS延時(shí)子程序并使P1.0交替輸出高、低電平的方波。

  /*------------程序名test.c------------*/

  #include P 晶振頻率12.000MHz<>

  /****************/

  void delay(void)//延時(shí)205μS

  {

  #pragma asm

  MOV R0,#100

  LOOP:

  DJNZ R0,LOOP

  #pragma endasm

  }

  /***************/

  void main (void)//主函數(shù),其功能使P1.0交替輸出高、低電平的方波

  {

  while(1)

  {P1_0=!P1_0;

  delay();}

  }

  具體實(shí)現(xiàn)過程為:

  1.先用匯編語言編制一段延時(shí)程序,在keil開發(fā)環(huán)境中編譯,然后進(jìn)行軟件仿真,晶振頻率的設(shè)置應(yīng)和你的要求相符。仿真時(shí)注意觀察左邊寄存器窗口內(nèi)的時(shí)間顯示,調(diào)整延時(shí)程序的參數(shù)可得到我們需要的精確延時(shí)。

  2.用C51編寫主程序及延時(shí)子程序的外殼(等待嵌入?yún)R編語言),假定此程序名稱為test.c。

  3.將第1步所得的匯編延時(shí)子程序放入C51編寫的延時(shí)子程序外殼中。注意在開始及結(jié)束時(shí)分別加上#pragma asm、#pragma endasm語句,這種方法是通過asm與endasm告訴C51編譯器,中間行不用編譯為匯編行。

  4.按照Keil的使用方法,建立工程文件并添加源程序。

  5.點(diǎn)擊含有匯編程序的C源程序后再右擊,在彈出的下拉菜單中選中Options for File ‘test.c’(圖1),這時(shí)出現(xiàn)圖2所示的界面,勾選Generate Assembler SRC File(生成匯編SRC文件)及Assembler SRC File(封裝匯編文件)使其有效。

  6. 根據(jù)項(xiàng)目的編譯模式加載封裝庫文件,通常在Small模式時(shí)為C51S.LIB(該文件在C:\Keil\C51\Lib\C51S.LIB),具體見圖3。

  7.點(diǎn)擊Rebuild target(重建所有目標(biāo)文件)即可得到編譯結(jié)果(圖4)。

  

  圖1

  

  圖2

  

  圖3

  

  圖4

  二。用軟件擴(kuò)展外部中斷

  大家知道,51單片機(jī)的外部中斷只有2個(gè),書本上曾介紹了一種擴(kuò)展外部中斷源的方法,但是需增加硬件開銷(見圖5)。經(jīng)或非門引入外中斷源輸入端(/INT0或/INT1),同時(shí)又連到某I/0口。這樣,每個(gè)“源”都可能引起中斷,在中斷服務(wù)程序中通過軟件查詢便可確定哪一個(gè)是正在申請(qǐng)的中斷源,其查詢的次序則由中斷源優(yōu)先級(jí)決定,這就可實(shí)現(xiàn)多個(gè)外部中斷源的擴(kuò)展。

  

  圖5

  這種方法盡管擴(kuò)展了外部中斷源,但也有不盡人意之處,如設(shè)計(jì)一個(gè)具有8個(gè)中斷源的電路,則需一個(gè)8輸入端的或非門(或門),顯然,對(duì)體積與成本都不利。這里介紹筆者設(shè)計(jì)的擴(kuò)展外部中斷源的方法,由純軟件實(shí)現(xiàn),不添加一個(gè)元件(見圖6)。

  

  圖6

  #include < P>

  static unsigned char data m;//m為全局變量

  /*-------延時(shí)子程序-------*/

  void delay(unsigned int k)

  {

  unsigned int i,j;

  for(i=0;i

  for(j=0;j<121;j++)

  {;}}

  }

  /*---外部中斷INT0子程序---*/

  void init0()interrupt 0

  {

  delay(10);//延時(shí)10mS抗抖動(dòng)干擾

  if(P3_2==0)

  {

  EX0=0;//關(guān)INT0中斷

  EA=0;//關(guān)總中斷

  P3_2=0;//置P3.2為低電平

  P2=0xff;//置P2口為全1

  m=P2;//讀取P2口狀態(tài)至m

  P2=0x00;//恢復(fù)P2口為全0

  P3_2=1; //置P3.2為高電平

  IT0=1;//置INT0為邊沿觸發(fā)

  EX0=1; //開INT0中斷

  EA=1;} //開總中斷

  }

  /********主程序*********/

  void main(void)

  {

  P2=0x00;// 置P2口為全0

  P3_2=1;// 置P3.2為高電平

  IT0=1;// 置INT0為邊沿觸發(fā)

  EX0=1;// 開INT0中斷

  EA=1; //開總中斷

  while(1)//無限循環(huán)

  {

  P0=m;//將全局變量m中的內(nèi)容輸出至P0口

  P3_0=!P3_0;//P3.0取反,指示程序狀態(tài)

  delay(500);//延時(shí)500mS

  }

  }

  程序解釋:無按鍵按下時(shí),P3.0的發(fā)光管閃亮,作程序狀態(tài)顯示。主程序初始化時(shí),置P2口為全0,置P3.2為高電平,同時(shí)置INT0為邊沿觸發(fā),并開放中斷。8個(gè)按鍵的任一個(gè)按下時(shí)都會(huì)引起INT0中斷,進(jìn)入中斷服務(wù)子程序后,首先關(guān)閉中斷,然后置P3.2為低電平,置P2口為全1,再讀取P2口狀態(tài)至m,通過查詢m的狀態(tài)字即可知道正在申請(qǐng)的中斷源。這里我們采用的方法是將m輸出至P0口點(diǎn)亮LED作指示。退出中斷時(shí),重新開放中斷。

此內(nèi)容為AET網(wǎng)站原創(chuàng),未經(jīng)授權(quán)禁止轉(zhuǎn)載。