2007-02-04 17:05:44 +00:00
|
|
|
/*
|
2009-01-19 15:46:40 +00:00
|
|
|
* Copyright (c) 2007 The FFmpeg Project
|
2007-02-04 17:05:44 +00:00
|
|
|
*
|
|
|
|
* This file is part of FFmpeg.
|
|
|
|
*
|
|
|
|
* FFmpeg is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* FFmpeg 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with FFmpeg; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
2008-08-31 07:39:47 +00:00
|
|
|
#ifndef AVFORMAT_NETWORK_H
|
|
|
|
#define AVFORMAT_NETWORK_H
|
2007-02-04 17:05:44 +00:00
|
|
|
|
2011-06-04 14:58:31 +00:00
|
|
|
#include <errno.h>
|
2012-10-13 21:57:23 +00:00
|
|
|
#include <stdint.h>
|
2011-06-04 14:58:31 +00:00
|
|
|
|
2009-01-24 14:52:46 +00:00
|
|
|
#include "config.h"
|
2011-06-04 14:58:31 +00:00
|
|
|
#include "libavutil/error.h"
|
2011-04-07 13:09:03 +00:00
|
|
|
#include "os_support.h"
|
2012-10-09 12:26:02 +00:00
|
|
|
#include "avio.h"
|
2013-05-31 01:05:13 +00:00
|
|
|
#include "url.h"
|
2009-01-24 14:52:46 +00:00
|
|
|
|
2012-06-24 21:39:57 +00:00
|
|
|
#if HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2009-01-13 23:44:16 +00:00
|
|
|
#if HAVE_WINSOCK2_H
|
2007-05-15 14:58:30 +00:00
|
|
|
#include <winsock2.h>
|
|
|
|
#include <ws2tcpip.h>
|
|
|
|
|
2012-06-23 12:00:17 +00:00
|
|
|
#ifndef EPROTONOSUPPORT
|
2011-02-19 18:14:11 +00:00
|
|
|
#define EPROTONOSUPPORT WSAEPROTONOSUPPORT
|
2012-01-29 09:45:22 +00:00
|
|
|
#endif
|
2012-06-23 12:00:17 +00:00
|
|
|
#ifndef ETIMEDOUT
|
2011-02-19 18:14:11 +00:00
|
|
|
#define ETIMEDOUT WSAETIMEDOUT
|
2012-01-29 09:45:22 +00:00
|
|
|
#endif
|
2012-06-23 12:00:17 +00:00
|
|
|
#ifndef ECONNREFUSED
|
2011-02-19 18:14:11 +00:00
|
|
|
#define ECONNREFUSED WSAECONNREFUSED
|
2012-01-29 09:45:22 +00:00
|
|
|
#endif
|
2012-06-23 12:00:17 +00:00
|
|
|
#ifndef EINPROGRESS
|
2011-02-19 18:14:11 +00:00
|
|
|
#define EINPROGRESS WSAEINPROGRESS
|
2012-06-23 12:00:17 +00:00
|
|
|
#endif
|
2019-12-11 12:18:43 +00:00
|
|
|
#ifndef ENOTCONN
|
|
|
|
#define ENOTCONN WSAENOTCONN
|
|
|
|
#endif
|
2012-06-23 12:00:17 +00:00
|
|
|
|
2012-06-18 20:39:30 +00:00
|
|
|
#define getsockopt(a, b, c, d, e) getsockopt(a, b, c, (char*) d, e)
|
|
|
|
#define setsockopt(a, b, c, d, e) setsockopt(a, b, c, (const char*) d, e)
|
2011-02-19 18:14:11 +00:00
|
|
|
|
2011-02-07 10:59:50 +00:00
|
|
|
int ff_neterrno(void);
|
2007-05-15 14:58:30 +00:00
|
|
|
#else
|
2007-02-04 17:05:44 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
2017-11-13 20:20:18 +00:00
|
|
|
#include <netinet/tcp.h>
|
2007-02-04 17:05:44 +00:00
|
|
|
#include <netdb.h>
|
|
|
|
|
2010-04-15 18:27:27 +00:00
|
|
|
#define ff_neterrno() AVERROR(errno)
|
2014-01-06 10:26:20 +00:00
|
|
|
#endif /* HAVE_WINSOCK2_H */
|
2007-05-15 14:58:30 +00:00
|
|
|
|
2009-01-13 23:44:16 +00:00
|
|
|
#if HAVE_ARPA_INET_H
|
2007-05-15 14:58:30 +00:00
|
|
|
#include <arpa/inet.h>
|
|
|
|
#endif
|
2007-04-27 00:35:54 +00:00
|
|
|
|
2011-04-04 16:17:12 +00:00
|
|
|
#if HAVE_POLL_H
|
|
|
|
#include <poll.h>
|
|
|
|
#endif
|
|
|
|
|
2007-04-27 00:41:50 +00:00
|
|
|
int ff_socket_nonblock(int socket, int enable);
|
|
|
|
|
2011-02-07 10:59:50 +00:00
|
|
|
int ff_network_init(void);
|
|
|
|
void ff_network_close(void);
|
|
|
|
|
2015-01-22 14:50:48 +00:00
|
|
|
int ff_tls_init(void);
|
2011-02-04 22:25:07 +00:00
|
|
|
void ff_tls_deinit(void);
|
|
|
|
|
2011-02-07 10:59:50 +00:00
|
|
|
int ff_network_wait_fd(int fd, int write);
|
2007-08-09 23:39:05 +00:00
|
|
|
|
2012-10-09 12:26:02 +00:00
|
|
|
/**
|
|
|
|
* This works similarly to ff_network_wait_fd, but waits up to 'timeout' microseconds
|
|
|
|
* Uses ff_network_wait_fd in a loop
|
|
|
|
*
|
2018-05-06 11:53:19 +00:00
|
|
|
* @param fd Socket descriptor
|
|
|
|
* @param write Set 1 to wait for socket able to be read, 0 to be written
|
|
|
|
* @param timeout Timeout interval, in microseconds. Actual precision is 100000 mcs, due to ff_network_wait_fd usage
|
2013-07-12 07:35:51 +00:00
|
|
|
* @param int_cb Interrupt callback, is checked before each ff_network_wait_fd call
|
2012-10-09 12:26:02 +00:00
|
|
|
* @return 0 if data can be read/written, AVERROR(ETIMEDOUT) if timeout expired, or negative error code
|
|
|
|
*/
|
|
|
|
int ff_network_wait_fd_timeout(int fd, int write, int64_t timeout, AVIOInterruptCB *int_cb);
|
|
|
|
|
2018-01-02 16:05:03 +00:00
|
|
|
/**
|
|
|
|
* Waits for up to 'timeout' microseconds. If the usert's int_cb is set and
|
|
|
|
* triggered, return before that.
|
2018-05-06 11:53:19 +00:00
|
|
|
* @param timeout Timeout in microseconds. Maybe have lower actual precision.
|
2018-01-02 16:05:03 +00:00
|
|
|
* @param int_cb Interrupt callback, is checked regularly.
|
|
|
|
* @return AVERROR(ETIMEDOUT) if timeout expirted, AVERROR_EXIT if interrupted by int_cb
|
|
|
|
*/
|
|
|
|
int ff_network_sleep_interruptible(int64_t timeout, AVIOInterruptCB *int_cb);
|
|
|
|
|
2010-01-11 17:42:35 +00:00
|
|
|
#if !HAVE_STRUCT_SOCKADDR_STORAGE
|
|
|
|
struct sockaddr_storage {
|
2010-01-20 17:26:14 +00:00
|
|
|
#if HAVE_STRUCT_SOCKADDR_SA_LEN
|
|
|
|
uint8_t ss_len;
|
|
|
|
uint8_t ss_family;
|
|
|
|
#else
|
|
|
|
uint16_t ss_family;
|
2014-01-06 10:26:20 +00:00
|
|
|
#endif /* HAVE_STRUCT_SOCKADDR_SA_LEN */
|
2010-01-20 17:26:14 +00:00
|
|
|
char ss_pad1[6];
|
|
|
|
int64_t ss_align;
|
|
|
|
char ss_pad2[112];
|
2010-01-11 17:42:35 +00:00
|
|
|
};
|
2014-01-06 10:26:20 +00:00
|
|
|
#endif /* !HAVE_STRUCT_SOCKADDR_STORAGE */
|
2010-01-11 17:42:35 +00:00
|
|
|
|
2014-09-15 23:00:49 +00:00
|
|
|
typedef union sockaddr_union {
|
|
|
|
struct sockaddr_storage storage;
|
|
|
|
struct sockaddr_in in;
|
|
|
|
#if HAVE_STRUCT_SOCKADDR_IN6
|
|
|
|
struct sockaddr_in6 in6;
|
|
|
|
#endif
|
|
|
|
} sockaddr_union;
|
|
|
|
|
2014-08-20 20:06:07 +00:00
|
|
|
#ifndef MSG_NOSIGNAL
|
|
|
|
#define MSG_NOSIGNAL 0
|
|
|
|
#endif
|
|
|
|
|
2010-01-11 17:27:07 +00:00
|
|
|
#if !HAVE_STRUCT_ADDRINFO
|
|
|
|
struct addrinfo {
|
|
|
|
int ai_flags;
|
|
|
|
int ai_family;
|
|
|
|
int ai_socktype;
|
|
|
|
int ai_protocol;
|
|
|
|
int ai_addrlen;
|
|
|
|
struct sockaddr *ai_addr;
|
|
|
|
char *ai_canonname;
|
|
|
|
struct addrinfo *ai_next;
|
|
|
|
};
|
2014-01-06 10:26:20 +00:00
|
|
|
#endif /* !HAVE_STRUCT_ADDRINFO */
|
2010-01-11 17:27:07 +00:00
|
|
|
|
|
|
|
/* getaddrinfo constants */
|
2012-06-25 09:44:18 +00:00
|
|
|
#ifndef EAI_AGAIN
|
|
|
|
#define EAI_AGAIN 2
|
|
|
|
#endif
|
|
|
|
#ifndef EAI_BADFLAGS
|
|
|
|
#define EAI_BADFLAGS 3
|
|
|
|
#endif
|
2010-01-11 17:27:07 +00:00
|
|
|
#ifndef EAI_FAIL
|
|
|
|
#define EAI_FAIL 4
|
|
|
|
#endif
|
2010-01-11 17:45:17 +00:00
|
|
|
#ifndef EAI_FAMILY
|
|
|
|
#define EAI_FAMILY 5
|
|
|
|
#endif
|
2012-06-25 09:44:18 +00:00
|
|
|
#ifndef EAI_MEMORY
|
|
|
|
#define EAI_MEMORY 6
|
|
|
|
#endif
|
|
|
|
#ifndef EAI_NODATA
|
|
|
|
#define EAI_NODATA 7
|
|
|
|
#endif
|
2010-01-11 17:45:17 +00:00
|
|
|
#ifndef EAI_NONAME
|
|
|
|
#define EAI_NONAME 8
|
|
|
|
#endif
|
2012-06-25 09:44:18 +00:00
|
|
|
#ifndef EAI_SERVICE
|
|
|
|
#define EAI_SERVICE 9
|
|
|
|
#endif
|
|
|
|
#ifndef EAI_SOCKTYPE
|
|
|
|
#define EAI_SOCKTYPE 10
|
|
|
|
#endif
|
2010-01-11 17:45:17 +00:00
|
|
|
|
2010-01-11 17:27:07 +00:00
|
|
|
#ifndef AI_PASSIVE
|
|
|
|
#define AI_PASSIVE 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AI_CANONNAME
|
|
|
|
#define AI_CANONNAME 2
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef AI_NUMERICHOST
|
|
|
|
#define AI_NUMERICHOST 4
|
|
|
|
#endif
|
|
|
|
|
2010-01-11 17:45:17 +00:00
|
|
|
#ifndef NI_NOFQDN
|
|
|
|
#define NI_NOFQDN 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef NI_NUMERICHOST
|
|
|
|
#define NI_NUMERICHOST 2
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef NI_NAMERQD
|
|
|
|
#define NI_NAMERQD 4
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef NI_NUMERICSERV
|
|
|
|
#define NI_NUMERICSERV 8
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef NI_DGRAM
|
|
|
|
#define NI_DGRAM 16
|
|
|
|
#endif
|
|
|
|
|
2010-01-11 17:27:07 +00:00
|
|
|
#if !HAVE_GETADDRINFO
|
|
|
|
int ff_getaddrinfo(const char *node, const char *service,
|
|
|
|
const struct addrinfo *hints, struct addrinfo **res);
|
|
|
|
void ff_freeaddrinfo(struct addrinfo *res);
|
2010-01-11 17:45:17 +00:00
|
|
|
int ff_getnameinfo(const struct sockaddr *sa, int salen,
|
|
|
|
char *host, int hostlen,
|
|
|
|
char *serv, int servlen, int flags);
|
2010-01-11 17:27:07 +00:00
|
|
|
#define getaddrinfo ff_getaddrinfo
|
|
|
|
#define freeaddrinfo ff_freeaddrinfo
|
2010-01-11 17:45:17 +00:00
|
|
|
#define getnameinfo ff_getnameinfo
|
2014-01-06 10:26:20 +00:00
|
|
|
#endif /* !HAVE_GETADDRINFO */
|
|
|
|
|
2012-06-25 09:50:13 +00:00
|
|
|
#if !HAVE_GETADDRINFO || HAVE_WINSOCK2_H
|
|
|
|
const char *ff_gai_strerror(int ecode);
|
2012-06-18 20:36:15 +00:00
|
|
|
#undef gai_strerror
|
2010-02-08 18:48:12 +00:00
|
|
|
#define gai_strerror ff_gai_strerror
|
2014-01-06 10:26:20 +00:00
|
|
|
#endif /* !HAVE_GETADDRINFO || HAVE_WINSOCK2_H */
|
2010-01-11 17:27:07 +00:00
|
|
|
|
2012-10-14 17:00:05 +00:00
|
|
|
#ifndef INADDR_LOOPBACK
|
|
|
|
#define INADDR_LOOPBACK 0x7f000001
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef INET_ADDRSTRLEN
|
|
|
|
#define INET_ADDRSTRLEN 16
|
|
|
|
#endif
|
|
|
|
|
2010-09-03 20:06:01 +00:00
|
|
|
#ifndef INET6_ADDRSTRLEN
|
|
|
|
#define INET6_ADDRSTRLEN INET_ADDRSTRLEN
|
|
|
|
#endif
|
|
|
|
|
2010-10-07 07:53:31 +00:00
|
|
|
#ifndef IN_MULTICAST
|
|
|
|
#define IN_MULTICAST(a) ((((uint32_t)(a)) & 0xf0000000) == 0xe0000000)
|
|
|
|
#endif
|
|
|
|
#ifndef IN6_IS_ADDR_MULTICAST
|
|
|
|
#define IN6_IS_ADDR_MULTICAST(a) (((uint8_t *) (a))[0] == 0xff)
|
|
|
|
#endif
|
|
|
|
|
2011-02-07 10:59:50 +00:00
|
|
|
int ff_is_multicast_address(struct sockaddr *addr);
|
2010-10-07 07:54:52 +00:00
|
|
|
|
2013-06-01 17:38:57 +00:00
|
|
|
#define POLLING_TIME 100 /// Time in milliseconds between interrupt check
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Bind to a file descriptor and poll for a connection.
|
|
|
|
*
|
|
|
|
* @param fd First argument of bind().
|
|
|
|
* @param addr Second argument of bind().
|
|
|
|
* @param addrlen Third argument of bind().
|
|
|
|
* @param timeout Polling timeout in milliseconds.
|
|
|
|
* @param h URLContext providing interrupt check
|
|
|
|
* callback and logging context.
|
|
|
|
* @return A non-blocking file descriptor on success
|
|
|
|
* or an AVERROR on failure.
|
|
|
|
*/
|
2013-05-29 23:08:51 +00:00
|
|
|
int ff_listen_bind(int fd, const struct sockaddr *addr,
|
2013-06-01 17:38:57 +00:00
|
|
|
socklen_t addrlen, int timeout,
|
|
|
|
URLContext *h);
|
|
|
|
|
2015-07-03 00:22:25 +00:00
|
|
|
/**
|
|
|
|
* Bind to a file descriptor to an address without accepting connections.
|
|
|
|
* @param fd First argument of bind().
|
|
|
|
* @param addr Second argument of bind().
|
|
|
|
* @param addrlen Third argument of bind().
|
|
|
|
* @return 0 on success or an AVERROR on failure.
|
|
|
|
*/
|
2022-01-10 16:34:04 +00:00
|
|
|
int ff_listen(int fd, const struct sockaddr *addr, socklen_t addrlen,
|
|
|
|
void *logctx);
|
2015-07-03 00:22:25 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Poll for a single connection on the passed file descriptor.
|
|
|
|
* @param fd The listening socket file descriptor.
|
|
|
|
* @param timeout Polling timeout in milliseconds.
|
|
|
|
* @param h URLContext providing interrupt check
|
|
|
|
* callback and logging context.
|
|
|
|
* @return A non-blocking file descriptor on success
|
|
|
|
* or an AVERROR on failure.
|
|
|
|
*/
|
|
|
|
int ff_accept(int fd, int timeout, URLContext *h);
|
|
|
|
|
2013-06-01 17:38:57 +00:00
|
|
|
/**
|
|
|
|
* Connect to a file descriptor and poll for result.
|
|
|
|
*
|
|
|
|
* @param fd First argument of connect(),
|
|
|
|
* will be set as non-blocking.
|
|
|
|
* @param addr Second argument of connect().
|
|
|
|
* @param addrlen Third argument of connect().
|
|
|
|
* @param timeout Polling timeout in milliseconds.
|
|
|
|
* @param h URLContext providing interrupt check
|
|
|
|
* callback and logging context.
|
2013-08-05 16:44:20 +00:00
|
|
|
* @param will_try_next Whether the caller will try to connect to another
|
|
|
|
* address for the same host name, affecting the form of
|
|
|
|
* logged errors.
|
2013-06-01 17:38:57 +00:00
|
|
|
* @return 0 on success, AVERROR on failure.
|
|
|
|
*/
|
2013-05-31 01:05:13 +00:00
|
|
|
int ff_listen_connect(int fd, const struct sockaddr *addr,
|
|
|
|
socklen_t addrlen, int timeout,
|
2013-08-05 16:44:20 +00:00
|
|
|
URLContext *h, int will_try_next);
|
2013-06-01 17:38:57 +00:00
|
|
|
|
2013-06-15 09:41:36 +00:00
|
|
|
int ff_http_match_no_proxy(const char *no_proxy, const char *hostname);
|
|
|
|
|
2022-01-10 16:34:04 +00:00
|
|
|
int ff_socket(int domain, int type, int protocol, void *logctx);
|
2013-08-03 13:00:11 +00:00
|
|
|
|
2018-08-04 09:48:15 +00:00
|
|
|
void ff_log_net_error(void *ctx, int level, const char* prefix);
|
|
|
|
|
network: Add RFC 8305 style "Happy Eyeballs"/"Fast Fallback" helper function
For cases with dual stack (IPv4 + IPv6) connectivity, but where one
stack potentially is less reliable, strive to trying to connect over
both protocols in parallel, using whichever address connected first.
In cases with a hostname resolving to multiple IPv4 and IPv6
addresses, the current connection mechanism would try all addresses
in the order returned by getaddrinfo (with all IPv6 addresses ordered
before the IPv4 addresses normally). If connection attempts to the
IPv6 addresses return quickly with an error, this was no problem, but
if they were unsuccessful leading up to timeouts, the connection process
would have to wait for timeouts on all IPv6 target addresses before
attempting any IPv4 address.
Similar to what RFC 8305 suggests, reorder the list of addresses to
try connecting to, interleaving address families. After starting one
connection attempt, start another one in parallel after a small delay
(200 ms as suggested by the RFC).
For cases with unreliable IPv6 but reliable IPv4, this should make
connection attempts work as reliably as with plain IPv4, with only an
extra 200 ms of connection delay.
Signed-off-by: Martin Storsjö <martin@martin.st>
2018-08-10 07:25:59 +00:00
|
|
|
/**
|
|
|
|
* Connect to any of the given addrinfo addresses, with multiple attempts
|
|
|
|
* running in parallel.
|
|
|
|
*
|
|
|
|
* @param addrs The list of addresses to try to connect to.
|
|
|
|
* This list will be mutated internally, but the list head
|
|
|
|
* will remain as such, so this doesn't affect the caller
|
|
|
|
* freeing the list afterwards.
|
|
|
|
* @param timeout_ms_per_address The number of milliseconds to wait for each
|
|
|
|
* connection attempt. Since multiple addresses are tried,
|
|
|
|
* some of them in parallel, the total run time will at most
|
|
|
|
* be timeout_ms_per_address*ceil(nb_addrs/parallel) +
|
|
|
|
* (parallel - 1) * NEXT_ATTEMPT_DELAY_MS.
|
|
|
|
* @param parallel The maximum number of connections to attempt in parallel.
|
|
|
|
* This is limited to an internal maximum capacity.
|
|
|
|
* @param h URLContext providing interrupt check
|
|
|
|
* callback and logging context.
|
|
|
|
* @param fd If successful, the connected socket is returned here.
|
|
|
|
* @param customize_fd Function that will be called for each socket created,
|
|
|
|
* to allow the caller to set socket options before calling
|
|
|
|
* connect() on it, may be NULL.
|
|
|
|
* @param customize_ctx Context parameter passed to customize_fd.
|
|
|
|
* @return 0 on success, AVERROR on failure.
|
|
|
|
*/
|
|
|
|
int ff_connect_parallel(struct addrinfo *addrs, int timeout_ms_per_address,
|
|
|
|
int parallel, URLContext *h, int *fd,
|
2023-03-03 09:49:26 +00:00
|
|
|
int (*customize_fd)(void *, int, int), void *customize_ctx);
|
network: Add RFC 8305 style "Happy Eyeballs"/"Fast Fallback" helper function
For cases with dual stack (IPv4 + IPv6) connectivity, but where one
stack potentially is less reliable, strive to trying to connect over
both protocols in parallel, using whichever address connected first.
In cases with a hostname resolving to multiple IPv4 and IPv6
addresses, the current connection mechanism would try all addresses
in the order returned by getaddrinfo (with all IPv6 addresses ordered
before the IPv4 addresses normally). If connection attempts to the
IPv6 addresses return quickly with an error, this was no problem, but
if they were unsuccessful leading up to timeouts, the connection process
would have to wait for timeouts on all IPv6 target addresses before
attempting any IPv4 address.
Similar to what RFC 8305 suggests, reorder the list of addresses to
try connecting to, interleaving address families. After starting one
connection attempt, start another one in parallel after a small delay
(200 ms as suggested by the RFC).
For cases with unreliable IPv6 but reliable IPv4, this should make
connection attempts work as reliably as with plain IPv4, with only an
extra 200 ms of connection delay.
Signed-off-by: Martin Storsjö <martin@martin.st>
2018-08-10 07:25:59 +00:00
|
|
|
|
2008-08-31 07:39:47 +00:00
|
|
|
#endif /* AVFORMAT_NETWORK_H */
|