blob: ba500a00210d9980bef07af30998159940769e2b [file] [log] [blame]
//===-- Linux implementation of tcsetattr ---------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "src/termios/tcsetattr.h"
#include "kernel_termios.h"
#include "src/__support/OSUtil/syscall.h"
#include "src/__support/common.h"
#include "src/errno/libc_errno.h"
#include <asm/ioctls.h> // Safe to include without the risk of name pollution.
#include <sys/syscall.h> // For syscall numbers
#include <termios.h>
namespace LIBC_NAMESPACE {
LLVM_LIBC_FUNCTION(int, tcsetattr,
(int fd, int actions, const struct termios *t)) {
struct kernel_termios kt;
long cmd;
switch (actions) {
case TCSANOW:
cmd = TCSETS;
break;
case TCSADRAIN:
cmd = TCSETSW;
break;
case TCSAFLUSH:
cmd = TCSETSF;
break;
default:
libc_errno = EINVAL;
return -1;
}
kt.c_iflag = t->c_iflag;
kt.c_oflag = t->c_oflag;
kt.c_cflag = t->c_cflag;
kt.c_lflag = t->c_lflag;
size_t nccs = KERNEL_NCCS <= NCCS ? KERNEL_NCCS : NCCS;
for (size_t i = 0; i < nccs; ++i)
kt.c_cc[i] = t->c_cc[i];
if (nccs < KERNEL_NCCS) {
for (size_t i = nccs; i < KERNEL_NCCS; ++i)
kt.c_cc[i] = 0;
}
int ret = LIBC_NAMESPACE::syscall_impl<int>(SYS_ioctl, fd, cmd, &kt);
if (ret < 0) {
libc_errno = -ret;
return -1;
}
return 0;
}
} // namespace LIBC_NAMESPACE