/* 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 #include #include namespace ableton { namespace platforms { namespace LINK_ASIO_NAMESPACE { template struct Socket { Socket(::LINK_ASIO_NAMESPACE::io_service& io, ::LINK_ASIO_NAMESPACE::ip::udp protocol) : mpImpl(std::make_shared(io, protocol)) { } Socket(const Socket&) = delete; Socket& operator=(const Socket&) = delete; Socket(Socket&& rhs) : mpImpl(std::move(rhs.mpImpl)) { } std::size_t send( const uint8_t* const pData, const size_t numBytes, const discovery::UdpEndpoint& to) { assert(numBytes < MaxPacketSize); return mpImpl->mSocket.send_to(::LINK_ASIO_NAMESPACE::buffer(pData, numBytes), to); } template void receive(Handler handler) { mpImpl->mHandler = std::move(handler); mpImpl->mSocket.async_receive_from( ::LINK_ASIO_NAMESPACE::buffer(mpImpl->mReceiveBuffer, MaxPacketSize), mpImpl->mSenderEndpoint, util::makeAsyncSafe(mpImpl)); } discovery::UdpEndpoint endpoint() const { return mpImpl->mSocket.local_endpoint(); } struct Impl { Impl(::LINK_ASIO_NAMESPACE::io_service& io, ::LINK_ASIO_NAMESPACE::ip::udp protocol) : mSocket(io, protocol) { } ~Impl() { // Ignore error codes in shutdown and close as the socket may // have already been forcibly closed ::LINK_ASIO_NAMESPACE::error_code ec; mSocket.shutdown(::LINK_ASIO_NAMESPACE::ip::udp::socket::shutdown_both, ec); mSocket.close(ec); } void operator()( const ::LINK_ASIO_NAMESPACE::error_code& error, const std::size_t numBytes) { if (!error && numBytes > 0 && numBytes <= MaxPacketSize) { const auto bufBegin = begin(mReceiveBuffer); mHandler(mSenderEndpoint, bufBegin, bufBegin + static_cast(numBytes)); } } discovery::UdpSocket mSocket; discovery::UdpEndpoint mSenderEndpoint; using Buffer = std::array; Buffer mReceiveBuffer; using ByteIt = typename Buffer::const_iterator; std::function mHandler; }; std::shared_ptr mpImpl; }; } // namespace LINK_ASIO_NAMESPACE } // namespace platforms } // namespace ableton