mirror of
https://sourceware.org/git/glibc.git
synced 2025-08-08 17:42:12 +03:00
Undefine TCGETS, TCGETS2, and related ioctl constants in the installed headers. Extract the correct constants (using the kernel type definitions) automatically from the UAPI headers. The kernel constants are available under KERNEL_* names during the glibc build, computed using assembler constant extraction mechanism. Alpha may have to use TCGETS instead of TCGETS2 because TCTGETS2 became available in Linux 4.20 only. Introduce ARCH_TCGETS to make this choice explict. To support emulation on powerpc, glibc versions of the termios constants are added to the emulation code in internal-ioctl.h. Reviewed-by: Adhemerval Zanella <adhemerval.zanella@linaro.org>
115 lines
3.8 KiB
C
115 lines
3.8 KiB
C
/* Copyright (C) 1993-2025 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it 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.
|
|
|
|
The GNU C Library 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 the GNU C Library; if not, see
|
|
<https://www.gnu.org/licenses/>. */
|
|
|
|
#include <termios_internals.h>
|
|
|
|
#define static_assert_equal(x,y) _Static_assert ((x) == (y), #x " != " #y)
|
|
|
|
static_assert_equal (sizeof (struct termios2), KERNEL_TERMIOS2_SIZE);
|
|
static_assert_equal (offsetof (struct termios2, c_cc),
|
|
KERNEL_TERMIOS2_CC_OFFSET);
|
|
static_assert_equal (offsetof (struct termios2, c_line),
|
|
KERNEL_TERMIOS2_LINE_OFFSET);
|
|
|
|
/* Set the state of FD to *TERMIOS_P. */
|
|
int
|
|
__tcsetattr (int fd, int optional_actions, const struct termios *termios_p)
|
|
{
|
|
struct termios2 k_termios;
|
|
unsigned long cmd;
|
|
|
|
memset (&k_termios, 0, sizeof k_termios);
|
|
|
|
k_termios.c_iflag = termios_p->c_iflag;
|
|
k_termios.c_oflag = termios_p->c_oflag;
|
|
k_termios.c_cflag = termios_p->c_cflag;
|
|
k_termios.c_lflag = termios_p->c_lflag;
|
|
k_termios.c_line = termios_p->c_line;
|
|
|
|
k_termios.c_ospeed = termios_p->c_ospeed;
|
|
k_termios.c_ispeed = termios_p->c_ispeed;
|
|
|
|
___termios2_canonicalize_speeds (&k_termios);
|
|
|
|
copy_c_cc (k_termios.c_cc, _TERMIOS2_NCCS, termios_p->c_cc, NCCS);
|
|
|
|
/*
|
|
* Choose the proper ioctl number to invoke.
|
|
*
|
|
* Alpha got TCSETS2 late (Linux 4.20), but has the same structure
|
|
* format, and it only needs TCSETS2 if either it needs to use
|
|
* __BOTHER or split speed. All other architectures have TCSETS2 as
|
|
* far back as the current glibc supports. Calling TCSETS with
|
|
* __BOTHER causes unpredictable results on old Alpha kernels and
|
|
* could even crash them.
|
|
*/
|
|
static_assert_equal(TCSADRAIN, TCSANOW + 1);
|
|
static_assert_equal(TCSAFLUSH, TCSANOW + 2);
|
|
static_assert_equal(KERNEL_TCSETSW2, KERNEL_TCSETS2 + 1);
|
|
static_assert_equal(KERNEL_TCSETSF2, KERNEL_TCSETS2 + 2);
|
|
static_assert_equal(KERNEL_TCSETSW, KERNEL_TCSETS + 1);
|
|
static_assert_equal(KERNEL_TCSETSF, KERNEL_TCSETS + 2);
|
|
|
|
cmd = (long)optional_actions - TCSANOW;
|
|
if (cmd > 2)
|
|
return INLINE_SYSCALL_ERROR_RETURN_VALUE (EINVAL);
|
|
|
|
if (__ASSUME_TERMIOS2 ||
|
|
k_termios.c_ospeed != k_termios.c_ispeed ||
|
|
cbaud (k_termios.c_cflag) == __BOTHER)
|
|
{
|
|
cmd += KERNEL_TCSETS2;
|
|
}
|
|
else
|
|
{
|
|
cmd += KERNEL_TCSETS;
|
|
k_termios.c_cflag &= ~CIBAUD;
|
|
}
|
|
|
|
return INLINE_SYSCALL_CALL (ioctl, fd, cmd, &k_termios);
|
|
}
|
|
libc_hidden_def (__tcsetattr)
|
|
|
|
#if _HAVE_STRUCT_OLD_TERMIOS && _TERMIOS_OLD_COMPAT
|
|
|
|
versioned_symbol (libc, __tcsetattr, tcsetattr, GLIBC_2_42);
|
|
|
|
/* Legacy version for shorter struct termios without speed fields */
|
|
int
|
|
attribute_compat_text_section
|
|
__old_tcsetattr (int fd, int optional_actions, const old_termios_t *termios_p)
|
|
{
|
|
struct termios new_termios;
|
|
|
|
memset (&new_termios, 0, sizeof (new_termios));
|
|
new_termios.c_iflag = termios_p->c_iflag;
|
|
new_termios.c_oflag = termios_p->c_oflag;
|
|
new_termios.c_cflag = termios_p->c_cflag;
|
|
new_termios.c_lflag = termios_p->c_lflag;
|
|
new_termios.c_line = termios_p->c_line;
|
|
copy_c_cc(new_termios.c_cc, NCCS, termios_p->c_cc, OLD_NCCS);
|
|
|
|
return __tcsetattr (fd, optional_actions, &new_termios);
|
|
}
|
|
compat_symbol (libc, __old_tcsetattr, tcsetattr, GLIBC_2_0);
|
|
|
|
#else
|
|
|
|
weak_alias (__tcsetattr, tcsetattr)
|
|
|
|
#endif
|