mirror of https://github.com/ArduPilot/ardupilot
194 lines
4.2 KiB
C++
194 lines
4.2 KiB
C++
#include "UARTDevice.h"
|
|
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <poll.h>
|
|
#include <stdio.h>
|
|
#include <sys/ioctl.h>
|
|
#include <asm/ioctls.h>
|
|
#include <asm/termbits.h>
|
|
#include <unistd.h>
|
|
|
|
#include <AP_HAL/AP_HAL.h>
|
|
|
|
UARTDevice::UARTDevice(const char *device_path):
|
|
_device_path(device_path)
|
|
{
|
|
}
|
|
|
|
UARTDevice::~UARTDevice()
|
|
{
|
|
}
|
|
|
|
bool UARTDevice::close()
|
|
{
|
|
if (_fd != -1) {
|
|
if (::close(_fd) < 0) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
_fd = -1;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool UARTDevice::open()
|
|
{
|
|
_fd = ::open(_device_path, O_RDWR | O_CLOEXEC | O_NOCTTY);
|
|
|
|
if (_fd < 0) {
|
|
::fprintf(stderr, "Failed to open UART device %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return false;
|
|
}
|
|
|
|
_disable_crlf();
|
|
|
|
return true;
|
|
}
|
|
|
|
ssize_t UARTDevice::read(uint8_t *buf, uint16_t n)
|
|
{
|
|
return ::read(_fd, buf, n);
|
|
}
|
|
|
|
ssize_t UARTDevice::write(const uint8_t *buf, uint16_t n)
|
|
{
|
|
struct pollfd fds;
|
|
fds.fd = _fd;
|
|
fds.events = POLLOUT;
|
|
fds.revents = 0;
|
|
|
|
int ret = 0;
|
|
|
|
if (poll(&fds, 1, 0) == 1) {
|
|
ret = ::write(_fd, buf, n);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void UARTDevice::set_blocking(bool blocking)
|
|
{
|
|
int flags = fcntl(_fd, F_GETFL, 0);
|
|
|
|
if (blocking) {
|
|
flags = flags & ~O_NONBLOCK;
|
|
} else {
|
|
flags = flags | O_NONBLOCK;
|
|
}
|
|
|
|
if (fcntl(_fd, F_SETFL, flags) < 0) {
|
|
::fprintf(stderr, "Failed to make UART nonblocking %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
}
|
|
|
|
}
|
|
|
|
void UARTDevice::_disable_crlf()
|
|
{
|
|
struct termios2 t = { 0 };
|
|
|
|
if (ioctl(_fd, TCGETS2, &t) != 0) {
|
|
::fprintf(stderr, "Failed to read serial options for %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return;
|
|
}
|
|
|
|
// disable LF -> CR/LF
|
|
t.c_iflag &= ~(BRKINT | ICRNL | IMAXBEL | IXON | IXOFF);
|
|
t.c_oflag &= ~(OPOST | ONLCR);
|
|
t.c_lflag &= ~(ISIG | ICANON | IEXTEN | ECHO | ECHOE | ECHOK | ECHOCTL | ECHOKE);
|
|
t.c_cc[VMIN] = 0;
|
|
|
|
if (ioctl(_fd, TCSETS2, &t) != 0) {
|
|
::fprintf(stderr, "Failed to disable crlf on %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return;
|
|
}
|
|
}
|
|
|
|
void UARTDevice::set_speed(uint32_t baudrate)
|
|
{
|
|
struct termios2 tio = { 0 };
|
|
|
|
if (ioctl(_fd, TCGETS2, &tio) != 0) {
|
|
::fprintf(stderr, "Failed to read serial options for %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return;
|
|
}
|
|
|
|
// use CBAUDEX to gain access to "non-standard" rates that are common for eg. RC receivers
|
|
tio.c_cflag &= ~CBAUD;
|
|
tio.c_cflag |= CBAUDEX;
|
|
tio.c_ispeed = baudrate;
|
|
tio.c_ospeed = baudrate;
|
|
|
|
if (ioctl(_fd, TCSETS2, &tio) != 0) {
|
|
::fprintf(stderr, "Failed to set serial baud to %d for %s - %s\n",
|
|
baudrate, _device_path, strerror(errno));
|
|
return;
|
|
}
|
|
}
|
|
|
|
void UARTDevice::set_flow_control(AP_HAL::UARTDriver::flow_control flow_control_setting)
|
|
{
|
|
if (_flow_control == flow_control_setting) {
|
|
return;
|
|
}
|
|
|
|
struct termios2 t = { 0 };
|
|
|
|
if (ioctl(_fd, TCGETS2, &t) != 0) {
|
|
::fprintf(stderr, "Failed to read serial options for %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return;
|
|
}
|
|
|
|
if (flow_control_setting != AP_HAL::UARTDriver::FLOW_CONTROL_DISABLE) {
|
|
t.c_cflag |= CRTSCTS;
|
|
} else {
|
|
t.c_cflag &= ~CRTSCTS;
|
|
}
|
|
|
|
if (ioctl(_fd, TCSETS2, &t) != 0) {
|
|
::fprintf(stderr, "Failed to set flow control for %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return;
|
|
}
|
|
|
|
_flow_control = flow_control_setting;
|
|
}
|
|
|
|
void UARTDevice::set_parity(int v)
|
|
{
|
|
struct termios2 t = { 0 };
|
|
|
|
if (ioctl(_fd, TCGETS2, &t) != 0) {
|
|
::fprintf(stderr, "Failed to read serial options for %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return;
|
|
}
|
|
|
|
if (v != 0) {
|
|
// enable parity
|
|
t.c_cflag |= PARENB;
|
|
if (v == 1) {
|
|
t.c_cflag |= PARODD;
|
|
} else {
|
|
t.c_cflag &= ~PARODD;
|
|
}
|
|
}
|
|
else {
|
|
// disable parity
|
|
t.c_cflag &= ~PARENB;
|
|
}
|
|
|
|
if (ioctl(_fd, TCSETS2, &t) != 0) {
|
|
::fprintf(stderr, "Failed to set parity for %s - %s\n",
|
|
_device_path, strerror(errno));
|
|
return;
|
|
}
|
|
}
|