1
0
mirror of https://gitlab.com/obbart/universal_robots_ros_driver.git synced 2026-04-12 11:00:47 +02:00

Major refactor

This commit is contained in:
Simon Rasmussen
2017-04-27 06:40:03 +02:00
parent 46f4e493cf
commit c59bfc78cc
22 changed files with 825 additions and 423 deletions

View File

@@ -6,139 +6,38 @@
#include "ur_modern_driver/log.h"
#include "ur_modern_driver/ur/stream.h"
bool URStream::connect()
bool URStream::write(const uint8_t* buf, size_t buf_len, size_t &written)
{
if (initialized_)
return false;
LOG_INFO("Connecting to UR @ %s:%d", host_.c_str(), port_);
// gethostbyname() is deprecated so use getadderinfo() as described in:
// http://www.beej.us/guide/bgnet/output/html/multipage/syscalls.html#getaddrinfo
std::string service = std::to_string(port_);
struct addrinfo hints, *result;
std::memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
hints.ai_flags = AI_PASSIVE;
if (getaddrinfo(host_.c_str(), service.c_str(), &hints, &result) != 0)
{
LOG_ERROR("Failed to get host name");
return false;
}
// loop through the list of addresses untill we find one that's connectable
for (struct addrinfo* p = result; p != nullptr; p = p->ai_next)
{
socket_fd_ = socket(p->ai_family, p->ai_socktype, p->ai_protocol);
if (socket_fd_ == -1) // socket error?
continue;
if (::connect(socket_fd_, p->ai_addr, p->ai_addrlen) != 0)
{
if (stopping_)
break;
else
continue; // try next addrinfo if connect fails
}
// disable Nagle's algorithm to ensure we sent packets as fast as possible
int flag = 1;
setsockopt(socket_fd_, IPPROTO_TCP, TCP_NODELAY, &flag, sizeof(flag));
initialized_ = true;
LOG_INFO("Connection successfully established");
break;
}
freeaddrinfo(result);
if (!initialized_)
LOG_ERROR("Connection failed");
return initialized_;
std::lock_guard<std::mutex> lock(write_mutex_);
return TCPSocket::write(buf, buf_len, written);
}
void URStream::disconnect()
bool URStream::read(uint8_t* buf, size_t buf_len, size_t &total)
{
if (!initialized_ || stopping_)
return;
std::lock_guard<std::mutex> lock(read_mutex_);
LOG_INFO("Disconnecting from %s:%d", host_.c_str(), port_);
stopping_ = true;
close(socket_fd_);
initialized_ = false;
}
void URStream::reconnect()
{
disconnect();
stopping_ = false;
connect();
}
ssize_t URStream::send(const uint8_t* buf, size_t buf_len)
{
if (!initialized_)
return -1;
if (stopping_)
return 0;
std::lock_guard<std::mutex> lock(send_mutex_);
size_t total = 0;
size_t remaining = buf_len;
// TODO: handle reconnect?
// handle partial sends
while (total < buf_len)
{
ssize_t sent = ::send(socket_fd_, buf + total, remaining, 0);
if (sent <= 0)
return stopping_ ? 0 : sent;
total += sent;
remaining -= sent;
}
return total;
}
ssize_t URStream::receive(uint8_t* buf, size_t buf_len)
{
if (!initialized_)
return -1;
if (stopping_)
return 0;
std::lock_guard<std::mutex> lock(receive_mutex_);
size_t remainder = sizeof(int32_t);
uint8_t* buf_pos = buf;
bool initial = true;
uint8_t* buf_pos = buf;
size_t remainder = sizeof(int32_t);
size_t read = 0;
do
while(remainder > 0 && TCPSocket::read(buf_pos, remainder, read))
{
ssize_t read = recv(socket_fd_, buf_pos, remainder, 0);
if (read <= 0) // failed reading from socket
return stopping_ ? 0 : read;
if (initial)
{
remainder = be32toh(*(reinterpret_cast<int32_t*>(buf)));
if (remainder >= (buf_len - sizeof(int32_t)))
{
LOG_ERROR("Packet size %zd is larger than buffer %zu, discarding.", remainder, buf_len);
return -1;
return false;
}
initial = false;
}
total += read;
buf_pos += read;
remainder -= read;
} while (remainder > 0);
return buf_pos - buf;
}
return remainder == 0;
}