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

Debounce routine not working

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



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

Debounce routine not working
PostPosted: Sat Mar 31, 2012 7:54 pm     Reply with quote

PIC16F887 //EDIT--The code I am trying is actually in the 3rd post. The following will not return a value as it is marked as "void" my mistake
I have a very bouncy switch in dire need of debouncing. I am trying the following to remedy it with out success.
Code:

#define DEBOUNCE_PERIOD_IN_MS  10
#define DEBOUNCE_COUNT  2

void wait_for_keypress(void)
{
int count = 0;

while(1)
  {
   if(!input(PIN_B2) == 1)
      count++;
   else
      count = 0;
   
   if(count == DEBOUNCE_COUNT)
      break; // I want to return to "scanfaces" and execute my "bitset"

   delay_ms(DEBOUNCE_PERIOD_IN_MS);
  }

}


int8 scanfaces(void)

   int led_output=0;
   int face;
   
    if(!input(pin_B2)) {    // Checking Face 1
    wait_for_keypress();

//I want to debounce here and then return to execute below.
   
      bit_set(led_output,0);         
                       }
    else{
       bit_clear(led_output,0);
        }   

This doesn't seem to be working. It compiles fine but.... I've even tried to return a value to "wait_for_keypress" and then use that value as a signal to execute, as in the following
Code:

void wait_for_keypress(void)
{
int key_confirmed=0;
int count = 0;
while(1)
  {
   if(!input(PIN_B2) == 1)
      count++;
   else
      count = 0;
   
   if(count == DEBOUNCE_COUNT){
    Key_confirmed=1;
     }
  else{
          delay_ms(DEBOUNCE_PERIOD_IN_MS);
         }
  return(key_confirmed);

}


 int8 scanfaces(void)

   int led_output=0;
   int face;
   
    if(!input(pin_B2))   
    wait_for_keypress();

   if(wait_for_keypress()==1)
      bit_set(led_output,0);         
    else{
       bit_clear(led_output,0);
        }

Does anyone have any advice or direction? I've been at it all day. I am not very experienced but am picking things up at a reasonable pace and am learning so much. I am very excited and truly have a passion for this.


Last edited by rems on Sat Mar 31, 2012 9:09 pm; edited 1 time in total
PCM programmer



Joined: 06 Sep 2003
Posts: 21708

View user's profile Send private message

PostPosted: Sat Mar 31, 2012 8:40 pm     Reply with quote

Quote:

void wait_for_keypress(void)
{


return(key_confirmed);
}

How do you return something when the function is declared as void ?
You are telling the compiler that it returns nothing.

I'm not convinced you even tested this code, because if I do a test
compile, I get this error message:
Quote:

Error 127 "PCM_Test.c" Line 10(1,7): Return value not allowed in void function
rems



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

PostPosted: Sat Mar 31, 2012 9:04 pm     Reply with quote

You are absolutely correct. I copied the wrong code out. The following is what I use and does in fact compile with no errors. I added a bitset to it.
Code:

int8 wait_for_keypress_1(void)
{
  char count=0;
  int button_press;
  int key_confirmed=0;
 
while(1)
  {
   if(input(pin_B2)==1)
      count++;
   
   if(count == DEBOUNCE_COUNT)
    {
     bit_set(key_confirmed,0);
    }
   else{
     
   delay_ms(DEBOUNCE_PERIOD_IN_MS);
   bit_clear(key_confirmed,0);
   }

   if (key_confirmed==0b00000001)
            button_press=1;

    return(button_press);
}

}


 int8 scanfaces(void)

   int led_output=0;
   int face;

   OUTPUT_low(pin_B1);     // Configuring cage to sense faces
   
   if(!input(pin_B2))    // Checking Face 1
    wait_for_keypress_1();

   if(wait_for_keypress_1()==1){
         bit_set(led_output,0);         
    }   

   else{
       bit_clear(led_output,0);
        }

Of course this is followed by main which runs the "scanfaces" function along with other things not needed for the sake of my question. I apologize for the "fools mission" I sent you on due to my error in copying.
PCM programmer



Joined: 06 Sep 2003
Posts: 21708

View user's profile Send private message

PostPosted: Sat Mar 31, 2012 10:29 pm     Reply with quote

My comments are placed above each section of code.
Code:

int8 wait_for_keypress_1(void)
{
  char count=0;
  int button_press;
  int key_confirmed=0;

  while(1)
  {


Below, you are incrementing a counter if the switch pin reads a
a high level. But in your schematic here, from a previous thread,
http://i1249.photobucket.com/albums/hh515/remsphotos/ccs_pic_schematic.jpg
you have pull-ups on the switch input pins to the PIC. You don't show it
in the schematic but I assume the bottom sides of the switches are tied
to ground. Are they ? If they are, then the switches are "active low".
So, when they're pressed, you should be doing the counting. So you
should be looking for a 0 level in the code below, not a 1.
Code:

   if(input(pin_B2)==1)
      count++;
 


This section below is needlessly complex. Look at the original code
again. No need to add complexity and additional variables.
Code:

   if(count == DEBOUNCE_COUNT)
    {
     bit_set(key_confirmed,0);
    }
   else{
     
   delay_ms(DEBOUNCE_PERIOD_IN_MS);
   bit_clear(key_confirmed,0);
   }

   if (key_confirmed==0b00000001)
            button_press=1;


Below, you have a return statement at the end of the while() loop.
That means the while() loop will return immediately after the first pass.
It will only do one pass. I'm sure you don't intend that. The original
code had a test, and used a 'break' statement to break out of the
while() loop if the test condition was satisfied.
Code:

    return(button_press);
}

}
rems



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

PostPosted: Sun Apr 01, 2012 6:49 am     Reply with quote

I can't believe I forgot my exclamation mark
"if(!input(pin_B2)==1)"

I know the code is probably needlessly complex. I don't know enough to make it simple. I figured I didn't need the "bitset" but when it wasn't working, I thought I'd better try something that I have seen work before. Meanwhile, I had made a silly mistake the whole time.

Thank you for pointing out my blunders. I learned a lot. I will go back to the original and try again. I just wasn't sure if the "break" would go back to where I wanted it to. So I took it out and tried using the "return" Now, I realize it will contradict the "while".

I will keep trying. Thank you so much for your time and advice.
ckielstra



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

View user's profile Send private message

PostPosted: Sun Apr 01, 2012 7:32 am     Reply with quote

rems wrote:
I can't believe I forgot my exclamation mark
"if(!input(pin_B2)==1)"

I know the code is probably needlessly complex. I don't know enough to make it simple.
Just a small suggestion to make things more simple is in the line referenced here. In my experience humans have troubles with reading logical NOT instructions, so your code would be easier to read when you change it to:
Code:
if (input(PIN_B2) == 0)
Functionality of this code is exactly the same but now it easier to see you are testing for a zero value (instead of 'not 1').

Note also that I changed the capitalization of pin_B2 to PIN_B2. It is good coding practice to write constants in all upper case letters. In fact, that is how this value was defined in the header file. You code compiles because the CCS compiler is case insensitive by default, this is different from the C standard and can be changed with the #case precompiler directive.
rems



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

PostPosted: Tue Apr 03, 2012 10:52 pm     Reply with quote

I appreciate the advice.
I have my debounce to where I think it really should be working but it isn't.

Does anyone have any thoughts or suggestions. The following is an excerpt from my code.
Code:

#define DEBOUNCE_PERIOD_IN_MS  10
#define DEBOUNCE_COUNT  2

void wait_for_keypress_1(void)
{
int count=0;
 
while(1)
  {
   if(!input(PIN_B2)==0)
      {
       count++;
      }
    else{
         count=0;
        }
   
   if(count == DEBOUNCE_COUNT)
    break;
 
   delay_ms(DEBOUNCE_PERIOD_IN_MS);
}

}


void wait_for_keypress_2(void)
{
int count=0;
 
while(1)
  {
   if(input(PIN_B3)==0)
      {
       count++;
      }
    else{
         count=0;
        }
   
   if(count == DEBOUNCE_COUNT)
    break;
 
   delay_ms(DEBOUNCE_PERIOD_IN_MS);
}

}


 int8 scanfaces(void)

   int led_output=0;
   int face;

 // ************************************Configuring cage to sense faces 

    output_low(PIN_B1);   
 
 //************************************ Checking Face 1 

    if(input(PIN_B2)==0) {   
    wait_for_keypress_1();
     bit_set(led_output,0);  // will the "break;" come back to this line?                   
     
}
    else {
         bit_clear(led_output,0);
    }     
                 
  //************************************ Checking Face 5             
    if(input(PIN_B3)==0) {
     wait_for_keypress_2();
      bit_set(led_output,1);  // will the "break;" come back to this line?           
       }
     else{
        bit_clear(led_output,1); 
         }
}

Will the "break;" automatically bring my program back to where it left? ie will the next instruction be my "bitset" ?
Is there something better that I should be looking at?
jeremiah



Joined: 20 Jul 2010
Posts: 1349

View user's profile Send private message

PostPosted: Wed Apr 04, 2012 6:17 am     Reply with quote

I don't expect wait_for_keypress_1() to work well the way you have it. Prior to calling it, you check:

Code:

if(input(PIN_B2)==0)


Then in the function, you increment count if:
Code:

if(!input(PIN_B2)==0)


The two if statements don't match, so I would expect a gnd'ed button to go into this and never fully increment the count in wait_for_keypress_1()
rems



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

PostPosted: Wed Apr 04, 2012 6:32 am     Reply with quote

It still doesn't work. I am concerned it may be due to how I think the "break;" will perform. Will it break to where I think?

void wait_for_keypress_1(void)
{
char count=0;


while(1)
{
if(input(pin_B2)==0)
{
count++;
}
else{
count=0;
}

if(count == DEBOUNCE_COUNT)
break;


delay_ms(DEBOUNCE_PERIOD_IN_MS);

}


}
void wait_for_keypress_2(void)
{
char count=0;


while(1)
{
if(input(pin_B3)==0)
{
count++;
}
else{
count=0;
}

if(count == DEBOUNCE_COUNT)
break;


delay_ms(DEBOUNCE_PERIOD_IN_MS);

}


}

int8 scanfaces(void)
{
int led_output=0;
int face;

// ************************************Configuring cage to sense faces

output_low(PIN_B1);

//************************************ Checking Face 1

if(input(PIN_B2)==0) {
wait_for_keypress_1();
bit_set(led_output,0); // will it break to here?

}
else {
bit_clear(led_output,0);
}

//************************************ Checking Face 5
if(input(PIN_B3)==0) {
wait_for_keypress_2();
bit_set(led_output,1); // will it break to here?


}
else{

bit_clear(led_output,1);
}
Thank you for your help
jeremiah



Joined: 20 Jul 2010
Posts: 1349

View user's profile Send private message

PostPosted: Wed Apr 04, 2012 6:49 am     Reply with quote

All break does is take you out of the most current loop or switch statement. So it will simply jump to right outside your loop and continue from there.
rems



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

PostPosted: Wed Apr 04, 2012 6:56 am     Reply with quote

Okay, Great...I get it...So no wonder it isn't working...So it would just start "scanfaces" again and go back to the "keypress" function. Just a big loop.
Maybe I should have the "wait_for_keypress" function return a value and then check for that value in my "scanfaces" function? But then I couldn't loop my _wait_for_keypress" function with the "while(1)". Would a "for" loop be better in my "wait_for_keypress" function to work with some sort of "return"?
jeremiah



Joined: 20 Jul 2010
Posts: 1349

View user's profile Send private message

PostPosted: Wed Apr 04, 2012 7:23 am     Reply with quote

The benefit of how you did it is that you skip the last delay_ms() call when you have a good number.

Just for clarity's sake, this is a visual of what break does:

Code:

while(TRUE){
   if(input(PIN_B2) == 0){
      count++;
   }else{
      count = 0;
   }

   if(count == DEBOUNCE_COUNT){
      break;
   }

   delay_ms(DEBOUNCE_PERIOD_IN_MS);

}

/* BREAK JUMPS YOU TO THIS SPOT */



Since you didn't provide any code other than your scanfaces(), wait_for_keypress_1(), and wait_for_keypress_2() methods, I can't really comment on the rest of your inquiry.

The way I understand how your code works is that you just loop infinitely until you get a debounced low. Is that the intention?
rems



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

PostPosted: Wed Apr 04, 2012 7:46 am     Reply with quote

I have a very bouncy switch. It is homeade and literally has a ball within a cage that has 6 possible positions to be in. I need to debounce each of those positions. When the ball is in position, an i/p is sensed "LOW" and the debounce should start. In the debounce, I want to check and recheck that the i/p is still low. I may need to do this several times or just increment the delay between checks. Like I said it's a very bouncy switch.

Here is my complete code.
Code:

#include <16F887.h>
#fuses INTRC_IO //use internal oscillator
#fuses NOWDT //No watch dog timer
#fuses PUT//
#fuses BROWNOUT//reset chip if voltage falls
#fuses MCLR//master clear enabled
#fuses NOLVP
#use delay(clock=8000000)


#define DEBOUNCE_PERIOD_IN_MS  100
#define DEBOUNCE_COUNT  2

#define resbutt pin_B5 //restart 'c' condition
                       //if condition is obtained turn on led
#define sleepbutt pin_D7


//**********************************Debounce Functions

void wait_for_keypress_1(void)
{
  int count=0;
 
 
while(1)
  {
   if(input(pin_B2)==0)
      {
       count++;
      }
    else{
         count=0;
        }
   
   if(count == DEBOUNCE_COUNT)
    break;
 
 
   delay_ms(DEBOUNCE_PERIOD_IN_MS);
     
}


}
void wait_for_keypress_2(void)
{
  int count=0;
 
 
while(1)
  {
   if(input(pin_B3)==0)
      {
       count++;
      }
    else{
         count=0;
        }
   
   if(count == DEBOUNCE_COUNT)
    break;
 
 
   delay_ms(DEBOUNCE_PERIOD_IN_MS);
     
}


}

void wait_for_keypress_3(void)
{
  int count=0;
 
 
while(1)
  {
   if(input(pin_B4)==0)
      {
       count++;
      }
    else{
         count=0;
        }
   
   if(count == DEBOUNCE_COUNT)
    break;
 
 
   delay_ms(DEBOUNCE_PERIOD_IN_MS);
     
}


}


/*****************************************************************Declare a function that will SCAN the faces of the puzzle box*******************/
 int8 scanfaces(void)

   int led_output=0;
   int face;

 // ************************************Configuring cage to sense faces 

    OUTPUT_low(PIN_B1);   
 
 //************************************ Checking Face 1 

    if(input(PIN_B2)==0) {   
    wait_for_keypress_1();
     bit_set(led_output,0);         
     
}
    else {
         bit_clear(led_output,0);
    }     
                 
  //************************************ Checking Face 5             
    if(input(PIN_B3)==0) {
      wait_for_keypress_2();
      bit_set(led_output,1);           
                           
       
                       }
     else{
       
        bit_clear(led_output,1); 
         }
//*************************************** Checking Face 4
    if(input(PIN_B4)==0) {
      wait_for_keypress_3();
      bit_set(led_output,2);             
                           
       
                       }
       else{
         
            bit_clear(led_output,2);
          }
// **************************************Configuring cage to sense second 3 i/p's

    OUTPUT_low(pin_B2);
         
//*************************************** Checking Face 2 
    if(input(PIN_B3)==0) {
       wait_for_keypress_2();
       bit_set(led_output,3);             
     
                       }
        else{
           
             bit_clear(led_output,3);
            }

//**************************************** Checking Face 6

    if(input(PIN_B4)==0) {
      wait_for_keypress_3();
        bit_set(led_output,4);             
                               
 
                       }
       else{
       
            bit_clear(led_output,4);
           }

//****************************************** Configuring cage to sense last i/p

     OUTPUT_LOW(PIN_B3); 
 
//****************************************** Checking Face 3 
   
     if(input(PIN_B4)==0)  {
      wait_for_keypress_3();
        bit_set(led_output,5);             
                                     
       
                        }                   
       else{
           
             bit_clear(led_output,5);
            }

     
//*********************************************Making 6 unique possibilities
         
         if (led_output==0b00000001)
            face=1;
         
         if (led_output==0b00001001)
            face=2;
         
         if (led_output==0b00100000)
            face=3;
         
         if (led_output==0b00110100)
            face=4;
         
         if (led_output==0b00001010)
            face=5;
         
         if (led_output==0b00110000)
            face=6;
                     
                 
               
     return(face);
}

//*****************************************Password = 512346
void main()
{
int sh1=1,sh2,sh3,sh4,sh5,sh6;   // sw1=1 for starting first condition.

 
 while(1){
         
     

   
   if(scanfaces()==5&&sh1==1) //if butt 1 = 1 AND sw1=1(which it does to start) then 
        sh2=1; //Allow second condition
       
       
       
       
                             

   if(scanfaces()==1&&sh2==1)//if butt 2=1 AND sw2=1(it only will if butt 1 was pressed) then
        sh3=1; //Allow third condition
         
       
       
       
         
   if(scanfaces()==2&&sh3==1)  //if the three condition are obtained turn the led on.
        sh4=1;
       
     

   if(scanfaces()==3&&sh4==1)  //if the three condition are obtained turn the led on.
        sh5=1;
       
     
       

   if(scanfaces()==4&&sh5==1)  //if the three condition are obtained turn the led on.
        sh6=1;
       
       

   if(scanfaces()==6&&sh6==1)   //if the three condition are obtained turn the led on.
      {
      output_high(pin_C5);
      sh6=0;
     
      }


   else if(scanfaces()==6&&sh6!=1) //if the algorithm is not executed wait for restart. (if butt 3=1 and sh3 does NOT =1 then )
      {
       sh1=0; //set all to zero
       sh2=0; //which ensures that only the proper combo(sequence) will activate
       sh3=0;
       sh4=0;
       sh5=0;
       sh6=0;
      }
         
   if(input(resbutt)==1) //restart entering sequence of buttons. if reset button pushed
      {
      output_low(pin_C5); //led will turn off
      sh1=1;   //allow condition start again
      }
}   
}

Because my "cage switch" shares common pins, I use my "scanfaces" to toggle i/p's and o/p's. I then give each a unique ID with the "bitset" and then return the value of each for use in the state machine, in my main program.. All works beautifully except of course for the debounce.
RF_Developer



Joined: 07 Feb 2011
Posts: 839

View user's profile Send private message

PostPosted: Wed Apr 04, 2012 9:39 am     Reply with quote

The basic idea is that your main while loop just goes round and round. It checks if things need doing: it checks buttons, it checks for incoming data that needs to be processed, it measures, it outputs stuff and so on. None of these things should ever wait anything other than a few tens of us or so. Instead they check and if nothing to do they do nothing in that loop.

Most of the time your main loop does nothing much. Only if something needs servicing: display needs updating, someones pressed a button, a message had been receved, a timed event has been flagged up and so on; does your main loop actually do any processing. That means you'll check (or poll) everything on every loop.

So. you never "wait for a key press". You note a key has been pressed; if so do stuff and set a counter so that the keys will be ignored for say 50ms worth of loops before looking at the keys again. Longer for your caged ball switch. Using a timer, or clock tick based software timer will help there.

With debouncing there actually isn't much point in checking the contact is still where you first detected it after a time. The bounce happens when a contact is first made or broken and then for a while un-makes or un-breaks repeatedly. So only the first detection is relevant, then hold off from making further detections for at least the longest time you'll get bounce, which for some switches (and it varies from switch to switch of one type and from one type of switch to another) may be longer than 100ms.

Your caged ball will be tricky as normal switches work with some form of spring arrangement that biases them in to on position or another. The ball however relies on gravity alone, and it may get dirty and not make good contact. Have you designed the ball switch to be break-before-make? If so then you should be able to check it goes to a "none connected" state between one face and another and so, if it does go into the "none" state and back then nothing has really happened... or has it? How you interpret that is dependant on your application. You almost ceratainly can't do that if you are multiplexing the pins with another function.

This is the simplest form of "co-operative multitasking". Where each thing that has to be done: e.g. detect keys, update display, read serial, measure voltages, send formatted data, is regarded as a seperate task and your loop is the most simple form of round-robin scheduler. No "task" should ever wait if it can help it, and certainly not if it has nothing to do. Even ADC reads can, if required, be started on on loop and read back on the next, eliminating the need to wait for the ADC to convert. I have one simple app that does this. Though I confess on my others waiting for the ADC typically forms the greatest single time in the overall loop execution time.

RF Developer
rems



Joined: 02 Feb 2012
Posts: 56

View user's profile Send private message

PostPosted: Thu Apr 05, 2012 6:50 am     Reply with quote

A lot to think about...That is some great stuff for me consider. With a lot of hard work and dedication to this stuff, I will eventually understand and appreciate everything you wrote.
I will keep at it.
Thank you for your insight.
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