2016-06-09 02:44:34 +03:00
|
|
|
// Commands for controlling hardware based pulse-width-modulator 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:44:34 +03:00
|
|
|
#include "board/gpio.h" // struct gpio_pwm
|
|
|
|
#include "command.h" // DECL_COMMAND
|
2017-03-11 06:12:05 +03:00
|
|
|
#include "sched.h" // sched_add_timer
|
2016-06-09 02:44:34 +03:00
|
|
|
|
|
|
|
struct pwm_out_s {
|
|
|
|
struct timer timer;
|
|
|
|
struct gpio_pwm pin;
|
|
|
|
uint32_t max_duration;
|
2019-03-29 14:35:19 +03:00
|
|
|
uint16_t value, default_value;
|
2016-06-09 02:44:34 +03:00
|
|
|
};
|
|
|
|
|
2017-08-06 18:19:30 +03:00
|
|
|
static uint_fast8_t
|
2016-06-09 02:44:34 +03:00
|
|
|
pwm_end_event(struct timer *timer)
|
|
|
|
{
|
|
|
|
shutdown("Missed scheduling of next hard pwm event");
|
|
|
|
}
|
|
|
|
|
2017-08-06 18:19:30 +03:00
|
|
|
static uint_fast8_t
|
2016-06-09 02:44:34 +03:00
|
|
|
pwm_event(struct timer *timer)
|
|
|
|
{
|
|
|
|
struct pwm_out_s *p = container_of(timer, struct pwm_out_s, timer);
|
|
|
|
gpio_pwm_write(p->pin, p->value);
|
|
|
|
if (p->value == p->default_value || !p->max_duration)
|
|
|
|
return SF_DONE;
|
|
|
|
p->timer.waketime += p->max_duration;
|
|
|
|
p->timer.func = pwm_end_event;
|
|
|
|
return SF_RESCHEDULE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
command_config_pwm_out(uint32_t *args)
|
|
|
|
{
|
2017-10-12 05:50:29 +03:00
|
|
|
struct gpio_pwm pin = gpio_pwm_setup(args[1], args[2], args[3]);
|
2019-02-27 21:49:31 +03:00
|
|
|
struct pwm_out_s *p = oid_alloc(args[0], command_config_pwm_out
|
|
|
|
, sizeof(*p));
|
2017-10-12 05:50:29 +03:00
|
|
|
p->pin = pin;
|
|
|
|
p->default_value = args[4];
|
|
|
|
p->max_duration = args[5];
|
2016-06-09 02:44:34 +03:00
|
|
|
}
|
|
|
|
DECL_COMMAND(command_config_pwm_out,
|
2017-10-12 05:50:29 +03:00
|
|
|
"config_pwm_out oid=%c pin=%u cycle_ticks=%u value=%hu"
|
|
|
|
" default_value=%hu max_duration=%u");
|
2016-06-09 02:44:34 +03:00
|
|
|
|
|
|
|
void
|
|
|
|
command_schedule_pwm_out(uint32_t *args)
|
|
|
|
{
|
2017-03-09 21:49:03 +03:00
|
|
|
struct pwm_out_s *p = oid_lookup(args[0], command_config_pwm_out);
|
2016-06-09 02:44:34 +03:00
|
|
|
sched_del_timer(&p->timer);
|
|
|
|
p->timer.func = pwm_event;
|
|
|
|
p->timer.waketime = args[1];
|
|
|
|
p->value = args[2];
|
2017-03-11 06:12:05 +03:00
|
|
|
sched_add_timer(&p->timer);
|
2016-06-09 02:44:34 +03:00
|
|
|
}
|
|
|
|
DECL_COMMAND(command_schedule_pwm_out,
|
2017-05-08 00:33:45 +03:00
|
|
|
"schedule_pwm_out oid=%c clock=%u value=%hu");
|
2016-06-09 02:44:34 +03:00
|
|
|
|
2017-05-26 16:14:26 +03:00
|
|
|
void
|
2016-06-09 02:44:34 +03:00
|
|
|
pwm_shutdown(void)
|
|
|
|
{
|
|
|
|
uint8_t i;
|
|
|
|
struct pwm_out_s *p;
|
|
|
|
foreach_oid(i, p, command_config_pwm_out) {
|
|
|
|
gpio_pwm_write(p->pin, p->default_value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DECL_SHUTDOWN(pwm_shutdown);
|
|
|
|
|
|
|
|
void
|
|
|
|
command_set_pwm_out(uint32_t *args)
|
|
|
|
{
|
|
|
|
gpio_pwm_setup(args[0], args[1], args[2]);
|
|
|
|
}
|
2019-02-27 21:49:31 +03:00
|
|
|
DECL_COMMAND(command_set_pwm_out,
|
|
|
|
"set_pwm_out pin=%u cycle_ticks=%u value=%hu");
|