關(guān) 閉

新聞中心

EEPW首頁(yè) > 工控自動(dòng)化 > 設(shè)計(jì)應(yīng)用 > 單片機(jī)監(jiān)控程序的實(shí)現(xiàn)

單片機(jī)監(jiān)控程序的實(shí)現(xiàn)

——
作者:柯 鋒 周永鵬 黃錦恩 時(shí)間:2006-07-23 來(lái)源:現(xiàn)代電子技術(shù) 收藏
  首先分析了單片機(jī)(80C196為例)監(jiān)控程序的實(shí)現(xiàn);介紹了在Win98平臺(tái)使用VisualC++6.0實(shí)現(xiàn)串行通信,以及如何利用串行通信將匯編后的二進(jìn)制用戶(hù)程序發(fā)送至單片機(jī)執(zhí)行。
關(guān)鍵詞 API函數(shù) 串行通信 VisualC++6.0 單片機(jī)

1  
  在調(diào)試單片機(jī)應(yīng)用系統(tǒng)時(shí),需要反復(fù)地修改用戶(hù)程序,為了避免頻繁地使用編程器寫(xiě)存儲(chǔ)芯片,可以編制單片機(jī)監(jiān)控程序,單片機(jī)的監(jiān)控程序接收來(lái)自PC機(jī)的用戶(hù)程序,PC機(jī)向單片機(jī)發(fā)送用戶(hù)程序。
2 用戶(hù)程序格式
  用戶(hù)將單片機(jī)源程序(.a(chǎn)sm文件)匯編后形成.hex格式的文件,該文件即為發(fā)送至單片機(jī)的十六進(jìn)制可執(zhí)行文件。該文件的結(jié)構(gòu)是:由多行構(gòu)成,行頭為起始符(:),然后是該行有效數(shù)據(jù)字節(jié)數(shù)(滿(mǎn)行時(shí)該數(shù)為10,即十進(jìn)數(shù)16),接下來(lái)為兩字節(jié)地址及00,接著是有效數(shù)據(jù),行尾是校驗(yàn)碼及換行符。為了簡(jiǎn)化單片機(jī)監(jiān)控程序,僅向單片機(jī)發(fā)送行字節(jié)數(shù)和有效數(shù)據(jù),可用下面的簡(jiǎn)單C語(yǔ)句從.hex文件中提取字節(jié)數(shù)和有效數(shù)據(jù):
  fscanf(fp,":%2x%4x00",&TranBytes,&Address);
  fscanf(fp,"%2x",&TranChar);
3 單片機(jī)監(jiān)控程序的實(shí)現(xiàn)
  單片機(jī)監(jiān)控程序?qū)崿F(xiàn)的功能為:接收來(lái)自PC機(jī)的用戶(hù)程序,將用戶(hù)程序放置在用戶(hù)程序段,當(dāng)用戶(hù)程序接收完畢后,跳轉(zhuǎn)至用戶(hù)程序段首地址以執(zhí)行用戶(hù)程序。接收采用單字節(jié)方式,即每次只接收一個(gè)字節(jié)。
    為了保證單片機(jī)接收用戶(hù)程序的準(zhǔn)確性,在接收用戶(hù)程序之前需要進(jìn)行握手。筆者所用的握手協(xié)議為:PC機(jī)發(fā)0x55,單片機(jī)收0x55后發(fā)回0x55給PC機(jī),上位PC機(jī)收0x55后再發(fā)0xaa,單片機(jī)收0xaa后發(fā)回,PC機(jī)收0xaa后握手成功,轉(zhuǎn)為發(fā)用戶(hù)程序,單片機(jī)轉(zhuǎn)為接收用戶(hù)程序。
  為了防止單片機(jī)監(jiān)控程序被破壞,需將其固化在EPROM里。筆者所用的80C196系統(tǒng),將地址2000H-7FFFFH固化,監(jiān)控程序從2080開(kāi)始。這樣一來(lái)又涉及到如何使用中斷向量的問(wèn)題。可以這樣解決:在中斷向量地址中放入8000之后的地址,舉例說(shuō)明,在串行中斷向量地址放入8030H,當(dāng)要使用串行中斷時(shí),在8030中PUSHF和LJMP指令,在8032中放入跳轉(zhuǎn)字節(jié)數(shù)。
  利用串行中斷,單片機(jī)接收用戶(hù)程序?yàn)橹鹦薪邮眨ㄒ?jiàn)前敘.hex文件的結(jié)構(gòu))。先接收本行要接收的字節(jié)個(gè)數(shù),然后才將接受的有效數(shù)據(jù)寫(xiě)入用戶(hù)程序段,當(dāng)接受的有效數(shù)據(jù)數(shù)等于該行要接收的字節(jié)個(gè)數(shù)時(shí),準(zhǔn)備接收下一行,如此反復(fù),如果某行要接收的字節(jié)數(shù)為0,則表明用戶(hù)程序已經(jīng)傳完,將用戶(hù)程序段首址壓入堆棧再?gòu)棾觯ǜ淖冎袛喾祷氐刂芳夹g(shù)),以執(zhí)行用戶(hù)程序,如圖1所示。


4 Win98平臺(tái)串行通信的實(shí)現(xiàn)
  現(xiàn)在PC機(jī)的應(yīng)用程序絕大多數(shù)都是基于Win98,在進(jìn)行串行通信時(shí)可以通過(guò)調(diào)用API函數(shù)來(lái)實(shí)現(xiàn)。API函數(shù)提供了對(duì)串口的各種操作。串口通信時(shí)通過(guò)CreateFile,GetCommState,SetCommState,WriteFile,ReadFile,CloseHandle以及超時(shí)函數(shù)GetCommTimeouts,SetCommTimeouts來(lái)實(shí)現(xiàn)。利用CreateFile函數(shù)打開(kāi)串口,獲取串口句柄,CloseHandle關(guān)閉串口句柄,利用GetComm-State和SetCommState對(duì)通信參數(shù)進(jìn)行設(shè)置,WriteFile及ReadFile可對(duì)串口進(jìn)行讀寫(xiě)。在TC環(huán)境下,對(duì)串口的操作方式有兩種:查詢(xún)方式和中斷方式。在VC環(huán)境下,對(duì)串口的操作方式可有多種:查詢(xún)方式,同步I/O方式,異步I/O方式,以及事件驅(qū)動(dòng)I/O方式。筆者采用的是異步I/O方式,它可以讓串口操作在后臺(tái)執(zhí)行。讓讀寫(xiě)串口操作有足夠的時(shí)間在后臺(tái)執(zhí)行。使用異步I/O方式時(shí),采用如下方式打開(kāi)串口:
  HANDLE m_hCom=CreateFile("COM2",GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN_EXISTING,F(xiàn)ILE_FLAG_OVERLAPPED,NULL);
  FILE_FLAG_OVERLAPPED指明串口為異步方式打開(kāi)。此時(shí)可以分別在WriteFile和ReadFile的最后一個(gè)參數(shù)中指定一個(gè)OVERLAPPED結(jié)構(gòu),如下所示:
    OVERLAPPED m_OverlappedWrite,     
  m_OverLappedRead;
  m_OverlappedWrite.hEvent=CreateEvent(NULL,TRUE,F(xiàn)ALSE,NULL);
  m_OverlappedRead.hEvent=CreateEvent(NULL,TRUE,F(xiàn)ALSE,NULL);
    CreateEvent函數(shù)創(chuàng)建一個(gè)有名或無(wú)名的事件對(duì)象,第一個(gè)參數(shù)為保密屬性,設(shè)為NULL,第二個(gè)參數(shù)為T(mén)RUE,指明要用ResetEvent函數(shù)將事件設(shè)為無(wú)信號(hào),若為FALSE,則在一個(gè)等待該事件的線(xiàn)程被釋放后系統(tǒng)自動(dòng)將其設(shè)置為無(wú)信號(hào),第三個(gè)參數(shù)設(shè)事件初始為無(wú)信號(hào),第四個(gè)參數(shù)設(shè)事件名為NULL。在讀寫(xiě)操作中使用事件:
  WriteFile(m_hCom,&WriteBuffer,nByteToWrite,&nByteWritten,&m_OverlappedWrite)

  ReadFile(m_hCom,&ReadBuffer,nByteToRead,&nByteRead,&m_OverlappedRead)

  當(dāng)WriteFile接手控制時(shí)m_OverlappedWrite.hEvent為無(wú)信號(hào),讀操作完成后,m_OverlappedWrite.hEvent變?yōu)榘l(fā)信號(hào)狀態(tài),寫(xiě)操作完成類(lèi)似。在使用ReadFile和WriteFile對(duì)串口進(jìn)行讀寫(xiě)時(shí)需要注意的是,這兩個(gè)函數(shù)均為立即返回型函數(shù),亦即,可能在實(shí)際的讀寫(xiě)操作還沒(méi)有完成時(shí)函數(shù)就返回,操作轉(zhuǎn)入后臺(tái),但這并不表明實(shí)際的操作失敗。如果返回錯(cuò)誤代碼為ERROR_IO_PENDING(通過(guò)調(diào)用GetLastError獲取),說(shuō)明讀寫(xiě)操作仍在進(jìn)行,這時(shí)事件仍然為無(wú)信號(hào),為了顯式地限定前臺(tái)等待操作的時(shí)間,可以進(jìn)行延時(shí)處理,調(diào)用GetTickCount,GetTickCount()函數(shù)獲取系統(tǒng)當(dāng)前時(shí)間,類(lèi)似于C中的biostime()函數(shù)??梢酝ㄟ^(guò)調(diào)用GetOverlappedResult獲取后臺(tái)的操作情況,該函數(shù)報(bào)告最近一次OVERLAPPED操作的結(jié)果,函數(shù)原型如下: 
    BOOLGetOverlappedResult(
    HANDLEhFile,     //文件句柄

    LPOVERLAPPEDlpOverlapped,
                   //OVERLAPPED結(jié)構(gòu)指針

    LPDWORD lpNumberOfBytesTransferred, 

                  ?。瘜?shí)際完成的字節(jié)數(shù)

    BOOLbWait

                   //等待標(biāo)志 );
    在進(jìn)行串口讀寫(xiě)時(shí),hFile為串口句柄,lpOverlapped是該函數(shù)等待的事件,lpNumberOfBytesTransferrd為實(shí)際讀寫(xiě)完成的字節(jié)數(shù),當(dāng)bWait為T(mén)RUE時(shí),該函數(shù)等待讀寫(xiě)操作完成后返回,bWait為FALSE時(shí)函數(shù)立即返回。關(guān)于以上兩個(gè)函數(shù)的使用,參看如下程序代碼:
if(!WriteFile(m_hCom,&WriteBuffer,nByteToWrite,&nByteWritten,&m_OverlappedWrite)){

if(GetLastError()==ERROR_IO_PENDING) 

    endtime=GetTickCount()+1000;?。訒r(shí)1000 ms  
    while(!GetOverlappedResult(m_hCom,&m_OverlappedWrite&nByteWritten,F(xiàn)ALSE)) 

{ 

   if(GetTickCount()>endtime) break;

  }

 } 
if(nByteWritten)    
    ?。幚硭x的字節(jié)

 }
  函數(shù)ReadFile的調(diào)用可以類(lèi)似地處理,進(jìn)行延時(shí)處理后,就可以等到讀寫(xiě)操作完成之后再執(zhí)行后續(xù)程序。為了應(yīng)用的方便,可以通過(guò)調(diào)用API函數(shù),編寫(xiě)自己的串口操作類(lèi)庫(kù)來(lái)完成實(shí)際的需要。
5 結(jié) 語(yǔ)

  本文介紹的單片機(jī)監(jiān)控程序及串行通信方法簡(jiǎn)單,易于實(shí)現(xiàn),程序運(yùn)行穩(wěn)定。

   獻(xiàn)

1 汪建,孫開(kāi)放,章述漢.MCS-96系列單片機(jī)原理及應(yīng)用技術(shù).武昌:華中理工大學(xué)出版社,1999
2 Peter Norton,Rob McGregor.MFC開(kāi)發(fā)Windows95/NT4應(yīng)用程序.北京:清華大學(xué)出版社,1998
3 譚浩強(qiáng).C程序設(shè)計(jì).北京:清華大學(xué)出版社,1998



評(píng)論


相關(guān)推薦

技術(shù)專(zhuān)區(qū)

關(guān)閉