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

Interrupt does not work with different PIC
Goto page 1, 2  Next
 
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion
View previous topic :: View next topic  
Author Message
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

Interrupt does not work with different PIC
PostPosted: Thu Feb 18, 2016 2:52 pm     Reply with quote

The following program works as shown with a PIC16F722 chip.
If I physically change the PIC to a 16F1938 in my hardware, change the first line to #include <16F1938>, and run the Project Wizard to change the chip to 16F1938 and recompile, the program does not work and seems to hang in the ISR, the LED stops blinking, and the “Keypress=x” never appears. The Welcome appears and the LED blinks until I press a key on the 4x4 keypad so there is an interrupt.
If I reverse the procedure back to a 16F722 then the program works again.
PCM 5.019, Win7, ICD3
I am starting to suspect the compiler but I am no expert so any suggestions would be appreciated.
Code:
//////////////////////////////////////////////////////////////////////////////////////////////
// File  test.c                                                                              //
// Digole Serial Adapter for 128x64 Graphic LCD.                                           //
// Port B is used for keyboard data entry, Port A for Analog input and control,             //
// Port C drives the display                                                                //
// RA4 drives an LED that is toggled every pass                                             //
// Purpose is to test the Keyboard interrupt ISR                                            //
// 18 Feb 2016                                                                              //
//////////////////////////////////////////////////////////////////////////////////////////////

/* Pre-processor directives */
 #include <16F1938.H>
// #include <math.h>
 #fuses INTRC_IO, NOWDT, PUT, NOPROTECT, BROWNOUT, MCLR
 #use delay (clock=2000000)
 #use I2C (master, SCL=PIN_C3, SDA=PIN_C4)
// #byte porta = 5
 #byte portb = 6
 #use fast_io(B)
 #byte portc = getenv("SFR:PORTC")
 
// global variables
static char key;         // character entered

// define I2C address
#define LCD_WRT_ADDR 0X4E               // LCD display
#define buff_size 22                    // characters per line plus one
   
// Function prototypes
    void clear_LCD (void);                      // Clear LCD
    void text_position (int line, int column);  // set start for next text
    void set_font (char size);                  // set font size
    void send_str(char buff[buff_size]);        // Send string to LCD

// Interrupt Service Routine
#INT_RB                  // any change on B4 to B7
void RB_isr(void)
{
// declare variables
 byte KeyPad [4] [4] = {'1','2','3','U',   // keypad array, row, col
                        '4','5','6','D',
                        '7','8','9','A',
                        'C','0','H','E'};
 int key_press, new_kp, old_kp, row, col, count;

// debounce key
new_kp = portb ^ 0xF0;                  // get and mask new keypress
old_kp = new_kp;                     // store
 if (new_kp == 0)                     // if not pressed
  {
   key = 0;
   return;                           // return false
  }
// adjust debounce here
 for (count = 1; count <= 10; count++)      // else confirm key still pressed
      delay_cycles (250);               // after 5 ms delay
 new_kp = portb ^ 0xF0;                  // get and mask new keypress again
 if (old_kp != new_kp)                  // if not legit
  {
   key = 0;
   return;                           // return false
  }
key_press = old_kp;

// Disable interrupts so release does not clear "key"
disable_interrupts (INT_RB);            // until enabled in main

// Find the column. A switch in the matrix is holding one of the RB7 thru RB4
// inputs low by connecting to one of the RB3 thru RB0 outputs. Find which of
// RB7 thru RB4 is low.
 switch (key_press)
  {
   case 0x10:
    col = 0;
    break;
   case 0x20:
    col = 1;
    break;
   case 0x40:
    col = 2;
    break;
   case 0x80:
    col = 3;
    break;
   default:
    col = 0;
    break;
  }
 
// FIND THE ROW. Now find which of RB3 thru RB0 is connected by the switch.
// Each of RB3 thru RB0 are brought high in turn to find the row.
// The interrupts-on-change must be disabled or this ISR will be triggered
// by actions in the following.

 output_high (PIN_B0);                  // scan for row 0
 new_kp = portb ^ 0xF1;
 if (new_kp == 0)
  row = 0;
 output_low (PIN_B0);

 output_high (PIN_B1);                  // scan for row 1
 new_kp = portb ^ 0xF2;
 if (new_kp == 0)
  row = 1;
 output_low (PIN_B1);

 output_high (PIN_B2);                  // scan for row 2
 new_kp = portb ^ 0xF4;
 if (new_kp == 0)
  row = 2;
 output_low (PIN_B2);

 output_high (PIN_B3);                  // scan for row 3. Pin_B3 will be set low
 new_kp = portb ^ 0xF8;                  // by the output_b(0xF0) statement below
 if (new_kp == 0)
  row = 3;

 key = KeyPad [row] [col];               // Get character from array. Key = True
 output_b (0xF0);                     // must leave outputs low
return;
}
// ************************************************************
// The main function
void main(void)
{
// setup ports
  set_tris_b (0xF0);                        // 11110000
  set_tris_c (0x00);                    // all outputs
  output_b(0xF0);                           // outputs low
// setup interrupts
   clear_interrupt(INT_RB);
   enable_interrupts(INT_RB);
   enable_interrupts(global);
   port_b_pullups(0xF0);      // pullups on inputs
// declare variables
  char buff[buff_size];
//  int i;
 
// setup display with welcome screen
  clear_LCD();                          // clear the LCD, sets default size,
  sprintf(buff, "Welcome");     // place text string in buffer
  text_position(3,0);                   // start first line, center
  send_str(buff);                       // display text array

// Turn on LED
  output_low(PIN_A4);                    // LOW = relay ON

// START Endless loop
    while (1)
 {
    delay_ms (500);
    output_toggle (pin_A4);             // flash for testing
// Check for key press
 if (key)
      {
    text_position(0,3);                             // start fourth line
    sprintf (buff, "Keypress= %c", key);            // send key to buffer
    send_str(buff);                                 // display text array
    key = false;                                    // clear the flag
    delay_ms(500);
   enable_interrupts(INT_RB);
   enable_interrupts(global);
      }     

}                       // end of while loop
}                        // end of main function

// Functions
// Clear Display
void clear_LCD (void)
   {
   I2C_START ();                // start I2C
   I2C_WRITE (LCD_WRT_ADDR);    // addr of LCD
   I2C_WRITE ('C');             // C CL to clear display
   I2C_WRITE ('L');             // L
   I2C_STOP ();                 // stop I2C
   }

// set position of next text
void text_position(int line, int column)
   {
   I2C_START ();                // start I2C
   I2C_WRITE (LCD_WRT_ADDR);    // addr of LCD
   I2C_WRITE ('T');            // T, TP set text position
   I2C_WRITE ('P');            // P
   I2C_WRITE (line);            // line position
   I2C_WRITE (column);          // column position
   I2C_STOP ();                 // stop I2C
   }
// Set Font Size
void set_font (char size)
   {
   I2C_START ();                // start I2C
   I2C_WRITE (LCD_WRT_ADDR);    // addr of LCD
   I2C_WRITE ('S');
   I2C_WRITE ('F');
   I2C_WRITE (size);
   I2C_STOP ();                 // stop I2C
   }

// send string to LCD
void send_str(char buff[buff_size])
   {
   int i;
   I2C_START ();                // start I2C
   I2C_WRITE (LCD_WRT_ADDR);    // addr of LCD
   I2C_WRITE('T');              // send TT for text coming
   I2C_WRITE('T');
    for (i=0; i<buff_size; i++)
   {
   I2C_WRITE(buff[i]);          // start with a Z
   }
   I2C_WRITE(0);
   I2C_STOP ();                 // stop I2C   
   }

// end[code][/code]
temtronic



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

View user's profile Send private message

PostPosted: Thu Feb 18, 2016 3:08 pm     Reply with quote

this...
#byte portb = 6
.. is wrong.
At least according to the datasheet I have PORTB is located at 00Dh, table 3-3 page 26.

What's _interesting_ to me is you have this...
#byte portc = getenv("SFR:PORTC")
... which is the preferred method of addressing addresses.

As you've just found out, not all PICs share a common, same register set.


What is happening is that your proram is NOT accessing PORTB instead it's looking at FSR1L.


Jay
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

PostPosted: Thu Feb 18, 2016 4:16 pm     Reply with quote

Wow! Ports A, B, and C have been on 5, 6, and 7 for all the chips I have ever used.
You are right but using #byte portb = getenv("SFR:PORTB") does not fix the problem.
It is strange that the analog and digital I/O on port A have been working with #byte porta = 5. I was trying to add an interrupt controlled keypad to a working project using the 16F1938 because it has EEPROM. I have noticed a drift in an ADC input.....
Any other suggestion?
asmboy



Joined: 20 Nov 2007
Posts: 2128
Location: albany ny

View user's profile Send private message AIM Address

PostPosted: Thu Feb 18, 2016 4:36 pm     Reply with quote

beyond the obvious - i fear that the NEWish 16F1938 chip
and your older compiler may be an imperfect fit..

its the complex INIT for all the IO functions would be my bet -

something extra needing to be done for the 1938 ....
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

PostPosted: Thu Feb 18, 2016 5:05 pm     Reply with quote

Thanks, I was thinking it was time to upgrade my compiler. I will let you know what happens.
temtronic



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

View user's profile Send private message

PostPosted: Thu Feb 18, 2016 7:00 pm     Reply with quote

re:
... I have noticed a drift in an ADC input.....

You should post a small program with JUST the ADC problem...

general comments though
ADC reference MUST be a stable supply( so NOT VDD)
The greater the bits the more 'unstable' the readings unless you're really good at PCB layout
Noise on VDD, supply lines can affect ADC readings if you use VDD as the ref.
EMI from bad PCB layout, no decoupling, wrong wiring/connections
Controlling relays(spikes, back EMF) cause bad reading
wrong selection for sample/hold capacitor
input sample to ADC is NOT stable.
'drift'? always one way ?? linear ?? based on time?? sampling rate??

Yeesh I could write a book on ADC design.Generally it's a process of elimination from observing a scope and following chip makers design specs( like bypass caps AT the chip ).

Jay
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

PostPosted: Thu Feb 18, 2016 7:41 pm     Reply with quote

I was just musing because I had the wrong address for port a but maybe it was never needed. I suspect the drift is the AD623 instrumentation Amp looking at a 30 amp shunt. They were bought cheap from China and I did a quick check that all showed zero by trying one at a time in the circuit without changing the output offset. They seemed good but occasionally I get an offset however the circuit is on a breadboard and wiggling wires sometimes eliminates it.

They are nice chips to use with a PIC because you can drive them from +/- 5v, feed them +/- mV (current) and have the output, in my case, as -30amps for 0v, 0amps for 2.5v, and +30amps for +5v so the ADC can handle it. The gain is set by a single resistor. You can't feed negative voltage into most PIC ADCs, at least not the ones I use. I use a resistor and the PIC diodes to protect the input but the input never gets to below zero anyway.

I trimmed pages of program to find out why the interrupt was hanging and just kept the minimum of code to simplify. The drift I see is not programming because it worked for years with a two op-amp version of the AD623A.

This is not my concern right now. I bought an update to my compiler but I can't download until tomorrow. I hope that solves the problem. If you got this far maybe you can explain a few things.

Why do I have to declare the address of a port? Why doesn't the compiler recognize something like portb or Port_B?
Why do I have to have enable_interrupts(INT_RB); as well as enable_interrupts(global); The program seems to need it but I really don't understand why.
Ttelmah



Joined: 11 Mar 2010
Posts: 19499

View user's profile Send private message

PostPosted: Fri Feb 19, 2016 12:05 am     Reply with quote

This is one of the problems of 'confused edges' to the PIC families.

You have the 'nominal' boundaries (PIC12, 16, 18 etc.), but there are chips that blur over these edges. So there are PIC16's, that have the more limited PIC12 instruction set, and also newer PIC16's, that have extended abilities, and as a result have a lot more basic registers,and some of these result in ones being relocated.
With all of the PIC's, if you are locating registers yourself, the old line 'read the data sheet' applies.....

The compiler does recognise the ports without you having to locate anything. The input functions reads the port. It is just that you are bypassing the compiler's functions, and talking to the port directly.

However as has already been pointed out the compiler will allow you to setup the #byte commands using a port name, and then the compiler will automatically locate these for you.

On the interrupts, the chip has two control bits. There is an individual bit for each interrupt, and a 'master' (GLOBAL) one for all the interrupts. Both have to be set to enable the interrupts.
The reason. Well imagine you have 10 interrupts all enabled, and you wanted to disable them all for a particular operation. If you only had the individual enables, you would have to turn them all off one by one. Instead disabling the 'GLOBAL' bit turns them all off in one operation.
However then imagine you just had the GLOBAL control. If there were no individual enables, turning this on, would enable every interrupt, including ones you don't want.
So there are separate 'individual' enables, and then a 'master' enable as well.

There is also another thing. If (for instance) you were using interrupts to wake from sleep. Here the global enable does not actually have to be 'on'. Any interrupt which has it's individual enable on, can wake the chip. If you leave GLOBAL off, this will then wake the chip, but won't result in an interrupt handler actually being called. Saves a lot of wasted time.
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

PostPosted: Fri Feb 19, 2016 1:05 pm     Reply with quote

Thanks to all above.
I now have the latest compiler installed. It does not solve the problem but I now get two warnings. I cannot see the problem. Here they are with the relevant code from the complete program above (with the addition of the correct address for portb)

>>> Warning 240 "test.c" Line 143(12,16): Pointer types do not match
>>> Warning 240 "test.c" Line 158(14,18): Pointer types do not match

Code:
#byte portb = getenv("SFR:PORTB")

139   // setup display with welcome screen
140   clear_LCD();                                // clear the LCD, sets default size,
141    sprintf(buff, "Welcome");               // place text string in buffer
142    text_position(3,0);                         // start first line, center
143   send_str(buff);                             // display text array

154   if (key)
155         {
156       text_position(0,3);                            // start fourth line
157       sprintf (buff, "Keypress= %c", key);           // send key to buffer
158       send_str(buff);                                    // display text array
159       key = false;                                          // clear the flag
160       delay_ms(500);
161   enable_interrupts(INT_RB);
   enable_interrupts(global);
         }     


I am now very suspicious of my use of portb and wonder if it is being read correctly. It would account for the program getting lost in the ISR that uses portb 6 times. Should I be using input_b() as in:

Code:
new_kp = input_b() ^ 0xF1;


if that is the correct way to do it? I don't think the pointer warnings are the problem but I would like to resolve them.
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

PostPosted: Fri Feb 19, 2016 1:16 pm     Reply with quote

OK I replaced the 6 portb with input_b() and the program now works (with the warnings).
An explanation of why portb does not work even when defined properly would be nice. Maybe the compiler still has a bug but I am more inclined to suspect my programming.
temtronic



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

View user's profile Send private message

PostPosted: Fri Feb 19, 2016 1:38 pm     Reply with quote

simply dump the listings for both the working and nonworking programs! Since you only made a slight change it should be obvious what is different.
Your program was hard for me to follow, thing like naming an array and a variable the same name (buff) is 'cloudy coding'. Self describing variable names makes it a lot easier. Also when using fast_io(), I like to have the tris() statements next in line though for 99% of the programs we've seen here standard_IO() works perfectly fine.

Also it might be better to put the port_b_pullups() before enabling the interrupt, maybe with say a 50ms delay, so to stabilze the inputs.


Jay
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

PostPosted: Fri Feb 19, 2016 2:01 pm     Reply with quote

Thanks Jay,
Where did I name buff a variable?
Should it always have the dimension attached as in
Code:
send_str(buff[buff_size]);

somehow that did not seem right. I will try your suggestions and see if I can figure the portb problem.
I am not a programmer and appreciate the help on this forum.
The functions talk to an LCD display.
PCM programmer



Joined: 06 Sep 2003
Posts: 21708

View user's profile Send private message

PostPosted: Fri Feb 19, 2016 2:04 pm     Reply with quote

Change it to this, and the pointer warnings will go away:
Code:

void send_str(char *buff);

void send_str(char *buff)
{


}
rovtech



Joined: 24 Sep 2006
Posts: 262

View user's profile Send private message AIM Address

PostPosted: Fri Feb 19, 2016 4:43 pm     Reply with quote

Thanks PCM Programmer,
The pointer to the array. I had forgotten about that.
Yes that cured that problem and I will try to get to the other later although with input_b() it is probably the way I should be doing it and it works.
I did not think using buff[] to show it was an array would work.
temtronic



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

View user's profile Send private message

PostPosted: Fri Feb 19, 2016 7:32 pm     Reply with quote

this...
143 send_str(buff); // display text array

I took 'buff' to be a single variable not an array of data

As PCM P pointed out the * pointer makes all the difference !

I like longer descriptive names like 'buffer_array' hoping to keep it organized in my mind as to what it is . I started programming when variable could only be 2 characters long and every line had a comment,even obvious ones!

Jay
Display posts from previous:   
Post new topic   Reply to topic    CCS Forum Index -> General CCS C Discussion All times are GMT - 6 Hours
Goto page 1, 2  Next
Page 1 of 2

 
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