288 lines
8.0 KiB
C++
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(_export_path);
|
|
free(_polarity_path);
|
|
free(_enable_path);
|
|
free(_duty_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)
|
|
{
|
|
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)
|
|
{
|
|
}
|
|
|
|
}
|