mirror of
https://github.com/slsdetectorgroup/aare.git
synced 2025-06-07 05:10:39 +02:00
restructure zmq socker interface
zmq socket can now return vector of frames. it knows end of transmission with header.data == 0 it can also send vector of frames
This commit is contained in:
parent
cd46f59b99
commit
a2499c96d6
@ -58,6 +58,14 @@ class Frame {
|
|||||||
other.m_data = nullptr;
|
other.m_data = nullptr;
|
||||||
other.m_rows = other.m_cols = other.m_bitdepth = 0;
|
other.m_rows = other.m_cols = other.m_bitdepth = 0;
|
||||||
}
|
}
|
||||||
|
// copy constructor
|
||||||
|
Frame(const Frame &other) {
|
||||||
|
m_rows = other.rows();
|
||||||
|
m_cols = other.cols();
|
||||||
|
m_bitdepth = other.bitdepth();
|
||||||
|
m_data = new std::byte[m_rows * m_cols * m_bitdepth / 8];
|
||||||
|
std::memcpy(m_data, other.m_data, m_rows * m_cols * m_bitdepth / 8);
|
||||||
|
}
|
||||||
|
|
||||||
template <typename T> NDView<T> view() {
|
template <typename T> NDView<T> view() {
|
||||||
std::vector<ssize_t> shape = {m_rows, m_cols};
|
std::vector<ssize_t> shape = {m_rows, m_cols};
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <array>
|
#include <array>
|
||||||
#include <fmt/format.h>
|
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <string_view>
|
#include <string_view>
|
||||||
#include <sys/types.h>
|
|
||||||
#include <variant>
|
#include <variant>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
@ -31,8 +31,7 @@ template <> DetectorType StringTo(std::string name) {
|
|||||||
else if (name == "ChipTestBoard")
|
else if (name == "ChipTestBoard")
|
||||||
return DetectorType::ChipTestBoard;
|
return DetectorType::ChipTestBoard;
|
||||||
else {
|
else {
|
||||||
auto msg = fmt::format("Could not decode dector from: \"{}\"", name);
|
throw std::runtime_error("Could not decode dector from: \"" + name + "\"");
|
||||||
throw std::runtime_error(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -42,8 +41,7 @@ template <> TimingMode StringTo(std::string mode) {
|
|||||||
else if (mode == "trigger")
|
else if (mode == "trigger")
|
||||||
return TimingMode::Trigger;
|
return TimingMode::Trigger;
|
||||||
else {
|
else {
|
||||||
auto msg = fmt::format("Could not decode timing mode from: \"{}\"", mode);
|
throw std::runtime_error("Could not decode timing mode from: \"" + mode + "\"");
|
||||||
throw std::runtime_error(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,27 +1,33 @@
|
|||||||
#include "aare/ZmqSocketReceiver.hpp"
|
#include "aare/network_io/ZmqSocketReceiver.hpp"
|
||||||
|
#include "aare/network_io/defs.hpp"
|
||||||
|
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
#include <fmt/core.h>
|
#include <fmt/core.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
using namespace aare;
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
|
// aare::logger::set_verbosity(aare::logger::DEBUG);
|
||||||
std::string endpoint = "tcp://localhost:5555";
|
std::string endpoint = "tcp://localhost:5555";
|
||||||
aare::ZmqSocketReceiver socket(endpoint);
|
aare::ZmqSocketReceiver socket(endpoint);
|
||||||
socket.connect();
|
socket.connect();
|
||||||
char *data = new char[1024 * 1024 * 10];
|
|
||||||
aare::ZmqHeader header;
|
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
int rc = socket.receive(header, reinterpret_cast<std::byte *>(data));
|
std::vector<ZmqFrame> v = socket.receive_n();
|
||||||
aare::logger::info("Received bytes", rc, "Received header: ", header.to_string());
|
aare::logger::info("Received ", v.size(), " frames");
|
||||||
auto *data_int = reinterpret_cast<uint32_t *>(data);
|
aare::logger::info("acquisition:", v[0].header.acqIndex);
|
||||||
for (uint32_t i = 0; i < header.npixelsx; i++) {
|
aare::logger::info("Header size:", v[0].header.to_string().size());
|
||||||
for (uint32_t j = 0; j < header.npixelsy; j++) {
|
aare::logger::info("Frame size:", v[0].frame.size());
|
||||||
// verify that the sent data is correct
|
aare::logger::info("Header:", v[0].header.to_string());
|
||||||
assert(data_int[i * header.npixelsy + j] == i + j);
|
|
||||||
|
for (ZmqFrame zmq_frame : v) {
|
||||||
|
auto &[header, frame] = zmq_frame;
|
||||||
|
for (int i = 0; i < 1024; i++) {
|
||||||
|
for (int j = 0; j < 1024; j++) {
|
||||||
|
assert(*(uint32_t *)frame.get(i, j) == (uint32_t)i + j);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
aare::logger::info("Frame verified");
|
||||||
}
|
}
|
||||||
aare::logger::info("Frame verified");
|
|
||||||
}
|
}
|
||||||
delete[] data;
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
@ -1,13 +1,17 @@
|
|||||||
#include "aare/Frame.hpp"
|
#include "aare/Frame.hpp"
|
||||||
#include "aare/ZmqSocketSender.hpp"
|
#include "aare/network_io/ZmqHeader.hpp"
|
||||||
|
#include "aare/network_io/ZmqSocketSender.hpp"
|
||||||
|
#include "aare/network_io/defs.hpp"
|
||||||
#include "aare/utils/logger.hpp"
|
#include "aare/utils/logger.hpp"
|
||||||
|
|
||||||
|
#include <ctime> // std::time
|
||||||
#include <fmt/core.h>
|
#include <fmt/core.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <unistd.h> // sleep
|
#include <unistd.h> // sleep
|
||||||
using namespace aare;
|
using namespace aare;
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
|
std::srand(std::time(nullptr));
|
||||||
std::string endpoint = "tcp://*:5555";
|
std::string endpoint = "tcp://*:5555";
|
||||||
aare::ZmqSocketSender socket(endpoint);
|
aare::ZmqSocketSender socket(endpoint);
|
||||||
socket.bind();
|
socket.bind();
|
||||||
@ -23,13 +27,24 @@ int main() {
|
|||||||
header.imageSize = sizeof(uint32_t) * 1024 * 1024;
|
header.imageSize = sizeof(uint32_t) * 1024 * 1024;
|
||||||
header.dynamicRange = 32;
|
header.dynamicRange = 32;
|
||||||
|
|
||||||
int i = 0;
|
std::vector<ZmqFrame> zmq_frames;
|
||||||
while (true) {
|
// send two exact frames
|
||||||
aare::logger::info("Sending frame:", i++);
|
|
||||||
aare::logger::info("Header size:", sizeof(header.to_string()));
|
|
||||||
aare::logger::info("Frame size:", frame.size(), "\n");
|
|
||||||
|
|
||||||
int rc = socket.send(header, frame.data(), frame.size());
|
int acqid = 0;
|
||||||
|
while (true) {
|
||||||
|
zmq_frames.clear();
|
||||||
|
header.acqIndex = acqid++;
|
||||||
|
size_t n_frames = std::rand() % 10 + 1;
|
||||||
|
|
||||||
|
aare::logger::info("acquisition:", header.acqIndex);
|
||||||
|
aare::logger::info("Header size:", header.to_string().size());
|
||||||
|
aare::logger::info("Frame size:", frame.size());
|
||||||
|
aare::logger::info("Number of frames:", n_frames);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < n_frames; i++) {
|
||||||
|
zmq_frames.push_back({header, frame});
|
||||||
|
}
|
||||||
|
size_t rc = socket.send(zmq_frames);
|
||||||
aare::logger::info("Sent bytes", rc);
|
aare::logger::info("Sent bytes", rc);
|
||||||
sleep(1);
|
sleep(1);
|
||||||
}
|
}
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "aare/NumpyHelpers.hpp"
|
#include "aare/NumpyHelpers.hpp"
|
||||||
|
#include <iterator>
|
||||||
|
|
||||||
namespace aare {
|
namespace aare {
|
||||||
|
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
#include "aare/SubFile.hpp"
|
#include "aare/SubFile.hpp"
|
||||||
#include "aare/utils/logger.hpp"
|
#include "aare/utils/logger.hpp"
|
||||||
|
#include <cstring> // memcpy
|
||||||
|
#include <fmt/core.h>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
// #include <filesystem>
|
// #include <filesystem>
|
||||||
|
|
||||||
@ -79,7 +81,7 @@ template <typename DataType> size_t SubFile::read_impl_flip(std::byte *buffer) {
|
|||||||
auto src = &tmp[0];
|
auto src = &tmp[0];
|
||||||
|
|
||||||
for (int i = 0; i != this->m_rows; ++i) {
|
for (int i = 0; i != this->m_rows; ++i) {
|
||||||
memcpy(dst, src, row_size);
|
std::memcpy(dst, src, row_size);
|
||||||
dst -= row_size;
|
dst -= row_size;
|
||||||
src += row_size;
|
src += row_size;
|
||||||
}
|
}
|
||||||
|
@ -1,12 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
#include "ZmqHeader.hpp"
|
|
||||||
#include "ZmqSocket.hpp"
|
|
||||||
|
|
||||||
namespace aare {
|
|
||||||
class ZmqSocketSender : public ZmqSocket {
|
|
||||||
public:
|
|
||||||
ZmqSocketSender(const std::string &endpoint);
|
|
||||||
void bind();
|
|
||||||
size_t send(ZmqHeader &header, const std::byte *data, size_t size, bool serialize_header = false);
|
|
||||||
};
|
|
||||||
} // namespace aare
|
|
@ -1,4 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "aare/Frame.hpp"
|
||||||
#include "aare/utils/logger.hpp"
|
#include "aare/utils/logger.hpp"
|
||||||
|
|
||||||
#include "simdjson.h"
|
#include "simdjson.h"
|
||||||
#include <array>
|
#include <array>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
@ -135,4 +138,5 @@ struct ZmqHeader {
|
|||||||
// compare operator
|
// compare operator
|
||||||
bool operator==(const ZmqHeader &other) const;
|
bool operator==(const ZmqHeader &other) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace aare
|
} // namespace aare
|
@ -1,7 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <array>
|
|
||||||
#include <map>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
// Socket to receive data from a ZMQ publisher
|
// Socket to receive data from a ZMQ publisher
|
@ -1,11 +1,11 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "ZmqHeader.hpp"
|
#include "aare/Frame.hpp"
|
||||||
#include "ZmqSocket.hpp"
|
#include "aare/network_io/ZmqHeader.hpp"
|
||||||
|
#include "aare/network_io/ZmqSocket.hpp"
|
||||||
|
#include "aare/network_io/defs.hpp"
|
||||||
|
|
||||||
#include <array>
|
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <map>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
// Socket to receive data from a ZMQ publisher
|
// Socket to receive data from a ZMQ publisher
|
||||||
@ -20,7 +20,13 @@ class ZmqSocketReceiver : public ZmqSocket {
|
|||||||
public:
|
public:
|
||||||
ZmqSocketReceiver(const std::string &endpoint);
|
ZmqSocketReceiver(const std::string &endpoint);
|
||||||
void connect();
|
void connect();
|
||||||
size_t receive(ZmqHeader &header, std::byte *data, bool serialized_header = false);
|
ZmqFrame receive(ZmqFrame &zmq_frame);
|
||||||
|
std::vector<ZmqFrame> receive_n();
|
||||||
|
|
||||||
|
private:
|
||||||
|
int receive_data(std::byte *data, size_t size);
|
||||||
|
ZmqFrame receive_zmqframe();
|
||||||
|
ZmqHeader receive_header();
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace aare
|
} // namespace aare
|
16
network_io/include/aare/network_io/ZmqSocketSender.hpp
Normal file
16
network_io/include/aare/network_io/ZmqSocketSender.hpp
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "aare/Frame.hpp"
|
||||||
|
#include "aare/network_io/ZmqHeader.hpp"
|
||||||
|
#include "aare/network_io/ZmqSocket.hpp"
|
||||||
|
#include "aare/network_io/defs.hpp"
|
||||||
|
|
||||||
|
namespace aare {
|
||||||
|
class ZmqSocketSender : public ZmqSocket {
|
||||||
|
public:
|
||||||
|
ZmqSocketSender(const std::string &endpoint);
|
||||||
|
void bind();
|
||||||
|
size_t send(const ZmqHeader &header, const std::byte *data, size_t size);
|
||||||
|
size_t send(const ZmqFrame &zmq_frame);
|
||||||
|
size_t send(const std::vector<ZmqFrame> &zmq_frames);
|
||||||
|
};
|
||||||
|
} // namespace aare
|
34
network_io/include/aare/network_io/defs.hpp
Normal file
34
network_io/include/aare/network_io/defs.hpp
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "aare/Frame.hpp"
|
||||||
|
#include "aare/network_io/ZmqHeader.hpp"
|
||||||
|
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace aare {
|
||||||
|
/**
|
||||||
|
* @brief ZmqFrame structure
|
||||||
|
* wrapper class to contain a ZmqHeader and a Frame
|
||||||
|
*/
|
||||||
|
struct ZmqFrame {
|
||||||
|
ZmqHeader header;
|
||||||
|
Frame frame;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace network_io {
|
||||||
|
/**
|
||||||
|
* @brief NetworkError exception class
|
||||||
|
*/
|
||||||
|
class NetworkError : public std::runtime_error {
|
||||||
|
private:
|
||||||
|
const char *m_msg;
|
||||||
|
|
||||||
|
public:
|
||||||
|
NetworkError(const char *msg) : std::runtime_error(msg), m_msg(msg) {}
|
||||||
|
NetworkError(const std::string msg) : std::runtime_error(msg) { m_msg = strdup(msg.c_str()); }
|
||||||
|
virtual const char *what() const noexcept override { return m_msg; }
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace network_io
|
||||||
|
|
||||||
|
} // namespace aare
|
@ -1,5 +1,5 @@
|
|||||||
|
|
||||||
#include "aare/ZmqHeader.hpp"
|
#include "aare/network_io/ZmqHeader.hpp"
|
||||||
|
|
||||||
#include "simdjson.h"
|
#include "simdjson.h"
|
||||||
|
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
#include "aare/ZmqSocket.hpp"
|
#include "aare/network_io/ZmqSocket.hpp"
|
||||||
#include <fmt/core.h>
|
|
||||||
#include <zmq.h>
|
#include <zmq.h>
|
||||||
|
|
||||||
namespace aare {
|
namespace aare {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
#include "aare/ZmqSocketReceiver.hpp"
|
#include "aare/network_io/ZmqSocketReceiver.hpp"
|
||||||
#include "aare/utils/logger.hpp"
|
#include "aare/utils/logger.hpp"
|
||||||
|
|
||||||
#include <fmt/core.h>
|
#include <fmt/core.h>
|
||||||
@ -6,69 +6,109 @@
|
|||||||
|
|
||||||
namespace aare {
|
namespace aare {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Construct a new ZmqSocketReceiver object
|
||||||
|
*/
|
||||||
ZmqSocketReceiver::ZmqSocketReceiver(const std::string &endpoint) {
|
ZmqSocketReceiver::ZmqSocketReceiver(const std::string &endpoint) {
|
||||||
m_endpoint = endpoint;
|
m_endpoint = endpoint;
|
||||||
memset(m_header_buffer, 0, m_max_header_size);
|
memset(m_header_buffer, 0, m_max_header_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Connect to the given endpoint
|
||||||
|
* subscribe to a Zmq published
|
||||||
|
*/
|
||||||
void ZmqSocketReceiver::connect() {
|
void ZmqSocketReceiver::connect() {
|
||||||
m_context = zmq_ctx_new();
|
m_context = zmq_ctx_new();
|
||||||
m_socket = zmq_socket(m_context, ZMQ_SUB);
|
m_socket = zmq_socket(m_context, ZMQ_SUB);
|
||||||
fmt::print("Setting ZMQ_RCVHWM to {}\n", m_zmq_hwm);
|
fmt::print("Setting ZMQ_RCVHWM to {}\n", m_zmq_hwm);
|
||||||
int rc = zmq_setsockopt(m_socket, ZMQ_RCVHWM, &m_zmq_hwm, sizeof(m_zmq_hwm)); // should be set before connect
|
int rc = zmq_setsockopt(m_socket, ZMQ_RCVHWM, &m_zmq_hwm, sizeof(m_zmq_hwm)); // should be set before connect
|
||||||
if (rc)
|
if (rc)
|
||||||
throw std::runtime_error(fmt::format("Could not set ZMQ_RCVHWM: {}", strerror(errno)));
|
throw network_io::NetworkError(fmt::format("Could not set ZMQ_RCVHWM: {}", strerror(errno)));
|
||||||
|
|
||||||
int bufsize = m_potential_frame_size * m_zmq_hwm;
|
int bufsize = m_potential_frame_size * m_zmq_hwm;
|
||||||
fmt::print("Setting ZMQ_RCVBUF to: {} MB\n", bufsize / (1024 * 1024));
|
fmt::print("Setting ZMQ_RCVBUF to: {} MB\n", bufsize / (1024 * 1024));
|
||||||
rc = zmq_setsockopt(m_socket, ZMQ_RCVBUF, &bufsize, sizeof(bufsize));
|
rc = zmq_setsockopt(m_socket, ZMQ_RCVBUF, &bufsize, sizeof(bufsize));
|
||||||
if (rc)
|
if (rc)
|
||||||
throw std::runtime_error(fmt::format("Could not set ZMQ_RCVBUF: {}", strerror(errno)));
|
throw network_io::NetworkError(fmt::format("Could not set ZMQ_RCVBUF: {}", strerror(errno)));
|
||||||
|
|
||||||
zmq_connect(m_socket, m_endpoint.c_str());
|
zmq_connect(m_socket, m_endpoint.c_str());
|
||||||
zmq_setsockopt(m_socket, ZMQ_SUBSCRIBE, "", 0);
|
zmq_setsockopt(m_socket, ZMQ_SUBSCRIBE, "", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t ZmqSocketReceiver::receive(ZmqHeader &header, std::byte *data, bool serialized_header) {
|
/**
|
||||||
|
* @brief receive a ZmqHeader
|
||||||
size_t data_bytes_received{};
|
* @return ZmqHeader
|
||||||
|
*/
|
||||||
if (serialized_header)
|
ZmqHeader ZmqSocketReceiver::receive_header() {
|
||||||
throw std::runtime_error("Not implemented");
|
|
||||||
|
|
||||||
|
// receive string ZmqHeader
|
||||||
size_t header_bytes_received = zmq_recv(m_socket, m_header_buffer, m_max_header_size, 0);
|
size_t header_bytes_received = zmq_recv(m_socket, m_header_buffer, m_max_header_size, 0);
|
||||||
|
|
||||||
// receive header
|
|
||||||
m_header_buffer[header_bytes_received] = '\0'; // make sure we zero terminate
|
m_header_buffer[header_bytes_received] = '\0'; // make sure we zero terminate
|
||||||
if (header_bytes_received < 0) {
|
if (header_bytes_received < 0) {
|
||||||
fmt::print("Error receiving header: {}\n", strerror(errno));
|
throw network_io::NetworkError(LOCATION + "Error receiving header");
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
aare::logger::debug("Bytes: ", header_bytes_received, ", Header: ", m_header_buffer);
|
aare::logger::debug("Bytes: ", header_bytes_received, ", Header: ", m_header_buffer);
|
||||||
|
|
||||||
// parse header
|
// parse header
|
||||||
|
ZmqHeader header;
|
||||||
try {
|
try {
|
||||||
std::string header_str(m_header_buffer);
|
std::string header_str(m_header_buffer);
|
||||||
header.from_string(header_str);
|
header.from_string(header_str);
|
||||||
} catch (const simdjson::simdjson_error &e) {
|
} catch (const simdjson::simdjson_error &e) {
|
||||||
aare::logger::error(LOCATION + "Error parsing header: ", e.what());
|
throw network_io::NetworkError(LOCATION + "Error parsing header: " + e.what());
|
||||||
return -1;
|
}
|
||||||
|
return header;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief receive data following a ZmqHeader
|
||||||
|
* @param data pointer to data
|
||||||
|
* @param size size of data
|
||||||
|
* @return ZmqHeader
|
||||||
|
*/
|
||||||
|
int ZmqSocketReceiver::receive_data(std::byte *data, size_t size) {
|
||||||
|
int data_bytes_received = zmq_recv(m_socket, data, size, 0);
|
||||||
|
if (data_bytes_received == -1)
|
||||||
|
network_io::NetworkError("Got half of a multipart msg!!!");
|
||||||
|
aare::logger::debug("Bytes: ", data_bytes_received);
|
||||||
|
|
||||||
|
return data_bytes_received;
|
||||||
|
}
|
||||||
|
|
||||||
|
ZmqFrame ZmqSocketReceiver::receive_zmqframe() {
|
||||||
|
// receive header from zmq and parse it
|
||||||
|
ZmqHeader header = receive_header();
|
||||||
|
if (!header.data) {
|
||||||
|
// no data following header
|
||||||
|
return {header, Frame(0, 0, 0)};
|
||||||
}
|
}
|
||||||
|
|
||||||
// do we have a multipart message (data following header)?
|
// receive frame data
|
||||||
int more;
|
Frame frame(header.npixelsx, header.npixelsy, header.dynamicRange);
|
||||||
size_t more_size = sizeof(more);
|
int bytes_received = receive_data(frame.data(), frame.size());
|
||||||
zmq_getsockopt(m_socket, ZMQ_RCVMORE, &more, &more_size);
|
if (bytes_received == -1) {
|
||||||
if (!more) {
|
throw network_io::NetworkError(LOCATION + "Error receiving frame");
|
||||||
return 0; // no data following header
|
|
||||||
} else {
|
|
||||||
|
|
||||||
data_bytes_received = zmq_recv(m_socket, data, header.imageSize, 0); // TODO! configurable size!!!!
|
|
||||||
if (data_bytes_received == -1)
|
|
||||||
throw std::runtime_error("Got half of a multipart msg!!!");
|
|
||||||
aare::logger::debug("Bytes: ", data_bytes_received);
|
|
||||||
}
|
}
|
||||||
return data_bytes_received + header_bytes_received;
|
if ((uint32_t)bytes_received != header.imageSize) {
|
||||||
|
throw network_io::NetworkError(
|
||||||
|
fmt::format("{} Expected {} bytes but received {}", LOCATION, header.imageSize, bytes_received));
|
||||||
|
}
|
||||||
|
return {header, std::move(frame)};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<ZmqFrame> ZmqSocketReceiver::receive_n() {
|
||||||
|
std::vector<ZmqFrame> frames;
|
||||||
|
while (true) {
|
||||||
|
// receive header and frame
|
||||||
|
ZmqFrame zmq_frame = receive_zmqframe();
|
||||||
|
if (!zmq_frame.header.data) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
frames.push_back(zmq_frame);
|
||||||
|
}
|
||||||
|
return frames;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace aare
|
} // namespace aare
|
||||||
|
@ -1,11 +1,18 @@
|
|||||||
#include "aare/ZmqSocketSender.hpp"
|
#include "aare/network_io/ZmqSocketSender.hpp"
|
||||||
|
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
#include <zmq.h>
|
#include <zmq.h>
|
||||||
|
|
||||||
namespace aare {
|
namespace aare {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor
|
||||||
|
* @param endpoint ZMQ endpoint
|
||||||
|
*/
|
||||||
ZmqSocketSender::ZmqSocketSender(const std::string &endpoint) { m_endpoint = endpoint; }
|
ZmqSocketSender::ZmqSocketSender(const std::string &endpoint) { m_endpoint = endpoint; }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* bind to the given port
|
||||||
|
*/
|
||||||
void ZmqSocketSender::bind() {
|
void ZmqSocketSender::bind() {
|
||||||
m_context = zmq_ctx_new();
|
m_context = zmq_ctx_new();
|
||||||
m_socket = zmq_socket(m_context, ZMQ_PUB);
|
m_socket = zmq_socket(m_context, ZMQ_PUB);
|
||||||
@ -13,15 +20,23 @@ void ZmqSocketSender::bind() {
|
|||||||
assert(rc == 0);
|
assert(rc == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t ZmqSocketSender::send(ZmqHeader &header, const std::byte *data, size_t size, bool serialize_header) {
|
/**
|
||||||
|
* send a header and data
|
||||||
|
* @param header
|
||||||
|
* @param data pointer to data
|
||||||
|
* @param size size of data
|
||||||
|
* @return number of bytes sent
|
||||||
|
*/
|
||||||
|
size_t ZmqSocketSender::send(const ZmqHeader &header, const std::byte *data, size_t size) {
|
||||||
size_t rc;
|
size_t rc;
|
||||||
if (serialize_header) {
|
// if (serialize_header) {
|
||||||
rc = zmq_send(m_socket, &header, sizeof(ZmqHeader), ZMQ_SNDMORE);
|
// rc = zmq_send(m_socket, &header, sizeof(ZmqHeader), ZMQ_SNDMORE);
|
||||||
assert(rc == sizeof(ZmqHeader));
|
// assert(rc == sizeof(ZmqHeader));
|
||||||
} else {
|
std::string header_str = header.to_string();
|
||||||
std::string header_str = header.to_string();
|
rc = zmq_send(m_socket, header_str.c_str(), header_str.size(), ZMQ_SNDMORE);
|
||||||
rc = zmq_send(m_socket, header_str.c_str(), header_str.size(), ZMQ_SNDMORE);
|
assert(rc == header_str.size());
|
||||||
assert(rc == header_str.size());
|
if (data == nullptr) {
|
||||||
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t rc2 = zmq_send(m_socket, data, size, 0);
|
size_t rc2 = zmq_send(m_socket, data, size, 0);
|
||||||
@ -29,4 +44,42 @@ size_t ZmqSocketSender::send(ZmqHeader &header, const std::byte *data, size_t si
|
|||||||
return rc + rc2;
|
return rc + rc2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a frame with a header
|
||||||
|
* @param ZmqFrame that contains a header and a frame
|
||||||
|
* @return number of bytes sent
|
||||||
|
*/
|
||||||
|
size_t ZmqSocketSender::send(const ZmqFrame &zmq_frame) {
|
||||||
|
const Frame &frame = zmq_frame.frame;
|
||||||
|
// send frame
|
||||||
|
size_t rc = send(zmq_frame.header, frame.data(), frame.size());
|
||||||
|
// send end of message header
|
||||||
|
ZmqHeader end_header = zmq_frame.header;
|
||||||
|
end_header.data = false;
|
||||||
|
size_t rc2 = send(end_header, nullptr, 0);
|
||||||
|
|
||||||
|
return rc + rc2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a vector of headers and frames
|
||||||
|
* @param zmq_frames vector of ZmqFrame
|
||||||
|
* @return number of bytes sent
|
||||||
|
*/
|
||||||
|
size_t ZmqSocketSender::send(const std::vector<ZmqFrame> &zmq_frames) {
|
||||||
|
size_t rc = 0;
|
||||||
|
for (size_t i = 0; i < zmq_frames.size(); i++) {
|
||||||
|
const ZmqHeader &header = zmq_frames[i].header;
|
||||||
|
const Frame &frame = zmq_frames[i].frame;
|
||||||
|
// send header and frame
|
||||||
|
if (i < zmq_frames.size() - 1) {
|
||||||
|
// send header and frame
|
||||||
|
rc += send(header, frame.data(), frame.size());
|
||||||
|
} else {
|
||||||
|
// send header, frame and end of message header
|
||||||
|
rc += send({header, frame});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
} // namespace aare
|
} // namespace aare
|
@ -1,4 +1,4 @@
|
|||||||
#include "aare/ZmqHeader.hpp"
|
#include "aare/network_io/ZmqHeader.hpp"
|
||||||
#include "aare/utils/logger.hpp"
|
#include "aare/utils/logger.hpp"
|
||||||
#include <catch2/catch_test_macros.hpp>
|
#include <catch2/catch_test_macros.hpp>
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user