Cube Mini Solution

Dependencies:   mbed QEI MPU6050 BLE_API nRF51822 MCP4725 eMPL_MPU6050

Revision:
16:ff375f62a95f
Parent:
15:3395ad948f44
Child:
17:04eebb7c29b5
--- a/main.cpp	Fri May 29 11:34:21 2020 +0000
+++ b/main.cpp	Fri May 29 12:16:32 2020 +0000
@@ -12,7 +12,7 @@
 /*
     Settings for Maxon ESCON controller (upload via ESCON Studio)
     Escon Studio file Location: ...
-    
+
     Hardware Connections
         Serial PC Communication
             UART_TX     p9
@@ -27,13 +27,18 @@
         Current output to Escon occurs through I2C connection to a DAC (Digital to Analogue Converter). DAC outputs Analogue voltage to Escon
             Pin SDA p3
             Pin SCL P4
-            
+
+    Escon Mapping of voltage to current and rotational velocity to voltage
+        Escon Linear Mapping of input current from voltage is 0V is -15A and 5V is 15A, hence 2.5V is 0A !!!
+        Escon Linear Mapping of output rotational velocity (in rpm) to voltage is -4000rpm is 0V and 4000rpm is 3.0V, hence 1.5V is 0 RPM !!!
+
     Notes
         The Maximum output Current allowed is 13A and the Minimum is -13A !!!
+        Escon Linear Mapping of input current from voltage is 0V is -15A and 5V is 15A, hence 2.5V is 0A !!!
         All needed Libraries for the IMU (Accelerometer and Gyroscope), DAC are included.
         The PID is to be written by students. Use PID_Cntrl.cpp and PID_Cntrl.h as templates
         ...
-            
+
 
 */
 // Libraries
@@ -73,7 +78,7 @@
 // -------------------------------
 
 // Sample time of main loops
-float Ts = 0.007; // Around 143 Hz. Do not change. THis is the lowest value possible if printing to a serial connection is needed.
+float Ts = 0.007; // Around 143 Hz. Do not change. This is the lowest value possible if printing to a serial connection is needed.
 
 // MPU 6050 Variables - Acceleration and Gyroscope Raw and Converted Data Variables
 int16_t AccX_Raw, AccY_Raw, AccZ_Raw;
@@ -89,35 +94,59 @@
 //  Variables: Here define variables such as gains etc.
 // -------------------------------
 
-
 // Accelerometer and Gyroscope
 
 
 // Cube Angle and Speed Variables
-
+double Cuboid_Angle_Radians, Cuboid_Angle_Degrees;
+double Cuboid_Angle_Speed_Degrees = 0.0;
 
 // Low pass filter variables
-
+float t = 0.5f;
 
 // Flywheel Position and Velocity variables
+double Velocity_Input_Voltage = 0.0f;
+double Velocity, Velocity_Voltage, Velocity_rpm, Velocity_Voltage_Read;
 
+// User Button
+int Button_Status = 0;                  // User Button Status
+Timer T_Button;                         // define timer for button
 
 // -------------------------------
 //  Controller Variables: Such as SS Controller Values and Saturation Limits
 // -------------------------------
-// Variables concerning the Controller Design is in reference to the Matlab and Simulink Files .............. 
+// Variables concerning the Controller Design is in reference to the Matlab and Simulink Files ..............
+
+// User Input
+float Desired_input = 0.0f;
+// Initial System input in Amperes
+float Sys_input_Amps = 0.0f;
 
 // Sate Space Controller Values
-
+float K_SS_Controller [2] = {-57.1176*0.3, -2.6398*1.5}; // From Matlab
 
 // Controller Variables
-
+float Loop1_output; // Loop 1 controller output
+float Loop2_output;  // Loop 2 controller output
+float PID_Input, PID_Output, PID_Input2, PID_Output2;
 
 // PID (PI Parameters)
 
+// PID 1 - Velocity control after lift-up
+float Kp_1 = -0.09;
+float Ki_1 = -0.09*0.5*0.5;
+float Kd_1 = 0; // No D-Part
+float Tf_1 = 1; // No D-Part
+// Controller Loop (PI-Part) in Case 2 (breaking case)
+float Kp_2 = 0.25;
+float Ki_2 = 0.25;
+float Kd_2 = 0; // No D-Part
+float Tf_2 = 1; // No D-Part
 
 // Saturation Parameters
-
+// PI Controller Limits
+const float uMin1 = -13.0f;
+const float uMax1= 13.0f;
 
 // Cuboid Escon Input Limits in Amps
 const float uMin = -13.0f;        // Minimum Current Allowed
@@ -128,6 +157,28 @@
 //  User-Defined Functions
 // -------------------------------
 
+// USer Button
+void pressed(void);                     // user Button pressed
+void released(void);                    // user Button released
+
+// Controller loop (via interrupt)
+void updateControllers(void);           // controller loop (via interrupt)
+
+// Linear Scaler - Follow the mapping mentioned in the Notes above
+// Linear Scaler
+LinearCharacteristics CurrentToVoltage(-15.0f, 15.0f, 0.0f, 5.0f);
+LinearCharacteristics VoltageToVelocity(0.0f, 3.0f, -4000.0f, 4000.0f);
+
+// PID (PI) Controller (My PID Controller is fine but needs clarity updates)
+PID_Cntrl  C1(Kp_1,Ki_1,Kd_1,Tf_1,Ts,uMin1,uMax1);   // Defining the 1st Loop Controller (PI-Part)
+PID_Cntrl  C2(Kp_2,Ki_2,Kd_2,Tf_2,Ts,uMin1,uMax1);   // Defining the PI Controller for Chase (State 2) to keep motor velocity at zero
+PID_Cntrl  C3(Kp_1*2,Ki_1,Kd_2,Tf_2,Ts,uMin1,uMax1); // Safety Implementation in Case 4 PID
+
+// Accelerometer and Gyroscope Filters
+IIR_filter FilterAccX(t, Ts, 1.0f);
+IIR_filter FilterAccY(t, Ts, 1.0f);
+IIR_filter FilterGyro(t, Ts, t);
+
 // Interrupts
 Ticker  ControllerLoopTimer;            // Interrupt for control loop
 
@@ -146,18 +197,6 @@
 // Timers
 Timer Loop;
 
-// Accelerometer and Gyroscope Filters
-
-// PID (PI) Controller
-
-
-// ----- User defined functions -----------
-
-// Controller loop (via interrupt)
-void updateControllers(void);   
-
-// Linear Scaler - Follow the mapping mentioned in the Notes above
-
 //******************************************************************************
 //----------------------------- Main Loop --------------------------------------
 //******************************************************************************
@@ -171,16 +210,27 @@
     mpu.setDLPFMode(MPU6050_DLPF_BW_42); // Set Low Pass Filter Bandwidth to 44Hz (4.9ms) for the Acc and 42Hz (4.8ms) for the Gyroscope
     mpu.setFullScaleGyroRange(2u);  // Change the scale of the Gyroscope to +/- 1000 degrees/sec
     mpu.setFullScaleAccelRange(MPU6050_ACCEL_FS_4); // Change the scale of the Accelerometer to +/- 4g - Sensitivity: 4096 LSB/mg
-    
+
     // Serial Communication
     pc.baud(115200);
-    
+
     VoltageOut.write(2.5); // Output Zero Current to the Motor
 
     // Reset Filters and PID Controller
+    FilterAccX.reset(0.0f);
+    FilterAccY.reset(0.0f);
+    FilterGyro.reset(0.0f);
+    C1.reset(0.0f);
+    C2.reset(0.0f);
+    C3.reset(0.0f);
 
+    pc.printf("Hello World!\n\r");
+    wait(5);
     // Control Loop Interrupt at Ts, 200Hz Rate
     ControllerLoopTimer.attach(&updateControllers, Ts);
+    Button.fall(&pressed);          // attach key pressed function
+    Button.rise(&released);         // attach key pressed function
+
 
 }
 //******************************************************************************
@@ -188,10 +238,12 @@
 //******************************************************************************
 void updateControllers(void)
 {
-    
+
     // Acquire Velocity
     Velocity_Voltage = 3.0*1.1978917*(Velocity_Voltage_Input.read()); // *1.2V because the Vref is 1.2V // *3 because the prescaling is 1/3 // *1.1978917 is for Calibration purposes instead of 1.2
     // Velocity = .... Refer to mapping
+    Velocity_rpm = VoltageToVelocity(Velocity_Voltage);
+    Velocity = Velocity_rpm*2.0*pi/60.0;
 
 
     // Aquire Raw Acceleration and Gyro Data form the IMU
@@ -200,25 +252,111 @@
     // -------------- Convert Raw data to SI Units --------------------
 
     //Convert Acceleration Raw Data to (ms^-2) - (Settings of +/- 4g)
-    AccX_g = AccX_Raw / 8192.0f;    AccY_g = AccY_Raw / 8192.0f;    AccZ_g = AccZ_Raw / 8192.0f;
+    AccX_g = AccX_Raw / 8192.0f;
+    AccY_g = AccY_Raw / 8192.0f;
+    AccZ_g = AccZ_Raw / 8192.0f;
 
     //Convert Gyroscope Raw Data to Degrees per second - (2^15/1000 = 32.768)
-    GyroX_Degrees = GyroX_Raw / 32.768f;  GyroY_Degrees = GyroY_Raw / 32.768f;  GyroZ_Degrees = GyroZ_Raw / 32.768f;  
+    GyroX_Degrees = GyroX_Raw / 32.768f;
+    GyroY_Degrees = GyroY_Raw / 32.768f;
+    GyroZ_Degrees = GyroZ_Raw / 32.768f;
 
     //Convert Gyroscope Raw Data to Degrees per second
     GyroZ_RadiansPerSecond = (GyroZ_Raw / 32.768f)* pi/180.0f;
 
     // ----- Combine Accelerometer Data and Gyro Data to Get Angle ------
+    Cuboid_Angle_Radians =  -1*atan2(-FilterAccX(AccX_g), FilterAccY(AccY_g)) + 0.7854f + FilterGyro(GyroZ_RadiansPerSecond);
+    Cuboid_Angle_Degrees = Cuboid_Angle_Radians*180.0f/pi;
 
-    
 
     // ------------------------- Controller -----------------------------
 
     // Switch Statement Maybe?......
+    switch(Button_Status) {
+
+        case 0: // Output of 0 Amps
+
+            VoltageOut.write(CurrentToVoltage(0));
+            break;
+
+        case 1: // Breaking
+
+            // PI Controller
+            PID_Input = 0.0f - Velocity;
+            PID_Output = C2.update(PID_Input);
+
+            // System input
+            Sys_input_Amps = PID_Output;
+            // Scaling the controller output from -15 A --> 15 A to 0 V --> 5 V
+            VoltageOut.write(CurrentToVoltage(Sys_input_Amps));
+            break;
+
+        case 2: // Balancing and lift-up
+
+            // Current Input Updater - In Amps
+            // Loop 1
+            Loop1_output = Cuboid_Angle_Radians*K_SS_Controller[0];
+            // Loop 2
+            Loop2_output = GyroZ_RadiansPerSecond*K_SS_Controller[1];
+            // PI Controller
+            PID_Input = Desired_input - Velocity;
+            PID_Output = C1.update(PID_Input);
+
+            // System input
+            Sys_input_Amps = PID_Output - Loop1_output - Loop2_output;
+            // Scaling the controller output from -15 A --> 15 A to 0 V --> 5 V
+            VoltageOut.write(CurrentToVoltage(Sys_input_Amps));
+
+
+            // Do Not Modify - This is implemented to prevent the Cube from continuously speeding up while being still for after falling due to the user interrupting its movement
+            if ( abs(Velocity) > 250.0f && (abs(Cuboid_Angle_Degrees)<50.0f && abs(Cuboid_Angle_Degrees)>40.0f) ) {
+                Button_Status = 4;
+                break;
+            }
+            break;
+
+        case 3: // Fall
+            
+            VoltageOut.write(CurrentToVoltage(0.0f));
+            // Not implemented Yet :)
+            break;
+            
+        case 4: // Lift up when stuck
+        // Do Not Modify - This is implemented to prevent the Cube from continuously speeding up while being still for after falling due to the user interrupting its movement
+
+            PID_Input2 = 0.0f - Velocity;
+            PID_Output2 = C3.update(-1*PID_Input2);
+
+            // System input
+            Sys_input_Amps = PID_Output2;
+            if (Sys_input_Amps > 1.0f) {
+                Sys_input_Amps = 1.0f;
+            }
+            if (Sys_input_Amps < -1.0f) {
+                Sys_input_Amps = -1.0f;
+            }
+            VoltageOut.write(CurrentToVoltage(Sys_input_Amps));
+            //pc.printf("%0.7f, %0.7f, \n\r", abs(Cuboid_Angle_Degrees), abs(Velocity));
+
+            if (abs(Velocity) < 20.0f) {
+
+                Button_Status = 2;
+                break;
+            }
+            break;
+    }
+    // End of Switch Statement
     
+    if (Sys_input_Amps > uMax) {
+        Sys_input_Amps = uMax;
+    }
+    if (Sys_input_Amps < uMin) {
+        Sys_input_Amps = uMin;
+    }
+
     // ----------------
 
-    // Print Data
+    // Print Data // Printing Rate (in this case) is every Ts*100 = 0.007*100 = every 0.7 seconds
     if(++k >= 100) {
         k = 0;
         pc.printf("Some Outputs: %0.6f, %0.6f\n\r", Sys_input_Amps, Velocity);
@@ -231,3 +369,23 @@
 //******************************************************************************
 
 //...
+// start timer as soon as Button is pressed
+void pressed()
+{
+    T_Button.start();
+}
+// Falling edge of button: enable/disable controller
+void released()
+{
+    // readout, stop and reset timer
+    float ButtonTime = T_Button.read();
+    T_Button.stop();
+    T_Button.reset();
+    if(ButtonTime > 0.05f) {
+        Button_Status = Button_Status + 1;
+        if (Button_Status > 3.0f) {
+            Button_Status = 1.0;
+        }
+        pc.printf("Button Status: %i\r\n", Button_Status);
+    }
+}