103
103
#else
104
104
#include < pthread.h>
105
105
#include < sys/resource.h> // getrlimit, setrlimit
106
+ #include < termios.h> // tcgetattr, tcsetattr
106
107
#include < unistd.h> // STDIN_FILENO, STDERR_FILENO
107
108
#endif
108
109
@@ -191,7 +192,7 @@ void WaitForInspectorDisconnect(Environment* env) {
191
192
192
193
#ifdef __POSIX__
193
194
void SignalExit (int signo, siginfo_t * info, void * ucontext) {
194
- uv_tty_reset_mode ();
195
+ ResetStdio ();
195
196
raise (signo);
196
197
}
197
198
#endif // __POSIX__
@@ -451,7 +452,7 @@ void TrapWebAssemblyOrContinue(int signo, siginfo_t* info, void* ucontext) {
451
452
if (prev != nullptr ) {
452
453
prev (signo, info, ucontext);
453
454
} else {
454
- uv_tty_reset_mode ();
455
+ ResetStdio ();
455
456
raise (signo);
456
457
}
457
458
}
@@ -481,6 +482,16 @@ void RegisterSignalHandler(int signal,
481
482
482
483
#endif // __POSIX__
483
484
485
+ #ifdef __POSIX__
486
+ static struct {
487
+ int flags;
488
+ bool isatty;
489
+ struct stat stat;
490
+ struct termios termios;
491
+ } stdio[1 + STDERR_FILENO];
492
+ #endif // __POSIX__
493
+
494
+
484
495
inline void PlatformInit () {
485
496
#ifdef __POSIX__
486
497
#if HAVE_INSPECTOR
@@ -491,16 +502,18 @@ inline void PlatformInit() {
491
502
#endif // HAVE_INSPECTOR
492
503
493
504
// Make sure file descriptors 0-2 are valid before we start logging anything.
494
- for (int fd = STDIN_FILENO; fd <= STDERR_FILENO; fd += 1 ) {
495
- struct stat ignored ;
496
- if (fstat (fd, &ignored ) == 0 )
505
+ for (auto & s : stdio ) {
506
+ const int fd = &s - stdio ;
507
+ if (fstat (fd, &s. stat ) == 0 )
497
508
continue ;
498
509
// Anything but EBADF means something is seriously wrong. We don't
499
510
// have to special-case EINTR, fstat() is not interruptible.
500
511
if (errno != EBADF)
501
512
ABORT ();
502
513
if (fd != open (" /dev/null" , O_RDWR))
503
514
ABORT ();
515
+ if (fstat (fd, &s.stat ) != 0 )
516
+ ABORT ();
504
517
}
505
518
506
519
#if HAVE_INSPECTOR
@@ -523,6 +536,27 @@ inline void PlatformInit() {
523
536
}
524
537
#endif // !NODE_SHARED_MODE
525
538
539
+ // Record the state of the stdio file descriptors so we can restore it
540
+ // on exit. Needs to happen before installing signal handlers because
541
+ // they make use of that information.
542
+ for (auto & s : stdio) {
543
+ const int fd = &s - stdio;
544
+ int err;
545
+
546
+ do
547
+ s.flags = fcntl (fd, F_GETFL);
548
+ while (s.flags == -1 && errno == EINTR); // NOLINT
549
+ CHECK_NE (s.flags , -1 );
550
+
551
+ if (!isatty (fd)) continue ;
552
+ s.isatty = true ;
553
+
554
+ do
555
+ err = tcgetattr (fd, &s.termios );
556
+ while (err == -1 && errno == EINTR); // NOLINT
557
+ CHECK_EQ (err, 0 );
558
+ }
559
+
526
560
RegisterSignalHandler (SIGINT, SignalExit, true );
527
561
RegisterSignalHandler (SIGTERM, SignalExit, true );
528
562
@@ -576,6 +610,54 @@ inline void PlatformInit() {
576
610
#endif // _WIN32
577
611
}
578
612
613
+
614
+ // Safe to call more than once and from signal handlers.
615
+ void ResetStdio () {
616
+ uv_tty_reset_mode ();
617
+ #ifdef __POSIX__
618
+ for (auto & s : stdio) {
619
+ const int fd = &s - stdio;
620
+
621
+ struct stat tmp;
622
+ if (-1 == fstat (fd, &tmp)) {
623
+ CHECK_EQ (errno, EBADF); // Program closed file descriptor.
624
+ continue ;
625
+ }
626
+
627
+ bool is_same_file =
628
+ (s.stat .st_dev == tmp.st_dev && s.stat .st_ino == tmp.st_ino );
629
+ if (!is_same_file) continue ; // Program reopened file descriptor.
630
+
631
+ int flags;
632
+ do
633
+ flags = fcntl (fd, F_GETFL);
634
+ while (flags == -1 && errno == EINTR); // NOLINT
635
+ CHECK_NE (flags, -1 );
636
+
637
+ // Restore the O_NONBLOCK flag if it changed.
638
+ if (O_NONBLOCK & (flags ^ s.flags )) {
639
+ flags &= ~O_NONBLOCK;
640
+ flags |= s.flags & O_NONBLOCK;
641
+
642
+ int err;
643
+ do
644
+ err = fcntl (fd, F_SETFL, flags);
645
+ while (err == -1 && errno == EINTR); // NOLINT
646
+ CHECK_NE (err, -1 );
647
+ }
648
+
649
+ if (s.isatty ) {
650
+ int err;
651
+ do
652
+ err = tcsetattr (fd, TCSANOW, &s.termios );
653
+ while (err == -1 && errno == EINTR); // NOLINT
654
+ CHECK_NE (err, -1 );
655
+ }
656
+ }
657
+ #endif // __POSIX__
658
+ }
659
+
660
+
579
661
int ProcessGlobalArgs (std::vector<std::string>* args,
580
662
std::vector<std::string>* exec_args,
581
663
std::vector<std::string>* errors,
@@ -831,7 +913,7 @@ void Init(int* argc,
831
913
}
832
914
833
915
InitializationResult InitializeOncePerProcess (int argc, char ** argv) {
834
- atexit ([] () { uv_tty_reset_mode (); } );
916
+ atexit (ResetStdio );
835
917
PlatformInit ();
836
918
per_process::node_start_time = uv_hrtime ();
837
919
0 commit comments