1 /* 2 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. 3 * 4 * Use of this source code is governed by a BSD-style license 5 * that can be found in the LICENSE file in the root of the source 6 * tree. An additional intellectual property rights grant can be found 7 * in the file PATENTS. All contributing project authors may 8 * be found in the AUTHORS file in the root of the source tree. 9 */ 10 11 #ifndef WEBRTC_TEST_CHANNEL_TRANSPORT_UDP_TRANSPORT_H_ 12 #define WEBRTC_TEST_CHANNEL_TRANSPORT_UDP_TRANSPORT_H_ 13 14 #include "webrtc/common_types.h" 15 #include "webrtc/transport.h" 16 #include "webrtc/typedefs.h" 17 18 /* 19 * WARNING 20 * This code is not use in production/testing and might have security issues 21 * for example: http://code.google.com/p/webrtc/issues/detail?id=1028 22 * 23 */ 24 25 #define SS_MAXSIZE 128 26 #define SS_ALIGNSIZE (sizeof (uint64_t)) 27 #define SS_PAD1SIZE (SS_ALIGNSIZE - sizeof(int16_t)) 28 #define SS_PAD2SIZE (SS_MAXSIZE - (sizeof(int16_t) + SS_PAD1SIZE +\ 29 SS_ALIGNSIZE)) 30 31 // BSD requires use of HAVE_STRUCT_SOCKADDR_SA_LEN 32 namespace webrtc { 33 namespace test { 34 35 struct SocketAddressIn { 36 // sin_family should be either AF_INET (IPv4) or AF_INET6 (IPv6) 37 #ifdef HAVE_STRUCT_SOCKADDR_SA_LEN 38 int8_t sin_length; 39 int8_t sin_family; 40 #else 41 int16_t sin_family; 42 #endif 43 uint16_t sin_port; 44 uint32_t sin_addr; 45 int8_t sin_zero[8]; 46 }; 47 48 struct Version6InAddress { 49 union { 50 uint8_t _s6_u8[16]; 51 uint32_t _s6_u32[4]; 52 uint64_t _s6_u64[2]; 53 } Version6AddressUnion; 54 }; 55 56 struct SocketAddressInVersion6 { 57 // sin_family should be either AF_INET (IPv4) or AF_INET6 (IPv6) 58 #ifdef HAVE_STRUCT_SOCKADDR_SA_LEN 59 int8_t sin_length; 60 int8_t sin_family; 61 #else 62 int16_t sin_family; 63 #endif 64 // Transport layer port number. 65 uint16_t sin6_port; 66 // IPv6 traffic class and flow info or ip4 address. 67 uint32_t sin6_flowinfo; 68 // IPv6 address 69 struct Version6InAddress sin6_addr; 70 // Set of interfaces for a scope. 71 uint32_t sin6_scope_id; 72 }; 73 74 struct SocketAddressStorage { 75 // sin_family should be either AF_INET (IPv4) or AF_INET6 (IPv6) 76 #ifdef HAVE_STRUCT_SOCKADDR_SA_LEN 77 int8_t sin_length; 78 int8_t sin_family; 79 #else 80 int16_t sin_family; 81 #endif 82 int8_t __ss_pad1[SS_PAD1SIZE]; 83 uint64_t __ss_align; 84 int8_t __ss_pad2[SS_PAD2SIZE]; 85 }; 86 87 struct SocketAddress { 88 union { 89 struct SocketAddressIn _sockaddr_in; 90 struct SocketAddressInVersion6 _sockaddr_in6; 91 struct SocketAddressStorage _sockaddr_storage; 92 }; 93 }; 94 95 // Callback class that receives packets from UdpTransport. 96 class UdpTransportData { 97 public: 98 virtual ~UdpTransportData() {}; 99 100 virtual void IncomingRTPPacket(const int8_t* incomingRtpPacket, 101 const size_t rtpPacketLength, 102 const char* fromIP, 103 const uint16_t fromPort) = 0; 104 105 virtual void IncomingRTCPPacket(const int8_t* incomingRtcpPacket, 106 const size_t rtcpPacketLength, 107 const char* fromIP, 108 const uint16_t fromPort) = 0; 109 }; 110 111 class UdpTransport : public Transport { 112 public: 113 enum 114 { 115 kIpAddressVersion6Length = 64, 116 kIpAddressVersion4Length = 16 117 }; 118 enum ErrorCode 119 { 120 kNoSocketError = 0, 121 kFailedToBindPort = 1, 122 kIpAddressInvalid = 2, 123 kAddressInvalid = 3, 124 kSocketInvalid = 4, 125 kPortInvalid = 5, 126 kTosInvalid = 6, 127 kMulticastAddressInvalid = 7, 128 kQosError = 8, 129 kSocketAlreadyInitialized = 9, 130 kIpVersion6Error = 10, 131 FILTER_ERROR = 11, 132 kStartReceiveError = 12, 133 kStopReceiveError = 13, 134 kCannotFindLocalIp = 14, 135 kTosError = 16, 136 kNotInitialized = 17, 137 kPcpError = 18 138 }; 139 140 // Factory method. Constructor disabled. 141 static UdpTransport* Create(const int32_t id, uint8_t& numSocketThreads); 142 static void Destroy(UdpTransport* module); 143 144 // Prepares the class for sending RTP packets to ipAddr:rtpPort and RTCP 145 // packets to ipAddr:rtpPort+1 if rtcpPort is zero. Otherwise to 146 // ipAddr:rtcpPort. 147 virtual int32_t InitializeSendSockets(const char* ipAddr, 148 const uint16_t rtpPort, 149 const uint16_t rtcpPort = 0) = 0; 150 151 // Register packetCallback for receiving incoming packets. Set the local 152 // RTP port to rtpPort. Bind local IP address to ipAddr. If ipAddr is NULL 153 // bind to local IP ANY. Set the local rtcp port to rtcpPort or rtpPort + 1 154 // if rtcpPort is 0. 155 virtual int32_t InitializeReceiveSockets( 156 UdpTransportData* const packetCallback, 157 const uint16_t rtpPort, 158 const char* ipAddr = NULL, 159 const char* multicastIpAddr = NULL, 160 const uint16_t rtcpPort = 0) = 0; 161 162 // Set local RTP port to rtpPort and RTCP port to rtcpPort or rtpPort + 1 if 163 // rtcpPort is 0. These ports will be used for sending instead of the local 164 // ports set by InitializeReceiveSockets(..). 165 virtual int32_t InitializeSourcePorts(const uint16_t rtpPort, 166 const uint16_t rtcpPort = 0) = 0; 167 168 // Retrieve local ports used for sending if other than the ports specified 169 // by InitializeReceiveSockets(..). rtpPort is set to the RTP port. 170 // rtcpPort is set to the RTCP port. 171 virtual int32_t SourcePorts(uint16_t& rtpPort, 172 uint16_t& rtcpPort) const = 0; 173 174 // Set ipAddr to the IP address that is currently being listened on. rtpPort 175 // to the RTP port listened to. rtcpPort to the RTCP port listened on. 176 // multicastIpAddr to the multicast IP address group joined (the address 177 // is NULL terminated). 178 virtual int32_t ReceiveSocketInformation( 179 char ipAddr[kIpAddressVersion6Length], 180 uint16_t& rtpPort, 181 uint16_t& rtcpPort, 182 char multicastIpAddr[kIpAddressVersion6Length]) const = 0; 183 184 // Set ipAddr to the IP address being sent from. rtpPort to the local RTP 185 // port used for sending and rtcpPort to the local RTCP port used for 186 // sending. 187 virtual int32_t SendSocketInformation(char ipAddr[kIpAddressVersion6Length], 188 uint16_t& rtpPort, 189 uint16_t& rtcpPort) const = 0; 190 191 // Put the IP address, RTP port and RTCP port from the last received packet 192 // into ipAddr, rtpPort and rtcpPort respectively. 193 virtual int32_t RemoteSocketInformation( 194 char ipAddr[kIpAddressVersion6Length], 195 uint16_t& rtpPort, 196 uint16_t& rtcpPort) const = 0; 197 198 // Enable/disable quality of service if QoS is true or false respectively. 199 // Set the type of service to serviceType, max bitrate in kbit/s to 200 // maxBitrate and override DSCP if overrideDSCP is not 0. 201 // Note: Must be called both InitializeSendSockets() and 202 // InitializeReceiveSockets() has been called. 203 virtual int32_t SetQoS(const bool QoS, 204 const int32_t serviceType, 205 const uint32_t maxBitrate = 0, 206 const int32_t overrideDSCP = 0, 207 const bool audio = false) = 0; 208 209 // Set QoS to true if quality of service has been turned on. If QoS is true, 210 // also set serviceType to type of service and overrideDSCP to override 211 // DSCP. 212 virtual int32_t QoS(bool& QoS, 213 int32_t& serviceType, 214 int32_t& overrideDSCP) const = 0; 215 216 // Set type of service. 217 virtual int32_t SetToS(const int32_t DSCP, 218 const bool useSetSockOpt = false) = 0; 219 220 // Get type of service configuration. 221 virtual int32_t ToS(int32_t& DSCP, 222 bool& useSetSockOpt) const = 0; 223 224 // Set Priority Code Point (IEEE 802.1Q) 225 // Note: for Linux this function will set the priority for the socket, 226 // which then can be mapped to a PCP value with vconfig. 227 virtual int32_t SetPCP(const int32_t PCP) = 0; 228 229 // Get Priority Code Point 230 virtual int32_t PCP(int32_t& PCP) const = 0; 231 232 // Enable IPv6. 233 // Note: this API must be called before any call to 234 // InitializeReceiveSockets() or InitializeSendSockets(). It is not 235 // possible to go back to IPv4 (default) after this call. 236 virtual int32_t EnableIpV6() = 0; 237 238 // Return true if IPv6 has been enabled. 239 virtual bool IpV6Enabled() const = 0; 240 241 // Only allow packets received from filterIPAddress to be processed. 242 // Note: must be called after EnableIPv6(), if IPv6 is used. 243 virtual int32_t SetFilterIP( 244 const char filterIPAddress[kIpAddressVersion6Length]) = 0; 245 246 // Write the filter IP address (if any) to filterIPAddress. 247 virtual int32_t FilterIP( 248 char filterIPAddress[kIpAddressVersion6Length]) const = 0; 249 250 // Only allow RTP packets from rtpFilterPort and RTCP packets from 251 // rtcpFilterPort be processed. 252 // Note: must be called after EnableIPv6(), if IPv6 is used. 253 virtual int32_t SetFilterPorts(const uint16_t rtpFilterPort, 254 const uint16_t rtcpFilterPort) = 0; 255 256 // Set rtpFilterPort to the filter RTP port and rtcpFilterPort to the 257 // filter RTCP port (if filtering based on port is enabled). 258 virtual int32_t FilterPorts(uint16_t& rtpFilterPort, 259 uint16_t& rtcpFilterPort) const = 0; 260 261 // Set the number of buffers that the socket implementation may use for 262 // receiving packets to numberOfSocketBuffers. I.e. the number of packets 263 // that can be received in parallell. 264 // Note: this API only has effect on Windows. 265 virtual int32_t StartReceiving(const uint32_t numberOfSocketBuffers) = 0; 266 267 // Stop receive incoming packets. 268 virtual int32_t StopReceiving() = 0; 269 270 // Return true incoming packets are received. 271 virtual bool Receiving() const = 0; 272 273 // Return true if send sockets have been initialized. 274 virtual bool SendSocketsInitialized() const = 0; 275 276 // Return true if local ports for sending has been set. 277 virtual bool SourcePortsInitialized() const = 0; 278 279 // Return true if receive sockets have been initialized. 280 virtual bool ReceiveSocketsInitialized() const = 0; 281 282 // Send data with size length to ip:portnr. The same port as the set 283 // with InitializeSendSockets(..) is used if portnr is 0. The same IP 284 // address as set with InitializeSendSockets(..) is used if ip is NULL. 285 // If isRTCP is true the port used will be the RTCP port. 286 virtual int32_t SendRaw(const int8_t* data, 287 size_t length, 288 int32_t isRTCP, 289 uint16_t portnr = 0, 290 const char* ip = NULL) = 0; 291 292 // Send RTP data with size length to the address specified by to. 293 virtual int32_t SendRTPPacketTo(const int8_t* data, 294 size_t length, 295 const SocketAddress& to) = 0; 296 297 298 // Send RTCP data with size length to the address specified by to. 299 virtual int32_t SendRTCPPacketTo(const int8_t* data, 300 size_t length, 301 const SocketAddress& to) = 0; 302 303 // Send RTP data with size length to ip:rtpPort where ip is the ip set by 304 // the InitializeSendSockets(..) call. 305 virtual int32_t SendRTPPacketTo(const int8_t* data, 306 size_t length, 307 uint16_t rtpPort) = 0; 308 309 310 // Send RTCP data with size length to ip:rtcpPort where ip is the ip set by 311 // the InitializeSendSockets(..) call. 312 virtual int32_t SendRTCPPacketTo(const int8_t* data, 313 size_t length, 314 uint16_t rtcpPort) = 0; 315 316 // Set the IP address to which packets are sent to ipaddr. 317 virtual int32_t SetSendIP( 318 const char ipaddr[kIpAddressVersion6Length]) = 0; 319 320 // Set the send RTP and RTCP port to rtpPort and rtcpPort respectively. 321 virtual int32_t SetSendPorts(const uint16_t rtpPort, 322 const uint16_t rtcpPort = 0) = 0; 323 324 // Retreive the last registered error code. 325 virtual ErrorCode LastError() const = 0; 326 327 // Put the local IPv4 address in localIP. 328 // Note: this API is for IPv4 only. 329 static int32_t LocalHostAddress(uint32_t& localIP); 330 331 // Put the local IP6 address in localIP. 332 // Note: this API is for IPv6 only. 333 static int32_t LocalHostAddressIPV6(char localIP[16]); 334 335 // Return a copy of hostOrder (host order) in network order. 336 static uint16_t Htons(uint16_t hostOrder); 337 338 // Return a copy of hostOrder (host order) in network order. 339 static uint32_t Htonl(uint32_t hostOrder); 340 341 // Return IPv4 address in ip as 32 bit integer. 342 static uint32_t InetAddrIPV4(const char* ip); 343 344 // Convert the character string src into a network address structure in 345 // the af address family and put it in dst. 346 // Note: same functionality as inet_pton(..) 347 static int32_t InetPresentationToNumeric(int32_t af, 348 const char* src, 349 void* dst); 350 351 // Set ip and sourcePort according to address. As input parameter ipSize 352 // is the length of ip. As output parameter it's the number of characters 353 // written to ip (not counting the '\0' character). 354 // Note: this API is only implemented on Windows and Linux. 355 static int32_t IPAddress(const SocketAddress& address, 356 char* ip, 357 uint32_t& ipSize, 358 uint16_t& sourcePort); 359 360 // Set ip and sourcePort according to address. As input parameter ipSize 361 // is the length of ip. As output parameter it's the number of characters 362 // written to ip (not counting the '\0' character). 363 // Note: this API is only implemented on Windows and Linux. 364 // Additional note: this API caches the address of the last call to it. If 365 // address is likley to be the same for multiple calls it may be beneficial 366 // to call this API instead of IPAddress(). 367 virtual int32_t IPAddressCached(const SocketAddress& address, 368 char* ip, 369 uint32_t& ipSize, 370 uint16_t& sourcePort) = 0; 371 372 // Return true if ipaddr is a valid IP address. 373 // If ipV6 is false ipaddr is interpreted as an IPv4 address otherwise it 374 // is interptreted as IPv6. 375 static bool IsIpAddressValid(const char* ipaddr, const bool ipV6); 376 }; 377 378 } // namespace test 379 } // namespace webrtc 380 381 #endif // WEBRTC_TEST_CHANNEL_TRANSPORT_UDP_TRANSPORT_H_ 382