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

External Interrupts

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



Joined: 23 Mar 2017
Posts: 4
Location: Guanajuato

View user's profile Send private message

External Interrupts
PostPosted: Thu Mar 23, 2017 5:34 pm     Reply with quote

Hi guys:
I've just started working with microcontrollers and we're learning how to make external interrupts work out.
Allegedly, this is supposed to function as it is written but it does anything.
This code modifies the velocity of how the leds change, it is done with a potentiometer. It goes from 10us to 1s per Led, that part works pretty good but when we try to change the sequence nothing happens.

I know the code looks pretty cheap but we're on the road to learn how to do this and improve.

If anyone could help me to understand where is my mistake, I'd be very thankful. Please!



Code:
#include <18f4550.h>

#FUSES NOWDT                    //No Watch Dog Timer
#FUSES WDT128                   //Watch Dog Timer uses 1:128 Postscale
#FUSES PUT                      //Power Up Timer
#FUSES NOBROWNOUT               //No brownout reset
#FUSES NOPBADEN                 //PORTB pins are configured as digital I/O on RESET
#FUSES NOLVP                    //No low voltage prgming, B3(PIC16) or B5(PIC18) used for I/O
#FUSES NOXINST                  //Extended set extension and Indexed Addressing mode disabled (Legacy mode)
#fuses INTRC_IO                 //Internal RC Osc, no CLKOUT   
#use delay(internal=8Mhz)

#use standard_io(B) //Habilitar puerto B

#define use_portd_lcd TRUE
#define LCD_RS        PIN_D0
#define LCD_E         PIN_D1
#define LCD_DB4       PIN_D4
#define LCD_DB5       PIN_D5
#define LCD_DB6       PIN_D6
#define LCD_DB7       PIN_D7


#include <lcd1.c>

double ciclos, N=0;
float lectura,tiempo;
int i=0, entrada=1;

#INT_EXT
void  EXT_isr(void) //Inicio la interrupcion externa
{
  if(entrada<5) //4casos
    {
    ++entrada;
    }
  else
  {
    entrada=0;
  }
}

#INT_TIMER1
void  TIMER1_isr()  //Inicio interrupcion interna
{
 lectura=read_adc(); //lectura digital del pot
 delay_us(1);
 ciclos=(3.90234375)*(lectura); //relacion entre el numero de ciclos y el tiempo
 if(N>=ciclos) //
 {
  N=0; //Valor inicial de N
  i++; //Incremento para las secuencias de leds
 }
 N=N+1; //Va aumentando el numero de ciclos
 set_timer1(0xFC18);//le pone un valor al timer dentro de la variable y cuando se desborda llama a la interrupcion
}


void main()
{
port_B_pullups(0xFF);  //Se habilitan las resistencias internas del puerto B
setup_oscillator(osc_8mhz|osc_intrc); //Se habilita el oscilador interno

setup_timer_1(T1_INTERNAL | T1_DIV_BY_2);  //Configuracion del timer1 para interrupcion interna
enable_interrupts(INT_TIMER1); //Se habilitan interrupciones internas por timer 1
enable_interrupts(INT_EXT); //Se habilitan interrupciones externas
enable_interrupts(GLOBAL);

set_tris_d(0x00);//Se habilita puerto D como salidas
set_tris_c(0x00);//Se habilitan puerto C como salidas
set_tris_b(0xff); //Puerto B como entradas

lcd_init(); //Se inicia la LCD
 
//!//float read;
//!//int salida=0;
setup_adc(ADC_CLOCK_INTERNAL); //Se habilita el reloj ADC interno para lectura digital del puerto analogico       
set_adc_channel(0);//Se lee desde el puerto A0
setup_adc_ports(ALL_ANALOG); //Todos los pines A como analogicos
delay_us(15); //Tiempo de de espera de 15us

//set_timer1(0xFFEC);  //65516, así comienza desde los 10us
 
 while(TRUE)
 {
 switch(entrada) //Inicio de switch con la variable entrada de las interrupciones externas
 {
 
 case 1:
 
 if(i>7) //Reinicia la variable, solo para la primera secuencia
 {
  i=0;
 }
 
 if(i==0)  //------------------------------------------------------------------Secuencia 1
 {
  output_high(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==1)
 {
  output_low(PIN_C2);
  output_high(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==2)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_high(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==3)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_high(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==4)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_high(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==5)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_high(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==6)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_high(PIN_C7);
 }                      //-----------------------------------------------------Secuencia 1
 
 tiempo=N*(0.001);
 printf(lcd_putc, "tiempo:%04f", tiempo); //Impresion enn lcd del tiempo en el que se encuentra
 break;
 
 
 case 2:
 
 if(i==12) //Reinicia la variable, para las demas secuencias
 {
  i=0;
 }
 
 if(i==0)  //------------------------------------------------------------------Secuencia2
 {
  output_high(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==1)
 {
  output_low(PIN_C2);
  output_high(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==2)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_high(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==3)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_high(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==4)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_high(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==5)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_high(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==6)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_high(PIN_C7);
 }         
 
 if(i==7)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_high(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==8)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_high(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==9)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_high(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==10)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_high(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==11)
 {
  output_low(PIN_C2);
  output_high(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==12) 
 {
  output_high(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }  //-------------------------------------------------------------------------Secuencia2
 break;
 
 case 3:
 
 if(i==12) //Reinicia la variable, para las demas secuencias
 {
  i=0;
 }
 
 if(i==0)  //------------------------------------------------------------------Secuencia3
 {
  output_high(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==1)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==2)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==3)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==4)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==5)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==6)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_high(PIN_C7);
 }         
 
 if(i==7)
 {
  output_low(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_high(PIN_C7);
 }
 
 if(i==8)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_high(PIN_C7);
 }
 
 if(i==9)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_high(PIN_C7);
 }
 
 if(i==10)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_high(PIN_C7);
 }
 
 if(i==11)
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_high(PIN_C6);
  output_high(PIN_C7);
 }
 
  if(i==12) 
 {
  output_low(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_high(PIN_C7);
 }    //-----------------------------------------------------------------------Secuencia3
 break;
 
 case 4:
 
 if(i==12) //Reinicia la variable, para las demas secuencias
 {
  i=0;
 }
 
 if(i==0)  //------------------------------------------------------------------Secuencia4
 {
  output_high(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==1)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==2)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==3)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==4)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==5)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==6)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_high(PIN_C7);
 }         
 
 if(i==7)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_high(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==8)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_high(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==9)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_high(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==10)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_high(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
 if(i==11)
 {
  output_high(PIN_C2);
  output_high(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }
 
  if(i==12) 
 {
  output_high(PIN_C2);
  output_low(PIN_C1);
  output_low(PIN_C0);
  output_low(PIN_D2);
  output_low(PIN_D3);
  output_low(PIN_C6);
  output_low(PIN_C7);
 }     //----------------------------------------------------------------------Secuencia4
 break;
 
 default:
 entrada=0;
 delay_ms(1000); //Espera entre cada secuencia
 break;
 }
 
 }
}
PCM programmer



Joined: 06 Sep 2003
Posts: 21708

View user's profile Send private message

PostPosted: Thu Mar 23, 2017 7:58 pm     Reply with quote

What does your external interrupt circuit look like ? It should be similar
to the schematic below. Pin B0 is held at a logic high level by the pullup.
When you press the button it goes to a logic low level.
Code:

           +5v
            |
            <
            > 4.7K       
            <         ___  Switch 
To          |        _|_|_
PIC -----------------o   o------
pin B0                         |             
                              --- GND
                               -

You didn't set the edge for the external interrupt. It defaults to the rising
edge. The push-button circuit is intended to work with the falling edge.
Example of selecting the falling edge for the External interrupt:
Code:
ext_int_edge(H_TO_L);
ErickCuevas



Joined: 23 Mar 2017
Posts: 4
Location: Guanajuato

View user's profile Send private message

PostPosted: Thu Mar 23, 2017 8:21 pm     Reply with quote

Let me try that and I'll let you know what happens, thank you so much for your time.
ErickCuevas



Joined: 23 Mar 2017
Posts: 4
Location: Guanajuato

View user's profile Send private message

PostPosted: Thu Mar 23, 2017 9:10 pm     Reply with quote

PCM programmer wrote:
What does your external interrupt circuit look like ? It should be similar
to the schematic below. Pin B0 is held at a logic high level by the pullup.
When you press the button it goes to a logic low level.
Code:

           +5v
            |
            <
            > 4.7K       
            <         ___  Switch 
To          |        _|_|_
PIC -----------------o   o------
pin B0                         |             
                              --- GND
                               -

You didn't set the edge for the external interrupt. It defaults to the rising
edge. The push-button circuit is intended to work with the falling edge.
Example of selecting the falling edge for the External interrupt:
Code:
ext_int_edge(H_TO_L);


I did what you said but now it only does the second sequence, I wrote the missing line and put the push in that position.
jeremiah



Joined: 20 Jul 2010
Posts: 1358

View user's profile Send private message

PostPosted: Fri Mar 24, 2017 6:58 am     Reply with quote

I think you should take a step back and start a new side project that just lets you play with the external interrupt. Don't include all the LED code, just a few small bits to test out the external interrupt. Get your example paired down to something manageable. Get that working and then you can integrate it into your actual code. At that point you will know the issue comes from the logic and not the external interrupt if you still have trouble.

As an aside, you can clean up a lot of the LED code by making a function like this:



Code:

void set_leds(unsigned int8 mask){
   output_bit(PIN_C2,bit_test(mask,6));
   output_bit(PIN_C1,bit_test(mask,5));
   output_bit(PIN_C0,bit_test(mask,4));
   output_bit(PIN_D2,bit_test(mask,3));
   output_bit(PIN_D3,bit_test(mask,2));
   output_bit(PIN_C6,bit_test(mask,1));
   output_bit(PIN_C7,bit_test(mask,0));
}


Then you can just call that with a bit mask to set the LED's appropriately

You can even get fancy then and make a table of all your patterns:
Code:

#define NUM_PATTERN_SETS  5
#define MAX_SET_SIZE     13

unsigned int8 pattern_table[NUM_PATTERN_SETS][MAX_SET_SIZE] = {
   { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 },
   { 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 },
   { 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40 },
   /*do other patterns..I got too lazy to fill in the rest*/
};


At that point you can use your logic to choose the right indices for the pattern you want and just do a call to set_leds() passing in an element from that table:

Code:

set_leds(pattern_table[entrada][i]);


That said, you do have a lot of logic issues in your code as well as possibly some interrupt vs main code variable sharing timing issues. You have a lot of situations where you either don't update your LED's anymore or you repeat the same pattern multiple times in a row. Also, I think your timer interrupt is probably too complex. There is a lot of slow things happening in there that shouldn't be (ADC reads, floating point math, etc.). There are better ways to do that. As it stands now, you'll have a lot of bugs to fill out.
temtronic



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

View user's profile Send private message

PostPosted: Fri Mar 24, 2017 8:13 am     Reply with quote

a comment..

Whenever using 'push buttons' you need to 'debounce' the signal. Add a small value cap, say .1 mfd, in parallel with the switch. It along with the pullup will dampen the scratchy noise generated by the switch.
if you have access to an oscilloscope, attach and SEE what it looks like.

Where you can get into 'trouble' is having a noisey push button and a very fast PIC. The PIC sees the PB, triggers the ISR,carrys on, the PB still sends a 'false' closure, PIC see this as another signal, ISR fires.......In the good old days of 2Mhz CPUs this wasn't a problem... but a PIC at 64 MHz......it can be
ErickCuevas



Joined: 23 Mar 2017
Posts: 4
Location: Guanajuato

View user's profile Send private message

PostPosted: Mon Mar 27, 2017 6:34 pm     Reply with quote

Thank you all for your help, it finally worked. I can't thank you enough 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