2016-06-09 02:32:06 +03:00
|
|
|
// Commands for controlling GPIO analog-to-digital input pins
|
|
|
|
//
|
|
|
|
// Copyright (C) 2016 Kevin O'Connor <kevin@koconnor.net>
|
|
|
|
//
|
|
|
|
// This file may be distributed under the terms of the GNU GPLv3 license.
|
|
|
|
|
2017-03-09 21:49:03 +03:00
|
|
|
#include "basecmd.h" // oid_alloc
|
2016-06-09 02:32:06 +03:00
|
|
|
#include "board/gpio.h" // struct gpio_adc
|
2016-06-09 03:51:00 +03:00
|
|
|
#include "board/irq.h" // irq_disable
|
2016-06-09 02:32:06 +03:00
|
|
|
#include "command.h" // DECL_COMMAND
|
|
|
|
#include "sched.h" // DECL_TASK
|
|
|
|
|
|
|
|
struct analog_in {
|
|
|
|
struct timer timer;
|
|
|
|
uint32_t rest_time, sample_time, next_begin_time;
|
|
|
|
uint16_t value, min_value, max_value;
|
|
|
|
struct gpio_adc pin;
|
2018-07-02 20:47:22 +03:00
|
|
|
uint8_t invalid_count, range_check_count;
|
2016-06-09 02:32:06 +03:00
|
|
|
uint8_t state, sample_count;
|
|
|
|
};
|
|
|
|
|
2017-07-27 19:05:29 +03:00
|
|
|
static struct task_wake analog_wake;
|
|
|
|
|
2016-12-23 22:49:52 +03:00
|
|
|
static uint_fast8_t
|
2016-06-09 02:32:06 +03:00
|
|
|
analog_in_event(struct timer *timer)
|
|
|
|
{
|
|
|
|
struct analog_in *a = container_of(timer, struct analog_in, timer);
|
2016-11-03 00:30:34 +03:00
|
|
|
uint32_t sample_delay = gpio_adc_sample(a->pin);
|
|
|
|
if (sample_delay) {
|
|
|
|
a->timer.waketime += sample_delay;
|
2016-06-09 02:32:06 +03:00
|
|
|
return SF_RESCHEDULE;
|
|
|
|
}
|
|
|
|
uint16_t value = gpio_adc_read(a->pin);
|
|
|
|
uint8_t state = a->state;
|
|
|
|
if (state >= a->sample_count) {
|
|
|
|
state = 0;
|
|
|
|
} else {
|
|
|
|
value += a->value;
|
|
|
|
}
|
|
|
|
a->value = value;
|
|
|
|
a->state = state+1;
|
|
|
|
if (a->state < a->sample_count) {
|
|
|
|
a->timer.waketime += a->sample_time;
|
|
|
|
return SF_RESCHEDULE;
|
|
|
|
}
|
2018-07-02 20:47:22 +03:00
|
|
|
if (likely(a->value >= a->min_value && a->value <= a->max_value)) {
|
|
|
|
a->invalid_count = 0;
|
|
|
|
} else {
|
|
|
|
a->invalid_count++;
|
|
|
|
if (a->invalid_count >= a->range_check_count) {
|
|
|
|
try_shutdown("ADC out of range");
|
|
|
|
a->invalid_count = 0;
|
|
|
|
}
|
|
|
|
}
|
2017-07-27 19:05:29 +03:00
|
|
|
sched_wake_task(&analog_wake);
|
2016-06-09 02:32:06 +03:00
|
|
|
a->next_begin_time += a->rest_time;
|
|
|
|
a->timer.waketime = a->next_begin_time;
|
|
|
|
return SF_RESCHEDULE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
command_config_analog_in(uint32_t *args)
|
|
|
|
{
|
2019-01-13 19:03:30 +03:00
|
|
|
struct gpio_adc pin = gpio_adc_setup(args[1]);
|
2017-03-09 21:49:03 +03:00
|
|
|
struct analog_in *a = oid_alloc(
|
2016-06-09 02:32:06 +03:00
|
|
|
args[0], command_config_analog_in, sizeof(*a));
|
|
|
|
a->timer.func = analog_in_event;
|
2019-01-13 19:03:30 +03:00
|
|
|
a->pin = pin;
|
2016-06-09 02:32:06 +03:00
|
|
|
a->state = 1;
|
|
|
|
}
|
|
|
|
DECL_COMMAND(command_config_analog_in, "config_analog_in oid=%c pin=%u");
|
|
|
|
|
|
|
|
void
|
|
|
|
command_query_analog_in(uint32_t *args)
|
|
|
|
{
|
2017-03-09 21:49:03 +03:00
|
|
|
struct analog_in *a = oid_lookup(args[0], command_config_analog_in);
|
2016-06-09 02:32:06 +03:00
|
|
|
sched_del_timer(&a->timer);
|
2016-11-03 00:30:34 +03:00
|
|
|
gpio_adc_cancel_sample(a->pin);
|
2016-06-09 02:32:06 +03:00
|
|
|
a->next_begin_time = args[1];
|
|
|
|
a->timer.waketime = a->next_begin_time;
|
|
|
|
a->sample_time = args[2];
|
|
|
|
a->sample_count = args[3];
|
|
|
|
a->state = a->sample_count + 1;
|
|
|
|
a->rest_time = args[4];
|
|
|
|
a->min_value = args[5];
|
|
|
|
a->max_value = args[6];
|
2018-07-02 20:47:22 +03:00
|
|
|
a->range_check_count = args[7];
|
2016-06-09 02:32:06 +03:00
|
|
|
if (! a->sample_count)
|
|
|
|
return;
|
2017-03-11 06:12:05 +03:00
|
|
|
sched_add_timer(&a->timer);
|
2016-06-09 02:32:06 +03:00
|
|
|
}
|
|
|
|
DECL_COMMAND(command_query_analog_in,
|
|
|
|
"query_analog_in oid=%c clock=%u sample_ticks=%u sample_count=%c"
|
2018-07-02 20:47:22 +03:00
|
|
|
" rest_ticks=%u min_value=%hu max_value=%hu range_check_count=%c");
|
2016-06-09 02:32:06 +03:00
|
|
|
|
2017-05-26 16:14:26 +03:00
|
|
|
void
|
2016-06-09 02:32:06 +03:00
|
|
|
analog_in_task(void)
|
|
|
|
{
|
2017-07-27 19:05:29 +03:00
|
|
|
if (!sched_check_wake(&analog_wake))
|
2016-06-09 02:32:06 +03:00
|
|
|
return;
|
|
|
|
uint8_t oid;
|
|
|
|
struct analog_in *a;
|
|
|
|
foreach_oid(oid, a, command_config_analog_in) {
|
|
|
|
if (a->state != a->sample_count)
|
|
|
|
continue;
|
2016-06-09 03:51:00 +03:00
|
|
|
irq_disable();
|
2016-06-09 02:32:06 +03:00
|
|
|
if (a->state != a->sample_count) {
|
2016-06-09 03:51:00 +03:00
|
|
|
irq_enable();
|
2016-06-09 02:32:06 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
uint16_t value = a->value;
|
|
|
|
uint32_t next_begin_time = a->next_begin_time;
|
|
|
|
a->state++;
|
2016-06-09 03:51:00 +03:00
|
|
|
irq_enable();
|
2016-06-09 02:32:06 +03:00
|
|
|
sendf("analog_in_state oid=%c next_clock=%u value=%hu"
|
|
|
|
, oid, next_begin_time, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DECL_TASK(analog_in_task);
|
|
|
|
|
2017-05-26 16:14:26 +03:00
|
|
|
void
|
2016-06-09 02:32:06 +03:00
|
|
|
analog_in_shutdown(void)
|
|
|
|
{
|
|
|
|
uint8_t i;
|
|
|
|
struct analog_in *a;
|
|
|
|
foreach_oid(i, a, command_config_analog_in) {
|
2016-11-03 00:30:34 +03:00
|
|
|
gpio_adc_cancel_sample(a->pin);
|
2017-10-12 05:16:02 +03:00
|
|
|
if (a->sample_count) {
|
|
|
|
a->state = a->sample_count + 1;
|
|
|
|
a->next_begin_time += a->rest_time;
|
|
|
|
a->timer.waketime = a->next_begin_time;
|
|
|
|
sched_add_timer(&a->timer);
|
|
|
|
}
|
2016-06-09 02:32:06 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
DECL_SHUTDOWN(analog_in_shutdown);
|