extended UI

master
Ondřej Hruška 4 weeks ago
parent 27e4cd2f32
commit 80c33a3007
  1. 2
      Makefile
  2. 7
      src/global_state.h
  3. 69
      src/graphic_loading.c
  4. 2
      src/high_voltage.c
  5. 37
      src/main.c
  6. 101
      src/radiation.c
  7. 35
      src/radiation.h
  8. 2
      src/sevenseg.c
  9. 2
      src/sevenseg.h
  10. 5
      src/time_base.c
  11. 104
      src/user_interface.c

@ -29,7 +29,7 @@ CFLAGS = -std=gnu99 -mmcu=$(MCU) -DF_CPU=$(F_CPU)UL
CFLAGS += -funsigned-char -funsigned-bitfields -fpack-struct -fshort-enums CFLAGS += -funsigned-char -funsigned-bitfields -fpack-struct -fshort-enums
CFLAGS += -Wall -Wno-main -Wno-strict-prototypes -Wno-comment CFLAGS += -Wall -Wno-main -Wno-strict-prototypes -Wno-comment
CFLAGS += -g2 -Wextra -Wfatal-errors -Wno-unused-but-set-variable CFLAGS += -g2 -Wextra -Wfatal-errors -Wno-unused-but-set-variable
CFLAGS += -ffunction-sections -fdata-sections -Os -Wno-array-bounds CFLAGS += -ffunction-sections -fdata-sections -O2 -Wno-array-bounds
LFLAGS = -Wl,--gc-sections -Wl,--relax LFLAGS = -Wl,--gc-sections -Wl,--relax
#-Llib/libssd1306/bld/ -l:libssd1306.a #-Llib/libssd1306/bld/ -l:libssd1306.a

@ -9,7 +9,12 @@
#include <stdint.h> #include <stdint.h>
extern volatile bool req_update_display; extern volatile bool req_update_display;
extern volatile bool is_weak_battery; extern volatile bool disp_show_tick_mark;
extern volatile bool status_weak_battery;
/// timestamp counted in units of 100ms
extern volatile uint16_t timestamp_100ms; extern volatile uint16_t timestamp_100ms;
/// sub-second counter, counts 0-9 then overflows
extern volatile uint16_t subsec_counter;
#endif //CPROJ_GLOBAL_VARIABLES_H #endif //CPROJ_GLOBAL_VARIABLES_H

@ -16,50 +16,41 @@ void show_loading_screen(uint8_t progress_percent, bool clear)
// bar in a box // bar in a box
#define hei 20 #define hei 16
#define wid DISPLAY_W #define wid DISPLAY_W
#define thic 2
#define inpad 3
#define ofsx ((DISPLAY_W - wid)/2) #define ofsx ((DISPLAY_W - wid)/2)
#define ofsy ((DISPLAY_H - hei)/2) #define ofsy 8
const uint8_t rects[4][4] = { ssd1306_setColor(0xFFFF);
// top
{ lcdint_t xx = (uint8_t)(((uint16_t)wid * (uint16_t)progress_percent)/(uint16_t)100);
0, 0,
wid - 1, thic - 1 ssd1306_fillRect(
}, ofsx,
// left ofsy,
{ ofsx + xx,
0, thic, ofsy + hei
thic - 1, hei - thic - 1 );
},
// right ssd1306_fillRect(
{ ofsx + xx + 1,
wid - thic, thic, ofsy,
wid - 1, hei - thic - 1 ofsx + wid - 2,
}, ofsy
// bot );
{
0, hei - thic, ssd1306_fillRect(
wid - 1, hei - 1 ofsx + xx + 1,
}, ofsy + hei,
}; ofsx + wid - 2,
ofsy + hei
);
for (int i = 0; i < 4; i++) {
ssd1306_fillRect(
ofsx + rects[i][0],
ofsy + rects[i][1],
ofsx + rects[i][2],
ofsy + rects[i][3]
);
}
//
ssd1306_fillRect( ssd1306_fillRect(
ofsx + thic + inpad, ofsx + wid - 1,
ofsy + thic + inpad, ofsy,
ofsx + (uint8_t)(((uint16_t)wid * (uint16_t)progress_percent)/(uint16_t)100) - thic - inpad - 1, ofsx + wid - 1,
ofsy + hei - thic - inpad - 1 ofsy + hei
); );
} }

@ -72,7 +72,7 @@ ISR(ADC_vect)
// If fail to reach target voltage in reasonable time, // If fail to reach target voltage in reasonable time,
// show weak battery icon and stop trying. // show weak battery icon and stop trying.
if ((timestamp_100ms - ts_cap_charged) > 10 && analog < ADCVAL_MINIMAL) { if ((timestamp_100ms - ts_cap_charged) > 10 && analog < ADCVAL_MINIMAL) {
is_weak_battery = true; status_weak_battery = true;
} }
} }
} else if (analog < ADCVAL_TURN_ON) { } else if (analog < ADCVAL_TURN_ON) {

@ -7,10 +7,12 @@
#include "high_voltage.h" #include "high_voltage.h"
#include "radiation.h" #include "radiation.h"
#include "user_interface.h" #include "user_interface.h"
#include "ssd1306.h"
/* globals */ /* globals */
volatile bool req_update_display = false; volatile bool req_update_display = false;
volatile bool is_weak_battery = false; volatile bool status_weak_battery = false;
volatile uint8_t disp_show_tick_mark = 0;
static void shutdown_due_to_weak_battery(); static void shutdown_due_to_weak_battery();
@ -31,20 +33,33 @@ void __attribute__((noreturn)) main()
init_high_voltage(); init_high_voltage();
for (;;) { for (;;) {
_delay_ms(100); // one tick makes the marker lit for 10 ms (more in practice, plus display ghosting latency)
for (uint8_t i = 0; i < 10; i++) {
_delay_ms(10);
if (disp_show_tick_mark == 1) {
ssd1306_setColor(0xFFFF);
ssd1306_fillRect(124, 0, 127, 3);
disp_show_tick_mark = 2;
} else if (disp_show_tick_mark == 2) {
disp_show_tick_mark = 0;
ssd1306_setColor(0x0000);
ssd1306_fillRect(124, 0, 127, 3);
ssd1306_setColor(0xFFFF);
}
}
// check if redraw is needed // check if redraw is needed
bool update_display = false; // bool update_display = true;
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) { // ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
update_display = req_update_display; // update_display = req_update_display;
req_update_display = false; // req_update_display = false;
} // }
if (update_display) { // if (update_display) {
show_current_radiation(); show_current_radiation();
} // }
if (is_weak_battery) { if (status_weak_battery) {
shutdown_due_to_weak_battery(); shutdown_due_to_weak_battery();
} }
} }

@ -14,15 +14,15 @@
#include <math.h> #include <math.h>
#define MEAS_BIN_COUNT 120 #define MEAS_BIN_COUNT 120
typedef uint8_t ticks_t; typedef uint16_t ticks_t;
typedef uint8_t binnum_t; typedef uint8_t binnum_t;
#define TICKS_MAX 255 #define TICKS_MAX 0xFFFF
#define NEEDED_TICKS_FOR_VALUE_DISPLAY 5 #define NEEDED_TICKS_FOR_VALUE_DISPLAY 5
#define TICKS_NEEDED_FOR_AUTORANGE 15 #define TICKS_NEEDED_FOR_AUTORANGE 10
uint16_t get_tick_count_in_bins(binnum_t bincount); uint32_t get_tick_count_in_bins(binnum_t bincount);
/// measurements in the last X seconds /// measurements in the last X seconds
static volatile ticks_t measurements[MEAS_BIN_COUNT]; static volatile ticks_t measurements[MEAS_BIN_COUNT];
@ -40,6 +40,7 @@ ISR(INT0_vect)
measurements[meas_bin]++; measurements[meas_bin]++;
} }
req_update_display = true; req_update_display = true;
disp_show_tick_mark = 1;
} }
void second_callback_irq() void second_callback_irq()
@ -82,9 +83,10 @@ void second_callback_irq()
} }
} }
void init_radiation() { void init_radiation()
{
// clear the measurement buffer // clear the measurement buffer
memset((void*) measurements, 0, sizeof(measurements)); memset((void *) measurements, 0, sizeof(measurements));
as_input(D2); as_input(D2);
@ -93,34 +95,30 @@ void init_radiation() {
EIMSK = _BV(INT0); EIMSK = _BV(INT0);
} }
uint8_t rad_get_progress() { uint8_t rad_get_progress()
ticks_t ticks = get_tick_count_in_bins(num_live_bins); {
uint16_t ticks = get_tick_count_in_bins(num_live_bins);
if (ticks >= NEEDED_TICKS_FOR_VALUE_DISPLAY) { if (ticks >= NEEDED_TICKS_FOR_VALUE_DISPLAY) {
return 100; return 100;
} }
return (((uint16_t)ticks * 100) / (uint16_t)NEEDED_TICKS_FOR_VALUE_DISPLAY); return (((uint16_t) ticks * 100) / (uint16_t) NEEDED_TICKS_FOR_VALUE_DISPLAY);
} }
uint16_t get_tick_count_in_bins(binnum_t bincount) { uint32_t get_tick_count_in_bins(binnum_t bincount)
{
if (bincount == 0) { if (bincount == 0) {
return 0; return 0;
} }
uint16_t all_ticks = 0; uint32_t all_ticks = 0;
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) { ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
// going backwards // going backwards
binnum_t bin = meas_bin; binnum_t bin = meas_bin;
for (;;) { for (;;) {
ticks_t in_bin = measurements[bin]; all_ticks += (uint32_t) measurements[bin];
if ((0xFFFF - all_ticks) < in_bin) {
all_ticks = 0xFFFF;
} else {
all_ticks += in_bin;
}
if (bin == meas_oldest_bin) { if (bin == meas_oldest_bin) {
break; break;
@ -144,37 +142,56 @@ uint16_t get_tick_count_in_bins(binnum_t bincount) {
return all_ticks; return all_ticks;
} }
uint16_t rad_get_cpm() { void rad_get_reading(struct rad_results *out)
// bins take 1 second {
if (!out) { return; }
float count = get_tick_count_in_bins(num_observed_bins); // bins take 1 second
float cpm = (count / (float)num_observed_bins) * 60.0f; const float count = (float) get_tick_count_in_bins(num_observed_bins);
return (uint16_t) roundf(cpm * 10.0f); const float cpm = (count / (float) num_observed_bins) * 60.0f;
}
uint16_t rad_get_usvh(uint8_t *out_decimals) { out->cpm_x10 = (uint32_t) roundf(cpm * 10.0f);
float count = get_tick_count_in_bins(num_observed_bins);
float cpm = (count / (float)num_observed_bins) * 60.0f;
// https://sites.google.com/site/diygeigercounter/technical/gm-tubes-supported // https://sites.google.com/site/diygeigercounter/technical/gm-tubes-supported
float usvh = (cpm / 153.8f); const float usvh = (cpm / 153.8f);
if (usvh < 99.0f) { if (usvh < 2.0f) {
*out_decimals = 3; out->danger_level = RAD_LEVEL_0_SAFE;
return (uint16_t) roundf(usvh * 1000.0f); } else if (usvh < 10.0f) {
out->danger_level = RAD_LEVEL_1_ELEVATED;
} else if (usvh < 20.0f) {
out->danger_level = RAD_LEVEL_2_DANGER;
} else if (usvh < 1000.0f) {
out->danger_level = RAD_LEVEL_3_HIGH_DANGER;
} else {
out->danger_level = RAD_LEVEL_4_SEVERE;
} }
if (usvh < 999.0f) { // // for testing only
*out_decimals = 2; // if (usvh < 1.0f) {
return (uint16_t) roundf(usvh * 100.0f); // out->danger_level = RAD_LEVEL_0_SAFE;
} // } else if (usvh < 3.0f) {
// out->danger_level = RAD_LEVEL_1_ELEVATED;
// } else if (usvh < 6.0f) {
// out->danger_level = RAD_LEVEL_2_DANGER;
// } else if (usvh < 9.0f) {
// out->danger_level = RAD_LEVEL_3_HIGH_DANGER;
// } else {
// out->danger_level = RAD_LEVEL_4_SEVERE;
// }
if (usvh < 9999.0f) { if (usvh < 99.0f) {
*out_decimals = 1; out->usvh_decimals = 3;
return (uint16_t) roundf(usvh * 10.0f); out->usvh_num = (uint32_t) roundf(usvh * 1000.0f);
} else if (usvh < 999.0f) {
out->usvh_decimals = 2;
out->usvh_num = (uint32_t) roundf(usvh * 100.0f);
} else if (usvh < 9999.0f) {
out->usvh_decimals = 1;
out->usvh_num = (uint32_t) roundf(usvh * 10.0f);
} else {
out->usvh_decimals = 0;
out->usvh_num = (uint32_t) roundf(usvh);
} }
*out_decimals = 0;
return (uint16_t) roundf(usvh);
} }

@ -8,14 +8,41 @@
#include <stdbool.h> #include <stdbool.h>
#include <stdint.h> #include <stdint.h>
/// Init the radiation measuring subsystem
void init_radiation(); void init_radiation();
/// get progress 0-100, 100 = normal radiation resumed /// Get progress 0-100, 100 = normal radiation reading resumed.
/// We need some number of ticks after boot so the averaging results
/// are somewhat accurate.
uint8_t rad_get_progress(); uint8_t rad_get_progress();
enum rad_danger {
// 0-2 uSv
RAD_LEVEL_0_SAFE = 0,
// 2-15 uSv
RAD_LEVEL_1_ELEVATED,
// 10-20 uSv
RAD_LEVEL_2_DANGER,
// 20-1000 uSv
RAD_LEVEL_3_HIGH_DANGER,
// 1000+ uSv ("evacuate immediately")
RAD_LEVEL_4_SEVERE,
uint16_t rad_get_cpm(); RAD_LEVEL_MAX,
/// Get current radiation dose rate u uSv/h, x10 (one decimal place) };
uint16_t rad_get_usvh(uint8_t *out_decimals);
struct rad_results {
/// CPM (usvh before conversion), x10 (one decimal place)
uint32_t cpm_x10;
/// Current radiation dose rate uSv/h, adjusting scaling to fit in 5 digits
uint32_t usvh_num;
/// The position of the decimal point (places from the right)
uint8_t usvh_decimals;
/// Danger level enum
enum rad_danger danger_level;
};
void rad_get_reading(struct rad_results *out);
#endif //CPROJ_RADIATION_H #endif //CPROJ_RADIATION_H

@ -112,7 +112,7 @@ void sseg_period(struct SevenSeg *disp) {
disp->x += disp->thick + disp->spacing; disp->x += disp->thick + disp->spacing;
} }
void sseg_number(struct SevenSeg *disp, uint16_t num, uint8_t places, uint8_t decimals) { void sseg_number(struct SevenSeg *disp, uint32_t num, uint8_t places, uint8_t decimals) {
uint8_t digits[5] = {}; uint8_t digits[5] = {};
uint8_t pos = 4; uint8_t pos = 4;
while (num > 0) { while (num > 0) {

@ -25,7 +25,7 @@ void sseg_blank(struct SevenSeg *disp);
void sseg_period(struct SevenSeg *disp); void sseg_period(struct SevenSeg *disp);
void sseg_number(struct SevenSeg *disp, uint16_t num, uint8_t places, uint8_t decimals); void sseg_number(struct SevenSeg *disp, uint32_t num, uint8_t places, uint8_t decimals);
#endif //CPROJ_SEVENSEG_H #endif //CPROJ_SEVENSEG_H

@ -9,17 +9,14 @@
#include <stdbool.h> #include <stdbool.h>
#include <avr/interrupt.h> #include <avr/interrupt.h>
/// timestamp counted in units of 100ms
volatile uint16_t timestamp_100ms = 0; volatile uint16_t timestamp_100ms = 0;
/// sub-second counter, counts 0-9 then overflows volatile uint16_t subsec_counter = 0;
static volatile uint16_t subsec_counter = 0;
// 100ms counter // 100ms counter
ISR(TIMER1_COMPA_vect) ISR(TIMER1_COMPA_vect)
{ {
timestamp_100ms++; timestamp_100ms++;
req_update_display = true; req_update_display = true;
subsec_counter++; subsec_counter++;

@ -16,8 +16,6 @@ static struct SevenSeg sseg = {
.y0 = 0, .y0 = 0,
.charwidth = 17, .charwidth = 17,
.thick = 3, .thick = 3,
//.charwidth = 16,
//.thick = 1,
.spacing = 4, .spacing = 4,
}; };
@ -33,14 +31,27 @@ void init_user_interface()
ssd1306_clearScreen(); ssd1306_clearScreen();
} }
void turn_off_display() { void turn_off_display()
{
ssd1306_displayOff(); ssd1306_displayOff();
} }
static volatile uint16_t last_value = 0;
static volatile uint16_t last_value_d = 0;
void show_current_radiation() void show_current_radiation()
{ {
static struct rad_results last_results = {};
static struct rad_results results;
static uint8_t pending_dark_frames = 0;
static bool is_invert = false;
static bool was_dark = false;
// 12 step counter is better suited for display flashing effect
static uint8_t tick_counter_12 = 0;
tick_counter_12 += 1;
if (tick_counter_12 == 12) {
tick_counter_12 = 0;
}
// this is called from main, so we can sleep // this is called from main, so we can sleep
if (!ended_loading) { if (!ended_loading) {
@ -55,22 +66,81 @@ void show_current_radiation()
clear_screen(); clear_screen();
// start showing values // start showing values
} else { } else {
if (progress < 10) progress = 10; if (progress < 10) { progress = 10; }
show_loading_screen(progress, false); show_loading_screen(progress, false);
return; return;
} }
} }
// uint16_t cpm = rad_get_cpm(); if (pending_dark_frames > 0) {
// if (cpm != last_value) { pending_dark_frames -= 1;
// last_value = cpm; return;
// sseg_number(&sseg, cpm, 5, 1); }
// }
uint8_t decimals = 0; rad_get_reading(&results);
uint16_t usvh = rad_get_usvh(&decimals);
if (usvh != last_value || decimals != last_value_d) { // Severe inverts display to make it super obvious that shit's bad
last_value = usvh; if (results.danger_level >= RAD_LEVEL_4_SEVERE) {
last_value_d = decimals; if (!is_invert) {
sseg_number(&sseg, usvh, 5, decimals); ssd1306_invertMode();
is_invert = true;
}
} else {
if (is_invert) {
is_invert = false;
ssd1306_normalMode();
}
}
// blinking, higher radiation = faster
if (results.danger_level == RAD_LEVEL_1_ELEVATED) {
if (tick_counter_12 == 0) {
pending_dark_frames = 1;
}
}
if (results.danger_level == RAD_LEVEL_2_DANGER) {
if (tick_counter_12 == 0 || tick_counter_12 == 6) {
pending_dark_frames = 1;
}
}
if (results.danger_level >= RAD_LEVEL_3_HIGH_DANGER) {
if (tick_counter_12 == 0 || tick_counter_12 == 4 || tick_counter_12 == 8) {
pending_dark_frames = 1;
}
}
if (pending_dark_frames) {
clear_screen();
was_dark = true;
return;
}
if (was_dark
|| results.cpm_x10 != last_results.cpm_x10
|| results.usvh_decimals != last_results.usvh_decimals
|| results.danger_level != last_results.danger_level)
{
was_dark = false;
if (results.usvh_decimals != last_results.usvh_decimals || results.danger_level != last_results.danger_level) {
clear_screen();
}
memcpy(&last_results, &results, sizeof(struct rad_results));
// This is an optimized function that draws black as well as white lines as rectangles,
// so we don't need to redraw the whole screen
sseg_number(&sseg, results.usvh_num, 5, results.usvh_decimals);
ssd1306_setColor(0xFFFF);
for (uint8_t lv = 1; lv <= results.danger_level; lv++) {
uint8_t xx = 129 - lv * 3;
ssd1306_fillRect(
xx, 20, xx, 30
);
}
} }
} }

Loading…
Cancel
Save