public inbox for passt-dev@passt.top
 help / color / mirror / code / Atom feed
blob 592fafaab58a9e5e28cbb8a951252990749c9acf 2811 bytes (raw)

  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);
}

debug log:

solving 592fafaa ...
found 592fafaa in https://passt.top/passt

Code repositories for project(s) associated with this public inbox

	https://passt.top/passt

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for IMAP folder(s).