aboutgitcodebugslistschat
path: root/pif.c
blob: 592fafaab58a9e5e28cbb8a951252990749c9acf (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/* SPDX-License-Identifier: GPL-2.0-or-later
 * Copyright Red Hat
 * Author: David Gibson <david@gibson.dropbear.id.au>
 *
 * Passt/pasta interface types and IDs
 */

#include <stdint.h>
#include <assert.h>
#include <netinet/in.h>

#include "util.h"
#include "pif.h"
#include "siphash.h"
#include "ip.h"
#include "inany.h"
#include "passt.h"

const char *pif_type_str[] = {
	[PIF_NONE]		= "<none>",
	[PIF_HOST]		= "HOST",
	[PIF_TAP]		= "TAP",
	[PIF_SPLICE]		= "SPLICE",
};
static_assert(ARRAY_SIZE(pif_type_str) == PIF_NUM_TYPES,
	      "pif_type_str[] doesn't match enum pif_type");


/** pif_sockaddr() - Construct a socket address suitable for an interface
 * @c:		Execution context
 * @sa:		Pointer to sockaddr to fill in
 * @sl:		Updated to relevant length of initialised @sa
 * @pif:	Interface to create the socket address
 * @addr:	IPv[46] address
 * @port:	Port (host byte order)
 */
void pif_sockaddr(const struct ctx *c, union sockaddr_inany *sa, socklen_t *sl,
		  uint8_t pif, const union inany_addr *addr, in_port_t port)
{
	const struct in_addr *v4 = inany_v4(addr);

	ASSERT(pif_is_socket(pif));

	if (v4) {
		sa->sa_family = AF_INET;
		sa->sa4.sin_addr = *v4;
		sa->sa4.sin_port = htons(port);
		memset(&sa->sa4.sin_zero, 0, sizeof(sa->sa4.sin_zero));
		*sl = sizeof(sa->sa4);
	} else {
		sa->sa_family = AF_INET6;
		sa->sa6.sin6_addr = addr->a6;
		sa->sa6.sin6_port = htons(port);
		if (pif == PIF_HOST && IN6_IS_ADDR_LINKLOCAL(&addr->a6))
			sa->sa6.sin6_scope_id = c->ifi6;
		else
			sa->sa6.sin6_scope_id = 0;
		sa->sa6.sin6_flowinfo = 0;
		*sl = sizeof(sa->sa6);
	}
}

/** pif_sock_l4() - Open a socket bound to an address on a specified interface
 * @c:		Execution context
 * @type:	Socket epoll type
 * @pif:	Interface for this socket
 * @addr:	Address to bind to, or NULL for dual-stack any
 * @ifname:	Interface for binding, NULL for any
 * @port:	Port number to bind to (host byte order)
 * @data:	epoll reference portion for protocol handlers
 *
 * NOTE: For namespace pifs, this must be called having already entered the
 * relevant namespace.
 *
 * Return: newly created socket, negative error code on failure
 */
int pif_sock_l4(const struct ctx *c, enum epoll_type type, uint8_t pif,
		const union inany_addr *addr, const char *ifname,
		in_port_t port, uint32_t data)
{
	union sockaddr_inany sa = {
		.sa6.sin6_family = AF_INET6,
		.sa6.sin6_addr = in6addr_any,
		.sa6.sin6_port = htons(port),
	};
	socklen_t sl;

	ASSERT(pif_is_socket(pif));

	if (pif == PIF_SPLICE) {
		/* Sanity checks */
		ASSERT(!ifname);
		ASSERT(addr && inany_is_loopback(addr));
	}

	if (!addr)
		return sock_l4_sa(c, type, &sa, sizeof(sa.sa6),
				  ifname, false, data);

	pif_sockaddr(c, &sa, &sl, pif, addr, port);
	return sock_l4_sa(c, type, &sa, sl,
			  ifname, sa.sa_family == AF_INET6, data);
}