Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependencies: mbed QEI HIDScope biquadFilter MODSERIAL FastPWM
main.cpp
- Committer:
- paulstuiver
- Date:
- 2019-10-31
- Revision:
- 24:e87e4fcf6226
- Parent:
- 23:ff73ee119244
- Child:
- 25:45c131af2dba
File content as of revision 24:e87e4fcf6226:
// Operating mode might not go to next state when SW2 is pressed #include "mbed.h" #include "HIDScope.h" #include "BiQuad.h" #include "MODSERIAL.h" #include "FastPWM.h" #include "QEI.h" #include <cmath> // Included to use different math operations #include "Servo.h" // Included to control the servo motor Servo myservo(D13); // To control the servo motor DigitalIn but3(SW2); // To go to the next state or to choose one of two game modes when in state START_GAME DigitalIn but4(SW3); // To choose one of two game modes when in state START_GAME or to move the gripper AnalogIn S0(A0); AnalogIn S1(A1); AnalogIn S2(A2); AnalogIn S3(A3); DigitalOut motor1Direction(D7); FastPWM motor1Velocity(D6); DigitalOut motor2Direction(D4); FastPWM motor2Velocity(D5); // Encoders 1 and 2 respectively QEI Encoder1(D8,D9,NC,8400); QEI Encoder2(D11,D10,NC,8400); Ticker measurecontrol; // Ticker function for the measurements // Make arrays for the different variables for the motors //AnalogIn Shields[4] = {S0, S1, S2, S3}; //DigitalOut MotorDirections[2] = {MD0, MD1}; //FastPWM MotorVelocities[2] = {MV0, MV1}; //QEI Encoders[2] = {E0, E1}; Serial pc(USBTX, USBRX); double PI = 3.14159265358;//97932384626433832795028841971693993; volatile double timeinterval = 0.001f; // Time interval of the Ticker function volatile double frequency = 17000.0; // Set motorfrequency double period_signal = 1.0/frequency; // Convert to period of the signal double yendeffector = 10.6159; double xendeffector = 0; int ingedrukt = 0; int state_int; int previous_state_int; // Define the different states in which the robot can be enum States {MOTORS_OFF, EMG_CALIBRATION, START_GAME, DEMO_MODE, OPERATING_MODE, END_GAME }; // Default state is the state in which the motors are turned off States MyState = MOTORS_OFF; // Initialise the functions void motorsoff(); void measure_data(double &f_y0, double &f_y1, double &f_y2, double &f_y3); void det_max(double y, double &max_y); void emgcalibration(); void nothing() { // Do nothing } void startgame() ; void demo_mode(); void operating_mode(); void New_State(); void Set_State(); double PID_controller1(double error1); double PID_controller2(double error2); void getmeasuredposition(double & measuredposition1, double & measuredposition2); void getreferenceposition(double & referenceposition1, double & referenceposition2); void sendtomotor(double motorvalue1, double motorvalue2); void measureandcontrol(); int main() { pc.baud(115200); pc.printf("Starting...\r\n\r\n"); motor1Velocity.period(period_signal); // Set the period of the PWMfunction motor2Velocity.period(period_signal); // Set the period of the PWMfunction previous_state_int = (int)MyState; // Save previous state to compare with current state and possibly execute New_State() // in the while loop New_State(); // Execute the functions belonging to the current state while(true) { if ( (previous_state_int - (int)MyState) != 0 ) { // If current state is not previous state execute New_State() New_State(); } } } double PID_controller1(double error1) { // Define errors for motor 1 and 2 static double error_integral1 = 0; static double error_prev1 = error1; // Low-pass filter static BiQuad LowPassFilter(0.0640, 0.1279, 0.0640, -1.1683, 0.4241); // PID variables: we assume them to be the same for both motors double Kp = 63; double Ki = 3.64; double Kd = 5; //Proportional part: double u_k1 = Kp * error1; // Integreal part error_integral1 = error_integral1 + error1 * timeinterval; double u_i1 = Ki*error_integral1; // Derivate part double error_derivative1 = (error1 - error_prev1)/timeinterval; double filtered_error_derivative1 = LowPassFilter.step(error_derivative1); double u_d1 = Kd * filtered_error_derivative1; error_prev1 = error1; //sum all parts and return it return u_k1 + u_i1 + u_d1; } double PID_controller2(double error2) { // Define errors for motor 1 and 2 static double error_integral2 = 0; static double error_prev2 = error2; // Low-pass filter static BiQuad LowPassFilter(0.0640, 0.1279, 0.0640, -1.1683, 0.4241); // PID variables: we assume them to be the same for both motors double Kp = 63; double Ki = 3.64; double Kd = 5; //Proportional part: double u_k2 = Kp * error2; // Integreal part error_integral2 = error_integral2 + error2 * timeinterval; double u_i2 = Ki*error_integral2; // Derivate part double error_derivative2 = (error2 - error_prev2)/timeinterval; double filtered_error_derivative2 = LowPassFilter.step(error_derivative2); double u_d2 = Kd * filtered_error_derivative2; error_prev2 = error2; //sum all parts and return it return u_k2 + u_i2 + u_d2; } //get the measured position void getmeasuredposition(double & measuredposition1, double & measuredposition2) { // Obtain the counts of motors 1 and 2 from the encoder int countsmotor1; int countsmotor2; countsmotor1 = Encoder1.getPulses(); countsmotor2 = Encoder2.getPulses(); // Obtain the measured position for motor 1 and 2 measuredposition1 = ((double)countsmotor1) / 8400.0f * 2.0f; measuredposition2 = ((double)countsmotor2) / 8400.0f * 2.0f; } //get the reference of the void getreferenceposition(double & referenceposition1, double & referenceposition2) { //Measurements of the arm double L0=1.95; double L1=15; double L2=20; //Inverse kinematics: given the end position, what are the desired motor angles of 1 and 2 double desiredmotorangle1, desiredmotorangle2; desiredmotorangle1 = (atan2(yendeffector,(L0+xendeffector))*180.0/PI + acos((pow(L1,2)+pow(L0+xendeffector,2)+pow(yendeffector,2)-pow(L2,2))/(2*L1*sqrt(pow(L0+xendeffector,2)+pow(yendeffector,2))))*180.0/PI)-180.0; desiredmotorangle2 = (atan2(yendeffector,(L0-xendeffector))*180.0/PI + acos((pow(L1,2)+pow(L0-xendeffector,2)+pow(yendeffector,2)-pow(L2,2))/(2*L1*sqrt(pow(L0-xendeffector,2)+pow(yendeffector,2))))*180.0/PI)-180.0; //Convert motor angles to counts double desiredmotorrounds1; double desiredmotorrounds2; desiredmotorrounds1 = (desiredmotorangle1)/360.0; desiredmotorrounds2 = (desiredmotorangle2)/360.0; //Assign this to new variables because otherwise it doesn't work referenceposition1 = desiredmotorrounds1; referenceposition2 = desiredmotorrounds2; } //send value to motor // IT WAS "void sendtomotor(float & motorvalue1, float & motorvalue2)" BUT I REMOVED THE REFERENCE, BECAUSE I THOUGHT IT WAS NOT NEEDED void sendtomotor(double motorvalue1, double motorvalue2) { // Define the absolute motor values double absolutemotorvalue1; double absolutemotorvalue2; absolutemotorvalue1 = fabs(motorvalue1); absolutemotorvalue2 = fabs(motorvalue2); // If absolutemotorvalueocity is greater than 1, reduce to 1, otherwise remain absolutemotorvalue absolutemotorvalue1 = absolutemotorvalue1 > 1.0f ? 1.0f : absolutemotorvalue1; absolutemotorvalue2 = absolutemotorvalue2 > 1.0f ? 1.0f : absolutemotorvalue2; // Send the absolutemotorvalue to the motors motor1Velocity = absolutemotorvalue1; motor2Velocity = absolutemotorvalue2; // Determine the motor direction. Boolean output: true gives counterclockwise direction, false gives clockwise direction motor1Direction = (motorvalue1 > 0.0f); motor2Direction = (motorvalue2 > 0.0f); } // function to call reference absolutemotorvalueocity, measured absolutemotorvalueocity and controls motor with feedback void measureandcontrol() { // Get the reference positions of motor 1 and 2 double reference1, reference2; getreferenceposition(reference1, reference2); // Get the measured positions of motor 1 and 2 double measured1, measured2; getmeasuredposition(measured1, measured2); // Calculate the motor values double motorvalue1, motorvalue2; motorvalue1 = PID_controller1(reference1 - measured1); motorvalue2 = PID_controller2(reference2 - measured2); sendtomotor(motorvalue1, motorvalue2); } void motorsoff() { // Function to turn the motors off. First state that the robot has. Robot will stay in this state untill button SW2 is pressed. // Robot will not return to this state anymore unless the user sets it back to this state with the keyboard input. sendtomotor(0.0,0.0); // Set motor velocity to 0 state_int = 10; bool whileloop_boolean = true; // Boolean for the while loop while (whileloop_boolean) { if (but3.read() == 0) { // If button SW2 is pressed: MyState = (States)((int)MyState+1); // set MyState to EMG_CALIBRATION and exit the while loop whileloop_boolean = false; // by making whileloop_boolean equal to false wait(0.5f); } } } void measure_data(double &f_y0, double &f_y1, double &f_y2, double &f_y3) { // High pass double hb0 = 0.9169; // Coefficients from the following formula: double hb1 = -1.8338; // double hb2 = 0.9169; // b0 + b1 z^-1 + b2 z^-2 double ha0 = 1.0; // H(z) = ---------------------- double ha1 = -1.8268; // a0 + a1 z^-1 + a2 z^-2 double ha2 = 0.8407; // // Low pass double lb0 = 0.000083621; // Coefficients from the following formula: double lb1 = 0.0006724; // double lb2 = 0.000083621; // b0 + b1 z^-1 + b2 z^-2 double la0 = 1.0; // H(z) = ---------------------- double la1 = -1.9740; // a0 + a1 z^-1 + a2 z^-2 double la2 = 0.9743; // static double max_y0 = 0.001; static double max_y1 = 0.001; static double max_y2 = 0.001; static double max_y3 = 0.001; static BiQuad hFilter0(hb0,hb1,hb2,ha0,ha1,ha2); // Create 4 equal filters used for the different EMG signals static BiQuad hFilter1(hb0,hb1,hb2,ha0,ha1,ha2); static BiQuad hFilter2(hb0,hb1,hb2,ha0,ha1,ha2); static BiQuad hFilter3(hb0,hb1,hb2,ha0,ha1,ha2); static BiQuad lFilter0(lb0,lb1,lb2,la0,la1,la2); // Create 4 equal filters used for the different EMG signals static BiQuad lFilter1(lb0,lb1,lb2,la0,la1,la2); static BiQuad lFilter2(lb0,lb1,lb2,la0,la1,la2); static BiQuad lFilter3(lb0,lb1,lb2,la0,la1,la2); f_y0 = hFilter0.step(S0); // Apply filters on the different EMG signals f_y1 = hFilter1.step(S1); f_y2 = hFilter2.step(S2); f_y3 = hFilter3.step(S3); f_y0 = abs(f_y0); f_y1 = abs(f_y1); f_y2 = abs(f_y2); f_y3 = abs(f_y3); f_y0 = lFilter0.step(f_y0); f_y1 = lFilter1.step(f_y1); f_y2 = lFilter2.step(f_y2); f_y3 = lFilter3.step(f_y3); if (MyState == EMG_CALIBRATION) { det_max(f_y0, max_y0); // Determine the maximum RMS value of the EMG signals during the EMG_CALIBRATION state det_max(f_y1, max_y1); det_max(f_y2, max_y2); det_max(f_y3, max_y3); } else if ((int)MyState > 3) { f_y0 = f_y0/max_y0; // Normalise the RMS value by dividing by the maximum RMS value f_y1 = f_y1/max_y1; // This is done during the states with a value higher than 3, as this is when you start the operating mode f_y2 = f_y2/max_y2; f_y3 = f_y3/max_y3; } } void det_max(double y, double &max_y) { max_y = max_y < y ? y : max_y; // if max_rms is smaller than rms, set rms as new max_rms, otherwise keep max_rms } void emgcalibration() { double y0, y1, y2, y3; // RMS values of the different EMG signals measure_data(y0, y1, y2, y3); // Calculate RMS values double duration = 15.0; // Duration of the emgcalibration function, in this case 10 seconds int rounds = (duration / timeinterval); // Determine the amount of times this function has to run to run for the duration time // rounds is an integer so the value of duration / timeinterval is floored static int counter = 0; // Counter which keeps track of the amount of times the function has executed if (counter >= rounds) { MyState = START_GAME; // If counter is larger than rounds, change MyState to the next state measurecontrol.detach(); } else { counter++; // Else increase counter by 1 } } void startgame() { pc.printf("Please choose which game you would like to start:\r\n"); pc.printf("- Press button SW2 to start the demo mode\r\n Demo mode is a mode in which the different movements of the robot are shown.\r\n"); pc.printf(" It will move in straight lines to show that the robot meets the requirements.\r\n"); pc.printf(" It will also show how it is able to grab and lift objects which are on the board.\r\n\r\n"); pc.printf("- Press button SW3 to start the operating mode\r\n The operating mode is a mode in which you can control the robot by flexing the muscles to which the electrodes are attached.\r\n"); pc.printf(" You will be able to move the arm and use the gripper to try and grab and lift objects from the board to the container.\r\n\r\n"); while (MyState == START_GAME) { if (but3.read() == 0) { MyState = (States)((int)MyState+1); wait(0.5f); } else if (but4.read() == 0) { MyState = (States)((int)MyState+2); wait(0.5f); } } } void demo_mode() { state_int = 10; // 5 pre determined positions of the end effector to show it can move in straight lines xendeffector=-5; yendeffector=10.6159; wait(0.5); xendeffector=10; yendeffector=25.6159; wait(0.5); xendeffector=-14; yendeffector=21.6159; wait(0.5); xendeffector=10; yendeffector=11.6159; wait(0.5); // Last position is the start position xendeffector=0; yendeffector=10.6159; wait(0.5); MyState = START_GAME; // Go back to START_GAME mode } void operating_mode() { double y0,y1,y2,y3; measure_data(y0,y1,y2,y3); y0 = 0; y1 = 0; double threshold = 0.3; // When the analysed signal is above this threshold, the position of the end effector is changed double dr = 0.01; // The change in position with each step if(y0 > threshold && xendeffector < 16) { xendeffector=xendeffector+dr; } else if(y1 > threshold && xendeffector > -16) { xendeffector=xendeffector-dr; } if(y2 > threshold && yendeffector < 28) { yendeffector=yendeffector+dr; } else if(y3 > threshold && yendeffector > 8) { yendeffector=yendeffector-dr; } //servo besturing if(but4.read() == 0 && ingedrukt == 0) { for(int i=0; i<100; i++) { myservo = i/100.0; } ingedrukt = 1; } else if(but4.read() == 0 && ingedrukt == 1) { for(int i=100; i>0; i--) { myservo = i/100.0; } ingedrukt = 0; } if (but3.read() == 0) { pc.printf("The game has ended, will move the end effector to (0,0), put the motors off and will now return to the state START_GAME"); MyState = END_GAME; xendeffector=0.0; yendeffector=10.6159; } measureandcontrol(); } void endgame() { wait(1); measurecontrol.detach(); MyState = START_GAME; } void New_State() { previous_state_int = (int)MyState; // Change previous state to current state switch (MyState) { case MOTORS_OFF : pc.printf("\r\nState: Motors turned off\r\n"); motorsoff(); break; case EMG_CALIBRATION : pc.printf("\r\nState: EMG Calibration\r\n"); pc.printf("Emg calibration mode will run for 20 seconds. Try to flex the muscles to which the electrodes are attached as hard as possible during this time.\r\n"); measurecontrol.attach(emgcalibration,timeinterval); break; case START_GAME : pc.printf("\r\nState: Start game\r\n"); startgame(); break; case DEMO_MODE : pc.printf("\r\nState: Demo mode\r\n"); measurecontrol.attach(measureandcontrol,timeinterval); demo_mode(); measurecontrol.detach(); break; case OPERATING_MODE : pc.printf("\r\nState: Operating mode\r\n"); measurecontrol.attach(operating_mode,timeinterval); break; case END_GAME : pc.printf("\r\nState: End of the game\r\n"); endgame(); break; default : pc.printf("\r\nDefault state: Motors are turned off\r\n"); sendtomotor(0.0,0.0); break; } } void Set_State() { xendeffector=0.0; yendeffector=10.6159; wait(0.3f); sendtomotor(0.0,0.0); // Stop the motors // Stop the ticker function from running pc.printf("\r\nPress number: | To go to state:"); pc.printf("\r\n (0) | MOTORS_OFF: Set motorspeed just in case to 0 and wait till button SW2 is pressed"); pc.printf("\r\n (1) | EMG_CALIBRATION: Calibrate the maximum of the emg signals and normalise the emg signals with these maxima"); pc.printf("\r\n (2) | START_GAME: Determine by keyboard input if you want to go to the demo or operating mode"); pc.printf("\r\n (3) | DEMO_MODE: The demo mode will show the different motions that the robot can make"); pc.printf("\r\n (4) | OPERATING_MODE: Move the arms and gripper of the arm by flexing your muscles"); pc.printf("\r\n (5) | END_GAME: End effector returns to (0,0) and the motors are turned off, returns to START_GAME mode afterwards"); wait(0.5f); char a = '0'; char b = '5'; bool boolean = true; while (boolean) { char c = pc.getc(); if (c >= a && c <= b) { MyState = (States)(c-'0'); boolean = false; } else { pc.printf("\r\nPlease enter a number between 0 and 5\r\n"); } } }