blob: 0bc8aff8f586bc04921ded593cd1a2862c092b4d [file] [log] [blame]
btolsch9ccfa782018-07-26 00:16:08 -07001// Copyright 2018 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Jordan Baylesad2d2cd2019-05-22 14:49:40 -07005#ifndef PLATFORM_API_UDP_SOCKET_H_
6#define PLATFORM_API_UDP_SOCKET_H_
btolsch9ccfa782018-07-26 00:16:08 -07007
Ryan Keane83050af2019-08-26 08:58:09 -07008#include <atomic>
Yuri Wiitala10dea9f2019-02-04 20:19:17 -08009#include <cstdint>
Ryan Keanedeb48b32019-06-28 16:24:40 -070010#include <functional>
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080011#include <memory>
Ryan Keane9222e042019-08-27 10:44:13 -070012#include <mutex>
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080013
Yuri Wiitala82edd202018-10-31 18:42:58 -070014#include "platform/api/network_interface.h"
Jordan Baylesa26582d2019-07-10 14:44:58 -070015#include "platform/base/error.h"
16#include "platform/base/ip_address.h"
Yuri Wiitala977b6ef2019-11-07 17:08:25 -080017#include "platform/base/udp_packet.h"
btolsch9ccfa782018-07-26 00:16:08 -070018
19namespace openscreen {
btolsch9ccfa782018-07-26 00:16:08 -070020
Ryan Keanea4dfaa12019-08-19 10:30:20 -070021class TaskRunner;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080022class UdpSocket;
btolsch9ccfa782018-07-26 00:16:08 -070023
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080024// An open UDP socket for sending/receiving datagrams to/from either specific
25// endpoints or over IP multicast.
26//
27// Usage: The socket is created and opened by calling the Create() method. This
28// returns a unique pointer that auto-closes/destroys the socket when it goes
29// out-of-scope.
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080030class UdpSocket {
31 public:
Ryan Keanea4dfaa12019-08-19 10:30:20 -070032 // Client for the UdpSocket class.
33 class Client {
34 public:
35 virtual ~Client() = default;
36
37 // Method called on socket configuration operations when an error occurs.
38 // These specific APIs are:
39 // UdpSocket::Bind()
40 // UdpSocket::SetMulticastOutboundInterface(...)
41 // UdpSocket::JoinMulticastGroup(...)
42 // UdpSocket::SetDscp(...)
43 virtual void OnError(UdpSocket* socket, Error error) = 0;
44
45 // Method called when an error occurs during a SendMessage call.
46 virtual void OnSendError(UdpSocket* socket, Error error) = 0;
47
48 // Method called when a packet is read.
49 virtual void OnRead(UdpSocket* socket, ErrorOr<UdpPacket> packet) = 0;
50 };
51
Ryan Keane8f1c9252019-05-06 11:03:38 -070052 // Constants used to specify how we want packets sent from this socket.
53 enum class DscpMode : uint8_t {
54 // Default value set by the system on creation of a new socket.
55 kUnspecified = 0x0,
56
57 // Mode for Audio only.
58 kAudioOnly = 0xb8,
59
60 // Mode for Audio + Video.
61 kAudioVideo = 0x88,
62
63 // Mode for low priority operations such as trace log data.
64 kLowPriority = 0x20
65 };
66
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080067 using Version = IPAddress::Version;
btolsch9ccfa782018-07-26 00:16:08 -070068
Yuri Wiitalade01e532019-08-06 18:12:02 -070069 // Creates a new, scoped UdpSocket within the IPv4 or IPv6 family.
70 // |local_endpoint| may be zero (see comments for Bind()). This method must be
Yuri Wiitalad697dfc2019-10-23 16:00:16 -070071 // defined in the platform-level implementation. All |client| methods called
72 // will be queued on the provided |task_runner|. For this reason, the provided
73 // TaskRunner and Client must exist for the duration of the created socket's
Ryan Keanea4dfaa12019-08-19 10:30:20 -070074 // lifetime.
Yuri Wiitala2b02e322019-12-03 16:59:40 -080075 static ErrorOr<std::unique_ptr<UdpSocket>> Create(
76 TaskRunner* task_runner,
77 Client* client,
78 const IPEndpoint& local_endpoint);
btolsch9ccfa782018-07-26 00:16:08 -070079
Yuri Wiitalad697dfc2019-10-23 16:00:16 -070080 virtual ~UdpSocket();
81
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080082 // Returns true if |socket| belongs to the IPv4/IPv6 address family.
Ryan Keanedeb48b32019-06-28 16:24:40 -070083 virtual bool IsIPv4() const = 0;
84 virtual bool IsIPv6() const = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080085
Yuri Wiitalade01e532019-08-06 18:12:02 -070086 // Returns the current local endpoint's address and port. Initially, this will
87 // be the same as the value that was passed into Create(). However, it can
88 // later change after certain operations, such as Bind(), are executed.
89 virtual IPEndpoint GetLocalEndpoint() const = 0;
90
91 // Binds to the address specified in the constructor. If the local endpoint's
92 // address is zero, the operating system will bind to all interfaces. If the
93 // local endpoint's port is zero, the operating system will automatically find
Yuri Wiitalad697dfc2019-10-23 16:00:16 -070094 // a free local port and bind to it. Future calls to GetLocalEndpoint() will
Yuri Wiitalade01e532019-08-06 18:12:02 -070095 // reflect the resolved port.
Ryan Keane83e68322019-08-23 14:54:38 -070096 virtual void Bind() = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080097
98 // Sets the device to use for outgoing multicast packets on the socket.
Ryan Keane7e74bdd2019-08-27 08:15:21 -070099 virtual void SetMulticastOutboundInterface(NetworkInterfaceIndex ifindex) = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800100
101 // Joins to the multicast group at the given address, using the specified
102 // interface.
Ryan Keane7e74bdd2019-08-27 08:15:21 -0700103 virtual void JoinMulticastGroup(const IPAddress& address,
104 NetworkInterfaceIndex ifindex) = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800105
Yuri Wiitalad697dfc2019-10-23 16:00:16 -0700106 // Sends a message. If the message is not sent, Client::OnSendError() will be
107 // called to indicate this. Error::Code::kAgain indicates the operation would
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800108 // block, which can be expected during normal operation.
Ryan Keane71cacc22019-08-21 13:01:47 -0700109 virtual void SendMessage(const void* data,
110 size_t length,
111 const IPEndpoint& dest) = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800112
Ryan Keane8f1c9252019-05-06 11:03:38 -0700113 // Sets the DSCP value to use for all messages sent from this socket.
Ryan Keane7e74bdd2019-08-27 08:15:21 -0700114 virtual void SetDscp(DscpMode state) = 0;
Ryan Keanedeb48b32019-06-28 16:24:40 -0700115
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800116 protected:
Yuri Wiitalad697dfc2019-10-23 16:00:16 -0700117 UdpSocket();
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800118};
btolsch9ccfa782018-07-26 00:16:08 -0700119
btolsch9ccfa782018-07-26 00:16:08 -0700120} // namespace openscreen
121
Jordan Baylesad2d2cd2019-05-22 14:49:40 -0700122#endif // PLATFORM_API_UDP_SOCKET_H_