/* Copyright 2016, Ableton AG, Berlin. All rights reserved. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * If you would like to incorporate Link into a proprietary software application, * please contact . */ #pragma once #include #include namespace ableton { namespace discovery { inline UdpEndpoint multicastEndpointV4() { return {IpAddressV4::from_string("224.76.78.75"), 20808}; } inline UdpEndpoint multicastEndpointV6(uint64_t scopeId) { // This is a non-permanently-assigned link-local multicast address (RFC4291) return { ::LINK_ASIO_NAMESPACE::ip::make_address("ff12::8080%" + std::to_string(scopeId)), 20808}; } // Type tags for dispatching between unicast and multicast packets struct MulticastTag { }; struct UnicastTag { }; template class IpInterface { public: using Socket = typename util::Injected::type::template Socket; IpInterface(util::Injected io, const IpAddress& addr) : mIo(std::move(io)) , mMulticastReceiveSocket(mIo->template openMulticastSocket(addr)) , mSendSocket(mIo->template openUnicastSocket(addr)) { } IpInterface(const IpInterface&) = delete; IpInterface& operator=(const IpInterface&) = delete; IpInterface(IpInterface&& rhs) : mIo(std::move(rhs.mIo)) , mMulticastReceiveSocket(std::move(rhs.mMulticastReceiveSocket)) , mSendSocket(std::move(rhs.mSendSocket)) { } std::size_t send( const uint8_t* const pData, const size_t numBytes, const UdpEndpoint& to) { return mSendSocket.send(pData, numBytes, to); } template void receive(Handler handler, UnicastTag) { mSendSocket.receive(SocketReceiver{std::move(handler)}); } template void receive(Handler handler, MulticastTag) { mMulticastReceiveSocket.receive( SocketReceiver(std::move(handler))); } UdpEndpoint endpoint() const { return mSendSocket.endpoint(); } private: template struct SocketReceiver { SocketReceiver(Handler handler) : mHandler(std::move(handler)) { } template void operator()(const UdpEndpoint& from, const It messageBegin, const It messageEnd) { mHandler(Tag{}, from, messageBegin, messageEnd); } Handler mHandler; }; util::Injected mIo; Socket mMulticastReceiveSocket; Socket mSendSocket; }; template IpInterface makeIpInterface( util::Injected io, const IpAddress& addr) { return {std::move(io), addr}; } } // namespace discovery } // namespace ableton