blob: e668db12a504493cc9a612618182a8a702704d3b [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
mark a. foltz39dda172020-02-05 16:44:03 -08008#include <stddef.h> // size_t
9#include <stdint.h> // uint8_t
10
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080011#include <memory>
12
Yuri Wiitala82edd202018-10-31 18:42:58 -070013#include "platform/api/network_interface.h"
Jordan Baylesa26582d2019-07-10 14:44:58 -070014#include "platform/base/error.h"
15#include "platform/base/ip_address.h"
Yuri Wiitala977b6ef2019-11-07 17:08:25 -080016#include "platform/base/udp_packet.h"
btolsch9ccfa782018-07-26 00:16:08 -070017
18namespace openscreen {
btolsch9ccfa782018-07-26 00:16:08 -070019
Ryan Keanea4dfaa12019-08-19 10:30:20 -070020class TaskRunner;
btolsch9ccfa782018-07-26 00:16:08 -070021
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080022// An open UDP socket for sending/receiving datagrams to/from either specific
23// endpoints or over IP multicast.
24//
25// Usage: The socket is created and opened by calling the Create() method. This
26// returns a unique pointer that auto-closes/destroys the socket when it goes
27// out-of-scope.
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080028class UdpSocket {
29 public:
Ryan Keanea4dfaa12019-08-19 10:30:20 -070030 // Client for the UdpSocket class.
31 class Client {
32 public:
33 virtual ~Client() = default;
34
35 // Method called on socket configuration operations when an error occurs.
36 // These specific APIs are:
37 // UdpSocket::Bind()
38 // UdpSocket::SetMulticastOutboundInterface(...)
39 // UdpSocket::JoinMulticastGroup(...)
40 // UdpSocket::SetDscp(...)
41 virtual void OnError(UdpSocket* socket, Error error) = 0;
42
43 // Method called when an error occurs during a SendMessage call.
44 virtual void OnSendError(UdpSocket* socket, Error error) = 0;
45
46 // Method called when a packet is read.
47 virtual void OnRead(UdpSocket* socket, ErrorOr<UdpPacket> packet) = 0;
48 };
49
Ryan Keane8f1c9252019-05-06 11:03:38 -070050 // Constants used to specify how we want packets sent from this socket.
51 enum class DscpMode : uint8_t {
52 // Default value set by the system on creation of a new socket.
53 kUnspecified = 0x0,
54
55 // Mode for Audio only.
56 kAudioOnly = 0xb8,
57
58 // Mode for Audio + Video.
59 kAudioVideo = 0x88,
60
61 // Mode for low priority operations such as trace log data.
62 kLowPriority = 0x20
63 };
64
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080065 using Version = IPAddress::Version;
btolsch9ccfa782018-07-26 00:16:08 -070066
Yuri Wiitalade01e532019-08-06 18:12:02 -070067 // Creates a new, scoped UdpSocket within the IPv4 or IPv6 family.
68 // |local_endpoint| may be zero (see comments for Bind()). This method must be
Yuri Wiitalad697dfc2019-10-23 16:00:16 -070069 // defined in the platform-level implementation. All |client| methods called
70 // will be queued on the provided |task_runner|. For this reason, the provided
71 // TaskRunner and Client must exist for the duration of the created socket's
Ryan Keanea4dfaa12019-08-19 10:30:20 -070072 // lifetime.
Yuri Wiitala2b02e322019-12-03 16:59:40 -080073 static ErrorOr<std::unique_ptr<UdpSocket>> Create(
74 TaskRunner* task_runner,
75 Client* client,
76 const IPEndpoint& local_endpoint);
btolsch9ccfa782018-07-26 00:16:08 -070077
Yuri Wiitalad697dfc2019-10-23 16:00:16 -070078 virtual ~UdpSocket();
79
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080080 // Returns true if |socket| belongs to the IPv4/IPv6 address family.
Ryan Keanedeb48b32019-06-28 16:24:40 -070081 virtual bool IsIPv4() const = 0;
82 virtual bool IsIPv6() const = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080083
Yuri Wiitalade01e532019-08-06 18:12:02 -070084 // Returns the current local endpoint's address and port. Initially, this will
85 // be the same as the value that was passed into Create(). However, it can
86 // later change after certain operations, such as Bind(), are executed.
87 virtual IPEndpoint GetLocalEndpoint() const = 0;
88
89 // Binds to the address specified in the constructor. If the local endpoint's
90 // address is zero, the operating system will bind to all interfaces. If the
91 // local endpoint's port is zero, the operating system will automatically find
Yuri Wiitalad697dfc2019-10-23 16:00:16 -070092 // a free local port and bind to it. Future calls to GetLocalEndpoint() will
Yuri Wiitalade01e532019-08-06 18:12:02 -070093 // reflect the resolved port.
Ryan Keane83e68322019-08-23 14:54:38 -070094 virtual void Bind() = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080095
96 // Sets the device to use for outgoing multicast packets on the socket.
Ryan Keane7e74bdd2019-08-27 08:15:21 -070097 virtual void SetMulticastOutboundInterface(NetworkInterfaceIndex ifindex) = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -080098
99 // Joins to the multicast group at the given address, using the specified
100 // interface.
Ryan Keane7e74bdd2019-08-27 08:15:21 -0700101 virtual void JoinMulticastGroup(const IPAddress& address,
102 NetworkInterfaceIndex ifindex) = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800103
Yuri Wiitalad697dfc2019-10-23 16:00:16 -0700104 // Sends a message. If the message is not sent, Client::OnSendError() will be
105 // called to indicate this. Error::Code::kAgain indicates the operation would
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800106 // block, which can be expected during normal operation.
Ryan Keane71cacc22019-08-21 13:01:47 -0700107 virtual void SendMessage(const void* data,
108 size_t length,
109 const IPEndpoint& dest) = 0;
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800110
Ryan Keane8f1c9252019-05-06 11:03:38 -0700111 // Sets the DSCP value to use for all messages sent from this socket.
Ryan Keane7e74bdd2019-08-27 08:15:21 -0700112 virtual void SetDscp(DscpMode state) = 0;
Ryan Keanedeb48b32019-06-28 16:24:40 -0700113
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800114 protected:
Yuri Wiitalad697dfc2019-10-23 16:00:16 -0700115 UdpSocket();
Yuri Wiitala10dea9f2019-02-04 20:19:17 -0800116};
btolsch9ccfa782018-07-26 00:16:08 -0700117
btolsch9ccfa782018-07-26 00:16:08 -0700118} // namespace openscreen
119
Jordan Baylesad2d2cd2019-05-22 14:49:40 -0700120#endif // PLATFORM_API_UDP_SOCKET_H_