@shitiandmw/node-pty 1.1.0-agent.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (288) hide show
  1. package/LICENSE +69 -0
  2. package/README.md +170 -0
  3. package/binding.gyp +111 -0
  4. package/deps/.editorconfig +2 -0
  5. package/deps/winpty/.drone.yml +17 -0
  6. package/deps/winpty/.gitattributes +19 -0
  7. package/deps/winpty/LICENSE +21 -0
  8. package/deps/winpty/Makefile +166 -0
  9. package/deps/winpty/README.md +151 -0
  10. package/deps/winpty/RELEASES.md +280 -0
  11. package/deps/winpty/VERSION.txt +1 -0
  12. package/deps/winpty/configure +167 -0
  13. package/deps/winpty/misc/BufferResizeTests.cc +90 -0
  14. package/deps/winpty/misc/ChangeScreenBuffer.cc +53 -0
  15. package/deps/winpty/misc/ClearConsole.cc +72 -0
  16. package/deps/winpty/misc/ConinMode.cc +117 -0
  17. package/deps/winpty/misc/ConinMode.ps1 +116 -0
  18. package/deps/winpty/misc/ConoutMode.cc +113 -0
  19. package/deps/winpty/misc/DebugClient.py +42 -0
  20. package/deps/winpty/misc/DebugServer.py +63 -0
  21. package/deps/winpty/misc/DumpLines.py +5 -0
  22. package/deps/winpty/misc/EnableExtendedFlags.txt +46 -0
  23. package/deps/winpty/misc/Font-Report-June2016/CP437-Consolas.txt +528 -0
  24. package/deps/winpty/misc/Font-Report-June2016/CP437-Lucida.txt +633 -0
  25. package/deps/winpty/misc/Font-Report-June2016/CP932.txt +630 -0
  26. package/deps/winpty/misc/Font-Report-June2016/CP936.txt +630 -0
  27. package/deps/winpty/misc/Font-Report-June2016/CP949.txt +630 -0
  28. package/deps/winpty/misc/Font-Report-June2016/CP950.txt +630 -0
  29. package/deps/winpty/misc/Font-Report-June2016/MinimumWindowWidths.txt +16 -0
  30. package/deps/winpty/misc/Font-Report-June2016/Results.txt +4 -0
  31. package/deps/winpty/misc/Font-Report-June2016/Windows10SetFontBugginess.txt +144 -0
  32. package/deps/winpty/misc/FontSurvey.cc +100 -0
  33. package/deps/winpty/misc/FormatChar.h +21 -0
  34. package/deps/winpty/misc/FreezePerfTest.cc +62 -0
  35. package/deps/winpty/misc/GetCh.cc +20 -0
  36. package/deps/winpty/misc/GetConsolePos.cc +41 -0
  37. package/deps/winpty/misc/GetFont.cc +261 -0
  38. package/deps/winpty/misc/IdentifyConsoleWindow.ps1 +51 -0
  39. package/deps/winpty/misc/IsNewConsole.cc +87 -0
  40. package/deps/winpty/misc/MouseInputNotes.txt +90 -0
  41. package/deps/winpty/misc/MoveConsoleWindow.cc +34 -0
  42. package/deps/winpty/misc/Notes.txt +219 -0
  43. package/deps/winpty/misc/OSVersion.cc +27 -0
  44. package/deps/winpty/misc/ScreenBufferFreezeInactive.cc +101 -0
  45. package/deps/winpty/misc/ScreenBufferTest.cc +671 -0
  46. package/deps/winpty/misc/ScreenBufferTest2.cc +151 -0
  47. package/deps/winpty/misc/SelectAllTest.cc +45 -0
  48. package/deps/winpty/misc/SetBufferSize.cc +32 -0
  49. package/deps/winpty/misc/SetCursorPos.cc +10 -0
  50. package/deps/winpty/misc/SetFont.cc +145 -0
  51. package/deps/winpty/misc/SetWindowRect.cc +36 -0
  52. package/deps/winpty/misc/ShowArgv.cc +12 -0
  53. package/deps/winpty/misc/ShowConsoleInput.cc +40 -0
  54. package/deps/winpty/misc/Spew.py +5 -0
  55. package/deps/winpty/misc/TestUtil.cc +172 -0
  56. package/deps/winpty/misc/UnicodeDoubleWidthTest.cc +102 -0
  57. package/deps/winpty/misc/UnicodeWideTest1.cc +246 -0
  58. package/deps/winpty/misc/UnicodeWideTest2.cc +130 -0
  59. package/deps/winpty/misc/UnixEcho.cc +89 -0
  60. package/deps/winpty/misc/Utf16Echo.cc +46 -0
  61. package/deps/winpty/misc/VeryLargeRead.cc +122 -0
  62. package/deps/winpty/misc/VkEscapeTest.cc +56 -0
  63. package/deps/winpty/misc/Win10ResizeWhileFrozen.cc +52 -0
  64. package/deps/winpty/misc/Win10WrapTest1.cc +57 -0
  65. package/deps/winpty/misc/Win10WrapTest2.cc +30 -0
  66. package/deps/winpty/misc/Win32Echo1.cc +26 -0
  67. package/deps/winpty/misc/Win32Echo2.cc +19 -0
  68. package/deps/winpty/misc/Win32Test1.cc +46 -0
  69. package/deps/winpty/misc/Win32Test2.cc +70 -0
  70. package/deps/winpty/misc/Win32Test3.cc +78 -0
  71. package/deps/winpty/misc/Win32Write1.cc +44 -0
  72. package/deps/winpty/misc/WindowsBugCrashReader.cc +27 -0
  73. package/deps/winpty/misc/WriteConsole.cc +106 -0
  74. package/deps/winpty/misc/build32.sh +9 -0
  75. package/deps/winpty/misc/build64.sh +9 -0
  76. package/deps/winpty/misc/color-test.sh +212 -0
  77. package/deps/winpty/misc/font-notes.txt +300 -0
  78. package/deps/winpty/misc/winbug-15048.cc +201 -0
  79. package/deps/winpty/ship/build-pty4j-libpty.bat +36 -0
  80. package/deps/winpty/ship/common_ship.py +53 -0
  81. package/deps/winpty/ship/make_msvc_package.py +165 -0
  82. package/deps/winpty/ship/ship.py +108 -0
  83. package/deps/winpty/src/agent/Agent.cc +613 -0
  84. package/deps/winpty/src/agent/Agent.h +103 -0
  85. package/deps/winpty/src/agent/AgentCreateDesktop.cc +84 -0
  86. package/deps/winpty/src/agent/AgentCreateDesktop.h +28 -0
  87. package/deps/winpty/src/agent/ConsoleFont.cc +632 -0
  88. package/deps/winpty/src/agent/ConsoleFont.h +28 -0
  89. package/deps/winpty/src/agent/ConsoleInput.cc +852 -0
  90. package/deps/winpty/src/agent/ConsoleInput.h +109 -0
  91. package/deps/winpty/src/agent/ConsoleInputReencoding.cc +121 -0
  92. package/deps/winpty/src/agent/ConsoleInputReencoding.h +36 -0
  93. package/deps/winpty/src/agent/ConsoleLine.cc +152 -0
  94. package/deps/winpty/src/agent/ConsoleLine.h +41 -0
  95. package/deps/winpty/src/agent/Coord.h +87 -0
  96. package/deps/winpty/src/agent/DebugShowInput.cc +239 -0
  97. package/deps/winpty/src/agent/DebugShowInput.h +32 -0
  98. package/deps/winpty/src/agent/DefaultInputMap.cc +422 -0
  99. package/deps/winpty/src/agent/DefaultInputMap.h +28 -0
  100. package/deps/winpty/src/agent/DsrSender.h +30 -0
  101. package/deps/winpty/src/agent/EventLoop.cc +99 -0
  102. package/deps/winpty/src/agent/EventLoop.h +47 -0
  103. package/deps/winpty/src/agent/InputMap.cc +246 -0
  104. package/deps/winpty/src/agent/InputMap.h +114 -0
  105. package/deps/winpty/src/agent/LargeConsoleRead.cc +71 -0
  106. package/deps/winpty/src/agent/LargeConsoleRead.h +68 -0
  107. package/deps/winpty/src/agent/NamedPipe.cc +378 -0
  108. package/deps/winpty/src/agent/NamedPipe.h +125 -0
  109. package/deps/winpty/src/agent/Scraper.cc +699 -0
  110. package/deps/winpty/src/agent/Scraper.h +103 -0
  111. package/deps/winpty/src/agent/SimplePool.h +75 -0
  112. package/deps/winpty/src/agent/SmallRect.h +143 -0
  113. package/deps/winpty/src/agent/Terminal.cc +535 -0
  114. package/deps/winpty/src/agent/Terminal.h +69 -0
  115. package/deps/winpty/src/agent/UnicodeEncoding.h +157 -0
  116. package/deps/winpty/src/agent/UnicodeEncodingTest.cc +189 -0
  117. package/deps/winpty/src/agent/Win32Console.cc +107 -0
  118. package/deps/winpty/src/agent/Win32Console.h +67 -0
  119. package/deps/winpty/src/agent/Win32ConsoleBuffer.cc +193 -0
  120. package/deps/winpty/src/agent/Win32ConsoleBuffer.h +99 -0
  121. package/deps/winpty/src/agent/main.cc +114 -0
  122. package/deps/winpty/src/agent/subdir.mk +61 -0
  123. package/deps/winpty/src/configurations.gypi +60 -0
  124. package/deps/winpty/src/debugserver/DebugServer.cc +117 -0
  125. package/deps/winpty/src/debugserver/subdir.mk +41 -0
  126. package/deps/winpty/src/include/winpty.h +242 -0
  127. package/deps/winpty/src/include/winpty_constants.h +131 -0
  128. package/deps/winpty/src/libwinpty/AgentLocation.cc +75 -0
  129. package/deps/winpty/src/libwinpty/AgentLocation.h +28 -0
  130. package/deps/winpty/src/libwinpty/LibWinptyException.h +54 -0
  131. package/deps/winpty/src/libwinpty/WinptyInternal.h +72 -0
  132. package/deps/winpty/src/libwinpty/subdir.mk +46 -0
  133. package/deps/winpty/src/libwinpty/winpty.cc +970 -0
  134. package/deps/winpty/src/shared/AgentMsg.h +38 -0
  135. package/deps/winpty/src/shared/BackgroundDesktop.cc +122 -0
  136. package/deps/winpty/src/shared/BackgroundDesktop.h +73 -0
  137. package/deps/winpty/src/shared/Buffer.cc +103 -0
  138. package/deps/winpty/src/shared/Buffer.h +102 -0
  139. package/deps/winpty/src/shared/DebugClient.cc +187 -0
  140. package/deps/winpty/src/shared/DebugClient.h +38 -0
  141. package/deps/winpty/src/shared/GenRandom.cc +138 -0
  142. package/deps/winpty/src/shared/GenRandom.h +55 -0
  143. package/deps/winpty/src/shared/GetCommitHash.bat +13 -0
  144. package/deps/winpty/src/shared/Mutex.h +54 -0
  145. package/deps/winpty/src/shared/OsModule.h +63 -0
  146. package/deps/winpty/src/shared/OwnedHandle.cc +36 -0
  147. package/deps/winpty/src/shared/OwnedHandle.h +45 -0
  148. package/deps/winpty/src/shared/PrecompiledHeader.h +43 -0
  149. package/deps/winpty/src/shared/StringBuilder.h +227 -0
  150. package/deps/winpty/src/shared/StringBuilderTest.cc +114 -0
  151. package/deps/winpty/src/shared/StringUtil.cc +55 -0
  152. package/deps/winpty/src/shared/StringUtil.h +80 -0
  153. package/deps/winpty/src/shared/TimeMeasurement.h +63 -0
  154. package/deps/winpty/src/shared/UnixCtrlChars.h +45 -0
  155. package/deps/winpty/src/shared/UpdateGenVersion.bat +20 -0
  156. package/deps/winpty/src/shared/WindowsSecurity.cc +460 -0
  157. package/deps/winpty/src/shared/WindowsSecurity.h +104 -0
  158. package/deps/winpty/src/shared/WindowsVersion.cc +252 -0
  159. package/deps/winpty/src/shared/WindowsVersion.h +29 -0
  160. package/deps/winpty/src/shared/WinptyAssert.cc +55 -0
  161. package/deps/winpty/src/shared/WinptyAssert.h +64 -0
  162. package/deps/winpty/src/shared/WinptyException.cc +57 -0
  163. package/deps/winpty/src/shared/WinptyException.h +43 -0
  164. package/deps/winpty/src/shared/WinptyVersion.cc +42 -0
  165. package/deps/winpty/src/shared/WinptyVersion.h +27 -0
  166. package/deps/winpty/src/shared/winpty_snprintf.h +99 -0
  167. package/deps/winpty/src/subdir.mk +5 -0
  168. package/deps/winpty/src/tests/subdir.mk +28 -0
  169. package/deps/winpty/src/tests/trivial_test.cc +158 -0
  170. package/deps/winpty/src/unix-adapter/InputHandler.cc +114 -0
  171. package/deps/winpty/src/unix-adapter/InputHandler.h +56 -0
  172. package/deps/winpty/src/unix-adapter/OutputHandler.cc +80 -0
  173. package/deps/winpty/src/unix-adapter/OutputHandler.h +53 -0
  174. package/deps/winpty/src/unix-adapter/Util.cc +86 -0
  175. package/deps/winpty/src/unix-adapter/Util.h +31 -0
  176. package/deps/winpty/src/unix-adapter/WakeupFd.cc +70 -0
  177. package/deps/winpty/src/unix-adapter/WakeupFd.h +42 -0
  178. package/deps/winpty/src/unix-adapter/main.cc +729 -0
  179. package/deps/winpty/src/unix-adapter/subdir.mk +41 -0
  180. package/deps/winpty/src/winpty.gyp +234 -0
  181. package/deps/winpty/vcbuild.bat +83 -0
  182. package/lib/conpty_console_list_agent.js +16 -0
  183. package/lib/conpty_console_list_agent.js.map +1 -0
  184. package/lib/eventEmitter2.js +47 -0
  185. package/lib/eventEmitter2.js.map +1 -0
  186. package/lib/eventEmitter2.test.js +30 -0
  187. package/lib/eventEmitter2.test.js.map +1 -0
  188. package/lib/index.js +52 -0
  189. package/lib/index.js.map +1 -0
  190. package/lib/interfaces.js +7 -0
  191. package/lib/interfaces.js.map +1 -0
  192. package/lib/shared/conout.js +11 -0
  193. package/lib/shared/conout.js.map +1 -0
  194. package/lib/terminal.js +190 -0
  195. package/lib/terminal.js.map +1 -0
  196. package/lib/terminal.test.js +139 -0
  197. package/lib/terminal.test.js.map +1 -0
  198. package/lib/testUtils.test.js +28 -0
  199. package/lib/testUtils.test.js.map +1 -0
  200. package/lib/types.js +7 -0
  201. package/lib/types.js.map +1 -0
  202. package/lib/unixTerminal.js +346 -0
  203. package/lib/unixTerminal.js.map +1 -0
  204. package/lib/unixTerminal.test.js +351 -0
  205. package/lib/unixTerminal.test.js.map +1 -0
  206. package/lib/utils.js +39 -0
  207. package/lib/utils.js.map +1 -0
  208. package/lib/windowsConoutConnection.js +125 -0
  209. package/lib/windowsConoutConnection.js.map +1 -0
  210. package/lib/windowsPtyAgent.js +320 -0
  211. package/lib/windowsPtyAgent.js.map +1 -0
  212. package/lib/windowsPtyAgent.test.js +90 -0
  213. package/lib/windowsPtyAgent.test.js.map +1 -0
  214. package/lib/windowsTerminal.js +199 -0
  215. package/lib/windowsTerminal.js.map +1 -0
  216. package/lib/windowsTerminal.test.js +219 -0
  217. package/lib/windowsTerminal.test.js.map +1 -0
  218. package/lib/worker/conoutSocketWorker.js +22 -0
  219. package/lib/worker/conoutSocketWorker.js.map +1 -0
  220. package/package.json +68 -0
  221. package/prebuilds/darwin-arm64/pty.node +0 -0
  222. package/prebuilds/darwin-arm64/spawn-helper +0 -0
  223. package/prebuilds/darwin-x64/pty.node +0 -0
  224. package/prebuilds/darwin-x64/spawn-helper +0 -0
  225. package/prebuilds/win32-arm64/conpty/OpenConsole.exe +0 -0
  226. package/prebuilds/win32-arm64/conpty/conpty.dll +0 -0
  227. package/prebuilds/win32-arm64/conpty.node +0 -0
  228. package/prebuilds/win32-arm64/conpty.pdb +0 -0
  229. package/prebuilds/win32-arm64/conpty_console_list.node +0 -0
  230. package/prebuilds/win32-arm64/conpty_console_list.pdb +0 -0
  231. package/prebuilds/win32-arm64/pty.node +0 -0
  232. package/prebuilds/win32-arm64/pty.pdb +0 -0
  233. package/prebuilds/win32-arm64/winpty-agent.exe +0 -0
  234. package/prebuilds/win32-arm64/winpty-agent.pdb +0 -0
  235. package/prebuilds/win32-arm64/winpty.dll +0 -0
  236. package/prebuilds/win32-arm64/winpty.pdb +0 -0
  237. package/prebuilds/win32-x64/conpty/OpenConsole.exe +0 -0
  238. package/prebuilds/win32-x64/conpty/conpty.dll +0 -0
  239. package/prebuilds/win32-x64/conpty.node +0 -0
  240. package/prebuilds/win32-x64/conpty.pdb +0 -0
  241. package/prebuilds/win32-x64/conpty_console_list.node +0 -0
  242. package/prebuilds/win32-x64/conpty_console_list.pdb +0 -0
  243. package/prebuilds/win32-x64/pty.node +0 -0
  244. package/prebuilds/win32-x64/pty.pdb +0 -0
  245. package/prebuilds/win32-x64/winpty-agent.exe +0 -0
  246. package/prebuilds/win32-x64/winpty-agent.pdb +0 -0
  247. package/prebuilds/win32-x64/winpty.dll +0 -0
  248. package/prebuilds/win32-x64/winpty.pdb +0 -0
  249. package/scripts/gen-compile-commands.js +8 -0
  250. package/scripts/increment-version.js +54 -0
  251. package/scripts/post-install.js +99 -0
  252. package/scripts/prebuild.js +39 -0
  253. package/scripts/sync-prebuild.js +31 -0
  254. package/scripts/verify-darwin-fd-leak.js +63 -0
  255. package/src/conpty_console_list_agent.ts +15 -0
  256. package/src/eventEmitter2.test.ts +30 -0
  257. package/src/eventEmitter2.ts +48 -0
  258. package/src/index.ts +52 -0
  259. package/src/interfaces.ts +130 -0
  260. package/src/native.d.ts +54 -0
  261. package/src/shared/conout.ts +15 -0
  262. package/src/terminal.test.ts +119 -0
  263. package/src/terminal.ts +211 -0
  264. package/src/testUtils.test.ts +23 -0
  265. package/src/tsconfig.json +22 -0
  266. package/src/types.ts +15 -0
  267. package/src/unix/pty.cc +808 -0
  268. package/src/unix/spawn-helper.cc +23 -0
  269. package/src/unixTerminal.test.ts +367 -0
  270. package/src/unixTerminal.ts +388 -0
  271. package/src/utils.ts +29 -0
  272. package/src/win/conpty.cc +583 -0
  273. package/src/win/conpty.h +41 -0
  274. package/src/win/conpty_console_list.cc +44 -0
  275. package/src/win/path_util.cc +95 -0
  276. package/src/win/path_util.h +26 -0
  277. package/src/win/winpty.cc +333 -0
  278. package/src/windowsConoutConnection.ts +82 -0
  279. package/src/windowsPtyAgent.test.ts +94 -0
  280. package/src/windowsPtyAgent.ts +321 -0
  281. package/src/windowsTerminal.test.ts +229 -0
  282. package/src/windowsTerminal.ts +203 -0
  283. package/src/worker/conoutSocketWorker.ts +22 -0
  284. package/third_party/conpty/1.23.251008001/win10-arm64/OpenConsole.exe +0 -0
  285. package/third_party/conpty/1.23.251008001/win10-arm64/conpty.dll +0 -0
  286. package/third_party/conpty/1.23.251008001/win10-x64/OpenConsole.exe +0 -0
  287. package/third_party/conpty/1.23.251008001/win10-x64/conpty.dll +0 -0
  288. package/typings/node-pty.d.ts +211 -0
@@ -0,0 +1,729 @@
1
+ // Copyright (c) 2011-2015 Ryan Prichard
2
+ //
3
+ // Permission is hereby granted, free of charge, to any person obtaining a copy
4
+ // of this software and associated documentation files (the "Software"), to
5
+ // deal in the Software without restriction, including without limitation the
6
+ // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
7
+ // sell copies of the Software, and to permit persons to whom the Software is
8
+ // furnished to do so, subject to the following conditions:
9
+ //
10
+ // The above copyright notice and this permission notice shall be included in
11
+ // all copies or substantial portions of the Software.
12
+ //
13
+ // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14
+ // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15
+ // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16
+ // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17
+ // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
18
+ // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
19
+ // IN THE SOFTWARE.
20
+
21
+ // MSYS's sys/cygwin.h header only declares cygwin_internal if WINVER is
22
+ // defined, which is defined in windows.h. Therefore, include windows.h early.
23
+ #include <windows.h>
24
+
25
+ #include <assert.h>
26
+ #include <cygwin/version.h>
27
+ #include <errno.h>
28
+ #include <signal.h>
29
+ #include <stdio.h>
30
+ #include <stdlib.h>
31
+ #include <string.h>
32
+ #include <sys/ioctl.h>
33
+ #include <sys/select.h>
34
+ #include <sys/cygwin.h>
35
+ #include <termios.h>
36
+ #include <unistd.h>
37
+
38
+ #include <map>
39
+ #include <string>
40
+ #include <utility>
41
+ #include <vector>
42
+
43
+ #include <winpty.h>
44
+ #include "../shared/DebugClient.h"
45
+ #include "../shared/UnixCtrlChars.h"
46
+ #include "../shared/WinptyVersion.h"
47
+ #include "InputHandler.h"
48
+ #include "OutputHandler.h"
49
+ #include "Util.h"
50
+ #include "WakeupFd.h"
51
+
52
+ #define CSI "\x1b["
53
+
54
+ static WakeupFd *g_mainWakeup = NULL;
55
+
56
+ static WakeupFd &mainWakeup()
57
+ {
58
+ if (g_mainWakeup == NULL) {
59
+ static const char msg[] = "Internal error: g_mainWakeup is NULL\r\n";
60
+ write(STDERR_FILENO, msg, sizeof(msg) - 1);
61
+ abort();
62
+ }
63
+ return *g_mainWakeup;
64
+ }
65
+
66
+ struct SavedTermiosMode {
67
+ int count;
68
+ bool valid[3];
69
+ termios mode[3];
70
+ };
71
+
72
+ // Put the input terminal into non-canonical mode.
73
+ static SavedTermiosMode setRawTerminalMode(
74
+ bool allowNonTtys, bool setStdout, bool setStderr)
75
+ {
76
+ SavedTermiosMode ret;
77
+ const char *const kNames[3] = { "stdin", "stdout", "stderr" };
78
+
79
+ ret.valid[0] = true;
80
+ ret.valid[1] = setStdout;
81
+ ret.valid[2] = setStderr;
82
+
83
+ for (int i = 0; i < 3; ++i) {
84
+ if (!ret.valid[i]) {
85
+ continue;
86
+ }
87
+ if (!isatty(i)) {
88
+ ret.valid[i] = false;
89
+ if (!allowNonTtys) {
90
+ fprintf(stderr, "%s is not a tty\n", kNames[i]);
91
+ exit(1);
92
+ }
93
+ } else {
94
+ ret.valid[i] = true;
95
+ if (tcgetattr(i, &ret.mode[i]) < 0) {
96
+ perror("tcgetattr failed");
97
+ exit(1);
98
+ }
99
+ }
100
+ }
101
+
102
+ if (ret.valid[STDIN_FILENO]) {
103
+ termios buf;
104
+ if (tcgetattr(STDIN_FILENO, &buf) < 0) {
105
+ perror("tcgetattr failed");
106
+ exit(1);
107
+ }
108
+ buf.c_lflag &= ~(ECHO | ICANON | IEXTEN | ISIG);
109
+ buf.c_iflag &= ~(BRKINT | ICRNL | INPCK | ISTRIP | IXON);
110
+ buf.c_cflag &= ~(CSIZE | PARENB);
111
+ buf.c_cflag |= CS8;
112
+ buf.c_cc[VMIN] = 1; // blocking read
113
+ buf.c_cc[VTIME] = 0;
114
+ if (tcsetattr(STDIN_FILENO, TCSAFLUSH, &buf) < 0) {
115
+ fprintf(stderr, "tcsetattr failed\n");
116
+ exit(1);
117
+ }
118
+ }
119
+
120
+ for (int i = STDOUT_FILENO; i <= STDERR_FILENO; ++i) {
121
+ if (!ret.valid[i]) {
122
+ continue;
123
+ }
124
+ termios buf;
125
+ if (tcgetattr(i, &buf) < 0) {
126
+ perror("tcgetattr failed");
127
+ exit(1);
128
+ }
129
+ buf.c_cflag &= ~(CSIZE | PARENB);
130
+ buf.c_cflag |= CS8;
131
+ buf.c_oflag &= ~OPOST;
132
+ if (tcsetattr(i, TCSAFLUSH, &buf) < 0) {
133
+ fprintf(stderr, "tcsetattr failed\n");
134
+ exit(1);
135
+ }
136
+ }
137
+
138
+ return ret;
139
+ }
140
+
141
+ static void restoreTerminalMode(const SavedTermiosMode &original)
142
+ {
143
+ for (int i = 0; i < 3; ++i) {
144
+ if (!original.valid[i]) {
145
+ continue;
146
+ }
147
+ if (tcsetattr(i, TCSAFLUSH, &original.mode[i]) < 0) {
148
+ perror("error restoring terminal mode");
149
+ exit(1);
150
+ }
151
+ }
152
+ }
153
+
154
+ static void debugShowKey(bool allowNonTtys)
155
+ {
156
+ printf("\nPress any keys -- Ctrl-D exits\n\n");
157
+ const SavedTermiosMode saved =
158
+ setRawTerminalMode(allowNonTtys, false, false);
159
+ char buf[128];
160
+ while (true) {
161
+ const ssize_t len = read(STDIN_FILENO, buf, sizeof(buf));
162
+ if (len <= 0) {
163
+ break;
164
+ }
165
+ for (int i = 0; i < len; ++i) {
166
+ char ctrl = decodeUnixCtrlChar(buf[i]);
167
+ if (ctrl == '\0') {
168
+ putchar(buf[i]);
169
+ } else {
170
+ putchar('^');
171
+ putchar(ctrl);
172
+ }
173
+ }
174
+ for (int i = 0; i < len; ++i) {
175
+ unsigned char uch = buf[i];
176
+ printf("\t%3d %04o 0x%02x\n", uch, uch, uch);
177
+ fflush(stdout);
178
+ }
179
+ if (buf[0] == 4) {
180
+ // Ctrl-D
181
+ break;
182
+ }
183
+ }
184
+ restoreTerminalMode(saved);
185
+ }
186
+
187
+ static void terminalResized(int signo)
188
+ {
189
+ mainWakeup().set();
190
+ }
191
+
192
+ static void registerResizeSignalHandler()
193
+ {
194
+ struct sigaction resizeSigAct;
195
+ memset(&resizeSigAct, 0, sizeof(resizeSigAct));
196
+ resizeSigAct.sa_handler = terminalResized;
197
+ resizeSigAct.sa_flags = SA_RESTART;
198
+ sigaction(SIGWINCH, &resizeSigAct, NULL);
199
+ }
200
+
201
+ // Convert the path to a Win32 path if it is a POSIX path, and convert slashes
202
+ // to backslashes.
203
+ static std::string convertPosixPathToWin(const std::string &path)
204
+ {
205
+ char *tmp;
206
+ #if defined(CYGWIN_VERSION_CYGWIN_CONV) && \
207
+ CYGWIN_VERSION_API_MINOR >= CYGWIN_VERSION_CYGWIN_CONV
208
+ // MSYS2 and versions of Cygwin released after 2009 or so use this API.
209
+ // The original MSYS still lacks this API.
210
+ ssize_t newSize = cygwin_conv_path(CCP_POSIX_TO_WIN_A | CCP_ABSOLUTE,
211
+ path.c_str(), NULL, 0);
212
+ assert(newSize >= 0);
213
+ tmp = new char[newSize + 1];
214
+ ssize_t success = cygwin_conv_path(CCP_POSIX_TO_WIN_A | CCP_ABSOLUTE,
215
+ path.c_str(), tmp, newSize + 1);
216
+ assert(success == 0);
217
+ #else
218
+ // In the current Cygwin header file, this API is documented as deprecated
219
+ // because it's restricted to paths of MAX_PATH length. In the CVS version
220
+ // of MSYS, the newer API doesn't exist, and this older API is implemented
221
+ // using msys_p2w, which seems like it would handle paths larger than
222
+ // MAX_PATH, but there's no way to query how large the new path is.
223
+ // Hopefully, this is large enough.
224
+ tmp = new char[MAX_PATH + path.size()];
225
+ cygwin_conv_to_win32_path(path.c_str(), tmp);
226
+ #endif
227
+ for (int i = 0; tmp[i] != '\0'; ++i) {
228
+ if (tmp[i] == '/')
229
+ tmp[i] = '\\';
230
+ }
231
+ std::string ret(tmp);
232
+ delete [] tmp;
233
+ return ret;
234
+ }
235
+
236
+ static std::string resolvePath(const std::string &path)
237
+ {
238
+ char ret[PATH_MAX];
239
+ ret[0] = '\0';
240
+ if (realpath(path.c_str(), ret) != ret) {
241
+ return std::string();
242
+ }
243
+ return ret;
244
+ }
245
+
246
+ template <size_t N>
247
+ static bool endsWith(const std::string &path, const char (&suf)[N])
248
+ {
249
+ const size_t suffixLen = N - 1;
250
+ char actualSuf[N];
251
+ if (path.size() < suffixLen) {
252
+ return false;
253
+ }
254
+ strcpy(actualSuf, &path.c_str()[path.size() - suffixLen]);
255
+ for (size_t i = 0; i < suffixLen; ++i) {
256
+ actualSuf[i] = tolower(actualSuf[i]);
257
+ }
258
+ return !strcmp(actualSuf, suf);
259
+ }
260
+
261
+ static std::string findProgram(
262
+ const char *winptyProgName,
263
+ const std::string &prog)
264
+ {
265
+ std::string candidate;
266
+ if (prog.find('/') == std::string::npos &&
267
+ prog.find('\\') == std::string::npos) {
268
+ // XXX: It would be nice to use a lambda here (once/if old MSYS support
269
+ // is dropped).
270
+ // Search the PATH.
271
+ const char *const pathVar = getenv("PATH");
272
+ const std::string pathList(pathVar ? pathVar : "");
273
+ size_t elpos = 0;
274
+ while (true) {
275
+ const size_t elend = pathList.find(':', elpos);
276
+ candidate = pathList.substr(elpos, elend - elpos);
277
+ if (!candidate.empty() && *(candidate.end() - 1) != '/') {
278
+ candidate += '/';
279
+ }
280
+ candidate += prog;
281
+ candidate = resolvePath(candidate);
282
+ if (!candidate.empty()) {
283
+ int perm = X_OK;
284
+ if (endsWith(candidate, ".bat") || endsWith(candidate, ".cmd")) {
285
+ #ifdef __MSYS__
286
+ // In MSYS/MSYS2, batch files don't have the execute bit
287
+ // set, so just check that they're readable.
288
+ perm = R_OK;
289
+ #endif
290
+ } else if (endsWith(candidate, ".com") || endsWith(candidate, ".exe")) {
291
+ // Do nothing.
292
+ } else {
293
+ // Make the exe extension explicit so that we don't try to
294
+ // run shell scripts with CreateProcess/winpty_spawn.
295
+ candidate += ".exe";
296
+ }
297
+ if (!access(candidate.c_str(), perm)) {
298
+ break;
299
+ }
300
+ }
301
+ if (elend == std::string::npos) {
302
+ fprintf(stderr, "%s: error: cannot start '%s': Not found in PATH\n",
303
+ winptyProgName, prog.c_str());
304
+ exit(1);
305
+ } else {
306
+ elpos = elend + 1;
307
+ }
308
+ }
309
+ } else {
310
+ candidate = resolvePath(prog);
311
+ if (candidate.empty()) {
312
+ std::string errstr(strerror(errno));
313
+ fprintf(stderr, "%s: error: cannot start '%s': %s\n",
314
+ winptyProgName, prog.c_str(), errstr.c_str());
315
+ exit(1);
316
+ }
317
+ }
318
+ return convertPosixPathToWin(candidate);
319
+ }
320
+
321
+ // Convert argc/argv into a Win32 command-line following the escaping convention
322
+ // documented on MSDN. (e.g. see CommandLineToArgvW documentation)
323
+ static std::string argvToCommandLine(const std::vector<std::string> &argv)
324
+ {
325
+ std::string result;
326
+ for (size_t argIndex = 0; argIndex < argv.size(); ++argIndex) {
327
+ if (argIndex > 0)
328
+ result.push_back(' ');
329
+ const char *arg = argv[argIndex].c_str();
330
+ const bool quote =
331
+ strchr(arg, ' ') != NULL ||
332
+ strchr(arg, '\t') != NULL ||
333
+ *arg == '\0';
334
+ if (quote)
335
+ result.push_back('\"');
336
+ int bsCount = 0;
337
+ for (const char *p = arg; *p != '\0'; ++p) {
338
+ if (*p == '\\') {
339
+ bsCount++;
340
+ } else if (*p == '\"') {
341
+ result.append(bsCount * 2 + 1, '\\');
342
+ result.push_back('\"');
343
+ bsCount = 0;
344
+ } else {
345
+ result.append(bsCount, '\\');
346
+ bsCount = 0;
347
+ result.push_back(*p);
348
+ }
349
+ }
350
+ if (quote) {
351
+ result.append(bsCount * 2, '\\');
352
+ result.push_back('\"');
353
+ } else {
354
+ result.append(bsCount, '\\');
355
+ }
356
+ }
357
+ return result;
358
+ }
359
+
360
+ static wchar_t *heapMbsToWcs(const char *text)
361
+ {
362
+ // Calling mbstowcs with a NULL first argument seems to be broken on MSYS.
363
+ // Instead of returning the size of the converted string, it returns 0.
364
+ // Using strlen(text) * 2 is probably big enough.
365
+ size_t maxLen = strlen(text) * 2 + 1;
366
+ wchar_t *ret = new wchar_t[maxLen];
367
+ size_t len = mbstowcs(ret, text, maxLen);
368
+ assert(len != (size_t)-1 && len < maxLen);
369
+ return ret;
370
+ }
371
+
372
+ static char *heapWcsToMbs(const wchar_t *text)
373
+ {
374
+ // Calling wcstombs with a NULL first argument seems to be broken on MSYS.
375
+ // Instead of returning the size of the converted string, it returns 0.
376
+ // Using wcslen(text) * 3 is big enough for UTF-8 and probably other
377
+ // encodings. For UTF-8, codepoints that fit in a single wchar
378
+ // (U+0000 to U+FFFF) are encoded using 1-3 bytes. The remaining code
379
+ // points needs two wchar's and are encoded using 4 bytes.
380
+ size_t maxLen = wcslen(text) * 3 + 1;
381
+ char *ret = new char[maxLen];
382
+ size_t len = wcstombs(ret, text, maxLen);
383
+ if (len == (size_t)-1 || len >= maxLen) {
384
+ delete [] ret;
385
+ return NULL;
386
+ } else {
387
+ return ret;
388
+ }
389
+ }
390
+
391
+ static std::string wcsToMbs(const wchar_t *text)
392
+ {
393
+ std::string ret;
394
+ const char *ptr = heapWcsToMbs(text);
395
+ if (ptr != NULL) {
396
+ ret = ptr;
397
+ delete [] ptr;
398
+ }
399
+ return ret;
400
+ }
401
+
402
+ void setupWin32Environment()
403
+ {
404
+ std::map<std::string, std::string> varsToCopy;
405
+ const char *vars[] = {
406
+ "WINPTY_DEBUG",
407
+ "WINPTY_SHOW_CONSOLE",
408
+ NULL
409
+ };
410
+ for (int i = 0; vars[i] != NULL; ++i) {
411
+ const char *cstr = getenv(vars[i]);
412
+ if (cstr != NULL && cstr[0] != '\0') {
413
+ varsToCopy[vars[i]] = cstr;
414
+ }
415
+ }
416
+
417
+ #if defined(__MSYS__) && CYGWIN_VERSION_API_MINOR >= 48 || \
418
+ !defined(__MSYS__) && CYGWIN_VERSION_API_MINOR >= 153
419
+ // Use CW_SYNC_WINENV to copy the Unix environment to the Win32
420
+ // environment. The command performs special translation on some variables
421
+ // (such as PATH and TMP). It also copies the debugging environment
422
+ // variables.
423
+ //
424
+ // Note that the API minor versions have diverged in Cygwin and MSYS.
425
+ // CW_SYNC_WINENV was added to Cygwin in version 153. (Cygwin's
426
+ // include/cygwin/version.h says that CW_SETUP_WINENV was added in 153.
427
+ // The flag was renamed 8 days after it was added, but the API docs weren't
428
+ // updated.) The flag was added to MSYS in version 48.
429
+ //
430
+ // Also, in my limited testing, this call seems to be necessary with Cygwin
431
+ // but unnecessary with MSYS. Perhaps MSYS is automatically syncing the
432
+ // Unix environment with the Win32 environment before starting console.exe?
433
+ // It shouldn't hurt to call it for MSYS.
434
+ cygwin_internal(CW_SYNC_WINENV);
435
+ #endif
436
+
437
+ // Copy debugging environment variables from the Cygwin environment
438
+ // to the Win32 environment so the agent will inherit it.
439
+ for (std::map<std::string, std::string>::iterator it = varsToCopy.begin();
440
+ it != varsToCopy.end();
441
+ ++it) {
442
+ wchar_t *nameW = heapMbsToWcs(it->first.c_str());
443
+ wchar_t *valueW = heapMbsToWcs(it->second.c_str());
444
+ SetEnvironmentVariableW(nameW, valueW);
445
+ delete [] nameW;
446
+ delete [] valueW;
447
+ }
448
+
449
+ // Clear the TERM variable. The child process's immediate console/terminal
450
+ // environment is a Windows console, not the terminal that winpty is
451
+ // communicating with. Leaving the TERM variable set can break programs in
452
+ // various ways. (e.g. arrows keys broken in Cygwin less, IronPython's
453
+ // help(...) function doesn't start, misc programs decide they should
454
+ // output color escape codes on pre-Win10). See
455
+ // https://github.com/rprichard/winpty/issues/43.
456
+ SetEnvironmentVariableW(L"TERM", NULL);
457
+ }
458
+
459
+ static void usage(const char *program, int exitCode)
460
+ {
461
+ printf("Usage: %s [options] [--] program [args]\n", program);
462
+ printf("\n");
463
+ printf("Options:\n");
464
+ printf(" -h, --help Show this help message\n");
465
+ printf(" --mouse Enable terminal mouse input\n");
466
+ printf(" --showkey Dump STDIN escape sequences\n");
467
+ printf(" --version Show the winpty version number\n");
468
+ exit(exitCode);
469
+ }
470
+
471
+ struct Arguments {
472
+ std::vector<std::string> childArgv;
473
+ bool mouseInput;
474
+ bool testAllowNonTtys;
475
+ bool testConerr;
476
+ bool testPlainOutput;
477
+ bool testColorEscapes;
478
+ };
479
+
480
+ static void parseArguments(int argc, char *argv[], Arguments &out)
481
+ {
482
+ out.mouseInput = false;
483
+ out.testAllowNonTtys = false;
484
+ out.testConerr = false;
485
+ out.testPlainOutput = false;
486
+ out.testColorEscapes = false;
487
+ bool doShowKeys = false;
488
+ const char *const program = argc >= 1 ? argv[0] : "<program>";
489
+ int argi = 1;
490
+ while (argi < argc) {
491
+ std::string arg(argv[argi++]);
492
+ if (arg.size() >= 1 && arg[0] == '-') {
493
+ if (arg == "-h" || arg == "--help") {
494
+ usage(program, 0);
495
+ } else if (arg == "--mouse") {
496
+ out.mouseInput = true;
497
+ } else if (arg == "--showkey") {
498
+ doShowKeys = true;
499
+ } else if (arg == "--version") {
500
+ dumpVersionToStdout();
501
+ exit(0);
502
+ } else if (arg == "-Xallow-non-tty") {
503
+ out.testAllowNonTtys = true;
504
+ } else if (arg == "-Xconerr") {
505
+ out.testConerr = true;
506
+ } else if (arg == "-Xplain") {
507
+ out.testPlainOutput = true;
508
+ } else if (arg == "-Xcolor") {
509
+ out.testColorEscapes = true;
510
+ } else if (arg == "--") {
511
+ break;
512
+ } else {
513
+ fprintf(stderr, "Error: unrecognized option: '%s'\n",
514
+ arg.c_str());
515
+ exit(1);
516
+ }
517
+ } else {
518
+ out.childArgv.push_back(arg);
519
+ break;
520
+ }
521
+ }
522
+ for (; argi < argc; ++argi) {
523
+ out.childArgv.push_back(argv[argi]);
524
+ }
525
+ if (doShowKeys) {
526
+ debugShowKey(out.testAllowNonTtys);
527
+ exit(0);
528
+ }
529
+ if (out.childArgv.size() == 0) {
530
+ usage(program, 1);
531
+ }
532
+ }
533
+
534
+ static std::string errorMessageToString(DWORD err)
535
+ {
536
+ // Use FormatMessageW rather than FormatMessageA, because we want to use
537
+ // wcstombs to convert to the Cygwin locale, which might not match the
538
+ // codepage FormatMessageA would use. We need to convert using wcstombs,
539
+ // rather than print using %ls, because %ls doesn't work in the original
540
+ // MSYS.
541
+ wchar_t *wideMsgPtr = NULL;
542
+ const DWORD formatRet = FormatMessageW(
543
+ FORMAT_MESSAGE_FROM_SYSTEM |
544
+ FORMAT_MESSAGE_ALLOCATE_BUFFER |
545
+ FORMAT_MESSAGE_IGNORE_INSERTS,
546
+ NULL,
547
+ err,
548
+ MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
549
+ reinterpret_cast<wchar_t*>(&wideMsgPtr),
550
+ 0,
551
+ NULL);
552
+ if (formatRet == 0 || wideMsgPtr == NULL) {
553
+ return std::string();
554
+ }
555
+ std::string msg = wcsToMbs(wideMsgPtr);
556
+ LocalFree(wideMsgPtr);
557
+ const size_t pos = msg.find_last_not_of(" \r\n\t");
558
+ if (pos == std::string::npos) {
559
+ msg.clear();
560
+ } else {
561
+ msg.erase(pos + 1);
562
+ }
563
+ return msg;
564
+ }
565
+
566
+ static std::string formatErrorMessage(DWORD err)
567
+ {
568
+ char buf[64];
569
+ sprintf(buf, "error %#x", static_cast<unsigned int>(err));
570
+ std::string ret = errorMessageToString(err);
571
+ if (ret.empty()) {
572
+ ret += buf;
573
+ } else {
574
+ ret += " (";
575
+ ret += buf;
576
+ ret += ")";
577
+ }
578
+ return ret;
579
+ }
580
+
581
+ int main(int argc, char *argv[])
582
+ {
583
+ setlocale(LC_ALL, "");
584
+
585
+ g_mainWakeup = new WakeupFd();
586
+
587
+ Arguments args;
588
+ parseArguments(argc, argv, args);
589
+
590
+ setupWin32Environment();
591
+
592
+ winsize sz = { 0 };
593
+ sz.ws_col = 80;
594
+ sz.ws_row = 25;
595
+ ioctl(STDIN_FILENO, TIOCGWINSZ, &sz);
596
+
597
+ DWORD agentFlags = WINPTY_FLAG_ALLOW_CURPROC_DESKTOP_CREATION;
598
+ if (args.testConerr) { agentFlags |= WINPTY_FLAG_CONERR; }
599
+ if (args.testPlainOutput) { agentFlags |= WINPTY_FLAG_PLAIN_OUTPUT; }
600
+ if (args.testColorEscapes) { agentFlags |= WINPTY_FLAG_COLOR_ESCAPES; }
601
+ winpty_config_t *agentCfg = winpty_config_new(agentFlags, NULL);
602
+ assert(agentCfg != NULL);
603
+ winpty_config_set_initial_size(agentCfg, sz.ws_col, sz.ws_row);
604
+ if (args.mouseInput) {
605
+ winpty_config_set_mouse_mode(agentCfg, WINPTY_MOUSE_MODE_FORCE);
606
+ }
607
+
608
+ winpty_error_ptr_t openErr = NULL;
609
+ winpty_t *wp = winpty_open(agentCfg, &openErr);
610
+ if (wp == NULL) {
611
+ fprintf(stderr, "Error creating winpty: %s\n",
612
+ wcsToMbs(winpty_error_msg(openErr)).c_str());
613
+ exit(1);
614
+ }
615
+ winpty_config_free(agentCfg);
616
+ winpty_error_free(openErr);
617
+
618
+ HANDLE conin = CreateFileW(winpty_conin_name(wp), GENERIC_WRITE, 0, NULL,
619
+ OPEN_EXISTING, 0, NULL);
620
+ HANDLE conout = CreateFileW(winpty_conout_name(wp), GENERIC_READ, 0, NULL,
621
+ OPEN_EXISTING, 0, NULL);
622
+ assert(conin != INVALID_HANDLE_VALUE);
623
+ assert(conout != INVALID_HANDLE_VALUE);
624
+ HANDLE conerr = NULL;
625
+ if (args.testConerr) {
626
+ conerr = CreateFileW(winpty_conerr_name(wp), GENERIC_READ, 0, NULL,
627
+ OPEN_EXISTING, 0, NULL);
628
+ assert(conerr != INVALID_HANDLE_VALUE);
629
+ }
630
+
631
+ HANDLE childHandle = NULL;
632
+
633
+ {
634
+ // Start the child process under the console.
635
+ args.childArgv[0] = findProgram(argv[0], args.childArgv[0]);
636
+ std::string cmdLine = argvToCommandLine(args.childArgv);
637
+ wchar_t *cmdLineW = heapMbsToWcs(cmdLine.c_str());
638
+
639
+ winpty_spawn_config_t *spawnCfg = winpty_spawn_config_new(
640
+ WINPTY_SPAWN_FLAG_AUTO_SHUTDOWN,
641
+ NULL, cmdLineW, NULL, NULL, NULL);
642
+ assert(spawnCfg != NULL);
643
+
644
+ winpty_error_ptr_t spawnErr = NULL;
645
+ DWORD lastError = 0;
646
+ BOOL spawnRet = winpty_spawn(wp, spawnCfg, &childHandle, NULL,
647
+ &lastError, &spawnErr);
648
+ winpty_spawn_config_free(spawnCfg);
649
+
650
+ if (!spawnRet) {
651
+ winpty_result_t spawnCode = winpty_error_code(spawnErr);
652
+ if (spawnCode == WINPTY_ERROR_SPAWN_CREATE_PROCESS_FAILED) {
653
+ fprintf(stderr, "%s: error: cannot start '%s': %s\n",
654
+ argv[0],
655
+ cmdLine.c_str(),
656
+ formatErrorMessage(lastError).c_str());
657
+ } else {
658
+ fprintf(stderr, "%s: error: cannot start '%s': internal error: %s\n",
659
+ argv[0],
660
+ cmdLine.c_str(),
661
+ wcsToMbs(winpty_error_msg(spawnErr)).c_str());
662
+ }
663
+ exit(1);
664
+ }
665
+ winpty_error_free(spawnErr);
666
+ delete [] cmdLineW;
667
+ }
668
+
669
+ registerResizeSignalHandler();
670
+ SavedTermiosMode mode =
671
+ setRawTerminalMode(args.testAllowNonTtys, true, args.testConerr);
672
+
673
+ InputHandler inputHandler(conin, STDIN_FILENO, mainWakeup());
674
+ OutputHandler outputHandler(conout, STDOUT_FILENO, mainWakeup());
675
+ OutputHandler *errorHandler = NULL;
676
+ if (args.testConerr) {
677
+ errorHandler = new OutputHandler(conerr, STDERR_FILENO, mainWakeup());
678
+ }
679
+
680
+ while (true) {
681
+ fd_set readfds;
682
+ FD_ZERO(&readfds);
683
+ FD_SET(mainWakeup().fd(), &readfds);
684
+ selectWrapper("main thread", mainWakeup().fd() + 1, &readfds);
685
+ mainWakeup().reset();
686
+
687
+ // Check for terminal resize.
688
+ {
689
+ winsize sz2;
690
+ ioctl(STDIN_FILENO, TIOCGWINSZ, &sz2);
691
+ if (memcmp(&sz, &sz2, sizeof(sz)) != 0) {
692
+ sz = sz2;
693
+ winpty_set_size(wp, sz.ws_col, sz.ws_row, NULL);
694
+ }
695
+ }
696
+
697
+ // Check for an I/O handler shutting down (possibly indicating that the
698
+ // child process has exited).
699
+ if (inputHandler.isComplete() || outputHandler.isComplete() ||
700
+ (errorHandler != NULL && errorHandler->isComplete())) {
701
+ break;
702
+ }
703
+ }
704
+
705
+ // Kill the agent connection. This will kill the agent, closing the CONIN
706
+ // and CONOUT pipes on the agent pipe, prompting our I/O handler to shut
707
+ // down.
708
+ winpty_free(wp);
709
+
710
+ inputHandler.shutdown();
711
+ outputHandler.shutdown();
712
+ CloseHandle(conin);
713
+ CloseHandle(conout);
714
+
715
+ if (errorHandler != NULL) {
716
+ errorHandler->shutdown();
717
+ delete errorHandler;
718
+ CloseHandle(conerr);
719
+ }
720
+
721
+ restoreTerminalMode(mode);
722
+
723
+ DWORD exitCode = 0;
724
+ if (!GetExitCodeProcess(childHandle, &exitCode)) {
725
+ exitCode = 1;
726
+ }
727
+ CloseHandle(childHandle);
728
+ return exitCode;
729
+ }