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

Issue with time delays

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



Joined: 07 May 2011
Posts: 42
Location: Bulgaria

View user's profile Send private message

Issue with time delays
PostPosted: Tue Sep 04, 2012 11:03 pm     Reply with quote

I have an issue with the following code.
Compiler version is 4.120, MCU is PIC12F683 and clock is the internal 4MHz oscillator.
Problem is that every time the time delays are different. For example when i push button on ADC pin AN3) and the voltage is above a certain value, a pwm is started after a 5 seconds delay. This delay is different every time. When i release the button and the voltage disappears, a delay of 5 seconds should be present. But this delay is not the same every time. Sometimes is 3 seconds, sometimes more.

Code:
#include <12f683.h>
#device adc=10
#fuses INTRC_IO,NOWDT,NOMCLR,NOBROWNOUT
#fuses PROTECT   // code protect
#fuses CPD       // data eeprom code protect
#use delay(clock=4000000)

//time delay constants
#define delay_on 76 //5sec on delay
#define delay_off_5 76 //5sec off delay
#define delay_off_5_gab 15 //1sec delay

int8 ss=0,hs=0,status=0;
int16 on_counter,off_counter,off_gab_counter;
int16 kl15,mode,duty=0,set;

void soft_start()
{
 if(ss==1)
 { 
  duty++;
  if(duty>set)
  duty=set;   
 }
 else
 {
  duty=0;
 }
}

void hid_start()
{
 if(hs==1)
 { 
  duty=set;
 }
 else
 {
  duty=0;
 }
}


#int_RTCC
void RTCC_isr()
{

if(kl15>1017) //overvoltage protection (over 4.98V)
{
 setup_ccp1(CCP_OFF);
 output_low(PIN_A2);
 output_low(PIN_A4);
 duty=0;
}
else
{
 if(input(PIN_A3)==1) //if gab is on
 { 
  off_gab_counter--;
  if(!off_gab_counter )
  {
  off_gab_counter = delay_off_5_gab;
  setup_ccp1(CCP_OFF);
  output_low(PIN_A2);
  output_low(PIN_A4);
  duty=0;
  ss=0;
  hs=0;
  status=0;
  }
 }
 else
 {
    if(kl15>=700 && input(PIN_A5)==0 && status==0)
    {
      if(kl15>=820)
      {
       setup_ccp1(CCP_PWM);
       on_counter--;
       if(!on_counter )
       { 
        on_counter = delay_on;   
        ss=1;
        hs=1;
        output_high(PIN_A4);
       }
      }
    }
 
    else if(kl15>=700 && status==1)
    {
      if(kl15>=820)
      {
       setup_ccp1(CCP_PWM);
       on_counter--;
       if(!on_counter )
       { 
        on_counter = delay_on;   
        ss=1;
        hs=1;
        output_high(PIN_A4);
       }
      }
    }
   
    else if(kl15<700)
    {
       off_counter--;
       if(!off_counter )
       {
        off_counter = delay_off_5;   
        setup_ccp1(CCP_OFF);
        output_low(PIN_A2);
        output_low(PIN_A4);
        duty=0;
        ss=0;
        hs=0;
        status=0;
       }   
    }
    else
    {
    ss=0;
    hs=0;
    duty=0;
    status=0;
    setup_ccp1(CCP_OFF);
    output_low(PIN_A4);
    }
         
 }
       
}
 
}


void main()
{
on_counter = delay_on;
off_counter = delay_off_5;
off_gab_counter = delay_off_5_gab;
output_low(PIN_A2);
output_low(PIN_A4);
setup_timer_0(RTCC_INTERNAL|RTCC_DIV_256|RTCC_8_BIT);
enable_interrupts(INT_RTCC);
setup_adc_ports(sAN0 || sAN1);
setup_adc(ADC_CLOCK_DIV_8);
setup_ccp1(CCP_PWM);
set_pwm1_duty(0);
setup_timer_2(T2_DIV_BY_16, 249, 1); //250Hz
enable_interrupts(GLOBAL);

while(true)
{
set_pwm1_duty(duty);

set_adc_channel(0);
delay_us(25);
kl15=read_adc();

set_adc_channel(1);
delay_us(25);
mode=read_adc();

   if(mode>260 && mode<325)
   {
    set=250;
    status=0;
    soft_start();
   }
   
   if(mode>430 && mode<495)
   {
    set=300;
    status=0;
    soft_start();
   }
   
   if(mode>820 && mode<870)
   {
    set=700;
    status=0;
    soft_start();
   }
   
   if(mode>890 && mode<955)
   {
    set=1023;
    hid_start();
    status=1;
   }

delay_ms(10);

}

}
temtronic



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

View user's profile Send private message

PostPosted: Wed Sep 05, 2012 5:04 am     Reply with quote

You've got WAY too much code in the ISR !!

ISRs must be small and simple. Just reset/set a flag or two,set a few(<5) variables and get out fast!

In 'main', check the 'flag' variables and at accordingly..

Trying to do a lot of code inside the ISR will 'stall' or 'delay' the actual overall program or as you've found out , give random timings.

hth
jay
Mike Walne



Joined: 19 Feb 2004
Posts: 1785
Location: Boston Spa UK

View user's profile Send private message

PostPosted: Wed Sep 05, 2012 5:17 am     Reply with quote

Please explain how you think the 5s delays are created.

Mike
ckielstra



Joined: 18 Mar 2004
Posts: 3680
Location: The Netherlands

View user's profile Send private message

PostPosted: Wed Sep 05, 2012 5:32 am     Reply with quote

I do agree with Jay that your ISR should be made smaller and faster, but I doubt this is the source of your problem. Your ISR is only executed every 65536 instruction cycles, that should be long enough to allow some processing.

More likely the problem is in your decision tree inside the ISR. This spaghetti code of if-the-else constructions is very difficult to follow and easy to overlook one execution path. A better design is to use the technique known as 'State machine'. Basically it is a method to first draw on paper all your allowed 'stages' and allowed 'transitions'. Then explicitly convert this drawing to code. Search the internet for more information and code examples.
Problem with your approach is that all allowed transitions are _implicit_ in the code and this is for us humans hard to follow in all details.
tonkostz



Joined: 07 May 2011
Posts: 42
Location: Bulgaria

View user's profile Send private message

PostPosted: Wed Sep 05, 2012 11:20 am     Reply with quote

Mike Walne wrote:
Please explain how you think the 5s delays are created.

Mike

Timer0 will overflow after 65.5ms. So 76x65.5ms=4978ms. Am i wrong?
PCM programmer



Joined: 06 Sep 2003
Posts: 21708

View user's profile Send private message

PostPosted: Wed Sep 05, 2012 11:43 am     Reply with quote

Quote:

setup_adc_ports(sAN0 || sAN1);

The || operatior is not the same as | and will have undesired results,
such as one of the pins above not being configured for analog input.


Quote:

//time delay constants
#define delay_on 76 //5sec on delay
#define delay_off_5 76 //5sec off delay
#define delay_off_5_gab 15 //1sec delay

on_counter = delay_on;
off_counter = delay_off_5;
off_gab_counter = delay_off_5_gab;

The C style convention is to put constants in all caps. In the code above,
this is not done and the visual cue is missing. It appears to be loading
variables from other variables.
See section 11 - Naming Conventions on page 13:
http://www.literateprogramming.com/indhill-cstyle.pdf
Mike Walne



Joined: 19 Feb 2004
Posts: 1785
Location: Boston Spa UK

View user's profile Send private message

PostPosted: Thu Sep 06, 2012 9:29 am     Reply with quote

Quote:
Timer0 will overflow after 65.5ms. So 76x65.5ms=4978ms. Am i wrong?

Yes, I get the 65.5ms part.

BUT, somehow you are wanting to wait for 76 of these 65.5ms intervals.

You are complaining you get ~3s delays rather than 5s.

So, I presume that something goes wrong in the way you count off 76 of your 65.5ms intervals.

What I was asking for, was a lucid explantion of HOW the counting is achieved.

This problem is your's not mine. I don't want to decipher all of your code to arrive at the answer. I want you to either work it out for yourself, or give me enough guidance to be able to help you.

Mike
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