Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(184)

Side by Side Diff: util/test/multiprocess_exec_test_child.cc

Issue 808493003: win: port multiprocess_exec_test_child.cc (Closed) Base URL: https://chromium.googlesource.com/crashpad/crashpad@master
Patch Set: use NtQuerySystemInformation to get open handles, remove dep on base Created 5 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « util/test/multiprocess_exec_test.cc ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright 2014 The Crashpad Authors. All rights reserved. 1 // Copyright 2014 The Crashpad Authors. All rights reserved.
2 // 2 //
3 // Licensed under the Apache License, Version 2.0 (the "License"); 3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License. 4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at 5 // You may obtain a copy of the License at
6 // 6 //
7 // http://www.apache.org/licenses/LICENSE-2.0 7 // http://www.apache.org/licenses/LICENSE-2.0
8 // 8 //
9 // Unless required by applicable law or agreed to in writing, software 9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS, 10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and 12 // See the License for the specific language governing permissions and
13 // limitations under the License. 13 // limitations under the License.
14 14
15 #include <errno.h> 15 #include <errno.h>
16 #include <limits.h> 16 #include <limits.h>
17 #include <stdio.h> 17 #include <stdio.h>
18 #include <stdlib.h> 18 #include <stdlib.h>
19 #include <sys/types.h> 19 #include <sys/types.h>
20 #include <unistd.h>
21 20
22 #include <algorithm> 21 #include <algorithm>
23 22
23 #if defined(__APPLE__) || defined(__linux__)
24 #define OS_POSIX 1
25 #elif defined(_WIN32)
26 #define OS_WIN 1
27 #endif
28
29 #if defined(OS_POSIX)
30 #include <unistd.h>
31 #elif defined(OS_WIN)
32 #include <windows.h>
33 #endif
34
35 #if defined(OS_WIN)
36
37 // Various semi-documented NT internals to retrieve open handles.
38
39 typedef enum _SYSTEM_INFORMATION_CLASS {
40 SystemHandleInformation = 16
41 } SYSTEM_INFORMATION_CLASS;
42
43 typedef struct _SYSTEM_HANDLE_INFORMATION {
44 USHORT ProcessId;
45 USHORT CreatorBackTraceIndex;
46 UCHAR ObjectTypeNumber;
47 UCHAR Flags;
48 USHORT Handle;
49 PVOID Object;
50 ACCESS_MASK GrantedAccess;
51 } SYSTEM_HANDLE_INFORMATION, *PSYSTEM_HANDLE_INFORMATION;
52
53 typedef struct _SYSTEM_HANDLE_INFORMATION_EX {
54 ULONG NumberOfHandles;
55 SYSTEM_HANDLE_INFORMATION Information[1];
56 } SYSTEM_HANDLE_INFORMATION_EX, *PSYSTEM_HANDLE_INFORMATION_EX;
57
58 typedef NTSTATUS(WINAPI* NTQUERYSYSTEMINFORMATION)(
59 SYSTEM_INFORMATION_CLASS SystemInformationClass,
60 PVOID SystemInformation,
61 ULONG SystemInformationLength,
62 PULONG ReturnLength);
63
64 void EnsureOnlyStdioHandlesOpen() {
Mark Mentovai 2015/01/08 18:58:15 Unnamed namespace for this.
scottmg 2015/01/08 19:07:43 Done.
65 // Initialize the NTAPI functions we need.
66 HMODULE ntdll_handle = GetModuleHandle(L"ntdll.dll");
67 if (!ntdll_handle) {
68 fprintf(stderr, "GetModuleHandle ntdll.dll failed.\n");
69 abort();
70 }
71
72 NTQUERYSYSTEMINFORMATION NtQuerySystemInformation;
73 NtQuerySystemInformation = reinterpret_cast<NTQUERYSYSTEMINFORMATION>(
74 GetProcAddress(ntdll_handle, "NtQuerySystemInformation"));
75 if (!NtQuerySystemInformation) {
76 fprintf(stderr, "GetProcAddress NtQuerySystemInformation failed.\n");
77 abort();
78 }
79
80 // Get the number of handles on the system.
81 DWORD buffer_size = 0;
82 SYSTEM_HANDLE_INFORMATION_EX temp_info;
83 NTSTATUS status = NtQuerySystemInformation(
84 SystemHandleInformation, &temp_info, sizeof(temp_info), &buffer_size);
85 if (!buffer_size) {
86 fprintf(stderr,
87 "NtQuerySystemInformation for number of handles failed: 0x%lX\n",
88 status);
89 abort();
90 }
91
92 SYSTEM_HANDLE_INFORMATION_EX *system_handles =
93 reinterpret_cast<SYSTEM_HANDLE_INFORMATION_EX*>(new BYTE[buffer_size]);
94
95 // This is likely flaky as we're racing with other handles being created on
96 // the system between the size query above, and the actual retrieval here.
Mark Mentovai 2015/01/08 18:58:15 Well, hopefully we’re single-threaded here.
scottmg 2015/01/08 19:07:43 Unfortunately, NtQuerySystemInformation has no way
97 status = NtQuerySystemInformation(SystemHandleInformation, system_handles,
98 buffer_size, &buffer_size);
99 if (status != 0) {
100 fprintf(stderr, "Failed to get the handle list: 0x%lX\n", status);
101 delete[] system_handles;
102 abort();
103 }
104
105 for (ULONG i = 0; i < system_handles->NumberOfHandles; ++i) {
106 USHORT h = system_handles->Information[i].Handle;
107 if (system_handles->Information[i].ProcessId != GetCurrentProcessId())
108 continue;
109
110 // TODO(scottmg): This is probably insufficient, we'll need to allow having
111 // a few other standard handles open (for example, to the window station),
112 // or only check for handles of certain types.
113 HANDLE handle = reinterpret_cast<HANDLE>(h);
114 if (handle != GetStdHandle(STD_INPUT_HANDLE) &&
115 handle != GetStdHandle(STD_OUTPUT_HANDLE) &&
116 handle != GetStdHandle(STD_ERROR_HANDLE)) {
117 fprintf(stderr, "Handle 0x%lX is not stdio handle\n", handle);
118 abort();
119 }
120 }
121
122 delete [] system_handles;
123 }
124
125 #endif // OS_WIN
126
24 int main(int argc, char* argv[]) { 127 int main(int argc, char* argv[]) {
128 #if defined(OS_POSIX)
25 // Make sure that there’s nothing open at any FD higher than 3. All FDs other 129 // Make sure that there’s nothing open at any FD higher than 3. All FDs other
26 // than stdin, stdout, and stderr should have been closed prior to or at 130 // than stdin, stdout, and stderr should have been closed prior to or at
27 // exec(). 131 // exec().
28 int max_fd = std::max(static_cast<int>(sysconf(_SC_OPEN_MAX)), OPEN_MAX); 132 int max_fd = std::max(static_cast<int>(sysconf(_SC_OPEN_MAX)), OPEN_MAX);
29 max_fd = std::max(max_fd, getdtablesize()); 133 max_fd = std::max(max_fd, getdtablesize());
30 for (int fd = STDERR_FILENO + 1; fd < max_fd; ++fd) { 134 for (int fd = STDERR_FILENO + 1; fd < max_fd; ++fd) {
31 if (close(fd) == 0 || errno != EBADF) { 135 if (close(fd) == 0 || errno != EBADF) {
32 abort(); 136 abort();
33 } 137 }
34 } 138 }
35 139
36 // Read a byte from stdin, expecting it to be a specific value. 140 // Read a byte from stdin, expecting it to be a specific value.
37 char c; 141 char c;
38 ssize_t rv = read(STDIN_FILENO, &c, 1); 142 ssize_t rv = read(STDIN_FILENO, &c, 1);
39 if (rv != 1 || c != 'z') { 143 if (rv != 1 || c != 'z') {
40 abort(); 144 abort();
41 } 145 }
42 146
43 // Write a byte to stdout. 147 // Write a byte to stdout.
44 c = 'Z'; 148 c = 'Z';
45 rv = write(STDOUT_FILENO, &c, 1); 149 rv = write(STDOUT_FILENO, &c, 1);
46 if (rv != 1) { 150 if (rv != 1) {
47 abort(); 151 abort();
48 } 152 }
153 #elif defined(OS_WIN)
154 // Make sure there's nothing open other than stdin, stdout, and stderr.
155 EnsureOnlyStdioHandlesOpen();
156
157 // Read a byte from stdin, expecting it to be a specific value.
158 char c;
159 DWORD bytes_read;
160 if (!ReadFile(GetStdHandle(STD_INPUT_HANDLE), &c, 1, &bytes_read, nullptr) ||
161 bytes_read != 1 || c != 'z') {
162 abort();
163 }
164
165 // Write a byte to stdout.
166 c = 'Z';
167 DWORD bytes_written;
168 if (!WriteFile(
169 GetStdHandle(STD_OUTPUT_HANDLE), &c, 1, &bytes_written, nullptr) ||
170 bytes_written != 1) {
171 abort();
172 }
173 #endif // OS_POSIX
49 174
50 return 0; 175 return 0;
51 } 176 }
OLDNEW
« no previous file with comments | « util/test/multiprocess_exec_test.cc ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698