23:00

Dependencies:   biquadFilter MODSERIAL QEI Servo mbed

Fork of StateMachine_EMg_RKI_PID_MOTOR_DEMO_CLICK by Gaston Gabriël

Revision:
9:40c9a18c3430
Parent:
8:ec3ea0623620
Child:
10:2325e545ce11
--- a/main.cpp	Thu Nov 01 16:40:11 2018 +0000
+++ b/main.cpp	Thu Nov 01 17:09:18 2018 +0000
@@ -1,15 +1,33 @@
-// EMG, RKI, PID, MOTOR
+// EMG + KINEMATICS + PID + MOTOR CONTROL
+
+//----------------~INITIATING-------------------------
 #include "mbed.h"
+
+// EMG                 --      DEPENDENCIES
 #include <iostream>
 #include "BiQuad.h"
 #include "BiQuadchains_zelfbeun.h"
 #include "MODSERIAL.h"
 
+// KINEMATICS          --       DEPENDENCIES
+#include "stdio.h"
+#define _USE_MATH_DEFINES
+#include <math.h>
+#define M_PI    3.14159265358979323846 /* pi */
+
+// PID CONTROLLER      --      DEPENDENCIES
+#include "BiQuad.h"
+#include "QEI.h"
+//#include "HIDScope.h"
+
+
+// GENERAL PIN DEFENITIONS
 MODSERIAL pc(USBTX, USBRX);
 
+// EMG     --      PIN DEFENITIONS 
 DigitalOut gpo(D0);
 
-DigitalIn button2(SW3);
+DigitalIn button2(SW3);  
 DigitalIn button1(SW2); //or SW2
 
 DigitalOut led1(LED_GREEN);
@@ -22,18 +40,126 @@
 Timer t; //timer try out for Astrid
 Timer timer_calibration; //timer for EMG calibration
 
-
-
 //Input system
 AnalogIn emg1(A0); //right biceps
 AnalogIn emg2(A1); //right triceps
 AnalogIn emg3(A2); //left biceps
 AnalogIn emg4(A3); //left triceps
 
+
+// PID  CONTROLLER     --        PIN DEFENITIONS 
+//AnalogIn button2(A4);
+//AnalogIn button1(A3);
+//DigitalIn button3(SW2);
+//DigitalIn button4(SW3);
+
+DigitalOut directionpin1(D7);   // motor 1
+DigitalOut directionpin2(D4);   // motor 2
+DigitalOut directionpin3(D13);  // motor 3
+PwmOut pwmpin1(D6);             // motor 1
+PwmOut pwmpin2(D5);             // motor 2
+PwmOut pwmpin3(D12);            // motor 3
+
+QEI encoder1 (D9, D8, NC, 8400, QEI::X4_ENCODING);
+QEI encoder2 (D11, D10, NC, 8400, QEI::X4_ENCODING); // motor 2
+QEI encoder3 (D3, D2, NC, 8400, QEI::X4_ENCODING);  // motor 3
+// HIDScope scope(2);
+
+//  PID - TICKERS
+Ticker ref_rot;
+Ticker show_counts;
+Ticker Scope_Data;
+
+//----------------GLOBALS--------------------------
+// GLOBALS EMG
+
 //Filtered EMG signals from the end of the chains
 volatile double emg1_filtered, emg2_filtered, emg3_filtered, emg4_filtered;
 int i = 0;
 
+//Define doubles for calibration and ticker
+double ts = 0.001; //tijdsstap
+double calibration_time = 55; //time EMG calibration should take
+
+volatile double temp_highest_emg1 = 0; //highest detected value right biceps
+volatile double temp_highest_emg2 = 0;
+volatile double temp_highest_emg3 = 0;
+volatile double temp_highest_emg4 = 0;
+
+//Doubles for calculation threshold
+double biceps_p_t = 0.4; //set threshold at percentage of highest value
+double triceps_p_t = 0.5; //set threshold at percentage of highest value
+volatile double threshold1;
+volatile double threshold2;
+volatile double threshold3;
+volatile double threshold4;
+
+// thresholdreads bools
+int bicepsR;
+int tricepsR;
+int bicepsL;
+int tricepsL;
+
+// VARIABLES ROBOT KINEMATICS
+// constants
+const float la = 0.256;         // lengte actieve arm
+const float lp = 0.21;          // lengte passieve arm
+const float rp = 0.052;         // straal van midden end effector tot hoekpunt
+const float rm = 0.23;          // straal van global midden tot motor
+const float a = 0.09;           // zijde van de driehoek
+const float xas = 0.40;         // afstand van motor 1 tot motor 3
+const float yas = 0.346;        // afstand van xas tot motor 2
+const float thetap = 0;         // rotatiehoek van de end effector
+
+
+// motor locatie
+const int a1x = 0;              //x locatie motor 1
+const int a1y = 0;              //y locatie motor 1
+const float a2x = (0.5)*xas;    // x locatie motor 2
+const float a2y = yas;          // y locatie motor 2
+const float a3x = xas;          // x locatie motor 3
+const int a3y = 0;              // y locatie motor 3
+
+// script voor het bepalen van de desired position aan de hand van emg (1/0)
+
+//  EMG OUTPUT
+int EMGxplus;
+int EMGxmin ;
+int EMGyplus;
+int EMGymin ;
+
+// Dit moet experimenteel geperfectioneerd worden
+float tijdstap = 0.005;      //nu wss heel langzaam, kan miss omhoog KEER V GEEFT VERANDERING IN POSITIE
+float v = 0.1;                // snelheid kan wss ook hoger
+
+float px = 0.2;     //starting x    // BOUNDARIES
+float py = 0.155;   // starting y   // BOUNDARIES
+
+// verschil horizontale as met de actieve arm
+float da1 = 1.619685; // verschil a1 hoek en motor
+float da2 = -0.609780;
+float da3 = 3.372859;
+
+// limits (since no forward kinematics)
+float upperxlim = 0.275; //36, 0.04, 0.315, -0.085niet helemaal naar requierments ff kijken of ie groter kan
+float lowerxlim = 0.10;
+float upperylim = 0.225;
+float lowerylim = 0.03; //0.03 is goed
+
+// VARIABLES PID CONTROLLER
+double PI = M_PI;// CHANGE THIS INTO M_PI
+double Kp = 14; //200 , 50
+double Ki = 0;   //1, 0.5
+double Kd = 3; //200, 10
+double Ts = 0.1; // Sample time in seconds
+double reference_rotation; //define as radians
+double motor_position;
+bool AlwaysTrue;
+
+
+//----------------FUNCTIONS--------------------------
+
+// ~~~~~~~~~~~~~~~~~~~EMG FUNCTIONS~~~~~~~~~~~~~~~~~~
 void emgsample()
 {
     //All EMG signal through Highpass
@@ -64,28 +190,8 @@
     emg2_filtered = lowp2.step(emg2_abs);
     emg3_filtered = lowp3.step(emg3_abs);
     emg4_filtered = lowp4.step(emg4_abs);
-
-
 }
 
-
-//Define doubles for calibration and ticker
-double ts = 0.001; //tijdsstap
-double calibration_time = 55; //time EMG calibration should take
-
-volatile double temp_highest_emg1 = 0; //highest detected value right biceps
-volatile double temp_highest_emg2 = 0;
-volatile double temp_highest_emg3 = 0;
-volatile double temp_highest_emg4 = 0;
-
-//Doubles for calculation threshold
-double biceps_p_t = 0.4; //set threshold at percentage of highest value
-double triceps_p_t = 0.5; //set threshold at percentage of highest value
-volatile double threshold1;
-volatile double threshold2;
-volatile double threshold3;
-volatile double threshold4;
-
 void CalibrationEMG()
 {
     //static float samples = calibration_time/ts;
@@ -137,15 +243,13 @@
         led1=1;
         led2=1;
         led3=1;
-
-
     }
-
+/*
     pc.printf("Highest value right biceps= %f \r\n", temp_highest_emg1);
     pc.printf("Highest value right triceps= %f \r\n", temp_highest_emg2);
     pc.printf("Highest value left biceps= %f \r\n", temp_highest_emg3);
     pc.printf("Highest value left triceps= %f \r\n", temp_highest_emg4);
-
+*/
 
     threshold1 = temp_highest_emg1*biceps_p_t;  //Right Biceps
     threshold2 = temp_highest_emg2*triceps_p_t; //Right Triceps
@@ -154,11 +258,6 @@
 }
 
 //Check if emg_filtered has reached their threshold
-int bicepsR;
-int tricepsR;
-int bicepsL;
-int tricepsL;
-
 void threshold_check()
 {
 
@@ -193,11 +292,8 @@
     pc.printf("Biceps Left = %i", bicepsL);
     pc.printf("Triceps Left = %i", tricepsL);
     */
-
-
 }
 
-
 //Activate ticker for Movement state, filtering and Threshold checking
 void movement_ticker_activator()
 {
@@ -210,6 +306,178 @@
     threshold_check_ticker.detach();
 }
 
+// ~~~~~~~~~~~~~~ROBOT KINEMATICS ~~~~~~~~~~~~~~~~~~
+
+// functie x positie
+float positionx(int EMGxplus,int EMGxmin)
+{
+    float EMGx = EMGxplus - EMGxmin;
+
+    float verplaatsingx = EMGx * tijdstap * v;
+    float pxnieuw = px + verplaatsingx;
+    // x limit
+    if (pxnieuw <= upperxlim && pxnieuw >= lowerxlim) {
+        px = pxnieuw;
+    } else {
+        if (pxnieuw >= lowerxlim) {
+            px = upperxlim;
+        } else {
+            px = lowerxlim;
+        }
+    }
+//printf("X eindpunt (%f) en verplaatsing: (%f)\n\r",px,verplaatsingx);
+    return px;
+}
+
+
+// functie y positie
+float positiony(int EMGyplus,int EMGymin)
+{
+    float EMGy = EMGyplus - EMGymin;
+
+    float verplaatsingy = EMGy * tijdstap * v;
+    float pynieuw = py + verplaatsingy;
+
+    // y limit
+    if (pynieuw <= upperylim && pynieuw >= lowerylim) {
+        py = pynieuw;
+    } else {
+        if (pynieuw >= lowerylim) {
+            py = upperylim;
+        } else {
+            py = lowerylim;
+        }
+    }
+//printf("Y eindpunt (%f) en verplaatsing: (%f) \n\r",py,verplaatsingy);
+    return (py);
+}
+
+
+//~~~~~~~~~~~~CALCULATIING MOTOR ANGLES ~~~~~~~~
+// arm 1 --> reference angle motor 1
+float hoek1(float px, float py) // input: ref x, ref y
+{
+    float c1x =  px - rp * cos(thetap +(M_PI/6));       // x locatie hoekpunt end-effector
+    float c1y = py - rp*sin(thetap+(M_PI/6));           // y locatie hoekpunt end-effector
+    float alpha1 = atan2((c1y-a1y),(c1x-a1x));          // hoek tussen horizontaal en lijn van motor naar bijbehorende end-effector punt
+    float psi1 = acos(( pow(la,2)-pow(lp,2)+pow((c1x-a1x),2)+pow((c1y-a1y),2))/(2*la*sqrt(pow ((c1x-a1x),2)+pow((c1y-a1y),2) ))); //Hoek tussen lijn van motor naar bijbehorende end=effector punt en actieve arm
+    float a1 = alpha1 + psi1 - da1;                          //hoek tussen horizontaal en actieve arm
+    //printf("arm 1 = %f \n\r",a1);
+    return a1;
+}
+
+// arm 2 --> reference angle motor 2
+float hoek2(float px, float py)
+{
+    float c2x =  px - rp * cos(thetap -(M_PI/2));
+    float c2y = py - rp*sin(thetap-(M_PI/2));
+    float alpha2 = atan2((c2y-a2y),(c2x-a2x));
+    float psi2 = acos(( pow(la,2)-pow(lp,2)+pow((c2x-a2x),2)+pow((c2y-a2y),2))/(2*la*sqrt(pow ((c2x-a2x),2)+pow((c2y-a2y),2) )));
+    float a2 = alpha2 + psi2 - da2;
+    //printf("arm 2 = %f \n\r",a2);
+    return a2;
+}
+
+//arm 3 --> reference angle motor 3
+float hoek3(float px, float py)
+{
+    float c3x =  px - rp * cos(thetap +(5*M_PI/6));
+    float c3y = py - rp*sin(thetap+(5*M_PI/6));
+    float alpha3 = atan2((c3y-a3y),(c3x-a3x));
+    float psi3 = acos(( pow(la,2)-pow(lp,2)+pow((c3x-a3x),2)+pow((c3y-a3y),2))/(2*la*sqrt(pow ((c3x-a3x),2)+pow((c3y-a3y),2) )));
+    float a3 = alpha3 + psi3 - da3;
+    //printf("arm 3 = %f \n\r",a3);
+    return a3;
+}
+
+// ~~~~~~~~~~~~~~PID CONTROLLER~~~~~~~~~~~~~~~~~~
+
+double PID_controller(double error)
+{
+    static double error_integral = 0;
+    static double error_prev = error; // initialization with this value only done once!
+    static BiQuad LowPassFilter(0.0640, 0.1279, 0.0640, -1.1683, 0.4241);
+
+    // Proportional part:
+    double u_k = Kp * error;
+
+    // Integral part
+    error_integral = error_integral + error * Ts;
+    double u_i = Ki * error_integral;
+
+    // Derivative part
+    double error_derivative = (error - error_prev)/Ts;
+    double filtered_error_derivative = LowPassFilter.step(error_derivative);
+    double u_d = Kd * filtered_error_derivative;
+    error_prev = error;
+
+    // Sum all parts and return it
+    return u_k + u_i + u_d;
+}
+
+
+// DIRECTON AND SPEED CONTROL
+void moter_control(double u)
+{
+
+    directionpin1= u > 0.0f; //eithertrueor false
+    if (fabs(u)> 0.7f) {
+        u = 0.7f;
+    } else {
+        u= u;
+    }
+    pwmpin1= fabs(u); //pwmduty cycle canonlybepositive, floatingpoint absolute value
+}
+
+void moter2_control(double u)
+{
+    directionpin2= u > 0.0f; //eithertrueor false
+    if (fabs(u)> 0.7f) {
+        u = 0.7f;
+    } else {
+        u= u;
+    }
+    pwmpin2= fabs(u); //pwmduty cycle canonlybepositive, floatingpoint absolute value
+}
+
+void moter3_control(double u)
+{
+    directionpin3= u > 0.0f; //eithertrueor false
+    if (fabs(u)> 0.7f) {
+        u = 0.7f;
+    } else {
+        u= u;
+    }
+    pwmpin3 = fabs(u); //pwmduty cycle canonlybepositive, floatingpoint absolute value
+}
+
+// CONTROLLING THE MOTOR
+void Motor_mover()
+{
+    double motor_position = encoder1.getPulses(); //output in counts
+    double reference_rotation = hoek1(px, py);
+    double error = reference_rotation - motor_position*(2*PI)/8400;
+    double u = PID_controller(error);
+    moter_control(u);
+
+    double motor_position2 = encoder2.getPulses(); //output in counts
+    double reference_rotation2 = hoek2(px, py);
+    double error_2 = reference_rotation2 - motor_position2*(2*PI)/8400;
+    double u_2 = PID_controller(error_2);
+    moter2_control(u_2);
+
+    double motor_position3 = encoder3.getPulses(); //output in counts
+    double reference_rotation3 = hoek3(px, py);
+    double error_3 = reference_rotation3 - motor_position3*(2*PI)/8400;
+    double u_3 = PID_controller(error_3);
+    moter3_control(u_3);
+
+
+}
+
+
+
+//-------------------- STATE MACHINE --------------------------
 enum states {MOTORS_OFF,CALIBRATION,HOMING,DEMO,MOVEMENT,CLICK};
 states currentState = MOTORS_OFF; //Chosen startingposition for states
 bool stateChanged = true; // Make sure the initialization of first state is executed
@@ -430,6 +698,9 @@
     }
 }
 
+// --------------------------MAIN--------------------
+
+
 int main()
 {
     //BiQuad Chain add
@@ -453,6 +724,9 @@
     led1 = 1;
     led2 = 1;
     led3 = 1;
+    
+    pwmpin1.period_us(60); // setup motor
+        ref_rot.attach(Motor_mover, 0.001);// HAS TO GO TO STATE MACHINE
 
     while (true) {
         ProcessStateMachine();