2018-08-14 11:13:38 +03:00
|
|
|
/* vim: tabstop=4 shiftwidth=4 noexpandtab
|
|
|
|
* This file is part of ToaruOS and is released under the terms
|
|
|
|
* of the NCSA / University of Illinois License - see LICENSE.md
|
|
|
|
* Copyright (C) 2018 K. Lange
|
|
|
|
*
|
|
|
|
* getty - Manage a TTY.
|
|
|
|
*
|
|
|
|
* Wraps a serial port (or other dumb connection) with a pty
|
|
|
|
* and manages a login for it.
|
2018-07-06 04:40:49 +03:00
|
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
2018-08-02 07:13:27 +03:00
|
|
|
#include <pty.h>
|
2018-08-02 12:48:45 +03:00
|
|
|
#include <sys/wait.h>
|
2018-08-02 07:13:27 +03:00
|
|
|
#include <sys/fswait.h>
|
2018-07-06 04:40:49 +03:00
|
|
|
|
|
|
|
int main(int argc, char * argv[]) {
|
|
|
|
int fd_master, fd_slave, fd_serial;
|
|
|
|
char * file = "/dev/ttyS0";
|
2018-08-13 07:30:12 +03:00
|
|
|
char * user = NULL;
|
2018-07-06 04:40:49 +03:00
|
|
|
|
|
|
|
if (getuid() != 0) {
|
|
|
|
fprintf(stderr, "%s: only root can do that\n", argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-08-13 07:30:12 +03:00
|
|
|
int opt;
|
|
|
|
while ((opt = getopt(argc, argv, "a:")) != -1) {
|
|
|
|
switch (opt) {
|
|
|
|
case 'a':
|
|
|
|
user = optarg;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (optind < argc) {
|
|
|
|
file = argv[optind];
|
2018-08-02 12:48:45 +03:00
|
|
|
}
|
2018-07-06 04:40:49 +03:00
|
|
|
|
2018-08-02 12:48:45 +03:00
|
|
|
openpty(&fd_master, &fd_slave, NULL, NULL, NULL);
|
|
|
|
fd_serial = open(file, O_RDWR);
|
2018-07-06 04:40:49 +03:00
|
|
|
|
2018-08-02 12:48:45 +03:00
|
|
|
pid_t child = fork();
|
2018-07-06 04:40:49 +03:00
|
|
|
|
2018-08-02 12:48:45 +03:00
|
|
|
if (!child) {
|
|
|
|
dup2(fd_slave, 0);
|
|
|
|
dup2(fd_slave, 1);
|
|
|
|
dup2(fd_slave, 2);
|
2018-08-01 04:03:34 +03:00
|
|
|
|
2018-08-02 12:48:45 +03:00
|
|
|
system("ttysize -q");
|
2018-07-06 04:40:49 +03:00
|
|
|
|
2018-08-13 07:30:12 +03:00
|
|
|
char * tokens[] = {"/bin/login",NULL,NULL,NULL};
|
|
|
|
|
|
|
|
if (user) {
|
|
|
|
tokens[1] = "-f";
|
|
|
|
tokens[2] = user;
|
|
|
|
}
|
|
|
|
|
2018-08-02 12:48:45 +03:00
|
|
|
execvp(tokens[0], tokens);
|
|
|
|
exit(1);
|
|
|
|
} else {
|
2018-07-06 04:40:49 +03:00
|
|
|
|
2018-08-02 12:48:45 +03:00
|
|
|
int fds[2] = {fd_serial, fd_master};
|
2018-07-06 04:40:49 +03:00
|
|
|
|
2018-08-02 12:48:45 +03:00
|
|
|
while (1) {
|
|
|
|
int index = fswait2(2,fds,200);
|
|
|
|
char buf[1024];
|
|
|
|
int r;
|
|
|
|
switch (index) {
|
|
|
|
case 0: /* fd_serial */
|
|
|
|
r = read(fd_serial, buf, 1);
|
|
|
|
write(fd_master, buf, 1);
|
|
|
|
break;
|
|
|
|
case 1: /* fd_master */
|
|
|
|
r = read(fd_master, buf, 1024);
|
|
|
|
write(fd_serial, buf, r);
|
|
|
|
break;
|
|
|
|
default: /* timeout */
|
|
|
|
{
|
|
|
|
int result = waitpid(child, NULL, WNOHANG);
|
|
|
|
if (result > 0) {
|
|
|
|
/* Child login shell has returned (session ended) */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2018-07-06 04:40:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|