summaryrefslogtreecommitdiffstats
path: root/apps/sensor-watch-lite-test/app.c
blob: 42fb57168c61c9f0067cfb01826dc91323f7e344 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
#include <stdio.h>
#include <string.h>
#include "watch.h"

bool has_ticked = false;

// array of lcd pins from pins.h
const uint8_t lcd_pins[] = {
    SLCD26, // SEG23
    SLCD25, // SEG22
    SLCD24, // SEG21
    SLCD23, // SEG20
    SLCD22, // SEG19
    SLCD21, // SEG18
    SLCD20, // SEG17
    SLCD19, // SEG16
    SLCD18, // SEG15
    SLCD17, // SEG14
    SLCD16, // SEG13
    SLCD15, // SEG12
    SLCD14, // SEG11
    SLCD13, // SEG10
    SLCD12, // SEG9
    SLCD11, // SEG8
    SLCD10, // SEG7
    SLCD9,  // SEG6
    SLCD8,  // SEG5
    SLCD7,  // SEG4
    SLCD6,  // SEG3
    SLCD5,  // SEG2
    SLCD4,  // SEG1
    SLCD3,  // SEG0
    SLCD2,  // COM2
    SLCD1,  // COM1
    SLCD0,  // COM0
};

void cb_tick(void);
void cb_tick(void) {
    has_ticked = true;
    watch_rtc_disable_periodic_callback(8);
}

void pass_if(bool passed);
void pass_if(bool passed) {
    if (passed) {
        watch_uart_puts("P");
    } else {
        watch_uart_puts("F");
    }
}

void app_init(void) {
}

void app_wake_from_backup(void) {
}

void app_setup(void) {
    // Set up tick for RTC test
    watch_rtc_register_periodic_callback(cb_tick, 8);

    // Set up UART for communication with tester
    watch_enable_uart(A4, A1, 19200);

    // Set up LED pins
    watch_enable_leds();
    watch_enable_buzzer();

    // Set up buttons with pull-down resistors
    gpio_set_pin_direction(BTN_ALARM, GPIO_DIRECTION_IN);
    gpio_set_pin_pull_mode(BTN_ALARM, GPIO_PULL_DOWN);
    gpio_set_pin_direction(BTN_LIGHT, GPIO_DIRECTION_IN);
    gpio_set_pin_pull_mode(BTN_LIGHT, GPIO_PULL_DOWN);
    gpio_set_pin_direction(BTN_MODE, GPIO_DIRECTION_IN);
    gpio_set_pin_pull_mode(BTN_MODE, GPIO_PULL_DOWN);

    // Set up ADC for thermistor test    
    watch_enable_adc();
    watch_enable_analog_input(A2);
    // Pin A0 is the thermistor enable pin
    gpio_set_pin_direction(A0, GPIO_DIRECTION_OUT);

    watch_set_led_yellow();
}

void app_prepare_for_standby(void) {
}

void app_wake_from_standby(void) {
}

bool app_loop(void) {
    uint8_t buf[5] = {0};
    watch_storage_read(0, 0, buf, 4);
    printf("%s\n", (const char *)buf);
    if (strcmp((const char *)buf, "PASS") == 0) {
        watch_buzzer_play_note(BUZZER_NOTE_C5, 150);
        watch_buzzer_play_note(BUZZER_NOTE_REST, 25);
        watch_buzzer_play_note(BUZZER_NOTE_E5, 150);
        watch_buzzer_play_note(BUZZER_NOTE_REST, 25);
        watch_buzzer_play_note(BUZZER_NOTE_G5, 150);
        watch_buzzer_play_note(BUZZER_NOTE_REST, 25);
        watch_buzzer_play_note(BUZZER_NOTE_C6, 150);
        watch_set_led_green();
        return true;
    }

    char char_received = watch_uart_getc();

    if (char_received) {
        switch (char_received) {
            // - [X] UART echo
            case 'S':
                // Automatically passes if received by tester
                pass_if(true);
                break;
            // - [X] RTC
            case 'R':
                pass_if(has_ticked);
                break;
            // - [X] LCD pin continuity
            case 'O':
                // Set all LCD pins high
                for (int i = 0; i < 27; i++) {
                    gpio_set_pin_function(lcd_pins[i], GPIO_PIN_FUNCTION_OFF);
                    gpio_set_pin_direction(lcd_pins[i], GPIO_DIRECTION_OUT);
                    gpio_set_pin_level(lcd_pins[i], true);
                }
                // It is the tester's responsibility to check that the pins are high
                pass_if(true);
                break;
            case 'P':
                // Set all LCD pins low
                for (int i = 0; i < 27; i++) {
                    gpio_set_pin_function(lcd_pins[i], GPIO_PIN_FUNCTION_OFF);
                    gpio_set_pin_direction(lcd_pins[i], GPIO_DIRECTION_OUT);
                    gpio_set_pin_level(lcd_pins[i], false);
                }
                // It is the tester's responsibility to check that the pins are low
                pass_if(true);
                break;
            // - [X] LCD pin bridging
            case 'Q':
                {
                    bool passed = true;
                    // Pull all LCD pins up
                    for (int i = 0; i < 27; i++) {
                        gpio_set_pin_function(lcd_pins[i], GPIO_PIN_FUNCTION_OFF);
                        gpio_set_pin_direction(lcd_pins[i], GPIO_DIRECTION_IN);
                        gpio_set_pin_pull_mode(lcd_pins[i], GPIO_PULL_UP);
                    }
                    // SEG23 is adjacent to the red LED.
                    // setting the LED red drives RED low.
                    watch_set_led_red();
                    if (!gpio_get_pin_level(lcd_pins[0])) {
                        // If SEG23 is low, then it must be bridged to the red pin
                        pass_if(false);
                    }
                    watch_set_led_off();
                    // After this, all LCD pins are adjacent. Test if each pin is bridged to the previous one.
                    for (int i = 1; i < 27; i++) {
                        gpio_set_pin_direction(lcd_pins[i - 1], GPIO_DIRECTION_OUT);
                        gpio_set_pin_level(lcd_pins[i - 1], false);
                        if (!gpio_get_pin_level(lcd_pins[i])) {
                            passed = false;
                            break;
                        }
                        gpio_set_pin_direction(lcd_pins[i - 1], GPIO_DIRECTION_IN);
                        gpio_set_pin_pull_mode(lcd_pins[i - 1], GPIO_PULL_UP);
                    }
                    // Special cases: 
                    // SLCD0 neighbors VCC
                    gpio_set_pin_direction(SLCD0, GPIO_DIRECTION_IN);
                    gpio_set_pin_pull_mode(SLCD0, GPIO_PULL_DOWN);
                    if (gpio_get_pin_level(SLCD0)) {
                        passed = false;
                    }
                    // SLCD11 neighbors VCC
                    gpio_set_pin_direction(SLCD11, GPIO_DIRECTION_IN);
                    gpio_set_pin_pull_mode(SLCD11, GPIO_PULL_DOWN);
                    if (gpio_get_pin_level(SLCD11)) {
                        passed = false;
                    }
                    // SLCD21 neighbors VCC
                    gpio_set_pin_direction(SLCD21, GPIO_DIRECTION_IN);
                    gpio_set_pin_pull_mode(SLCD21, GPIO_PULL_DOWN);
                    if (gpio_get_pin_level(SLCD21)) {
                        passed = false;
                    }
                    watch_enable_display();
                    delay_ms(50);
                    // SLCD12 neighbors VLCD
                    gpio_set_pin_function(SLCD12, GPIO_PIN_FUNCTION_OFF);
                    gpio_set_pin_direction(SLCD12, GPIO_DIRECTION_IN);
                    gpio_set_pin_pull_mode(SLCD12, GPIO_PULL_DOWN);
                    if (gpio_get_pin_level(SLCD12)) {
                        passed = false;
                    }
                    for (int i = 0; i < 27; i++) {
                        gpio_set_pin_function(lcd_pins[i], GPIO_PIN_FUNCTION_OFF);
                        gpio_set_pin_direction(lcd_pins[i], GPIO_DIRECTION_IN);
                        gpio_set_pin_pull_mode(lcd_pins[i], GPIO_PULL_OFF);
                    }

                    pass_if(passed);
                }
                break;
            // - [X] Thermistor high
            case 'U':
                // Set A0 high and read the value of A2 via the ADC.
                // Pass if the value is near VCC.
                gpio_set_pin_level(A0, true);
                pass_if(watch_get_analog_pin_level(A2) > 65000);
                break;
            // - [X] Thermistor low
            case 'T':
                {
                    // Set A0 low and read the value of A2 via the ADC.
                    // Pass if the value is within the realm of reasonable temperatures.
                    // 15000 is a few minutes in the freezer, 45000 is holding it a few feet over
                    gpio_set_pin_level(A0, false);
                    uint16_t value = watch_get_analog_pin_level(A2);
                    pass_if(value < 45000 && value > 15000);
                }
                break;
            // - [X] VLCD low
            case 'V':
                watch_enable_display();
                SLCD->CTRLA.bit.ENABLE = 0;
                while(SLCD->SYNCBUSY.bit.ENABLE);
                SLCD->CTRLC.bit.CTST = 0x0;
                SLCD->CTRLA.bit.ENABLE = 1;
                while(SLCD->SYNCBUSY.bit.ENABLE);
                break;
            // - [X] VLCD high
            case 'W':
                watch_enable_display();
                SLCD->CTRLA.bit.ENABLE = 0;
                while(SLCD->SYNCBUSY.bit.ENABLE);
                SLCD->CTRLC.bit.CTST = 0xD;
                SLCD->CTRLA.bit.ENABLE = 1;
                while(SLCD->SYNCBUSY.bit.ENABLE);
                break;
            // - [X] Buttons
            case 'B':
                // Pass if all three buttons are low
                pass_if(!gpio_get_pin_level(BTN_ALARM) && !gpio_get_pin_level(BTN_LIGHT) && !gpio_get_pin_level(BTN_MODE));
                break;
            case 'L':
                // pass if BTN_LIGHT is high and the other two are low
                pass_if(gpio_get_pin_level(BTN_LIGHT) && !gpio_get_pin_level(BTN_ALARM) && !gpio_get_pin_level(BTN_MODE));
                watch_uart_puts("P");
                break;
            case 'A':
                // pass if BTN_ALARM is high and the other two are low
                pass_if(gpio_get_pin_level(BTN_ALARM) && !gpio_get_pin_level(BTN_LIGHT) && !gpio_get_pin_level(BTN_MODE));
                break;
            case 'M':
                // pass if BTN_MODE is high and the other two are low
                pass_if(gpio_get_pin_level(BTN_MODE) && !gpio_get_pin_level(BTN_ALARM) && !gpio_get_pin_level(BTN_LIGHT));
                break;

            // - [X] File system
            case 'F':
                watch_storage_erase(0);
                watch_storage_write(0, 0, (const char *)"PASS", 4);
                watch_storage_sync();
                watch_storage_read(0, 0, buf, 4);
                delay_ms(10);
                pass_if(strcmp((const char *)buf, (const char *)"PASS") == 0);
                break;

            // - [ ] Buzzer
            case 'Z':
                // reset the board
                break;

        }
    }

    return false;
}