diff options
author | Matthias P. Braendli <matthias.braendli@mpb.li> | 2024-07-01 13:20:19 +0200 |
---|---|---|
committer | Matthias P. Braendli <matthias.braendli@mpb.li> | 2024-07-01 13:20:19 +0200 |
commit | dcbf473dd5a0abedd26a0a668d04820d398c1e37 (patch) | |
tree | 0125ae0c94ecd2a59fa86b9a4c0267a7671c603b | |
parent | 31faad73568712e9e77cfbd0bf50f3b7cc2ede00 (diff) | |
download | ODR-AudioEnc-dcbf473dd5a0abedd26a0a668d04820d398c1e37.tar.gz ODR-AudioEnc-dcbf473dd5a0abedd26a0a668d04820d398c1e37.tar.bz2 ODR-AudioEnc-dcbf473dd5a0abedd26a0a668d04820d398c1e37.zip |
Common: 710f5e6, Socket multicast reception
-rw-r--r-- | contrib/Socket.cpp | 38 | ||||
-rw-r--r-- | contrib/Socket.h | 1 |
2 files changed, 33 insertions, 6 deletions
diff --git a/contrib/Socket.cpp b/contrib/Socket.cpp index b71c01e..1281066 100644 --- a/contrib/Socket.cpp +++ b/contrib/Socket.cpp @@ -193,6 +193,34 @@ void UDPSocket::reinit(int port, const std::string& name) } } +void UDPSocket::init_receive_multicast(int port, const string& local_if_addr, const string& mcastaddr) +{ + if (m_sock != INVALID_SOCKET) { + ::close(m_sock); + } + + m_port = port; + m_sock = ::socket(AF_INET, SOCK_DGRAM, 0); + + int reuse_setting = 1; + if (setsockopt(m_sock, SOL_SOCKET, SO_REUSEADDR, &reuse_setting, sizeof(reuse_setting)) == -1) { + throw runtime_error("Can't reuse address"); + } + + struct sockaddr_in la; + memset((char *) &la, 0, sizeof(la)); + la.sin_family = AF_INET; + la.sin_port = htons(port); + la.sin_addr.s_addr = INADDR_ANY; + if (::bind(m_sock, (struct sockaddr*)&la, sizeof(la))) { + throw runtime_error(string("Could not bind: ") + strerror(errno)); + } + + joinGroup(mcastaddr.c_str(), local_if_addr.c_str()); + +} + + void UDPSocket::close() { if (m_sock != INVALID_SOCKET) { @@ -276,7 +304,7 @@ void UDPSocket::joinGroup(const char* groupname, const char* if_addr) throw runtime_error("Cannot convert multicast group name"); } if (!IN_MULTICAST(ntohl(group.imr_multiaddr.s_addr))) { - throw runtime_error("Group name is not a multicast address"); + throw runtime_error(string("Group name '") + groupname + "' is not a multicast address"); } if (if_addr) { @@ -309,7 +337,7 @@ void UDPSocket::setMulticastTTL(int ttl) { if (setsockopt(m_sock, IPPROTO_IP, IP_MULTICAST_TTL, &ttl, sizeof(ttl)) == SOCKET_ERROR) { - throw runtime_error(string("Can't set multicast ttl") + strerror(errno)); + throw runtime_error(string("Can't set multicast ttl: ") + strerror(errno)); } } @@ -327,15 +355,13 @@ void UDPReceiver::add_receive_port(int port, const string& bindto, const string& UDPSocket sock; if (IN_MULTICAST(ntohl(inet_addr(mcastaddr.c_str())))) { - sock.reinit(port, mcastaddr); - sock.setMulticastSource(bindto.c_str()); - sock.joinGroup(mcastaddr.c_str(), bindto.c_str()); + sock.init_receive_multicast(port, bindto, mcastaddr); } else { sock.reinit(port, bindto); } - m_sockets.push_back(move(sock)); + m_sockets.push_back(std::move(sock)); } vector<UDPReceiver::ReceivedPacket> UDPReceiver::receive(int timeout_ms) diff --git a/contrib/Socket.h b/contrib/Socket.h index d8242e2..44f93d0 100644 --- a/contrib/Socket.h +++ b/contrib/Socket.h @@ -111,6 +111,7 @@ class UDPSocket /** Close the already open socket, and create a new one. Throws a runtime_error on error. */ void reinit(int port); void reinit(int port, const std::string& name); + void init_receive_multicast(int port, const std::string& local_if_addr, const std::string& mcastaddr); void close(void); void send(UDPPacket& packet); |