patches and low-level development discussion
 help / color / mirror / code / Atom feed
1abfff9e9adfb4058765167195894c45e3645a50 blob 2998 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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
 
// SPDX-License-Identifier: GPL-2.0-or-later
// SPDX-FileCopyrightText: 2020-2021 Alyssa Ross <hi@alyssa.is>

#define _GNU_SOURCE

#include <errno.h>
#include <inttypes.h>
#include <poll.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <stdnoreturn.h>
#include <sys/wait.h>
#include <sysexits.h>
#include <unistd.h>

#include "env.h"
#include "log.h"
#include "vsock.h"

noreturn static void ex_usage(void)
{
	if (verbosity)
		fprintf(stderr, "Usage: %s [ -1 ] [ -q | -Q | -v ] prog...\n",
			program_invocation_short_name);
	exit(EX_USAGE);
}

int main(int argc, char *argv[])
{
	bool notify = false;
	int opt;
	pid_t child;
	uint32_t lcid, lport, rcid, rport;
	struct pollfd poll_fd =
		{ .fd = STDIN_FILENO, .events = POLLIN, .revents = 0 };

	while ((opt = getopt(argc, argv, "+1qQv")) != -1) {
		switch (opt) {
		case '1':
			notify = true;
			break;
		case 'q':
		case 'Q':
		case 'v':
			set_verbosity(opt);
			break;
		default:
			ex_usage();
		}
	}

	// Check that there is something for us to exec into.
	if (optind > argc - 1)
		ex_usage();

	// Find out the CID and port of the socket.
	if (vsock_get_cid_and_port(STDIN_FILENO, &lcid, &lport)) {
		if (errno == ENOTSOCK || errno == EPROTOTYPE)
			die(EX_USAGE, "stdin is not an AF_VSOCK socket");
		diee(EX_OSERR, "getsockname");
	}

	// Notify the user of our port and indicate readiness.
	if (notify) {
		printf("%" PRIu32 "\n", lport);
		fclose(stdout);
	}

	// Set our CID and port as environment variables so `prog' can
	// know them.
	setenvf("VSOCKLOCALCID", 1, "%" PRIu32, lcid);
	setenvf("VSOCKLOCALPORT", 1, "%" PRIu32, lport);

	ilog("listening as %" PRIu32 " on port %" PRIu32, lcid, lport);

	// Wait for a connection.
	while (poll(&poll_fd, 1, -1) != -1) {
		// On Linux, conn will be blocking.  On other
		// platforms, this may not be the case.  If other
		// platforms are to be supported, we'd probably want
		// to set O_NONBLOCK here.
		int conn = vsock_accept(STDIN_FILENO, &rcid, &rport);
		if (conn == -1)
			diee(EX_OSERR, "accept");

		// Set the client's CID and port as environment
		// variables so `prog' can know them.
		setenvf("VSOCKREMOTECID", 1, "%" PRIu32, rcid);
		setenvf("VSOCKREMOTEPORT", 1, "%" PRIu32, rport);

		ilog("connection from %" PRIu32 " port %" PRIu32, rcid, rport);

		switch (child = fork()) {
		case -1: diee(EX_OSERR, "fork");
		case 0:
			// Set up the connection socket on prog's
			// stdin and stdout.  This has the happy side
			// effect of closing the listening socket in
			// the child, since it's also on stdin.
			if (dup2(conn, STDIN_FILENO) == -1)
				diee(EX_OSERR, "dup2");
			if (dup2(conn, STDOUT_FILENO) == -1)
				diee(EX_OSERR, "dup2");
			if (conn != STDIN_FILENO && conn != STDOUT_FILENO)
				close(conn);

			// exec into `prog'.
			execvp(argv[optind], &argv[optind]);
			diee(EX_OSERR, "exec");
		}

		if (waitpid(child, NULL, 0) == -1)
			diee(EX_OSERR, "waitpid");

		close(conn);
	}
	diee(EX_OSERR, "poll");
}
debug log:

solving 1abfff9 ...
found 1abfff9 in https://spectrum-os.org/git/ucspi-vsock

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

	https://spectrum-os.org/git/crosvm
	https://spectrum-os.org/git/doc
	https://spectrum-os.org/git/mktuntap
	https://spectrum-os.org/git/nixpkgs
	https://spectrum-os.org/git/spectrum
	https://spectrum-os.org/git/ucspi-vsock
	https://spectrum-os.org/git/www

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 NNTP newsgroup(s).