ardupilot/libraries/AP_HAL_Linux/PWM_Sysfs.cpp
Georgii Staroselskii d214a36e0a AP_HAL_Linux: reset duty cycle before setting period
On kernels 4.7+ duty_cycle should always be less than period.
Otherways, we'll get a EINVAL.

It makes sense to set duty_cycle to 0, as
duty_cycle doesn't really make sense without a proper period.

A proper way to handle these errors might be to call pwm_adjust_config
in every pwmchip backend but it's unrealistic to hope that all vendors
will do it quickly.
2017-10-17 13:28:34 -07:00

288 lines
8.0 KiB
C++

/*
* Copyright (C) 2015 Intel Corporation. All rights reserved.
*
* This file is free software: you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This file is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "PWM_Sysfs.h"
#include <errno.h>
#include <fcntl.h>
#include <inttypes.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <AP_HAL/AP_HAL.h>
#include <AP_Math/AP_Math.h>
static const AP_HAL::HAL &hal = AP_HAL::get_HAL();
namespace Linux {
PWM_Sysfs_Base::PWM_Sysfs_Base(char* export_path, char* polarity_path,
char* enable_path, char* duty_path,
char* period_path, uint8_t channel)
: _export_path(export_path)
, _polarity_path(polarity_path)
, _enable_path(enable_path)
, _duty_path(duty_path)
, _period_path(period_path)
, _channel(channel)
{
}
PWM_Sysfs_Base::~PWM_Sysfs_Base()
{
::close(_duty_cycle_fd);
free(_polarity_path);
free(_enable_path);
free(_period_path);
}
void PWM_Sysfs_Base::init()
{
if (_export_path == nullptr || _enable_path == nullptr ||
_period_path == nullptr || _duty_path == nullptr) {
AP_HAL::panic("PWM_Sysfs: export=%p enable=%p period=%p duty=%p"
" required path is NULL", _export_path, _enable_path,
_period_path, _duty_path);
}
/* Not checking the return of write_file since it will fail if
* the pwm has already been exported
*/
Util::from(hal.util)->write_file(_export_path, "%u", _channel);
free(_export_path);
_duty_cycle_fd = ::open(_duty_path, O_RDWR | O_CLOEXEC);
if (_duty_cycle_fd < 0) {
AP_HAL::panic("LinuxPWM_Sysfs:Unable to open file %s: %s",
_duty_path, strerror(errno));
}
free(_duty_path);
}
void PWM_Sysfs_Base::enable(bool value)
{
if (Util::from(hal.util)->write_file(_enable_path, "%u", value) < 0) {
hal.console->printf("LinuxPWM_Sysfs: %s Unable to %s\n",
_enable_path, value ? "enable" : "disable");
}
}
bool PWM_Sysfs_Base::is_enabled()
{
unsigned int enabled;
if (Util::from(hal.util)->read_file(_enable_path, "%u", &enabled) < 0) {
hal.console->printf("LinuxPWM_Sysfs: %s Unable to get status\n",
_enable_path);
}
return enabled;
}
void PWM_Sysfs_Base::set_period(uint32_t nsec_period)
{
set_duty_cycle(0);
if (Util::from(hal.util)->write_file(_period_path, "%u", nsec_period) < 0) {
hal.console->printf("LinuxPWM_Sysfs: %s Unable to set period\n",
_period_path);
}
}
uint32_t PWM_Sysfs_Base::get_period()
{
uint32_t nsec_period;
if (Util::from(hal.util)->read_file(_period_path, "%u", &nsec_period) < 0) {
hal.console->printf("LinuxPWM_Sysfs: %s Unable to get period\n",
_period_path);
nsec_period = 0;
}
return nsec_period;
}
void PWM_Sysfs_Base::set_freq(uint32_t freq)
{
set_period(hz_to_nsec(freq));
}
uint32_t PWM_Sysfs_Base::get_freq()
{
return nsec_to_hz(get_period());
}
bool PWM_Sysfs_Base::set_duty_cycle(uint32_t nsec_duty_cycle)
{
/* Don't log fails since this could spam the console */
if (dprintf(_duty_cycle_fd, "%u", nsec_duty_cycle) < 0) {
return false;
}
_nsec_duty_cycle_value = nsec_duty_cycle;
return true;
}
uint32_t PWM_Sysfs_Base::get_duty_cycle()
{
return _nsec_duty_cycle_value;
}
void PWM_Sysfs_Base::set_polarity(PWM_Sysfs_Base::Polarity polarity)
{
if (Util::from(hal.util)->write_file(_polarity_path, "%s",
polarity == NORMAL ?
"normal" : "inversed") < 0) {
hal.console->printf("LinuxPWM_Sysfs: %s Unable to set polarity\n",
_polarity_path);
}
}
PWM_Sysfs_Base::Polarity PWM_Sysfs_Base::get_polarity()
{
char polarity[16];
if (Util::from(hal.util)->read_file(_polarity_path, "%s", polarity) < 0) {
hal.console->printf("LinuxPWM_Sysfs: %s Unable to get polarity\n",
_polarity_path);
return NORMAL;
}
return strncmp(polarity, "normal", sizeof(polarity)) ? INVERSE : NORMAL;
}
/* PWM Sysfs api for mainline kernel */
char *PWM_Sysfs::_generate_export_path(uint8_t chip)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwmchip%u/export", chip);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate export path\n");
}
return path;
}
char *PWM_Sysfs::_generate_polarity_path(uint8_t chip, uint8_t channel)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwmchip%u/pwm%u/polarity",
chip, channel);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate polarity path\n");
}
return path;
}
char *PWM_Sysfs::_generate_enable_path(uint8_t chip, uint8_t channel)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwmchip%u/pwm%u/enable",
chip, channel);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate enable path\n");
}
return path;
}
char *PWM_Sysfs::_generate_duty_path(uint8_t chip, uint8_t channel)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwmchip%u/pwm%u/duty_cycle",
chip, channel);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate duty path\n");
}
return path;
}
char *PWM_Sysfs::_generate_period_path(uint8_t chip, uint8_t channel)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwmchip%u/pwm%u/period",
chip, channel);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate period path\n");
}
return path;
}
PWM_Sysfs::PWM_Sysfs(uint8_t chip, uint8_t channel) :
PWM_Sysfs_Base(_generate_export_path(chip),
_generate_polarity_path(chip, channel),
_generate_enable_path(chip, channel),
_generate_duty_path(chip, channel),
_generate_period_path(chip, channel),
channel)
{
}
/* PWM Sysfs api for bebop kernel */
char *PWM_Sysfs_Bebop::_generate_export_path()
{
return strdup("/sys/class/pwm/export");
}
char *PWM_Sysfs_Bebop::_generate_enable_path(uint8_t channel)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwm_%u/run",
channel);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate enable path\n");
}
return path;
}
char *PWM_Sysfs_Bebop::_generate_duty_path(uint8_t channel)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwm_%u/duty_ns",
channel);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate duty path\n");
}
return path;
}
char *PWM_Sysfs_Bebop::_generate_period_path(uint8_t channel)
{
char *path;
int r = asprintf(&path, "/sys/class/pwm/pwm_%u/period_ns",
channel);
if (r == -1) {
AP_HAL::panic("LinuxPWM_Sysfs :"
"couldn't allocate period path\n");
}
return path;
}
PWM_Sysfs_Bebop::PWM_Sysfs_Bebop(uint8_t channel) :
PWM_Sysfs_Base(_generate_export_path(),
nullptr,
_generate_enable_path(channel),
_generate_duty_path(channel),
_generate_period_path(channel),
channel)
{
}
}