getty-generator.c revision 543407517e05688915128cfc544c5d7f97f240ef
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
/***
This file is part of systemd.
Copyright 2010 Lennart Poettering
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.
systemd 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 systemd; If not, see <http://www.gnu.org/licenses/>.
***/
#include <string.h>
#include <errno.h>
#include <unistd.h>
#include <fcntl.h>
#include "log.h"
#include "util.h"
#include "mkdir.h"
#include "unit-name.h"
#include "virt.h"
#include "fileio.h"
#include "path-util.h"
static const char *arg_dest = "/tmp";
int r;
if (!from)
return log_oom();
if (!to)
return log_oom();
if (r < 0) {
/* In case console=hvc0 is passed this will very likely result in EEXIST */
return 0;
else {
return -errno;
}
}
return 0;
}
static int add_serial_getty(const char *tty) {
_cleanup_free_ char *n = NULL;
if (!n)
return log_oom();
return add_symlink("serial-getty@.service", n);
}
static int add_container_getty(const char *tty) {
_cleanup_free_ char *n = NULL;
if (!n)
return log_oom();
return add_symlink("container-getty@.service", n);
}
static int verify_tty(const char *name) {
const char *p;
/* Some TTYs are weird and have been enumerated but don't work
* when you try to use them, such as classic ttyS0 and
* friends. Let's check that and open the device and run
* isatty() on it. */
/* O_NONBLOCK is essential here, to make sure we don't wait
* for DCD */
if (fd < 0)
return -errno;
errno = 0;
return 0;
}
static const char virtualization_consoles[] =
"hvc0\0"
"xvc0\0"
"hvsi0\0"
"sclp_line0\0"
"ttysclp0\0";
const char *j;
int r;
log_error("This program takes three or no arguments.");
return EXIT_FAILURE;
}
if (argc > 1)
log_open();
umask(0022);
if (detect_container(NULL) > 0) {
log_debug("Automatically adding console shell.");
return EXIT_FAILURE;
/* When $container_ttys is set for PID 1, spawn
* gettys on all ptys named therein. Note that despite
* the variable name we only support ptys here. */
if (r > 0) {
char *w, *state;
size_t l;
const char *t;
char tty[l + 1];
tty[l] = 0;
/* First strip off /dev/ if it is specified */
if (!t)
t = tty;
/* Then, make sure it's actually a pty */
t = path_startswith(t, "pts/");
if (!t)
continue;
if (add_container_getty(t) < 0)
return EXIT_FAILURE;
}
}
/* Don't add any further magic if we are in a container */
return EXIT_SUCCESS;
}
char *w, *state;
size_t l;
/* Automatically add in a serial getty on all active
* kernel consoles */
if (!tty) {
log_oom();
return EXIT_FAILURE;
}
continue;
if (verify_tty(tty) < 0)
continue;
/* We assume that gettys on virtual terminals are
* started via manual configuration and do this magic
* only for non-VC terminals. */
if (add_serial_getty(tty) < 0)
return EXIT_FAILURE;
}
}
/* Automatically add in a serial getty on the first
* virtualizer console */
_cleanup_free_ char *p = NULL;
if (!p) {
log_oom();
return EXIT_FAILURE;
}
continue;
if (add_serial_getty(j) < 0)
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}