ESE519 Lab6 Part3
Dependencies: MPU6050_Lab6_Part3 mbed
Fork of BroBot_v2 by
Diff: main.cpp
- Revision:
- 6:ae3e6aefe908
- Parent:
- 4:2512939c10f0
--- a/main.cpp Tue Oct 18 20:46:01 2016 +0000 +++ b/main.cpp Thu Nov 10 19:20:55 2016 +0000 @@ -1,16 +1,30 @@ -//BroBot V3 +//Balance Bot V4 //Author: Carter Sharer //Date: 10/13/2016 +//ESE519 Lab 6 Part 3 Skeleton Code -//BroBot Begin +/******************************* README USAGE ******************************* +* This robot must be powered on while it is laying down flat on a still table +* This allows the robot to calibrate the IMU (~5 seconds) +* The motors are DISABLED when the robot tilts more then +-45 degrees from +* vertical. To ENABLE the motors you must lift the robot to < +- 45 degres and +* press the joystick button. +* To reset the motor positions you must press the josystick button anytime. +******************************************************************************/ + +//Balance Bot Begin #include "pin_assignments.h" #include "I2Cdev.h" #include "JJ_MPU6050_DMP_6Axis.h" -#include "BroBot.h" -#include "BroBot_IMU.h" +#include "balance_bot.h" +#include "balance_bot_IMU.h" #include "stepper_motors.h" #include "MRF24J40.h" +//Angle Offset is used to set the natural balance point of your robot. +//You should adjust this offset so that your robots balance point is near 0 +#define ANGLE_OFFSET 107 + //For RF Communication #define JSTICK_H 8 #define JSTICK_V 9 @@ -34,71 +48,36 @@ #define POTH p20 //PID -#define MAX_THROTTLE 580 -#define MAX_STEERING 150 +#define MAX_THROTTLE 100 #define MAX_TARGET_ANGLE 12 -#define KP 0.19 -#define KD 28 -#define KP_THROTTLE 0.01 //0.07 -#define KI_THROTTLE 0//0.04 -#define ITERM_MAX_ERROR 25 // Iterm windup constants for PI control //40 -#define ITERM_MAX 8000 // 5000 - -//MRF24J40 -PinName mosi(SDI); //SDI -PinName miso(SDO); //SDO -PinName sck(SCK); //SCK -PinName cs(CS); //CS -PinName reset(RESET); //RESET -// RF tranceiver to link with handheld. -MRF24J40 mrf(mosi, miso, sck, cs, reset); -uint8_t txBuffer[128]= {1, 8, 0, 0xA1, 0xB2, 0xC3, 0xD4, 0x00}; -uint8_t rxBuffer[128]; -uint8_t rxLen; +//PID Default control values from constant definitions +float Kp1; +float Kd1; +float Kp2; +float Kd2; //Controller Values uint8_t knob1, knob2, knob3, knob4; int8_t jstick_h, jstick_v; - -//PID Default control values from constant definitions -float Kp = KP; -float Kd = KD; -float Kp_thr = KP_THROTTLE; -float Ki_thr = KI_THROTTLE; -float Kd_thr; //Added for CS Pos contorl -float Kp_user = KP; -float Kd_user = KD; -float Kp_thr_user = KP_THROTTLE; -float Ki_thr_user = KI_THROTTLE; -bool newControlParameters = false; -bool modifing_control_parameters = false; -float PID_errorSum; -float PID_errorOld = 0; -float PID_errorOld2 = 0; -float setPointOld = 0; +//Control Variables float target_angle; -float throttle = 0; -float steering = 0; -float max_throttle = MAX_THROTTLE; -float max_steering = MAX_STEERING; +float throttle = 0; //From joystick +float steering = 0; //From joystick float max_target_angle = MAX_TARGET_ANGLE; -float control_output; -int16_t actual_robot_speed; // overall robot speed (measured from steppers speed) -int16_t actual_robot_speed_old; -float estimated_speed_filtered; // Estimated robot speed -int robot_pos = 0; +int robot_pos = 0; //Robots position +bool fallen = true; Timer timer; -int timer_value; //maybe make this a long -int timer_old; //maybe make this a long +int timer_value; +int timer_old; int dt; +//Loop Counters uint8_t slow_loop_counter; uint8_t medium_loop_counter; uint8_t loop_counter; - Serial pc(USBTX, USBRX); // LEDs @@ -109,48 +88,8 @@ //Button bool button; - -// ============================================================================= -// === PD controller implementation(Proportional, derivative) === -// ============================================================================= -// PD controller implementation(Proportional, derivative). DT is in miliseconds -// stabilityPDControl( dt, angle, target_angle, Kp, Kd); -float stabilityPDControl(float DT, float input, float setPoint, float Kp, float Kd) -{ - float error; - float output; - - error = setPoint - input; - - - // Kd is implemented in two parts - // The biggest one using only the input (sensor) part not the SetPoint input-input(t-2) - // And the second using the setpoint to make it a bit more agressive setPoint-setPoint(t-1) - output = Kp * error; //+ (Kd * (setPoint - setPointOld) - Kd * (input - PID_errorOld2)) / DT; +#include "communication.h" - PID_errorOld2 = PID_errorOld; - PID_errorOld = input; // error for Kd is only the input component - setPointOld = setPoint; - return output; - - -} - -// PI controller implementation (Proportional, integral). DT is in miliseconds -float speedPIControl(float DT, float input, float setPoint, float Kp, float Ki) -{ - float error; - float output; - - error = setPoint - input; - PID_errorSum += CAP(error, ITERM_MAX_ERROR); - PID_errorSum = CAP(PID_errorSum, ITERM_MAX); - - //Serial.println(PID_errorSum); - - output = Kp * error + Ki * PID_errorSum * DT * 0.001; // DT is in miliseconds... - return (output); -} // ================================================================ // === INITIAL SETUP === // ================================================================ @@ -197,27 +136,13 @@ // ================================================================ // === MAIN PROGRAM LOOP === // ================================================================ -//CS PID CONTROLLER TEST -float target_angle_old = 0; -float change_in_target_angle = 0; -float change_in_angle = 0; -float angle_old1 = 0; -float angle_old2 = 0; -float kp_term = 0; -float kd_term = 0; -float error; -//For Position controller -float pos_error = 0; -float kp_pos_term = 0; -float kd_pos_term = 0; -float change_in_target_pos; -float target_pos, target_pos_old; -float change_in_pos; -float robot_pos_old, robot_pos_old1, robot_pos_old2; - int main() { - pc.baud(230400); + //Set the Channel. 0 is default, 15 is max + uint8_t channel = 2; + mrf.SetChannel(channel); + + pc.baud(115200); pc.printf("Start\r\n"); init_imu(); timer.start(); @@ -230,11 +155,7 @@ timer_M2.attach_us(&ISR2, ZERO_SPEED); step_M1 = 1; dir_M1 = 1; - enable = ENABLE; //Enable Motors - - //Set Gains - Kp_thr = 0; //0.15; - Ki_thr = 0; //0.15; + enable = DISABLE; //Disable Motors //Attach Interupt for IMU checkpin.rise(&dmpDataReady); @@ -242,172 +163,126 @@ //Used to set angle upon startup, filter bool FILTER_DISABLE = true; + //Enable Motors + enable = ENABLE; + while(1) { + //Led 4 to indicate if robot it STANDING or FALLEN + led4 = !fallen; + //Led 2 to indicate a button press + led2 = button; + + //If button is pressed reset motor position if(button) { - pos_M1 = 0; - pos_M2 = 0; - target_pos = 0; + pos_M1 = 0; //Reset position of Motor 1 + pos_M2 = 0; //Reset position of motor 2 + fallen = false; //Reset fallen flag } - while(!mpuInterrupt) { // && fifoCount < packetSize) { - //led4 = led4^1; - //pc.printf("In while comp loop \r\n"); + //This is the main while loop, all your code goes here + while(!mpuInterrupt) { + //Timer timer_value = timer.read_us(); - //Set Gainz with knobs - Kp = ((float)knob1) / 1000.0; - Kd = ((float)knob2) / 1.0; - Kp_thr = ((float)knob3) / 1000.0; - Kd_thr = ((float)knob4) / 100.0; + //Set gainz with knobs + Kp1 = knob1; + Kd1 = knob2; + Kp2 = knob3; + Kd2 = knob4; //Joystick control - throttle = (float)jstick_v /10.0; - steering = (float)jstick_h / 10.0; + throttle = jstick_v; + steering = jstick_h; - //Update Values + /**** Update Values DO NOT MODIFY ********/ loop_counter++; slow_loop_counter++; medium_loop_counter++; dt = (timer_value - timer_old); timer_old = timer_value; angle_old = angle; - - // Motor contorl - if((angle < 45) && (angle > -45)) { + /*****************************************/ - //PID CONTROL MAGIC GOES HERE - // We calculate the estimated robot speed: - // Estimated_Speed = angular_velocity_of_stepper_motors(combined) - angular_velocity_of_robot(angle measured by IMU) - actual_robot_speed_old = actual_robot_speed; - actual_robot_speed = (speed_M1 + speed_M2) / 2; // Positive: forward - int16_t angular_velocity = (angle - angle_old) * 90.0; // 90 is an empirical extracted factor to adjust for real units - int16_t estimated_speed = -actual_robot_speed_old - angular_velocity; // We use robot_speed(t-1) or (t-2) to compensate the delay - estimated_speed_filtered = estimated_speed_filtered * 0.95 + (float)estimated_speed * 0.05; // low pass filter on estimated speed - // SPEED CONTROL: This is a PI controller. - // input:user throttle, variable: estimated robot speed, output: target robot angle to get the desired speed - //CS target_angle = speedPIControl(dt, estimated_speed_filtered, throttle, Kp_thr, Ki_thr); - //CS target_angle = CAP(target_angle, max_target_angle); // limited output - //target_angle = 0; - // Stability control: This is a PD controller. - // input: robot target angle(from SPEED CONTROL), variable: robot angle, output: Motor speed - // We integrate the output (sumatory), so the output is really the motor acceleration, not motor speed. + //STANDING: Motor Control Enabled + if(((angle < 45) && (angle > -45)) && (fallen == false)) { - //pc.printf("dt: %f, angle: %f, target_angle: %f, Kp: %f, Kd: %f \r\n", dt, angle, target_angle, Kp, Kd); - //control_output = stabilityPDControl(dt, angle, target_angle, Kp, Kd); + //Enable Motor + enable = ENABLE; - //CS Pd Target Angle Contoller Goes Here - target_pos += throttle; - robot_pos = (pos_M1 + pos_M2) / 2; - //KP Term - pos_error = robot_pos - target_pos; //robot_pos - target_pos; - kp_pos_term = -Kp_thr * pos_error; - - //KD Term - change_in_target_pos = target_pos - target_pos_old; - change_in_pos = robot_pos - robot_pos_old2; - kd_pos_term = ((-Kd_thr * change_in_target_pos) - (-Kd_thr*change_in_pos)) /dt; - target_angle = kp_pos_term + kd_pos_term; - target_angle = CAP(target_angle, MAX_TARGET_ANGLE); + /* This is where you want to impliment your controlers + * Start off with a simple P controller. + * + * float error = angle - 0; //should be balanced at 0 + * motor1 = error * Kp; + * motor2 = error * Kp; */ - //Update values - target_pos_old = target_pos; - robot_pos_old2 = robot_pos_old1; - robot_pos_old1 = robot_pos_old; + //Calculate motor inputs + motor1 = int16_t(throttle/2 + steering/8); + motor2 = int16_t(throttle/2 - steering/8); - //CS PD Stability CONTROLLER HERE - error = target_angle - angle; - kp_term = Kp * error; - - change_in_target_angle = target_angle - target_angle_old; //add - change_in_angle = angle - angle_old2; //add - kd_term = ((Kd * change_in_target_angle) - Kd*(change_in_angle)) / dt; - - //Control Output - control_output += kp_term + kd_term; - control_output = CAP(control_output, MAX_CONTROL_OUTPUT); // Limit max output from control - motor1 = (int16_t)(control_output + (steering/4)); - motor2 = (int16_t)(control_output - (steering/4)); + //Cap the max and min values [-100, 100] motor1 = CAP(motor1, MAX_CONTROL_OUTPUT); motor2 = CAP(motor2, MAX_CONTROL_OUTPUT); - - //Update variables - target_angle_old = target_angle; - angle_old2 = angle_old1; - angle_old1 = angle; + + //Set Motor Speed here + setMotor1Speed(motor1); + setMotor2Speed(motor2); - //Enable Motors - enable = ENABLE; - setMotor1Speed(-motor1); - setMotor2Speed(-motor2); - robot_pos += (-motor1 + -motor2) / 2; - //pc.printf("m1: %d m2: %d angle: %0.1f, controlout: %f tAngle: %f dt: %f timer: %d \r\n", motor1, motor2, angle, control_output, target_angle, dt, timer_value); - } else { + } else { //FALLEN: Motor Control Disabled //Disable Motors enable = DISABLE; - //Set Motor Speed 0 - PID_errorSum = 0; // Reset PID I term + + //Set fallen flag + fallen = true; } - - //Fast Loop + + /* Here are some loops that trigger at different intervals, this + * will allow you to do things at a slower rate, thus saving speed + * it is important to keep this fast so we dont miss IMU readings */ + + //Fast Loop: Good for printing to serial monitor if(loop_counter >= 5) { loop_counter = 0; - //pc.printf("angle: %d horz: %d verti: %d knob1: %d knob2: %d knob3: %d knob4: %d \r\n", int16_t(angle-ANGLE_OFFSET), jstick_h, jstick_v, knob1, knob2, knob3, knob4); - //setMotor1Speed(int16_t(angle)); - //setMotor2Speed(int16_t(angle)); - //pc.printf("horz: %d verti: %d knob1: %d angle: %d \r\n", jstick_h, jstick_v, knob1, (int)angle); - //pc.printf("angle: %d \r\n", (int)angle); - pc.printf("angle:%d Kp: %0.3f Kd: %0.2f Kp_thr: %0.2f Kd_thr: %0.3f tang: %0.2f dt:%d pos_M1:%d pos_M2:%d rob_pos: %d\r\n", (int)angle, Kp, Kd, Kp_thr, Ki_thr, target_angle, dt, pos_M1, pos_M2, robot_pos); + pc.printf("angle:%d Kp1:%0.3f Kd1:%0.2f Kp2:%0.2f Kd2:%0.3f pos_M1:%d pos_M2:%d \r\n", (int)angle, Kp1, Kd1, Kp2, Kd2, pos_M1, pos_M2, robot_pos); } - - //Meduim Loop + //Meduim Loop: Good for sending and receiving if (medium_loop_counter >= 10) { medium_loop_counter = 0; // Read status - led2 = led2^1; //Recieve Data - rxLen = mrf.Receive(rxBuffer, 128); - if(rxLen) { - if((rxBuffer[0] == (uint8_t)1) && (rxBuffer[1] == (uint8_t)8) && (rxBuffer[2]==(uint8_t)0)) { - jstick_h = (int8_t)rxBuffer[JSTICK_H] - JSTICK_OFFSET; - jstick_v = (int8_t)rxBuffer[JSTICK_V] - JSTICK_OFFSET; - knob1 = rxBuffer[KNOB1]; - knob2 = rxBuffer[KNOB2]; - knob3 = rxBuffer[KNOB3]; - knob4 = rxBuffer[KNOB4]; - button = rxBuffer[BUTTON]; - led1= led1^1; //flash led for debuggin - led4 = button; - } - } else { - mrf.Reset(); + rxLen = rf_receive(rxBuffer, 128); + if(rxLen > 0) { + led1 = led1^1; + //Process data with our protocal + communication_protocal(rxLen); } + } // End of medium loop - - //Slow Loop + + //Slow Loop: Good for sending if speed is not an issue if(slow_loop_counter >= 99) { slow_loop_counter = 0; - //Send Data - txBuffer[ANGLE] = (uint8_t)(angle + TX_ANGLE_OFFSET); - mrf.Send(txBuffer, TX_BUFFER_LEN); + /* Send Data To Controller goes here * + * */ } //End of Slow Loop //Reattach interupt checkpin.rise(&dmpDataReady); } //END WHILE + + /********************* All IMU Handling DO NOT MODIFY *****************/ //Disable IRQ checkpin.rise(NULL); - led3 = led3^1; - //pc.printf("taking care of imu stuff angle: %f \r\n", angle); - //All IMU stuff - // reset interrupt flag and get INT_STATUS byte + + //reset interrupt flag and get INT_STATUS byte mpuInterrupt = false; mpuIntStatus = mpu.getIntStatus(); - // get current FIFO count + //get current FIFO count fifoCount = mpu.getFIFOCount(); // check for overflow (this should never happen unless our code is too inefficient) @@ -416,32 +291,29 @@ mpu.resetFIFO(); pc.printf("FIFO overflow!"); - // otherwise, check for DMP data ready interrupt (this should happen frequently) + //otherwise, check for DMP data ready interrupt (this should happen frequently) } else if (mpuIntStatus & 0x02) { - // wait for correct available data length, should be a VERY short wait + //wait for correct available data length, should be a VERY short wait while (fifoCount < packetSize) fifoCount = mpu.getFIFOCount(); - // read a packet from FIFO + //read a packet from FIFO mpu.getFIFOBytes(fifoBuffer, packetSize); - // track FIFO count here in case there is > 1 packet available - // (this lets us immediately read more without waiting for an interrupt) + //track FIFO count here in case there is > 1 packet available + //(this lets us immediately read more without waiting for an interrupt) fifoCount -= packetSize; //Read new angle from IMU new_angle = (float)(dmpGetPhi() - ANGLE_OFFSET); dAngle = new_angle - angle; - //Filter out angle readings larger then MAX_ANGLE_DELTA if( ((dAngle < 15) && (dAngle > -15)) || FILTER_DISABLE) { angle = new_angle; FILTER_DISABLE = false; //turn of filter disabler - } else { - pc.printf("\t\t\t filtered angle \r\n"); - } - //END IMU STUFF + } + } + /********************* All IMU Handling DO NOT MODIFY *****************/ - } } //end main loop } //End Main() \ No newline at end of file