util-linux/lib/ttyutils.c
Karel Zak 84a8f5edb2 ttyutils: improve get_terminal_default_type() code
Signed-off-by: Karel Zak <kzak@redhat.com>
2023-05-31 09:18:06 +02:00

220 lines
4.3 KiB
C

/*
* No copyright is claimed. This code is in the public domain; do with
* it what you wish.
*
* Written by Karel Zak <kzak@redhat.com>
*/
#include <ctype.h>
#include <unistd.h>
#ifdef HAVE_GETTTYNAM
# include <ttyent.h>
#endif
#include "c.h"
#include "ttyutils.h"
#ifdef __linux__
# ifndef DEFAULT_VCTERM
# define DEFAULT_VCTERM "linux"
# endif
# if defined (__s390__) || defined (__s390x__)
# define DEFAULT_TTYS0 "dumb"
# define DEFAULT_TTY32 "ibm327x"
# define DEFAULT_TTYS1 "vt220"
# endif
# ifndef DEFAULT_STERM
# define DEFAULT_STERM "vt102"
# endif
#elif defined(__GNU__)
# ifndef DEFAULT_VCTERM
# define DEFAULT_VCTERM "hurd"
# endif
# ifndef DEFAULT_STERM
# define DEFAULT_STERM "vt102"
# endif
#else
# ifndef DEFAULT_VCTERM
# define DEFAULT_VCTERM "vt100"
# endif
# ifndef DEFAULT_STERM
# define DEFAULT_STERM "vt100"
# endif
#endif
static int get_env_int(const char *name)
{
const char *cp = getenv(name);
if (cp) {
char *end = NULL;
long x;
errno = 0;
x = strtol(cp, &end, 10);
if (errno == 0 && end && *end == '\0' && end > cp &&
x > 0 && x <= INT_MAX)
return x;
}
return -1;
}
int get_terminal_dimension(int *cols, int *lines)
{
int c = 0, l = 0;
#if defined(TIOCGWINSZ)
struct winsize w_win;
if (ioctl (STDOUT_FILENO, TIOCGWINSZ, &w_win) == 0) {
c = w_win.ws_col;
l = w_win.ws_row;
}
#elif defined(TIOCGSIZE)
struct ttysize t_win;
if (ioctl (STDOUT_FILENO, TIOCGSIZE, &t_win) == 0) {
c = t_win.ts_cols;
l = t_win.ts_lines;
}
#endif
if (cols) {
if (c <= 0)
c = get_env_int("COLUMNS");
*cols = c;
}
if (lines) {
if (l <= 0)
l = get_env_int("LINES");
*lines = l;
}
return 0;
}
int get_terminal_width(int default_width)
{
int width = 0;
get_terminal_dimension(&width, NULL);
return width > 0 ? width : default_width;
}
int get_terminal_stdfd(void)
{
if (isatty(STDIN_FILENO))
return STDIN_FILENO;
if (isatty(STDOUT_FILENO))
return STDOUT_FILENO;
if (isatty(STDERR_FILENO))
return STDERR_FILENO;
return -EINVAL;
}
int get_terminal_name(const char **path,
const char **name,
const char **number)
{
const char *tty;
const char *p;
int fd;
if (name)
*name = NULL;
if (path)
*path = NULL;
if (number)
*number = NULL;
fd = get_terminal_stdfd();
if (fd < 0)
return fd; /* error */
tty = ttyname(fd);
if (!tty)
return -1;
if (path)
*path = tty;
if (name || number)
tty = strncmp(tty, "/dev/", 5) == 0 ? tty + 5 : tty;
if (name)
*name = tty;
if (number) {
for (p = tty; p && *p; p++) {
if (isdigit(*p)) {
*number = p;
break;
}
}
}
return 0;
}
int get_terminal_type(const char **type)
{
*type = getenv("TERM");
if (*type)
return -EINVAL;
return 0;
}
char *get_terminal_default_type(const char *ttyname, int is_serial)
{
if (ttyname) {
#ifdef HAVE_GETTTYNAM
struct ttyent *ent = getttynam(ttyname);
if (ent && ent->ty_type)
return strdup(ent->ty_type);
#endif
#if defined (__s390__) || defined (__s390x__)
/*
* Special terminal on first serial line on a S/390(x) which
* is due legacy reasons a block terminal of type 3270 or
* higher. Whereas the second serial line on a S/390(x) is
* a real character terminal which is compatible with VT220.
*/
if (strcmp(ttyname, "ttyS0") == 0) /* linux/drivers/s390/char/con3215.c */
return strdup(DEFAULT_TTYS0);
else if (strncmp(ttyname, "3270/tty", 8) == 0) /* linux/drivers/s390/char/con3270.c */
return strdup(DEFAULT_TTY32);
else if (strcmp(ttyname, "ttyS1") == 0) /* linux/drivers/s390/char/sclp_vt220.c */
return strdup(DEFAULT_TTYS1);
#endif
}
return strdup(is_serial ? DEFAULT_STERM : DEFAULT_VCTERM);
}
#ifdef TEST_PROGRAM_TTYUTILS
# include <stdlib.h>
int main(void)
{
const char *path, *name, *num;
int c, l;
if (get_terminal_name(&path, &name, &num) == 0) {
char *term;
fprintf(stderr, "tty path: %s\n", path);
fprintf(stderr, "tty name: %s\n", name);
fprintf(stderr, "tty number: %s\n", num);
fprintf(stderr, "tty term: %s\n", getenv("TERM"));
term = get_terminal_default_type(name, 0);
fprintf(stderr, "tty dflt term: %s\n", term);
free(term);
}
get_terminal_dimension(&c, &l);
fprintf(stderr, "tty cols: %d\n", c);
fprintf(stderr, "tty lines: %d\n", l);
return EXIT_SUCCESS;
}
#endif /* TEST_PROGRAM_TTYUTILS */