Step 3: Specify and Implement a Sequence of Orientation State

Dependencies:   MMA8451Q

Fork of Accelerometer_LAB4_STEP2 by Soheil Novinfard

Files at this revision

API Documentation at this revision

Comitter:
novinfard
Date:
Thu Feb 22 13:08:47 2018 +0000
Parent:
2:dd33a9d29c63
Commit message:
Finalise Project

Changed in this revision

main.cpp Show annotated file Show diff for this revision Revisions of this file
diff -r dd33a9d29c63 -r 1b27f39fe4e8 main.cpp
--- a/main.cpp	Fri Feb 16 20:11:08 2018 +0000
+++ b/main.cpp	Thu Feb 22 13:08:47 2018 +0000
@@ -2,55 +2,264 @@
 #include "rtos.h"
 #include "MMA8451Q.h"
 
-  PinName const SDA = PTE25;
-  PinName const SCL = PTE24;
+PinName const SDA = PTE25;
+PinName const SCL = PTE24;
 
 #define MMA8451_I2C_ADDRESS (0x1d<<1)
 
+Thread isTriggeredStatus ;
+
+enum states { FLAT, RIGHT, LEFT, DOWN, UP, OVER, INTERMEDIATE };
+
+Serial pc(USBTX, USBRX); // tx, rx
+
+volatile states positionState = INTERMEDIATE;
+
+void ifTrigerred()
+{
+    DigitalOut rled(LED1);
+    DigitalOut gled(LED2);
+
+    // I decided to separate each of the "flat", "right" and "up" state into to two different states : Begin and End
+    // It is the only difference from the diagram I've already uploaded last week.
+    enum spaceStates { flatBegin, flatEnd, rightBegin, rightEnd, upBegin, upEnd, flatTriggered, stateTimeError, stateSequenceError };
+    Timer timer;
+
+    rled = 1;
+    gled = 1;
+    spaceStates state = flatBegin;
+
+    while (true) {
+        if (positionState == INTERMEDIATE) {
+            Thread::wait(100);
+            continue;
+        }
+
+
+        switch (state) {
+            case flatBegin:
+                if (positionState == FLAT) {
+                    pc.printf("flatBegin \n\r");
+                    timer.start();
+                    pc.printf("%d", timer.read_ms());
+                    if (timer.read() >= 10) {
+                        timer.stop();
+                        timer.reset();
+                        state = flatEnd;
+                    }
+                } else {
+                    timer.stop();
+                    timer.reset();
+                    if (positionState == RIGHT) {
+                        state = stateTimeError;
+                    } else {
+                        state = stateSequenceError;
+                    }
+                }
+                break;
+
+            case flatEnd:
+                rled = 1;
+                pc.printf("FlatEnd \n\r");
+                if (positionState == RIGHT) {
+                    state = rightBegin;
+                } else {
+                    if (positionState != FLAT) {
+                        state = stateSequenceError;
+                    }
+                }
+                break;
+
+            case rightBegin:
+                if (positionState == RIGHT) {
+                    pc.printf("RightBegin \n\r");
+                    timer.start();
+                    pc.printf("%d", timer.read_ms());
+                    if (timer.read() >= 2) {
+                        state = rightEnd;
+                    }
+                } else {
+                    timer.stop();
+                    timer.reset();
+                    if (positionState == UP) {
+                        state = stateTimeError;
+                    } else {
+                        state = stateSequenceError;
+                    }
+                }
+                break;
+
+            case rightEnd:
+                pc.printf("RightEnd \n\r");
+                pc.printf("%d", timer.read_ms());
+                if (timer.read() <= 6) {
+                    if (positionState == UP) {
+                        timer.stop();
+                        timer.reset();
+                        state = upBegin;
+                    } else if (positionState != RIGHT) {
+                        timer.stop();
+                        timer.reset();
+                        state = stateSequenceError;
+                    }
+                } else {
+                    timer.stop();
+                    timer.reset();
+                    state = stateTimeError;
+                }
+                break;
+
+            case upBegin:
+                if (positionState == UP) {
+                    pc.printf("UpStart \n\r");
+                    timer.start();
+                    pc.printf("%d", timer.read_ms());
+                    if (timer.read() >= 4) {
+                        state = upEnd;
+                    }
+                } else {
+                    timer.stop();
+                    timer.reset();
+                    if (positionState == FLAT) {
+                        state = stateTimeError;
+                    } else {
+                        state = stateSequenceError;
+                    }
+                }
+                break;
+
+            case upEnd:
+                pc.printf("upEnd \n\r");
+                pc.printf("%d", timer.read_ms());
+                if (timer.read() <= 8) {
+                    if (positionState == FLAT) {
+                        timer.stop();
+                        timer.reset();
+                        state = flatTriggered;
+                    } else if (positionState != UP) {
+                        timer.stop();
+                        timer.reset();
+                        state = stateSequenceError;
+                    }
+                } else {
+                    timer.stop();
+                    timer.reset();
+                    state = stateTimeError;
+                }
+                break;
+
+            case flatTriggered:
+                gled = 0;
+                pc.printf("flatTriggered \n\r");
+                break;
+
+            case stateSequenceError:
+                rled = 0;
+                pc.printf("State sequence error \n\r");
+                if (positionState == FLAT) {
+                    state = flatBegin;
+                }
+                break;
+
+            case stateTimeError:
+                rled = 0;
+                pc.printf("State time error \n\r");
+                if (positionState == FLAT) {
+                    state = flatBegin;
+                }
+                break;
+        }
+
+        Thread::wait(100);
+    }
+}
+
 int main(void)
 {
     MMA8451Q acc(SDA, SCL, MMA8451_I2C_ADDRESS);
-    PwmOut rled(LED1);
-    PwmOut gled(LED2);
-    PwmOut bled(LED3);
-    Serial pc(USBTX, USBRX); // tx, rx
 
 
-    pc.printf("MMA8451 ID: %d\n", acc.getWhoAmI());
+    states state = INTERMEDIATE;
+    states lastStableState = INTERMEDIATE;
+    states pastState = INTERMEDIATE;
+    int stateCounter = 0;
+    int const stableState = 3;
+
+    pc.printf("MMA8451 ID: %d\n\r", acc.getWhoAmI());
+
+    // is triggered or not
+    isTriggeredStatus.start(callback(ifTrigerred));
 
     while (true) {
         float x, y, z, ts;
+
         //threshold
-        ts = 0.2;
+        ts = 0.15;
+
         x = acc.getAccX();
         y = acc.getAccY();
         z = acc.getAccZ();
-        rled = 1.0f - abs(x);
-        gled = 1.0f - abs(y);
-        bled = 1.0f - abs(z);
-        
-        // black
-        rled = 1;
-        gled = 1;
-        bled = 1;
-        
-        if( abs(x) <= 1+ts && abs(x)>= 1-ts) {
-            rled = 0   ;
+
+        if (z > 1-ts && z < 1+ts) {
+            state = FLAT;
+        } else if (z > -1-ts && z < -1+ts) {
+            state = OVER;
+        } else if (y > -1-ts && y < -1+ts) {
+            state = RIGHT;
+        } else if (y > 1-ts && y < 1+ts) {
+            state = LEFT;
+        }  else if (x > -1-ts && x < -1+ts) {
+            state = UP;
+        } else if (x > 1-ts && x < 1+ts) {
+            state = DOWN;
+        }  else {
+            state = INTERMEDIATE;
         }
-        if( abs(y) <= 1+ts && abs(y)>= 1-ts) {
-            gled = 0   ;
+
+        if (state != INTERMEDIATE) {
+            if (state == pastState) {
+                if (stateCounter < stableState) {
+                    stateCounter ++;
+                }
+            } else {
+                pastState = state;
+                stateCounter = 0;
+            }
         }
-        if( abs(z) <= 1+ts && abs(z)>= 1-ts) {
-            bled = 0   ;
+
+        // state has changed and is stable
+        if (stateCounter == stableState && lastStableState != state) {
+            lastStableState = state;
+
+            switch (state) {
+                case FLAT:
+                    pc.printf("Flat state \n\r");
+                    break;
+
+                case LEFT:
+                    pc.printf("Left state \n\r");
+                    break;
+
+                case RIGHT:
+                    pc.printf("Right state \n\r");
+                    break;
+
+                case UP:
+                    pc.printf("Up state \n\r");
+                    break;
+
+                case DOWN:
+                    pc.printf("Down state \n\r");
+                    break;
+
+                case OVER:
+                    pc.printf("Over state \n\r");
+                    break;
+            }
+
+            positionState = state;
         }
-        
-        if(rled == 0 && gled == 0 && bled ==1)
-            pc.printf("Flat");
-        else if (rled == 0 && gled == 0 && bled ==1)
-            pc.printf("Flat");
-        
-        
-        Thread::wait(300);
-       pc.printf("X: %1.2f, Y: %1.2f, Z: %1.2f\n\r", x, y, z);
+
+        Thread::wait(200);
     }
-}
+}
\ No newline at end of file