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:
@@ -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;
|
||||
}
|
||||
Reference in New Issue
Block a user