<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 關(guān)于單片機(jī)的軟復(fù)位

          關(guān)于單片機(jī)的軟復(fù)位

          作者: 時(shí)間:2016-11-18 來(lái)源:網(wǎng)絡(luò) 收藏
          一、數(shù)組定位

          main()
          {
          unsignedcharcoderst[]={0xe4,0xc0,0xe0,0xc0,0xe0,0x32};//復(fù)位代碼
          (*((void(*)())(rst)))();//執(zhí)行上一行代碼,將rst數(shù)組當(dāng)函數(shù)調(diào)用
          }
          本來(lái)我告訴他嵌入如下代碼:
          clra
          pushacc
          pushacc
          reti
          結(jié)果他卻玩了前面哪一段,而數(shù)組rst[]中的內(nèi)容恰恰是上面的匯編機(jī)器碼,他的做法是將
          rst數(shù)組的數(shù)據(jù)當(dāng)作代碼保存,然后采用絕對(duì)地址方式指向該數(shù)組,將該數(shù)組中的代碼當(dāng)作
          函數(shù)來(lái)運(yùn)行。居然通過(guò)了!

          本文引用地址:http://www.ex-cimer.com/article/201611/315943.htm

          l單片機(jī)復(fù)位的更好方法
          帖子中匯編語(yǔ)言解釋如下:
          clra//清除ACC=0
          pushacc//壓0到堆棧——8位
          pushacc//再壓0到堆棧——再8位
          reti//返回到0地址,從而執(zhí)行。
          本句的分析方法同上,但更加精煉,沒(méi)有多余的匯編語(yǔ)句。

          軟件復(fù)位跟真正上電復(fù)位有很大差別:上電復(fù)位時(shí)大部分寄存器都有確定的復(fù)位值;軟件
          復(fù)位則只相當(dāng)于從0地址開(kāi)始執(zhí)行而已,寄存器不會(huì)變?yōu)榇_定的復(fù)位值。

          例:修改出錯(cuò)處理如下:
          ERR: CLR EA ;正確的軟件復(fù)位入口
          MOV 66H,#0AAH ;重建上電標(biāo)志
          MOV 67H,#55H
          MOV DPTR,#ERR1 ;準(zhǔn)備第一次返回地址
          PUSH DPL
          PUSH DPH
          RETI ;清除高級(jí)中斷激活標(biāo)志
          ERR1: CLR A
          PUSH ACC
          PUSH ACC
          RETI ;清除低級(jí)中斷激活標(biāo)志
          這時(shí),必須執(zhí)行兩次RETI,才能到達(dá)0000H,以保證清除全部中斷激活標(biāo)志,達(dá)到和硬件復(fù)位相同的效果。同樣,軟件陷井也必須由下列三條指令
          NOP
          NOP
          LJMP STAT
          改成:
          NOP
          NOP
          LJMP ERR
          才能達(dá)到目的。
          當(dāng)主程序受到干擾被軟件陷阱捕獲時(shí),中斷標(biāo)志并未置位,執(zhí)行ERR過(guò)程中,RETI指令等效于RET指令,同樣可以達(dá)到軟件復(fù)位的目的。有興趣的讀者可以將軟件陷阱代替死循環(huán),分別用LJMP STAT和LJMP ERR1來(lái)替代LJMP ERR,再將干擾檢測(cè)分別設(shè)在低級(jí)中斷和主程序中,實(shí)驗(yàn)結(jié)果必然證明同:只有LJMP ERR才能萬(wàn)無(wú)一失地實(shí)現(xiàn)軟件復(fù)位,使系統(tǒng)擺脫干擾同,恢復(fù)正常。在MCS-51單片機(jī)的軟件復(fù)位過(guò)程中,必須連續(xù)執(zhí)行兩次中斷返回指令RETI才能確保系統(tǒng)恢復(fù)正常。

          2、C語(yǔ)言復(fù)位

          void reset (void){((void (code *) (void)) 0x0000) ();

          復(fù)位程序并不能清除8051的中斷系統(tǒng)和某些8051的外圍設(shè)備,當(dāng)您在中斷程序中調(diào)用上面的軟件復(fù)位程序后,中斷將再不能觸發(fā)。因此,以上的軟復(fù)位程序不能在中斷子程序中調(diào)用。

          可以使用下面的程序跳到0000H實(shí)現(xiàn)軟復(fù)位,下面的程序?qū)嶋H上是一個(gè)函數(shù)指針,指針指向了0000H地址。((void (code *) (void)) 0x0000) ();

          下面的例子將實(shí)現(xiàn)軟件自復(fù)位
          void reset (void)
          {
          ((void (code *) (void)) 0x0000) ();
          }
          void main (void)
          {
          reset ();
          }

          你可能注意到以上的軟復(fù)位程序并不能清除8051的中斷系統(tǒng)和某些8051的外圍設(shè)備,當(dāng)您在中斷程序中調(diào)用上面的軟件復(fù)位程序后,中斷將再不能觸發(fā)。因此,以上的軟復(fù)位程序不能在中斷子程序中調(diào)用。

          下面的小段匯編函數(shù)可以在中斷程序或主程序中調(diào)用,該函數(shù)將0x0000壓棧,然后通過(guò)“RETI”出棧,這將清除中斷環(huán)境并讓程序從0000H重新開(kāi)始運(yùn)行。

          ?PR?RESET  SEGMENT CODERSEG ?PR?RESET
          ; C prototype:  void reset (void);PUBLIC resetreset: POP  ACC  ; pop return addressPOP  ACCCLR  A    ; push 0 as newPUSH ACC  ; return address to stackPUSH ACCRETI      ; execute return of interruptEND以上程序在選擇bank 0寄存器組時(shí)工作良好,假如選擇的不是bank0寄存器組,那么可能無(wú)法獲得預(yù)料的結(jié)果。你應(yīng)該在以上的程序或啟動(dòng)代碼中加上“MOV PSW, #0”來(lái)選擇bank 0寄存器組。

          我的軟復(fù)位,用了多年沒(méi)問(wèn)題
          clrEA
          callreset
          callreset;用了多少中斷優(yōu)先級(jí),就調(diào)用幾次reset
          jmpstart;跳轉(zhuǎn)至0000h



          關(guān)鍵詞: 單片機(jī)軟復(fù)

          評(píng)論


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

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();