CCS C Software and Maintenance Offers
FAQFAQ   FAQForum Help   FAQOfficial CCS Support   SearchSearch  RegisterRegister 

ProfileProfile   Log in to check your private messagesLog in to check your private messages   Log inLog in 

CCS does not monitor this forum on a regular basis.

Please do not post bug reports on this forum. Send them to CCS Technical Support

priority interrupts in PIC16F877A

 
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion
View previous topic :: View next topic  
Author Message
PIC457



Joined: 01 Jan 2012
Posts: 3

View user's profile Send private message

priority interrupts in PIC16F877A
PostPosted: Sun Jan 01, 2012 4:06 pm     Reply with quote

Hi everybody ! I'm making a project about DC servo (count Pulse encode)... and I have met with difficulties. I do not know what is happening. It took me a long time ... But in the end I did not solve the problem....

This in my code:
Code:

#include <16F877A.h>
#include <DEF_877A.h>
#device *=16 adc=10
#FUSES NOWDT, HS, NOPUT, NOPROTECT, NOLVP
#use delay(clock=20000000)

//#define LCD_DATA_PORT getenv("SFR:PORTD") 
#define LCD_ENABLE_PIN  PIN_E0                                    ////
#define LCD_RS_PIN      PIN_E1                                    ////
#define LCD_RW_PIN      PIN_E2   
#define LCD_DATA4       PIN_D4                                    ////
#define LCD_DATA5       PIN_D5                                    ////
#define LCD_DATA6       PIN_D6                                    ////
#define LCD_DATA7       PIN_D7     
#include <lcd.c>


int16 h=0,chuc,tram,h1,donvi,k=0;
int16 count=0;

#int_RB
 void int_RB()
    {   
     disable_interrupts(int_rb);
     disable_interrupts(int_timer0);
     GIE =1;
     if(RBIF)
      {
        if(RB0==1||RB0==0)              // check L_To_H  and H_To_L
        h++;                                     // count
        RBIF=0;
      }
    } 

#int_timer0
void int_timer0()
 {
   disable_interrupts(int_timer0);
   GIE =1;
   if(tmr0if)
   {
     ++count;
     if(count==5000)                        // period display LCD
       {
         count=0;
         h1=h;       
         k = h1+10;
         h=0;                                    // reset count
         
         set_timer0(6);
         tmr0if=0;
       }
   }
 }

void main()
{
set_tris_b(0b11110001);
lcd_init();
delay_ms(100);

enable_interrupts(int_timer0);
setup_timer_0(RTCC_INTERNAL|RTCC_DIV_2);
set_timer0(6);// T_dt = 2*(256 - 6)*1us = 500us

enable_interrupts(int_rb);
ext_int_edge(L_to_H);
enable_interrupts(global);

   while(1)
    {
     tram =h1/100;
     h1=h1%100;
     chuc=h1/10;
     donvi=h1%10;

     lcd_gotoxy(1,1);
     LCD_Putc(tram+0x30);
     LCD_Putc(chuc+0x30);
     LCD_Putc(donvi+0x30);
     
     tram =k/100;
     k=k%100;
     chuc=k/10;
     donvi=k%10;

     lcd_gotoxy(1,2);
     LCD_Putc(tram+0x30);
     LCD_Putc(chuc+0x30);
     LCD_Putc(donvi+0x30);
    }

}

And Result :


Hope the help from everyone !!!
temtronic



Joined: 01 Jul 2010
Posts: 9268
Location: Greensville,Ontario

View user's profile Send private message

PostPosted: Sun Jan 01, 2012 4:54 pm     Reply with quote

quick comments...
1) you do not have to disable/enable interrupts inside ISRs, compiler takes care of that.

2) you do not have to use the set_tris......() functions, again the compiler does that

3) with respect to the schematic, there are lots of errors and if wired as such will never work. Pins like osc pins, Vdd,Vss,_mclr, Vee of LCD.

4) If the 'device' attached to RB4 is a 'servo', do we assume it's a hobby type RC servo ? If so, it'll never work as +3V is NOT the normal voltage they require.

OR is the 'device' really a mechanical quadrature encoder? In that case, it'll never work as again +3V is applied to it, yet the PIC requires +5V for correct operation.Also no pin use is shown....

As for the code, you need to add comments on every line( they're free - don't take up space) to describe WHAT you're trying to do...not to many people here will try to figure out uncommented code. A brief description at the beginning would help as the schematic isn't complete.

The more information you supplied the better and faster we can help.Saying 'doesn't work' means nothing as we don't know what is supposed to happen.

this is a start......
PIC457



Joined: 01 Jan 2012
Posts: 3

View user's profile Send private message

PostPosted: Sun Jan 01, 2012 6:43 pm     Reply with quote

Thanks temtronic ! Thanks you very much !!!

I've revised my code as yours guide .... and now it is good ... but I have a question that In PIC16F is only interrupts vector ... In case 2 interrupts coincides .... then what happens ??? And how to fix ????

Code:


#int_RB                                  // interrupts RB4-RB7
  {
      if((RBIF)&&(RBIE))            // Bit interrupts flag RB and Bit interrupts    enable RB
         {
            //code here
            RBIF=0;                     // Clear bit RBIF
         }
  }

#int_timer0                             // interrupts timer0
  {
     if(tmr0if)                           // Bit interrupts flag timer 0
        {
           // code here
           tmr0if=0;                    // clear bit  flag
         }
   }
Ttelmah



Joined: 11 Mar 2010
Posts: 19585

View user's profile Send private message

PostPosted: Mon Jan 02, 2012 2:51 am     Reply with quote

Basically nothing......

The hardware only supports one interrupt 'level'. In CCS, the interrupt defined _first_, or the one listed first in a 'priority' statement, is always checked first. This is the interrupt 'priority' for a 16 family Chip. Remember the interrupt flag, once set, remains set till the handler is called, so what will happen is that if two interrupts trigger together, the one with the highest priority gets serviced first, and then the lower priority one is called.

On your code, you don't need to touch the interrupt flag, or check it. This is all already done for you. The 'RB' interrupt code will _only_ be called, if RBIF is set, and RBIE is set. Checking it again, is a waste of time. The compiler also automatically clears the interrupt flag when you leave the routine. Key thing in INT_RB, is that you _must_ read the port. The RBIF bit _cannot_ be cleared, until the port is read. It'll reset itself on the very next instruction, and 'INT_RB' will execute for ever.
Similarly on the timer code, tmr0if, _is_ set if the code is called, and will be cleared automatically when you leave the routine.
Critical thing throughout, is get out of the handlers quickly. Since the next interrupt won't be handled until you leave the handler code.

Best Wishes
PIC457



Joined: 01 Jan 2012
Posts: 3

View user's profile Send private message

PostPosted: Mon Jan 02, 2012 9:26 am     Reply with quote

Thank you for the useful advice !!! Very Happy Very Happy Very Happy
Display posts from previous:   
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion All times are GMT - 6 Hours
Page 1 of 1

 
Jump to:  
You cannot post new topics in this forum
You cannot reply to topics in this forum
You cannot edit your posts in this forum
You cannot delete your posts in this forum
You cannot vote in polls in this forum


Powered by phpBB © 2001, 2005 phpBB Group