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

Help me take a look on my code what error in it

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



Joined: 17 Jun 2011
Posts: 28
Location: Malaysia

View user's profile Send private message Send e-mail

Help me take a look on my code what error in it
PostPosted: Fri Jul 15, 2011 3:15 am     Reply with quote

Code:
#include<18F4520.h>
#DEVICE ADC=8
#fuses HS,NOWDT,NOPROTECT,NOLVP, BROWNOUT, PUT
#use delay(clock=4000000)
#include "C:\Program Files (x86)\PICC\Drivers\flex_lcd.c"

#define data_width 3

int8 AccData[data_width];

void main(void)
{    
   int i;
   //long value;

   output_high(LCD_POWER);  // Turn on power to LCD
   output_low(LCD_RW);
   lcd_init();
   
   setup_adc(ADC_CLOCK_INTERNAL);//enables the a/d module
   //and sets the clock to internal adc clock
   setup_adc_ports(AN0_TO_AN2); //set the adc pins to analog

   output_low(PIN_C1);   // Set CCP2 output low
   output_low(PIN_C2);   // Set CCP1 output low

   setup_ccp1(CCP_PWM);  // Configure CCP1 as a PWM
   setup_ccp2(CCP_PWM);  // Configure CCP2 as a PWM
   enable_interrupts(int_timer2);
   enable_interrupts(GLOBAL);
   setup_timer_2(T2_DIV_BY_16, 250, 5); 

   while(1)
   {
   disable_interrupts(int_timer2);
   disable_interrupts(GLOBAL);
      for(i = 0; i < data_width; i++)
      {
         delay_us(10);          //a small delay is required after setting the channel
           set_adc_channel(i);
         AccData[i] = read_adc(); //starts the conversion   
      }

      lcd_gotoxy(1,1);
      printf(lcd_putc, "x = %u  y = %u  ", AccData[0], AccData[1]);
      lcd_gotoxy(1,2);
      printf(lcd_putc, "z = %u ", AccData[2]);
      delay_ms(10);
   enable_interrupts(int_timer2);
   enable_interrupts(GLOBAL);
   
      output_low(PIN_D0);
      output_low(PIN_D1);
  //    set_pwm1_duty(125);                   // 50% duty cycle on pin C2
   //   set_pwm2_duty(63);                    // 25% duty cycle on pin C1
   }
}

#int_timer2
void ISR_Timer2()
{
   output_high(PIN_D0);   
   delay_us(2000);   
   output_high(PIN_D1);   
   delay_us(500);         
}

When I program it into the PIC board, the LCD show the x,y,z of the accelerometer. When I connect the SERVO MOTOR to pin D1/D0, the LCD will show black screen, all value gone.
SherpaDoug



Joined: 07 Sep 2003
Posts: 1640
Location: Cape Cod Mass USA

View user's profile Send private message

PostPosted: Fri Jul 15, 2011 6:38 am     Reply with quote

Tell us more about this servo motor. What type is it? Can you give us a link to a datasheet? How does it connect to the PIC?

Also why do you have long delays inside your timer interrupt? This is usually a bad idea. Interrupts want to do something quick and get out as fast as possible.
_________________
The search for better is endless. Instead simply find very good and get the job done.
pacman91



Joined: 17 Jun 2011
Posts: 28
Location: Malaysia

View user's profile Send private message Send e-mail

PostPosted: Fri Jul 15, 2011 7:00 am     Reply with quote

This is C36R servo motor, A7260 accelerometer.
http://www.cytron.com.my/usr_attachment/RC_Servo_User_Manual.pdf
I connect the brown wire to ground, red to 5v, orange to PWM signal (I connect to D0 or D1).
pacman91



Joined: 17 Jun 2011
Posts: 28
Location: Malaysia

View user's profile Send private message Send e-mail

PostPosted: Fri Jul 15, 2011 7:04 am     Reply with quote

Why you said that is a bad idea? Please teach me, i'm newbie, learning the CCS in this forum, no people teach =.=
Code:
setup_timer_2(T2_DIV_BY_16, 250, 5); //4ms, postscale 5 = 5*4ms=20ms(see in the occilloscope)


Code:
#int_timer2
void ISR_Timer2()
{
   output_high(PIN_D0);   
   delay_us(2000);   
   output_high(PIN_D1);   
   delay_us(500);         
}


output is 20ms period, D0 is 2ms high, 18ms low, D1 is 0.5ms high 19.5ms low
Ttelmah



Joined: 11 Mar 2010
Posts: 19589

View user's profile Send private message

PostPosted: Fri Jul 15, 2011 9:52 am     Reply with quote

Think of it by analogy.
Interrupts are designed to give rapid responses to something happening (like perhaps you returning a tennis ball, when a serving machine fires it at you).
Imagine what would happen, if when you received the first ball, you stopped and had a cup of tea (equivalent to delays in the interrupt). The subsequent balls would always be missed, and you'd probably end up spilling the tea as well.....

Key with interrupts, is always to do the _minimum_ to react to the event, and get out fast. If you want a long delay (cup of tea), in the court, you'd probably turn off the serving machine. The same is the way to approach interrupts. doing the slow things in the main code outside.

As a separate comment, look at your own remarks. Note the one that says
"//a small delay is required after setting the channel ". Then why is the delay _before_ setting the channel?.

Best Wishes
pacman91



Joined: 17 Jun 2011
Posts: 28
Location: Malaysia

View user's profile Send private message Send e-mail

PostPosted: Fri Jul 15, 2011 12:15 pm     Reply with quote

Quote:

As a separate comment, look at your own remarks. Note the one that says
"//a small delay is required after setting the channel ". Then why is the delay _before_ setting the channel?.

Code:

for(i = 0; i < data_width; i++)
   {
    delay_us(10);    //a small delay is required after setting the channel
    set_adc_channel(i);
    AccData[i] = read_adc(); //starts the conversion   
   }

hmm? if i put the delay below the
set_adc_channel(i);
then the problem set?
Code:
for(i = 0; i < data_width; i++)
   {
 
    set_adc_channel(i);
   delay_us(10);   
    AccData[i] = read_adc();   
   }
Ttelmah



Joined: 11 Mar 2010
Posts: 19589

View user's profile Send private message

PostPosted: Fri Jul 15, 2011 2:35 pm     Reply with quote

_One_ problem that will prevent the ADC returning a correct reading is fixed. Not the main ones.

Best Wishes
pacman91



Joined: 17 Jun 2011
Posts: 28
Location: Malaysia

View user's profile Send private message Send e-mail

PostPosted: Fri Jul 15, 2011 8:35 pm     Reply with quote

Ttelmah wrote:
_One_ problem that will prevent the ADC returning a correct reading is fixed. Not the main ones.

Best Wishes


then what is the main 1=.=
pacman91



Joined: 17 Jun 2011
Posts: 28
Location: Malaysia

View user's profile Send private message Send e-mail

PostPosted: Sat Jul 16, 2011 7:00 am     Reply with quote

o...the main problem is serve the timer interrupt used too long time? then have any idea i can set the 20ms for motor?
Ttelmah



Joined: 11 Mar 2010
Posts: 19589

View user's profile Send private message

PostPosted: Sat Jul 16, 2011 8:15 am     Reply with quote

There are two or more separate issues:
First, the code will not cause the display to blank when the servo is attached. That this is happening, implies something in hardware - faulty servo, wrong connection, etc. etc.. Triple check the actual servo wiring - colour codes mean nothing. Normally the pin 'order' across the plug, is 0v, 5v, PWM. The manual you refer to, gives a different colour code to the one you are referring to (black red, white, not black red orange), so if the servo is not exactly the model listed check what pin order the manufacturer uses. Some have the supply and PWM signal reversed (Fujitsu typically). You have a hardware problem first.
Second, are the code issues. Use the PWM, or CTC, to generate the pulses from the main code. Don't actually 'worry' about the 20mSec. This is not important to the servo. It is the on pulse that matters, within reasonable limits (perhaps 10 to 50mSec!), servo's don't care about the duration of the off period.

Best Wishes
pacman91



Joined: 17 Jun 2011
Posts: 28
Location: Malaysia

View user's profile Send private message Send e-mail

PostPosted: Sat Jul 16, 2011 10:09 am     Reply with quote

Quote:
Triple check the actual servo wiring - colour codes mean nothing. Normally the pin 'order' across the plug, is 0v, 5v, PWM. The manual you refer to, gives a different colour code to the one you are referring to (black red, white, not black red orange), so if the servo is not exactly the model listed check what pin order the manufacturer uses. Some have the supply and PWM signal reversed (Fujitsu typically). You have a hardware problem first.


I connect the wire with its RC motor manual(weblink at top(C36R)), as below,
Wiring (Black/Brown Wire) Ground
Wiring (Red Wire) 4.8-6.0
Wiring (Orange/Other Wire) PWM signal

I was tried the accelerometer and servo motor 1 by 1 with different program (similiar as this program, I just combine those 2 code to this 1), there have no any error.

But i curious my servo motor, when my period is 2.35ms (PWM at max angle) turning anticlockwise, and min angle at 0.5ms, it turn clockwise, totally reverse with my manual, it that any error or just normal?
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