2020-10-13 19:25:31 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup drivers_saul
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief SAUL wrapper for PWM pins
|
|
|
|
*
|
|
|
|
* @author Christian Amsüss <chrysn@fsfe.org>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "saul.h"
|
|
|
|
#include "phydat.h"
|
|
|
|
#include "periph/pwm.h"
|
|
|
|
#include "saul/periph.h"
|
2020-09-30 23:55:29 +02:00
|
|
|
#include "bitarithm.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find factor and shiftback such that for each value entry in the phydat, the
|
|
|
|
* resulting PWM duty cycle would be (value * factor) >> shiftback.
|
2020-11-05 19:01:17 +01:00
|
|
|
*
|
|
|
|
* This also makes the maximum legal input value for that input (which allows
|
|
|
|
* clipping the input to a value that doesn't wrap during that multiplication,
|
|
|
|
* or just to err out). If future versions of this take the scale into account,
|
|
|
|
* they will adjust the maximum accordingly.
|
2020-09-30 23:55:29 +02:00
|
|
|
*/
|
2020-11-05 19:01:17 +01:00
|
|
|
static int extract_scaling(const phydat_t *state, int *factor, int *shiftback, int *max)
|
2020-09-30 23:55:29 +02:00
|
|
|
{
|
|
|
|
if (state->scale != 0) {
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Number of bits i by which we can shift the calculation (value * (255 << i)/100) >> i
|
|
|
|
* to get a better result than value * 2 (which would otherwise happen in integers) */
|
|
|
|
int shift100 = bitarithm_msb(INT_MAX) - bitarithm_msb(saul_pwm_resolution);
|
|
|
|
|
|
|
|
switch (state->unit) {
|
|
|
|
case UNIT_UNDEF:
|
|
|
|
case UNIT_NONE:
|
|
|
|
*factor = 1;
|
|
|
|
*shiftback = 0;
|
2020-11-05 19:01:17 +01:00
|
|
|
*max = saul_pwm_resolution;
|
2020-09-30 23:55:29 +02:00
|
|
|
break;
|
|
|
|
case UNIT_BOOL:
|
|
|
|
*factor = saul_pwm_resolution;
|
|
|
|
*shiftback = 0;
|
2020-11-05 19:01:17 +01:00
|
|
|
*max = 1;
|
2020-09-30 23:55:29 +02:00
|
|
|
break;
|
|
|
|
case UNIT_PERCENT:
|
|
|
|
*factor = ((int)saul_pwm_resolution << shift100) / 100;
|
|
|
|
*shiftback = shift100;
|
2020-11-05 19:01:17 +01:00
|
|
|
*max = 100;
|
2020-09-30 23:55:29 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2020-10-13 19:25:31 +02:00
|
|
|
|
|
|
|
static inline void setchan(const saul_pwm_channel_t *chan, uint16_t value)
|
|
|
|
{
|
|
|
|
pwm_set(chan->dev,
|
|
|
|
chan->channel,
|
|
|
|
(chan->flags & SAUL_PWM_INVERTED) ? saul_pwm_resolution - value : value);
|
|
|
|
}
|
|
|
|
|
2022-05-02 21:31:03 +02:00
|
|
|
static int write_dimmer(const void *dev, const phydat_t *state)
|
2020-10-13 19:25:31 +02:00
|
|
|
{
|
|
|
|
const saul_pwm_dimmer_params_t *p = dev;
|
|
|
|
|
2020-11-05 19:01:17 +01:00
|
|
|
int factor, shiftback, max;
|
2020-09-30 23:55:29 +02:00
|
|
|
|
2020-11-05 19:01:17 +01:00
|
|
|
int err = extract_scaling(state, &factor, &shiftback, &max);
|
2020-09-30 23:55:29 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-11-05 19:01:17 +01:00
|
|
|
if (state->val[0] < 0 || state->val[0] > max) {
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
|
2020-09-30 23:55:29 +02:00
|
|
|
setchan(&p->channel, (state->val[0] * factor) >> shiftback);
|
2020-10-13 19:25:31 +02:00
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
const saul_driver_t dimmer_saul_driver = {
|
2022-05-02 21:31:03 +02:00
|
|
|
.read = saul_read_notsup,
|
2020-10-13 19:25:31 +02:00
|
|
|
.write = write_dimmer,
|
|
|
|
.type = SAUL_ACT_DIMMER
|
|
|
|
};
|
|
|
|
|
2022-05-02 21:31:03 +02:00
|
|
|
static int write_rgb(const void *dev, const phydat_t *state)
|
2020-10-13 19:25:31 +02:00
|
|
|
{
|
|
|
|
const saul_pwm_rgb_params_t *p = dev;
|
|
|
|
|
2020-11-05 19:01:17 +01:00
|
|
|
int factor, shiftback, max;
|
2020-09-30 23:55:29 +02:00
|
|
|
|
2020-11-05 19:01:17 +01:00
|
|
|
int err = extract_scaling(state, &factor, &shiftback, &max);
|
2020-09-30 23:55:29 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2020-11-05 19:01:17 +01:00
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
if (state->val[i] < 0 || state->val[i] > max) {
|
|
|
|
return -ECANCELED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-13 19:25:31 +02:00
|
|
|
for (int i = 0; i < 3; ++i) {
|
2020-09-30 23:55:29 +02:00
|
|
|
setchan(&p->channels[i], (state->val[i] * factor) >> shiftback);
|
2020-10-13 19:25:31 +02:00
|
|
|
}
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
const saul_driver_t rgb_saul_driver = {
|
2022-05-02 21:31:03 +02:00
|
|
|
.read = saul_read_notsup,
|
2020-10-13 19:25:31 +02:00
|
|
|
.write = write_rgb,
|
|
|
|
.type = SAUL_ACT_LED_RGB
|
|
|
|
};
|