Цифровые часы с 7-сегментным дисплеем с pic18f445k22 - PullRequest
0 голосов
/ 29 июня 2019

Может кто-нибудь, пожалуйста, помогите мне?Я написал код ниже, чтобы запустить цифровые часы, но когда я симулирую в протеусе, светодиод только мигает, а на дисплее отображается только 00:00, и это никогда не считается.Я использовал таймер 0 и таймер 1, чтобы сгенерировать задержку в 1 секунду

Любое предложение ??

ФУНКЦИИ И ПЕРЕМЕННЫЕ

 int display(int num);
 int clock_PIC();

    /*VARIAVEIS GLOBAIS*/
    char controlador = 0x01;
    int unidade, dezenas, centenas, milhares;

    char  segundos   = 0x00;
    char  minutos    = 0x00;
    char  horas      = 0x00;
    char  clck_cont  = 0x00;
    char flags_min = 0x00;
    char flags_hor = 0x00;

ПРЕРЫВАНИЕ (ТАЙМЕР 0 И ТАЙМЕР 1)

void interrupt interrupcao ()
{
//    int a;
//    display(a);

    if(INTCONbits.TMR0IF)
        {           
            if(!digito_milhares && controlador == 1)
            {
                controlador    = 0x02;
                digito_unidade = 0x00;
                digito_dezenas  = 0x00;
                digito_centenas = 0x00;
                PORTC = 0x00;
                milhares = (horas%100)/10;
                digito_milhares = 0x01;
                PORTC = display(milhares);
            }//end if digito milhares


            else if(!digito_centenas && controlador == 2)
            {
                controlador    = 0x03;
                digito_unidade = 0x00;
                digito_dezenas  = 0x00;
                digito_milhares = 0x00;
                PORTC = 0x00;
                centenas = horas%10;
                digito_centenas = 0x01;
                PORTC = display(centenas);
            }//end if digito centenas


            else if(!digito_dezenas && controlador == 3)
            {
                controlador    = 0x04;
                digito_unidade = 0x00;
                digito_centenas = 0x00;
                digito_milhares = 0x00;
                PORTC = 0x00;
                dezenas = (minutos%100)/10;
                digito_dezenas = 0x01;
                PORTC = display(dezenas);
            }//end if digito dezenas


            else if(!digito_unidade && controlador == 4)
            {
                controlador    = 0x01;
                digito_dezenas  = 0x00;
                digito_centenas = 0x00;
                digito_milhares = 0x00;
                PORTC = 0x00;
                unidade = minutos%10;
                digito_unidade = 0x01;
                PORTC = display(unidade);
            }//end if digito unidade

            // clear the TMR0 interrupt flag
            INTCONbits.TMR0IF = 0x00;

        }//end if TMR0IF

    if(PIR1bits.TMR1IF)
    {
        LATBbits.LATB7 = ~ LATBbits.LATB7;

        clck_cont ++;

        if(clck_cont == 0x02)
        {
            clck_cont = 0x00;
            segundos++;

        }//end clck_cont 

        // clear the TMR1 interrupt flag
        PIR1bits.TMR1IF = 0x00;

        //TMR1H 11; 
        TMR1H = 0x0B;

        //TMR1L 220; 
        TMR1L = 0xDC;

    }//end TMR1IF

}//end interrupcao

ОСНОВНАЯ ФУНКЦИЯ

void main(void)
{

    PORTB = 0x4F;
    PORTC = 0x3F;

    /**
    TRISx registers
    */

    TRISB = 0x70;
    TRISC = 0x80;
    /**
    ANSELx registers
    */

    ANSELC = 0x00;
    ANSELB = 0x00;

    // SCS FOSC; IRCF 4MHz_HFINTOSC/4; IDLEN disabled; 
    OSCCON = 0x50;
    // PRISD enabled; SOSCGO disabled; MFIOSEL disabled; 
    OSCCON2 = 0x04;
    // INTSRC disabled; PLLEN disabled; TUN 0; 
    OSCTUNE = 0x00;


    // TMR0H 0; 
    TMR0H = 0x00;
    // TMR0L 6; 
    TMR0L = 0x06;
    // T0PS 1:16; T08BIT 8-bit; T0SE Increment_hi_lo; T0CS FOSC/4; TMR0ON enabled; PSA assigned; 
    T0CON = 0xD3;


    //T1GSS T1G_pin; TMR1GE disabled; T1GTM disabled; T1GPOL low; T1GGO done; T1GSPM disabled; 
    T1GCON = 0x00;

    //TMR1H 11; 
    TMR1H = 0x0B;

    //TMR1L 220; 
    TMR1L = 0xDC;

    // Clearing IF flag before enabling the interrupt.
//    PIR1bits.TMR1IF = 0;

    // T1CKPS 1:8; T1OSCEN disabled; T1SYNC do_not_synchronize; TMR1CS FOSC/4; TMR1ON enabled; T1RD16 enabled; 
    T1CON = 0x37;

    INTCONbits.GIE = 1;     //Habilita a interrupcao global
    PIE1bits.TMR1IE = 1;    //Habilita a interrupcao do timer 1
    INTCONbits.TMR0IE = 1;  //Habilita a interrupcao do timer 0  
    INTCON2bits.nRBPU = 0;  //Habilita o pull up interno do port B

    while (1)
    {  
        clock_PIC();
    }//end while
}//end void

ФУНКЦИЯ ЧАСОВ

int clock_PIC()
{

    if (segundos > 59)
    {
        segundos = 0x00;
        minutos++;

        if(minutos > 59)
        {
            minutos = 0x00;
            horas++;

                    if (horas > 23)
                    {
                        horas = 0x00;

                    }//end horas

        }//end minutos

    }//end segundos

    if(!butt_minutos) flags_min = 0x01;
    if(!butt_horas) flags_hor = 0x01;

    if(butt_minutos && flags_min)
    {
        flags_min = 0x00;
        minutos++;
        if(minutos > 59) minutos = 0x00;
    }

    if(butt_horas && flags_hor)
    {
        flags_hor = 0x00;
        horas++;
        if(horas > 23) horas = 0x00;
    }

}//end clock_PIC

ФУНКЦИЯ ДИСПЛЕЯ

int display(int num)
{
    int cathode;                               //armazena código BCD

// -- Vetor para o código BCD --
    int SEGMENTO[10] = {0x3F,                  //BCD zero   '0'
                        0x06,                  //BCD um     '1'
                        0x5B,                  //BCD dois   '2'
                        0x4F,                  //BCD três   '3'
                        0x66,                  //BCD quatro '4'
                        0x6D,                  //BCD cinco  '5'
                        0x7D,                  //BCD seis   '6'
                        0x07,                  //BCD sete   '7'
                        0x7F,                  //BCD oito   '8'
                        0x67};                 //BCD nove   '9'

    cathode = SEGMENTO[num];                   //para retornar o cathode

    return(cathode);                           //retorna o número BCD

} //end display

1 Ответ

0 голосов
/ 01 июля 2019

Я действительно не знаю, для какой конкретной PIC вы используете, но, как правило, во встроенных системах прерывания должны состоять из краткого лаконичного кода. Механизм прерываний вызовет этот код, и ожидается, что он завершится в течение короткого времени, так как другое прерывание может возникнуть и пропустить (даже один и тот же ISR). Таким образом, обычной практикой является небольшая обработка внутри обратного вызова ISR (например, пометка прерывания как посещенного) и изменение флага, помечающего, что прерывание было сгенерировано, и выполнение большей части работы в main() или другая функция. Это не всегда возможно, так как это похоже на пул, а не на обратный вызов, но это характерно для микроконтроллеров низкого уровня.

Помимо кода, большого для ISR, вы должны избегать вызова других функций внутри обратного вызова прерывания из-за накладных расходов, которые он добавляет. Кроме того, в данном конкретном случае вы используете оператор модуля %, который подразумевает деление: если PIC не имеет аппаратного делителя, это действительно медленная операция.

Что вам нужно сделать, это переместить код внутри ISR void interrupt interrupcao () в другую функцию и оставить только код, который должен изменить регистр внутри нее, и пометить, что прерывание было снято. Затем проверьте этот флаг внутри основного цикла и выполните обработку.


Еще одна вещь, которая когда-то все испортила, когда обратный вызов прерывания был длинным, это пространство, физическая память, которая была назначена функции. Увеличение его размера привело к переполнению кода в другое пространство прерываний; но я сомневаюсь, что это случилось с вами, так как я видел только кодирование в ассемблере (компилятор поместит код в нужное место, но я не знаю, как вы компилируете;)).

...