1

我试图让我的 PIC18F2550 上的 Timer1 每秒触发一次中断以进行倒计时。到目前为止,我已经能够正确建立中断优先级以及其他设置,但是我的 ISR 只触发一次。我已经设置了我的程序以将倒计时中剩余的秒数打印到 LCD 上,因此用户可以清楚地看到已经过去了多少次中断。我已确保清除 ISR 中正确的中断标志,因为对于大多数在 google 上遇到相同问题的人来说,这就是问题的根源,但这并没有解决问题。到目前为止,我的程序的其他所有内容都运行良好。

(为了测试是否触发了正确的优先级,低计数向上而不是向下计数)

这是代码片段

//First testing program for PIC18F2550

#include <p18f2550.h>
#include <stdlib.h>
#include <delays.h>

#define _XTAL_FREQ 4915200

#pragma config PLLDIV = 1
#pragma config CPUDIV = OSC1_PLL2
#pragma config FOSC = XT_XT
#pragma config MCLRE = ON 
#pragma config BOR = OFF
#pragma config WDT = OFF
#pragma config IESO = OFF
#pragma config PBADEN = OFF
#pragma config LVP = OFF

unsigned char Countdown = 30;   (other vars, fcns omitted)

#pragma code high_vector = 0x08
void highvectinterrupt(void){
    _asm goto highisr _endasm
}

#pragma code low_vector = 0x18
void lowvectinterrupt(void){
    _asm goto lowisr _endasm
}


(...)

void highisr(void){
    Countdown--;
    PIR1bits.TMR1IF = 0;
}

void lowisr(void){
    Countdown++;
    PIR1bits.TMR1IF = 0;
}

void main(void){
    UCONbits.USBEN=0;  //Setup I/O
    UCFGbits.UTRDIS=1;
        OSCCONbits.SCS1 = 0;
        OSCCONbits.SCS0 = 0;
    BAUDCONbits.TXCKP=0;
        SPBRG = 0x3F;  //1200 baud
    TRISA = 0x00;
    TRISB = 0xFF;
    TRISC = 0x00;     

    beephigh();    //Done, clear the line and beep speaker 1

    LATA = 0;               //reset
    LATC = 0;
    Delay10KTCYx(8);
    LATAbits.LATA2 = 1;     //load the reset
    Delay10KTCYx(8);
    LATAbits.LATA3 = 1;     //stop reset
    LATAbits.LATA4 = 1; 
    LATAbits.LATA2 = 0;
    Delay10KTCYx(123);

    initlcd();

    Delay10KTCYx(10);

    setupUI();

    LATAbits.LATA0 = 0;   //Done, clear the line and beep speaker 2
    beeplow();

    INTCON = 0;             //Reset high interrupt flags and disable interrupts
    T1CON = 0x11110100;     //16 bit r/w; internal system clock; 8x prescale; timer1 osc disabled
    TMR1H = 0;              //Clear the counting register
    TMR1L = 0;
    PIE1bits.TMR1IE = 1;    //Enable Interrupts on Timer1
    PIR1bits.TMR1IF = 0;    //Clear the flag if set
    RCONbits.IPEN = 1;      //Enable priority levels
    INTCON2bits.TMR0IP = 0; //Timer0 on low priority
    IPR1bits.TMR1IP = 0;    //Timer1 on low priority
    IPR1bits.RCIP = 1;      //RS232 on high priority
    INTCONbits.GIEL = 1;    //Enable all low priority interrupts
    INTCONbits.GIEH = 1;    //Enable all high priority (must be enabled for low priority)
    T1CONbits.TMR1ON = 1;   //Turn Timer1 on    

    while(1){  //Idle and refresh screen 
        Delay10KTCYx(8);
        bin2ascii(Countdown);
        cmdlcd(0xCE);
        putclcd(ConvRegTens);
        putclcd(ConvRegOnes);
    }
}

最终,我盯着一个上面有“31”的液晶显示器。可能是我没有正确设置一个标志,导致没有任何工作正常,但据我检查所有内容都已启用并正确清除。我究竟做错了什么?我会很感激任何帮助,这非常令人沮丧。

另外:我已经清除了所有断点,因此它不会在任何地方停止,并在编译时对最新版本进行编程。所以我已经排除了运行旧版本的代码。

编辑:为了澄清,此代码仅用于功能;它不会每秒触发一次,而是更快。一旦我让它工作,我将播放溢出计数以使其达到 1 秒。

4

1 回答 1

2

您没有正确完成 ISR 例程!

#pragma code low_vector = 0x18
void lowvectinterrupt(void){
    _asm goto lowisr _endasm
}

在 ISR 执行后,中断必须再次打开,因此lowvectinterrupt必须用RETFIE指令完成,并且至少必须恢复标志和 WREG 寄存器。

Normaly 在 ISR 例程中声明如下:

static void interrupt isr(void)
  {

  }
于 2013-03-03T00:36:03.600 回答