Framework for doing the figure 8 track.

Dependencies:   mbed

Revision:
0:ad375c052b4c
Child:
1:55e0aaf71bda
diff -r 000000000000 -r ad375c052b4c main.cpp
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/main.cpp	Fri Mar 20 08:59:05 2015 +0000
@@ -0,0 +1,273 @@
+#include "mbed.h"
+#include "stdlib.h"
+
+//Outputs
+DigitalOut led1(LED1);
+DigitalOut clk(PTD5);
+DigitalOut si(PTD4);
+PwmOut motor1(PTA4);
+PwmOut motor2(PTA12);
+DigitalOut break1(PTC12);
+DigitalOut break2(PTC13);
+PwmOut servo(PTA5);
+
+//Inputs
+AnalogIn camData(PTC2);
+
+//Encoder setup and variables
+InterruptIn interrupt(PTA13);
+
+//Line Tracking Variables --------------------------------
+float ADCdata [128];
+float minAccum;
+float minCount;
+float approxPos;
+float minVal;
+int minLoc;
+
+//Servo turning parameters
+float straight = 0.00155f;
+float hardLeft = 0.0013f;
+float slightLeft = 0.00145f;
+float hardRight = 0.0018f;
+float slightRight = 0.00165f;
+
+//End of Line Tracking Variables -------------------------
+
+//Encoder and Motor Driver Variables ---------------------
+
+//Intervals used during encoder data collection to measure velocity
+int interval1=0;
+int interval2=0;
+int interval3=0;
+int avg_interval=0;
+int lastchange1 = 0;
+int lastchange2 = 0;
+int lastchange3 = 0;
+int lastchange4 = 0;
+
+//Variables used to for velocity control
+float avg_speed = 0;
+float stall_check = 0;
+float tuning_val = 1;
+
+Timer t;
+Timer servoTimer;
+
+//Observed average speeds for each duty cycle
+const float DESIRED_SPEED = 0.5;
+const float TUNING_CONSTANT_10 = 1.10;
+const float TUNING_CONSTANT_20 = 3.00;
+const float TUNING_CONSTANT_30 = 4.30;
+const float TUNING_CONSTANT_50 = 6.880;
+const float PI = 3.14159;
+const float WHEEL_CIRCUMFERENCE = .05*PI;
+
+//Velocity Control Tuning Constants
+const float TUNE_THRESH = 0.5f;
+const float TUNE_AMT = 0.1f;
+
+//Parameters specifying sample sizes and delays for small and large average speed samples
+float num_samples_small = 3.0f;
+float delay_small = 0.05f;
+float num_samples_large = 100.0f;
+float delay_large = 0.1f;
+
+//Large and small arrays used to get average velocity values
+float large_avg_speed_list [100];
+float small_avg_speed_list [10];
+
+//End of Encoder and Motor Driver Variables ----------------------
+
+// Encoder and Motor Driver Functions ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+float get_speed(){
+    float revPerSec = (1.0f/((float)avg_interval*.000001))*.25f;
+    float linearSpeed = revPerSec * WHEEL_CIRCUMFERENCE;
+    return linearSpeed;
+}
+
+float get_avg_speed(float num_samples, float delay) {
+    
+    float avg_avg_speed = 0;
+    
+    for (int c = 0; c < num_samples; c++) {
+        if (num_samples == num_samples_small){
+            small_avg_speed_list[c] = get_speed();
+        } else if (num_samples == num_samples_large){
+            large_avg_speed_list[c] = get_speed();
+        }
+        //wait(delay);
+        }
+                
+        for (int c = 1; c < num_samples; c++) {
+            if (num_samples == num_samples_small){
+                avg_avg_speed += small_avg_speed_list[c];
+            } else if (num_samples == num_samples_large){
+                avg_avg_speed += large_avg_speed_list[c];
+            }
+        }
+    return avg_avg_speed/num_samples;
+}
+
+void velocity_control(float duty_cyc, float tuning_const) {
+    
+    avg_speed = get_speed();//get_avg_speed(num_samples_small, delay_small);
+    
+    if (avg_speed == stall_check) {
+        avg_speed = 0;
+        tuning_val += TUNE_AMT;
+    } else if((avg_speed - tuning_const) > TUNE_THRESH){
+        tuning_val -= TUNE_AMT;
+        stall_check = avg_speed;
+    } else if (tuning_const - avg_speed > TUNE_THRESH){
+        //tuning_val += TUNE_AMT;
+        stall_check = avg_speed;
+    } else {
+        //tuning_val = 1;
+        stall_check = avg_speed;
+    }
+    motor1.pulsewidth(.0025 * duty_cyc * tuning_val);
+    motor2.pulsewidth(.0025 * duty_cyc * tuning_val);
+    
+}
+
+void fallInterrupt(){
+    
+    int time = t.read_us();
+    interval1 = time - lastchange2;
+    interval2 = lastchange1-lastchange3;
+    interval3 = lastchange2 - lastchange4;
+    avg_interval = (interval1 + interval2 + interval3)/3;
+    
+    lastchange4 = lastchange3;
+    lastchange3 = lastchange2;
+    lastchange2 = lastchange1;
+    lastchange1 = time;
+}
+
+void riseInterrupt(){
+    int time = t.read_us();
+    interval1 = time - lastchange2;
+    interval2 = lastchange1-lastchange3;
+    interval3 = lastchange2 - lastchange4;
+    avg_interval = (interval1 + interval2 + interval3)/3;
+    
+    lastchange4 = lastchange3;
+    lastchange3 = lastchange2;
+    lastchange2 = lastchange1;
+    lastchange1 = time;
+}
+
+
+//End of Encoder and Motor Driver Functions ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+int main() {
+    //Line Tracker Initializations
+    int integrationCounter = 0;
+    
+    // Motor Driver Initializations
+    motor1.period(.0025);
+    motor2.period(.0025);
+    interrupt.fall(&fallInterrupt);
+    interrupt.rise(&riseInterrupt);
+
+    //wait(5);
+
+    motor1.pulsewidth(.0025*.1);
+    motor2.pulsewidth(.0025*.1);
+    break1 = 0;
+    break2 = 0;
+    
+    t.start();
+    
+    wait(5);
+    
+    while(1) {
+            
+        if(integrationCounter % 151== 0){
+            //Disable interrupts
+            //__disable_irq();
+            //interrupt.fall(NULL);
+            //interrupt.rise(NULL);
+            
+            //Send start of integration signal
+            si = 1;
+            clk = 1;
+
+            si = 0;
+            clk = 0;
+            
+            //Reset timing counter for integration
+            integrationCounter = 0;
+            
+            //Reset line tracking variables
+            minAccum = 0;
+            minCount = 0;
+            approxPos = 0;
+            
+            //Reset Encoder and Motor Driver Variables
+            interval1=0;
+            interval2=0;
+            interval3=0;
+            avg_interval=0;
+            lastchange1 = 0;
+            lastchange2 = 0;
+            lastchange3 = 0;
+            lastchange4 = 0;
+            
+            t.reset();
+                
+        }
+        else if (integrationCounter > 129){
+            //Enable interrupts
+            //__enable_irq();
+            //interrupt.fall(&fallInterrupt);
+            //interrupt.rise(&riseInterrupt);
+            
+            minVal = ADCdata[15];
+            for (int c = 15; c < 118; c++) {
+                if (ADCdata[c] < minVal){
+                    minVal = ADCdata[c];
+                    minLoc = c;
+                }
+            }
+            
+            for (int c = 15; c < 118; c++) {
+                if (ADCdata[c] >= minVal && ADCdata[c] - minVal < 0.04f && ADCdata[c] > 0.1f){
+                    minAccum += c;
+                    minCount++;
+                }
+            }
+            
+            approxPos = (float)minAccum/(float)minCount;
+            
+            if(approxPos > 0 && approxPos <= 20){
+                    servo.pulsewidth(hardLeft);
+            } else if (approxPos > 20 && approxPos <= 45){
+                    servo.pulsewidth(slightLeft);
+            } else if (approxPos > 45 && approxPos <= 90){
+                servo.pulsewidth(straight);
+            } else if (approxPos > 90 && approxPos <= 105){
+                servo.pulsewidth(slightRight);
+            } else if (approxPos > 105 && approxPos <= 128){
+                    servo.pulsewidth(hardRight);
+            }
+            
+            velocity_control(0.05f, DESIRED_SPEED);
+            
+            integrationCounter = 150;
+        }
+        else{
+            clk = 1;
+            wait_us(70);
+            ADCdata[integrationCounter - 1] = camData;
+            clk = 0;
+        }
+
+        //clk = 0;
+        integrationCounter++;
+        //camData.
+        
+    }
+}