OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 #include "platform/globals.h" | 5 #include "platform/globals.h" |
6 #if defined(TARGET_OS_MACOS) | 6 #if defined(TARGET_OS_MACOS) |
7 | 7 |
8 #include "bin/process.h" | 8 #include "bin/process.h" |
9 | 9 |
10 #include <errno.h> // NOLINT | 10 #include <errno.h> // NOLINT |
11 #include <fcntl.h> // NOLINT | 11 #include <fcntl.h> // NOLINT |
12 #include <poll.h> // NOLINT | 12 #include <poll.h> // NOLINT |
13 #include <signal.h> // NOLINT | 13 #include <signal.h> // NOLINT |
14 #include <stdio.h> // NOLINT | 14 #include <stdio.h> // NOLINT |
15 #include <stdlib.h> // NOLINT | 15 #include <stdlib.h> // NOLINT |
16 #include <string.h> // NOLINT | 16 #include <string.h> // NOLINT |
17 #include <unistd.h> // NOLINT | 17 #include <unistd.h> // NOLINT |
18 | 18 |
19 #include "bin/fdutils.h" | 19 #include "bin/fdutils.h" |
20 #include "bin/log.h" | 20 #include "bin/log.h" |
| 21 #include "bin/signal_blocker.h" |
21 #include "bin/thread.h" | 22 #include "bin/thread.h" |
22 | 23 |
23 extern char **environ; | 24 extern char **environ; |
24 | 25 |
25 | 26 |
26 namespace dart { | 27 namespace dart { |
27 namespace bin { | 28 namespace bin { |
28 | 29 |
29 // ProcessInfo is used to map a process id to the file descriptor for | 30 // ProcessInfo is used to map a process id to the file descriptor for |
30 // the pipe used to communicate the exit code of the process to Dart. | 31 // the pipe used to communicate the exit code of the process to Dart. |
(...skipping 598 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
629 | 630 |
630 void Process::TerminateExitCodeHandler() { | 631 void Process::TerminateExitCodeHandler() { |
631 ExitCodeHandler::TerminateExitCodeThread(); | 632 ExitCodeHandler::TerminateExitCodeThread(); |
632 } | 633 } |
633 | 634 |
634 | 635 |
635 intptr_t Process::CurrentProcessId() { | 636 intptr_t Process::CurrentProcessId() { |
636 return static_cast<intptr_t>(getpid()); | 637 return static_cast<intptr_t>(getpid()); |
637 } | 638 } |
638 | 639 |
| 640 |
| 641 static Mutex* signal_mutex = new Mutex(); |
| 642 static SignalInfo* signal_handlers = NULL; |
| 643 static const int kSignalsCount = 5; |
| 644 static const int kSignals[kSignalsCount] = { |
| 645 SIGINT, |
| 646 SIGWINCH, |
| 647 SIGTERM, |
| 648 SIGUSR1, |
| 649 SIGUSR2 |
| 650 }; |
| 651 |
| 652 |
| 653 SignalInfo::~SignalInfo() { |
| 654 VOID_TEMP_FAILURE_RETRY(close(fd_)); |
| 655 } |
| 656 |
| 657 |
| 658 static void SignalHandler(int signal) { |
| 659 MutexLocker lock(signal_mutex); |
| 660 const SignalInfo* handler = signal_handlers; |
| 661 while (handler != NULL) { |
| 662 if (handler->signal() == signal) { |
| 663 int value = 0; |
| 664 VOID_TEMP_FAILURE_RETRY(write(handler->fd(), &value, 1)); |
| 665 } |
| 666 handler = handler->next(); |
| 667 } |
| 668 } |
| 669 |
| 670 |
| 671 intptr_t Process::SetSignalHandler(intptr_t signal) { |
| 672 bool found = false; |
| 673 for (int i = 0; i < kSignalsCount; i++) { |
| 674 if (kSignals[i] == signal) { |
| 675 found = true; |
| 676 break; |
| 677 } |
| 678 } |
| 679 if (!found) return -1; |
| 680 int fds[2]; |
| 681 if (TEMP_FAILURE_RETRY_BLOCK_SIGNALS(pipe(fds)) != 0) { |
| 682 return -1; |
| 683 } |
| 684 if (!FDUtils::SetNonBlocking(fds[0])) { |
| 685 VOID_TEMP_FAILURE_RETRY_BLOCK_SIGNALS(close(fds[0])); |
| 686 VOID_TEMP_FAILURE_RETRY_BLOCK_SIGNALS(close(fds[1])); |
| 687 return -1; |
| 688 } |
| 689 ThreadSignalBlocker blocker(kSignalsCount, kSignals); |
| 690 MutexLocker lock(signal_mutex); |
| 691 SignalInfo* handler = signal_handlers; |
| 692 bool listen = true; |
| 693 while (handler != NULL) { |
| 694 if (handler->signal() == signal) { |
| 695 listen = false; |
| 696 break; |
| 697 } |
| 698 handler = handler->next(); |
| 699 } |
| 700 if (listen) { |
| 701 struct sigaction act; |
| 702 bzero(&act, sizeof(act)); |
| 703 act.sa_handler = SignalHandler; |
| 704 int status = TEMP_FAILURE_RETRY_BLOCK_SIGNALS( |
| 705 sigaction(signal, &act, NULL)); |
| 706 if (status < 0) { |
| 707 VOID_TEMP_FAILURE_RETRY_BLOCK_SIGNALS(close(fds[0])); |
| 708 VOID_TEMP_FAILURE_RETRY_BLOCK_SIGNALS(close(fds[1])); |
| 709 return -1; |
| 710 } |
| 711 } |
| 712 if (signal_handlers == NULL) { |
| 713 signal_handlers = new SignalInfo(fds[1], signal); |
| 714 } else { |
| 715 new SignalInfo(fds[1], signal, signal_handlers); |
| 716 } |
| 717 return fds[0]; |
| 718 } |
| 719 |
| 720 |
| 721 void Process::ClearSignalHandler(intptr_t signal) { |
| 722 ThreadSignalBlocker blocker(kSignalsCount, kSignals); |
| 723 MutexLocker lock(signal_mutex); |
| 724 SignalInfo* handler = signal_handlers; |
| 725 bool unlisten = true; |
| 726 while (handler != NULL) { |
| 727 bool remove = false; |
| 728 if (handler->signal() == signal) { |
| 729 if (handler->port() == Dart_GetMainPortId()) { |
| 730 if (signal_handlers == handler) signal_handlers = handler->next(); |
| 731 handler->Unlink(); |
| 732 remove = true; |
| 733 } else { |
| 734 unlisten = false; |
| 735 } |
| 736 } |
| 737 SignalInfo* next = handler->next(); |
| 738 if (remove) delete handler; |
| 739 handler = next; |
| 740 } |
| 741 if (unlisten) { |
| 742 struct sigaction act; |
| 743 bzero(&act, sizeof(act)); |
| 744 act.sa_handler = SIG_DFL; |
| 745 VOID_TEMP_FAILURE_RETRY_BLOCK_SIGNALS(sigaction(signal, &act, NULL)); |
| 746 } |
| 747 } |
| 748 |
639 } // namespace bin | 749 } // namespace bin |
640 } // namespace dart | 750 } // namespace dart |
641 | 751 |
642 #endif // defined(TARGET_OS_MACOS) | 752 #endif // defined(TARGET_OS_MACOS) |
OLD | NEW |