~ubuntu-branches/ubuntu/trusty/debian-installer-utils/trusty-proposed

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
121
122
123
124
125
126
#define _GNU_SOURCE /* for getopt_long */
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <getopt.h>
#include <syslog.h>
#include <debian-installer.h>

#ifdef __GNUC__
#  define ATTRIBUTE_UNUSED __attribute__((__unused__))
#else
#  define ATTRIBUTE_UNUSED
#endif

/* See below for why this empty handler exists. */
static void sigchld_handler(int signum ATTRIBUTE_UNUSED)
{
}

static int close_orig_stdout(pid_t pid ATTRIBUTE_UNUSED, void *user_data)
{
	int orig_stdout = *(int *) user_data;
	close(orig_stdout);
	return 0;
}

static int restore_orig_stdout(pid_t pid ATTRIBUTE_UNUSED, void *user_data)
{
	int orig_stdout = *(int *) user_data;
	if (dup2(orig_stdout, 1) == -1)
		return 1;
	close(orig_stdout);
	return 0;
}

static int logger(const char *buf, size_t len ATTRIBUTE_UNUSED, void *user_data)
{
	static int log_open = 0;

	if (!log_open) {
		const char *ident = (const char *) user_data;
		if (!ident)
			ident = "log-output";
		openlog(ident, 0, LOG_USER);
		log_open = 1;
	}

	syslog(LOG_NOTICE, "%s", buf);

	return 0;
}

static void usage(FILE *output)
{
	fprintf(output, "Usage: log-output -t TAG [--pass-stdout] PROGRAM [ARGUMENTS]\n");
}

int main(int argc, char **argv)
{
	char *tag = NULL;
	static int pass_stdout = 0;
	static struct option long_options[] = {
		{ "help", no_argument, NULL, 'h' },
		{ "pass-stdout", no_argument, &pass_stdout, 1 },
		{ NULL, 0, NULL, 0 }
	};
	struct sigaction sa;
	di_io_handler *stdout_handler = NULL, *stderr_handler = NULL;
	di_process_handler *parent_prepare_handler = NULL;
	di_process_handler *child_prepare_handler = NULL;
	void *prepare_user_data = NULL;
	int orig_stdout = -1;
	int status;

	for (;;) {
		int c = getopt_long(argc, argv, "+t:", long_options, NULL);
		if (c == -1)
			break;
		switch (c) {
			case 0: /* long option */
				break;
			case 'h':
				usage(stdout);
				break;
			case 't':
				tag = strdup(optarg);
				break;
			default:
				usage(stderr);
				exit(1);
		}
	}

	if (!argv[optind])
		return 0;

	/* It's possible for subsidiary processes to start daemons which
	 * forget to clean up their file descriptors properly, which means
	 * that polling the other ends of those file descriptors will never
	 * complete.  We install a no-op SIGCHLD handler to make sure that
	 * its poll() gets EINTR and gives up.
	 *
	 * Technically, this is exploiting a bug in di_exec, and a better
	 * solution would be nice ...
	 */
	sa.sa_handler = &sigchld_handler;
	sigemptyset(&sa.sa_mask);
	sa.sa_flags = SA_NOCLDSTOP;
	sigaction(SIGCHLD, &sa, NULL);

	if (pass_stdout) {
		orig_stdout = dup(1);
		parent_prepare_handler = &close_orig_stdout;
		child_prepare_handler = &restore_orig_stdout;
		prepare_user_data = &orig_stdout;
	} else
		stdout_handler = &logger;
	stderr_handler = &logger;
	status = di_exec_path_full(argv[optind], (const char **) &argv[optind],
		stdout_handler, stderr_handler, tag,
		parent_prepare_handler, prepare_user_data,
		child_prepare_handler, prepare_user_data);

	return di_exec_mangle_status(status);
}