You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							177 lines
						
					
					
						
							3.8 KiB
						
					
					
				
			
		
		
	
	
							177 lines
						
					
					
						
							3.8 KiB
						
					
					
				| #include <cassert>
 | |
| #include <cstring>
 | |
| #include <iostream>
 | |
| #include <cstdlib>
 | |
| #include <cerrno>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #include "msgq/impl_zmq.h"
 | |
| 
 | |
| static size_t fnv1a_hash(const std::string &str) {
 | |
|     const size_t fnv_prime = 0x100000001b3;
 | |
|     size_t hash_value = 0xcbf29ce484222325;
 | |
|     for (char c : str) {
 | |
|         hash_value ^= (unsigned char)c;
 | |
|         hash_value *= fnv_prime;
 | |
|     }
 | |
|     return hash_value;
 | |
| }
 | |
| 
 | |
| //FIXME: This is a hack to get the port number from the socket name, might have collisions
 | |
| static int get_port(std::string endpoint) {
 | |
|     size_t hash_value = fnv1a_hash(endpoint);
 | |
|     int start_port = 8023;
 | |
|     int max_port = 65535;
 | |
|     int port = start_port + (hash_value % (max_port - start_port));
 | |
|     return port;
 | |
| }
 | |
| 
 | |
| ZMQContext::ZMQContext() {
 | |
|   context = zmq_ctx_new();
 | |
| }
 | |
| 
 | |
| ZMQContext::~ZMQContext() {
 | |
|   zmq_ctx_term(context);
 | |
| }
 | |
| 
 | |
| void ZMQMessage::init(size_t sz) {
 | |
|   size = sz;
 | |
|   data = new char[size];
 | |
| }
 | |
| 
 | |
| void ZMQMessage::init(char * d, size_t sz) {
 | |
|   size = sz;
 | |
|   data = new char[size];
 | |
|   memcpy(data, d, size);
 | |
| }
 | |
| 
 | |
| void ZMQMessage::close() {
 | |
|   if (size > 0){
 | |
|     delete[] data;
 | |
|   }
 | |
|   size = 0;
 | |
| }
 | |
| 
 | |
| ZMQMessage::~ZMQMessage() {
 | |
|   this->close();
 | |
| }
 | |
| 
 | |
| 
 | |
| int ZMQSubSocket::connect(Context *context, std::string endpoint, std::string address, bool conflate, bool check_endpoint){
 | |
|   sock = zmq_socket(context->getRawContext(), ZMQ_SUB);
 | |
|   if (sock == NULL){
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   zmq_setsockopt(sock, ZMQ_SUBSCRIBE, "", 0);
 | |
| 
 | |
|   if (conflate){
 | |
|     int arg = 1;
 | |
|     zmq_setsockopt(sock, ZMQ_CONFLATE, &arg, sizeof(int));
 | |
|   }
 | |
| 
 | |
|   int reconnect_ivl = 500;
 | |
|   zmq_setsockopt(sock, ZMQ_RECONNECT_IVL_MAX, &reconnect_ivl, sizeof(reconnect_ivl));
 | |
| 
 | |
| 
 | |
|   full_endpoint = "tcp://" + address + ":";
 | |
|   if (check_endpoint){
 | |
|     full_endpoint += std::to_string(get_port(endpoint));
 | |
|   } else {
 | |
|     full_endpoint += endpoint;
 | |
|   }
 | |
| 
 | |
|   return zmq_connect(sock, full_endpoint.c_str());
 | |
| }
 | |
| 
 | |
| 
 | |
| Message * ZMQSubSocket::receive(bool non_blocking){
 | |
|   zmq_msg_t msg;
 | |
|   assert(zmq_msg_init(&msg) == 0);
 | |
| 
 | |
|   int flags = non_blocking ? ZMQ_DONTWAIT : 0;
 | |
|   int rc = zmq_msg_recv(&msg, sock, flags);
 | |
|   Message *r = NULL;
 | |
| 
 | |
|   if (rc >= 0){
 | |
|     // Make a copy to ensure the data is aligned
 | |
|     r = new ZMQMessage;
 | |
|     r->init((char*)zmq_msg_data(&msg), zmq_msg_size(&msg));
 | |
|   }
 | |
| 
 | |
|   zmq_msg_close(&msg);
 | |
|   return r;
 | |
| }
 | |
| 
 | |
| void ZMQSubSocket::setTimeout(int timeout){
 | |
|   zmq_setsockopt(sock, ZMQ_RCVTIMEO, &timeout, sizeof(int));
 | |
| }
 | |
| 
 | |
| ZMQSubSocket::~ZMQSubSocket(){
 | |
|   zmq_close(sock);
 | |
| }
 | |
| 
 | |
| int ZMQPubSocket::connect(Context *context, std::string endpoint, bool check_endpoint){
 | |
|   sock = zmq_socket(context->getRawContext(), ZMQ_PUB);
 | |
|   if (sock == NULL){
 | |
|     return -1;
 | |
|   }
 | |
| 
 | |
|   full_endpoint = "tcp://*:";
 | |
|   if (check_endpoint){
 | |
|     full_endpoint += std::to_string(get_port(endpoint));
 | |
|   } else {
 | |
|     full_endpoint += endpoint;
 | |
|   }
 | |
| 
 | |
|   // ZMQ pub sockets cannot be shared between processes, so we need to ensure pid stays the same
 | |
|   pid = getpid();
 | |
| 
 | |
|   return zmq_bind(sock, full_endpoint.c_str());
 | |
| }
 | |
| 
 | |
| int ZMQPubSocket::sendMessage(Message *message) {
 | |
|   assert(pid == getpid());
 | |
|   return zmq_send(sock, message->getData(), message->getSize(), ZMQ_DONTWAIT);
 | |
| }
 | |
| 
 | |
| int ZMQPubSocket::send(char *data, size_t size) {
 | |
|   assert(pid == getpid());
 | |
|   return zmq_send(sock, data, size, ZMQ_DONTWAIT);
 | |
| }
 | |
| 
 | |
| bool ZMQPubSocket::all_readers_updated() {
 | |
|   assert(false); // TODO not implemented
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| ZMQPubSocket::~ZMQPubSocket(){
 | |
|   zmq_close(sock);
 | |
| }
 | |
| 
 | |
| 
 | |
| void ZMQPoller::registerSocket(SubSocket * socket){
 | |
|   assert(num_polls + 1 < MAX_POLLERS);
 | |
|   polls[num_polls].socket = socket->getRawSocket();
 | |
|   polls[num_polls].events = ZMQ_POLLIN;
 | |
| 
 | |
|   sockets.push_back(socket);
 | |
|   num_polls++;
 | |
| }
 | |
| 
 | |
| std::vector<SubSocket*> ZMQPoller::poll(int timeout){
 | |
|   std::vector<SubSocket*> r;
 | |
| 
 | |
|   int rc = zmq_poll(polls, num_polls, timeout);
 | |
|   if (rc < 0){
 | |
|     return r;
 | |
|   }
 | |
| 
 | |
|   for (size_t i = 0; i < num_polls; i++){
 | |
|     if (polls[i].revents){
 | |
|       r.push_back(sockets[i]);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return r;
 | |
| }
 | |
| 
 |