mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 19:16:41 +00:00 
			
		
		
		
	The recent change to crt.S that aggressively aligns the system-provided
stack has been rolled back on non-Linux until we can find a better way,
since it can cause a segfault early in execution on several platforms.
This change fixes a regression in tcgetattr() and tcsetattr() on OpenBSD
and NetBSD caused by 4778cd4d27.
This change has been tested across the runitd test fleet which is green.
		
	
			
		
			
				
	
	
		
			141 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			141 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | |
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│
 | |
| ╞══════════════════════════════════════════════════════════════════════════════╡
 | |
| │ Copyright 2021 Justine Alexandra Roberts Tunney                              │
 | |
| │                                                                              │
 | |
| │ Permission to use, copy, modify, and/or distribute this software for         │
 | |
| │ any purpose with or without fee is hereby granted, provided that the         │
 | |
| │ above copyright notice and this permission notice appear in all copies.      │
 | |
| │                                                                              │
 | |
| │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL                │
 | |
| │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED                │
 | |
| │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE             │
 | |
| │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL         │
 | |
| │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR        │
 | |
| │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER               │
 | |
| │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │
 | |
| │ PERFORMANCE OF THIS SOFTWARE.                                                │
 | |
| ╚─────────────────────────────────────────────────────────────────────────────*/
 | |
| #include "libc/calls/internal.h"
 | |
| #include "libc/calls/struct/metatermios.internal.h"
 | |
| #include "libc/calls/struct/termios.h"
 | |
| #include "libc/calls/syscall-sysv.internal.h"
 | |
| #include "libc/calls/syscall_support-nt.internal.h"
 | |
| #include "libc/calls/termios.h"
 | |
| #include "libc/dce.h"
 | |
| #include "libc/fmt/itoa.h"
 | |
| #include "libc/intrin/strace.internal.h"
 | |
| #include "libc/mem/alloca.h"
 | |
| #include "libc/nt/comms.h"
 | |
| #include "libc/sysv/consts/termios.h"
 | |
| #include "libc/sysv/errfuns.h"
 | |
| 
 | |
| #define kNtPurgeTxabort 1
 | |
| #define kNtPurgeRxabort 2
 | |
| 
 | |
| #define TCXONC    0x0000540A  // linux
 | |
| #define TIOCSTOP  0x2000746f  // bsd
 | |
| #define TIOCSTART 0x2000746e  // bsd
 | |
| #define TIOCIXON  0x20007481  // xnu
 | |
| #define TIOCIXOFF 0x20007480  // xnu
 | |
| 
 | |
| static const char *DescribeFlow(char buf[12], int action) {
 | |
|   if (action == TCOOFF) return "TCOOFF";
 | |
|   if (action == TCOON) return "TCOON";
 | |
|   if (action == TCIOFF) return "TCIOFF";
 | |
|   if (action == TCION) return "TCION";
 | |
|   FormatInt32(buf, action);
 | |
|   return buf;
 | |
| }
 | |
| 
 | |
| static int sys_tcflow_bsd_write(int fd, int cc) {
 | |
|   unsigned char c;
 | |
|   struct termios_bsd term;
 | |
|   if (sys_ioctl(fd, TCGETS, &term) == -1) {
 | |
|     return -1;
 | |
|   }
 | |
|   if ((c = term.c_cc[cc]) != _POSIX_VDISABLE &&
 | |
|       sys_write(fd, &c, sizeof(c)) == -1) {
 | |
|     return -1;
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static int sys_tcflow_bsd(int fd, int action) {
 | |
|   switch (action) {
 | |
|     case TCOOFF:
 | |
|       return sys_ioctl(fd, TIOCSTOP, 0);
 | |
|     case TCOON:
 | |
|       return sys_ioctl(fd, TIOCSTART, 0);
 | |
|     case TCION:
 | |
|       if (IsXnu()) {
 | |
|         return sys_ioctl(fd, TIOCIXON, 0);
 | |
|       } else {
 | |
|         return sys_tcflow_bsd_write(fd, VSTART);
 | |
|       }
 | |
|     case TCIOFF:
 | |
|       if (IsXnu()) {
 | |
|         return sys_ioctl(fd, TIOCIXOFF, 0);
 | |
|       } else {
 | |
|         return sys_tcflow_bsd_write(fd, VSTOP);
 | |
|       }
 | |
|       return 0;
 | |
|     default:
 | |
|       return einval();
 | |
|   }
 | |
| }
 | |
| 
 | |
| static dontinline textwindows int sys_tcflow_nt(int fd, int action) {
 | |
|   bool32 ok;
 | |
|   int64_t h;
 | |
|   if (!__isfdopen(fd)) return ebadf();
 | |
|   h = g_fds.p[fd].handle;
 | |
|   switch (action) {
 | |
|     case TCOON:
 | |
|     case TCION:
 | |
|       ok = ClearCommBreak(h);
 | |
|       break;
 | |
|     case TCOOFF:
 | |
|       ok = PurgeComm(h, kNtPurgeTxabort);
 | |
|       break;
 | |
|     case TCIOFF:
 | |
|       ok = PurgeComm(h, kNtPurgeRxabort);
 | |
|       break;
 | |
|     default:
 | |
|       return einval();
 | |
|   }
 | |
|   return ok ? 0 : __winerr();
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Changes flow of teletypewriter data.
 | |
|  *
 | |
|  * @param fd is file descriptor of tty
 | |
|  * @param action may be one of:
 | |
|  *     - `TCOOFF` to suspend output
 | |
|  *     - `TCOON` to resume output
 | |
|  *     - `TCIOFF` to transmit a `STOP` character
 | |
|  *     - `TCION` to transmit a `START` character
 | |
|  * @return 0 on success, or -1 w/ errno
 | |
|  * @raise EINVAL if `action` is invalid
 | |
|  * @raise ENOSYS on Windows and Bare Metal
 | |
|  * @raise EBADF if `fd` isn't an open file descriptor
 | |
|  * @raise ENOTTY if `fd` is open but not a teletypewriter
 | |
|  * @raise EIO if process group of writer is orphoned, calling thread is
 | |
|  *     not blocking `SIGTTOU`, and process isn't ignoring `SIGTTOU`
 | |
|  * @asyncsignalsafe
 | |
|  */
 | |
| int tcflow(int fd, int action) {
 | |
|   int rc;
 | |
|   if (IsLinux()) {
 | |
|     rc = sys_ioctl(fd, TCXONC, action);
 | |
|   } else if (IsBsd()) {
 | |
|     rc = sys_tcflow_bsd(fd, action);
 | |
|   } else if (IsWindows()) {
 | |
|     rc = sys_tcflow_nt(fd, action);
 | |
|   } else {
 | |
|     rc = enosys();
 | |
|   }
 | |
|   STRACE("tcflow(%d, %s) → %d% m", fd, DescribeFlow(alloca(12), action), rc);
 | |
|   return rc;
 | |
| }
 |