mirror of https://github.com/nodejs/node.git
728 lines
19 KiB
C++
728 lines
19 KiB
C++
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
#include <node.h>
|
|
#include <node_stdio.h>
|
|
|
|
#include <v8.h>
|
|
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <io.h>
|
|
|
|
#include <platform_win32.h>
|
|
|
|
using namespace v8;
|
|
namespace node {
|
|
|
|
#define THROW_ERROR(msg) \
|
|
return ThrowException(Exception::Error(String::New(msg)));
|
|
#define THROW_BAD_ARGS \
|
|
return ThrowException(Exception::TypeError(String::New("Bad argument")));
|
|
|
|
#define KEY(scancode, name) \
|
|
scancodes[scancode] = name;
|
|
#define MAX_KEY VK_OEM_PERIOD
|
|
|
|
static const char* scancodes[MAX_KEY + 1] = {0};
|
|
|
|
static Persistent<String> name_symbol;
|
|
static Persistent<String> shift_symbol;
|
|
static Persistent<String> ctrl_symbol;
|
|
static Persistent<String> meta_symbol;
|
|
|
|
|
|
static void init_scancode_table() {
|
|
KEY(VK_CANCEL, "break")
|
|
KEY(VK_BACK, "backspace")
|
|
KEY(VK_TAB, "tab")
|
|
KEY(VK_CLEAR, "clear")
|
|
KEY(VK_RETURN, "enter")
|
|
KEY(VK_PAUSE, "pause")
|
|
KEY(VK_ESCAPE, "escape")
|
|
KEY(VK_SPACE, "space")
|
|
KEY(VK_PRIOR, "pageup")
|
|
KEY(VK_NEXT, "pagedown")
|
|
KEY(VK_END, "end")
|
|
KEY(VK_HOME, "home")
|
|
KEY(VK_LEFT, "left")
|
|
KEY(VK_UP, "up")
|
|
KEY(VK_RIGHT, "right")
|
|
KEY(VK_DOWN, "down")
|
|
KEY(VK_SELECT, "select")
|
|
KEY(VK_PRINT, "print")
|
|
KEY(VK_EXECUTE, "execute")
|
|
KEY(VK_SNAPSHOT, "printscreen")
|
|
KEY(VK_INSERT, "insert")
|
|
KEY(VK_DELETE, "delete")
|
|
KEY(VK_HELP, "help")
|
|
KEY(VK_LWIN, "lwin")
|
|
KEY(VK_RWIN, "rwin")
|
|
KEY(VK_APPS, "apps")
|
|
KEY(VK_SLEEP, "sleep")
|
|
KEY(VK_NUMPAD0, "numpad0")
|
|
KEY(VK_NUMPAD1, "numpad1")
|
|
KEY(VK_NUMPAD2, "numpad2")
|
|
KEY(VK_NUMPAD3, "numpad3")
|
|
KEY(VK_NUMPAD4, "numpad4")
|
|
KEY(VK_NUMPAD5, "numpad5")
|
|
KEY(VK_NUMPAD6, "numpad6")
|
|
KEY(VK_NUMPAD7, "numpad7")
|
|
KEY(VK_NUMPAD8, "numpad8")
|
|
KEY(VK_NUMPAD9, "numpad9")
|
|
KEY(VK_MULTIPLY, "numpad*")
|
|
KEY(VK_ADD, "numpad+")
|
|
KEY(VK_SEPARATOR, "numpad,")
|
|
KEY(VK_SUBTRACT, "numpad-")
|
|
KEY(VK_DECIMAL, "numpad.")
|
|
KEY(VK_DIVIDE, "numpad/")
|
|
KEY(VK_F1, "f1")
|
|
KEY(VK_F2, "f2")
|
|
KEY(VK_F3, "f3")
|
|
KEY(VK_F4, "f4")
|
|
KEY(VK_F5, "f5")
|
|
KEY(VK_F6, "f6")
|
|
KEY(VK_F7, "f7")
|
|
KEY(VK_F8, "f8")
|
|
KEY(VK_F9, "f9")
|
|
KEY(VK_F10, "f10")
|
|
KEY(VK_F11, "f11")
|
|
KEY(VK_F12, "f12")
|
|
KEY(VK_F13, "f13")
|
|
KEY(VK_F14, "f14")
|
|
KEY(VK_F15, "f15")
|
|
KEY(VK_F16, "f16")
|
|
KEY(VK_F17, "f17")
|
|
KEY(VK_F18, "f18")
|
|
KEY(VK_F19, "f19")
|
|
KEY(VK_F20, "f20")
|
|
KEY(VK_F21, "f21")
|
|
KEY(VK_F22, "f22")
|
|
KEY(VK_F23, "f23")
|
|
KEY(VK_F24, "f24")
|
|
KEY(VK_OEM_PLUS, "+")
|
|
KEY(VK_OEM_MINUS, "-")
|
|
KEY(VK_OEM_COMMA, ",")
|
|
KEY(VK_OEM_PERIOD, ".")
|
|
|
|
// Letter keys have the ascii code of their uppercase equivalent as a scan code
|
|
for (int i = 0; i < 26; i++) {
|
|
char *name = new char[2];
|
|
name[0] = 'a' + i;
|
|
name[1] = '\0';
|
|
KEY('A' + i, name)
|
|
}
|
|
|
|
// Number keys have their ascii code as scan code
|
|
for (int i = '0'; i <= '9'; i++) {
|
|
char *name = new char[2];
|
|
name[0] = i;
|
|
name[1] = '\0';
|
|
KEY(i, name)
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Flush stdout and stderr on node exit
|
|
* Not necessary on windows, so a no-op
|
|
*/
|
|
void Stdio::Flush() {
|
|
}
|
|
|
|
|
|
/*
|
|
* STDERR should always be blocking
|
|
*/
|
|
static Handle<Value> WriteError(const Arguments& args) {
|
|
HandleScope scope;
|
|
|
|
if (args.Length() < 1)
|
|
return Undefined();
|
|
|
|
String::Utf8Value msg(args[0]->ToString());
|
|
|
|
fprintf(stderr, "%s", reinterpret_cast<char*>(*msg));
|
|
|
|
return Undefined();
|
|
}
|
|
|
|
|
|
static Handle<Value> IsATTY(const Arguments& args) {
|
|
HandleScope scope;
|
|
int fd = args[0]->IntegerValue();
|
|
DWORD result;
|
|
int r = GetConsoleMode((HANDLE)_get_osfhandle(fd), &result);
|
|
return scope.Close(r ? True() : False());
|
|
}
|
|
|
|
|
|
/* Whether stdio is currently in raw mode */
|
|
/* -1 means that it has not been set */
|
|
static int rawMode = -1;
|
|
static DWORD naturalMode = 0;
|
|
|
|
static void setRawMode(int newMode) {
|
|
DWORD flags = 0;
|
|
BOOL result;
|
|
|
|
if(rawMode == -1) {
|
|
GetConsoleMode((HANDLE)_get_osfhandle(STDIN_FILENO), &naturalMode);
|
|
}
|
|
|
|
flags |= (naturalMode & ENABLE_QUICK_EDIT_MODE);
|
|
flags |= flags ? ENABLE_EXTENDED_FLAGS : 0;
|
|
|
|
if (newMode != rawMode) {
|
|
if (newMode) {
|
|
// raw input
|
|
flags |= ENABLE_WINDOW_INPUT;
|
|
} else {
|
|
// input not raw, but still processing enough messages to make the
|
|
// tty watcher work (this mode is not the windows default)
|
|
flags |= ENABLE_ECHO_INPUT | ENABLE_INSERT_MODE | ENABLE_LINE_INPUT |
|
|
ENABLE_EXTENDED_FLAGS | ENABLE_PROCESSED_INPUT | ENABLE_WINDOW_INPUT;
|
|
}
|
|
|
|
result = SetConsoleMode((HANDLE)_get_osfhandle(STDIN_FILENO), flags);
|
|
if (result) {
|
|
rawMode = newMode;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static Handle<Value> SetRawMode(const Arguments& args) {
|
|
HandleScope scope;
|
|
|
|
int newMode = !args[0]->IsFalse();
|
|
setRawMode(newMode);
|
|
|
|
if (newMode != rawMode) {
|
|
return ThrowException(ErrnoException(GetLastError(), "EnableRawMode"));
|
|
}
|
|
|
|
return scope.Close(rawMode ? True() : False());
|
|
}
|
|
|
|
|
|
|
|
void Stdio::DisableRawMode(int fd) {
|
|
if (rawMode == 1)
|
|
setRawMode(0);
|
|
}
|
|
|
|
|
|
static Handle<Value> OpenStdin(const Arguments& args) {
|
|
HandleScope scope;
|
|
setRawMode(0); // init into nonraw mode
|
|
return scope.Close(Integer::New(STDIN_FILENO));
|
|
}
|
|
|
|
|
|
static Handle<Value> IsStdinBlocking(const Arguments& args) {
|
|
// On windows stdin always blocks
|
|
return True();
|
|
}
|
|
|
|
|
|
static Handle<Value> IsStdoutBlocking(const Arguments& args) {
|
|
// On windows stdout always blocks
|
|
return True();
|
|
}
|
|
|
|
|
|
static Handle<Value> WriteTTY(const Arguments& args) {
|
|
HandleScope scope;
|
|
int fd, len;
|
|
DWORD written;
|
|
HANDLE handle;
|
|
|
|
if (!args[0]->IsNumber())
|
|
THROW_BAD_ARGS
|
|
|
|
fd = args[0]->IntegerValue();
|
|
handle = (HANDLE)_get_osfhandle(fd);
|
|
|
|
Handle<String> data = args[1]->ToString();
|
|
String::Value buf(data);
|
|
len = data->Length();
|
|
|
|
if (!WriteConsoleW(handle, reinterpret_cast<void*>(*buf), len, &written, NULL))
|
|
return ThrowException(ErrnoException(GetLastError(), "WriteConsole"));
|
|
|
|
return scope.Close(Integer::New(written));
|
|
}
|
|
|
|
|
|
static Handle<Value> CloseTTY(const Arguments& args) {
|
|
HandleScope scope;
|
|
|
|
int fd = args[0]->IntegerValue();
|
|
if (close(fd) < 0)
|
|
return ThrowException(ErrnoException(errno, "close"));
|
|
|
|
return Undefined();
|
|
}
|
|
|
|
|
|
// process.binding('stdio').getWindowSize(fd);
|
|
// returns [row, col]
|
|
static Handle<Value> GetWindowSize (const Arguments& args) {
|
|
HandleScope scope;
|
|
int fd;
|
|
HANDLE handle;
|
|
CONSOLE_SCREEN_BUFFER_INFO info;
|
|
|
|
if (!args[0]->IsNumber())
|
|
THROW_BAD_ARGS
|
|
fd = args[0]->IntegerValue();
|
|
handle = (HANDLE)_get_osfhandle(fd);
|
|
|
|
if (!GetConsoleScreenBufferInfo(handle, &info))
|
|
return ThrowException(ErrnoException(GetLastError(), "GetConsoleScreenBufferInfo"));
|
|
|
|
Local<Array> ret = Array::New(2);
|
|
ret->Set(0, Integer::New(static_cast<int>(info.dwSize.Y)));
|
|
ret->Set(1, Integer::New(static_cast<int>(info.dwSize.X)));
|
|
|
|
return scope.Close(ret);
|
|
}
|
|
|
|
|
|
/* moveCursor(fd, dx, dy) */
|
|
/* cursorTo(fd, x, y) */
|
|
template<bool relative>
|
|
static Handle<Value> SetCursor(const Arguments& args) {
|
|
HandleScope scope;
|
|
int fd;
|
|
COORD size, pos;
|
|
HANDLE handle;
|
|
CONSOLE_SCREEN_BUFFER_INFO info;
|
|
|
|
if (!args[0]->IsNumber())
|
|
THROW_BAD_ARGS
|
|
fd = args[0]->IntegerValue();
|
|
handle = (HANDLE)_get_osfhandle(fd);
|
|
|
|
if (!GetConsoleScreenBufferInfo(handle, &info))
|
|
return ThrowException(ErrnoException(GetLastError(), "GetConsoleScreenBufferInfo"));
|
|
|
|
pos = info.dwCursorPosition;
|
|
if (relative) {
|
|
if (args[1]->IsNumber())
|
|
pos.X += static_cast<short>(args[1]->Int32Value());
|
|
if (args[2]->IsNumber())
|
|
pos.Y += static_cast<short>(args[2]->Int32Value());
|
|
} else {
|
|
if (args[1]->IsNumber())
|
|
pos.X = static_cast<short>(args[1]->Int32Value());
|
|
if (args[2]->IsNumber())
|
|
pos.Y = static_cast<short>(args[2]->Int32Value());
|
|
}
|
|
|
|
size = info.dwSize;
|
|
if (pos.X >= size.X) pos.X = size.X - 1;
|
|
if (pos.X < 0) pos.X = 0;
|
|
if (pos.Y >= size.Y) pos.Y = size.Y - 1;
|
|
if (pos.Y < 0) pos.Y = 0;
|
|
|
|
if (!SetConsoleCursorPosition(handle, pos))
|
|
return ThrowException(ErrnoException(GetLastError(), "SetConsoleCursorPosition"));
|
|
|
|
return Undefined();
|
|
}
|
|
|
|
|
|
/*
|
|
* ClearLine(fd, direction)
|
|
* direction:
|
|
* -1: from cursor leftward
|
|
* 0: entire line
|
|
* 1: from cursor to right
|
|
*/
|
|
static Handle<Value> ClearLine(const Arguments& args) {
|
|
HandleScope scope;
|
|
int fd, dir;
|
|
short x1, x2, count;
|
|
WCHAR *buf;
|
|
COORD pos;
|
|
HANDLE handle;
|
|
CONSOLE_SCREEN_BUFFER_INFO info;
|
|
DWORD res, written, mode, oldmode;
|
|
|
|
if (!args[0]->IsNumber())
|
|
THROW_BAD_ARGS
|
|
fd = args[0]->IntegerValue();
|
|
handle = (HANDLE)_get_osfhandle(fd);
|
|
|
|
if (args[1]->IsNumber())
|
|
dir = args[1]->IntegerValue();
|
|
|
|
if (!GetConsoleScreenBufferInfo(handle, &info))
|
|
return ThrowException(ErrnoException(GetLastError(), "GetConsoleScreenBufferInfo"));
|
|
|
|
x1 = dir <= 0 ? 0 : info.dwCursorPosition.X;
|
|
x2 = dir >= 0 ? info.dwSize.X - 1: info.dwCursorPosition.X;
|
|
count = x2 - x1 + 1;
|
|
|
|
if (x1 != info.dwCursorPosition.X) {
|
|
pos.Y = info.dwCursorPosition.Y;
|
|
pos.X = x1;
|
|
if (!SetConsoleCursorPosition(handle, pos))
|
|
return ThrowException(ErrnoException(GetLastError(), "SetConsoleCursorPosition"));
|
|
}
|
|
|
|
if (!GetConsoleMode(handle, &oldmode))
|
|
return ThrowException(ErrnoException(GetLastError(), "GetConsoleMode"));
|
|
|
|
// Disable wrapping at eol because otherwise windows scrolls the console
|
|
// when clearing the last line of the console
|
|
mode = oldmode & ~ENABLE_WRAP_AT_EOL_OUTPUT;
|
|
if (!SetConsoleMode(handle, mode))
|
|
return ThrowException(ErrnoException(GetLastError(), "SetConsoleMode"));
|
|
|
|
buf = new WCHAR[count];
|
|
for (short i = 0; i < count; i++) {
|
|
buf[i] = L' ';
|
|
}
|
|
|
|
res = WriteConsoleW(handle, buf, count, &written, NULL);
|
|
|
|
delete[] buf;
|
|
|
|
if (!res)
|
|
return ThrowException(ErrnoException(GetLastError(), "WriteConsole"));
|
|
|
|
if (!SetConsoleCursorPosition(handle, info.dwCursorPosition))
|
|
return ThrowException(ErrnoException(GetLastError(), "SetConsoleCursorPosition"));
|
|
|
|
if (!SetConsoleMode(handle, oldmode))
|
|
return ThrowException(ErrnoException(GetLastError(), "SetConsoleMode"));
|
|
|
|
return Undefined();
|
|
}
|
|
|
|
|
|
/* TTY watcher data */
|
|
HANDLE tty_handle;
|
|
HANDLE tty_wait_handle;
|
|
void *tty_error_callback;
|
|
void *tty_keypress_callback;
|
|
void *tty_resize_callback;
|
|
static bool tty_watcher_initialized = false;
|
|
static bool tty_watcher_active = false;
|
|
static uv_async_t tty_avail_notifier;
|
|
|
|
|
|
static void CALLBACK tty_want_poll(void *context, BOOLEAN didTimeout) {
|
|
assert(!didTimeout);
|
|
uv_async_send(&tty_avail_notifier);
|
|
}
|
|
|
|
|
|
static void tty_watcher_arm() {
|
|
// Register a new wait handle before dropping the old one, because
|
|
// otherwise windows might destroy and recreate the wait thread.
|
|
// MSDN promises that thread pool threads are kept alive when they're idle,
|
|
// but apparently this does not apply to wait threads. Sigh.
|
|
|
|
HANDLE old_wait_handle = tty_wait_handle;
|
|
tty_wait_handle = NULL;
|
|
|
|
assert(tty_watcher_active);
|
|
|
|
if (!RegisterWaitForSingleObject(&tty_wait_handle, tty_handle, tty_want_poll, NULL,
|
|
INFINITE, WT_EXECUTEINWAITTHREAD | WT_EXECUTEONLYONCE)) {
|
|
ThrowException(ErrnoException(GetLastError(), "RegisterWaitForSingleObject"));
|
|
}
|
|
|
|
if (old_wait_handle != NULL) {
|
|
if (!UnregisterWait(old_wait_handle) && GetLastError() != ERROR_IO_PENDING)
|
|
ThrowException(ErrnoException(GetLastError(), "UnregisterWait"));
|
|
}
|
|
}
|
|
|
|
|
|
static void tty_watcher_disarm() {
|
|
DWORD result;
|
|
if (tty_wait_handle != NULL) {
|
|
result = UnregisterWait(tty_wait_handle);
|
|
tty_wait_handle = NULL;
|
|
if (!result && GetLastError() != ERROR_IO_PENDING)
|
|
ThrowException(ErrnoException(GetLastError(), "UnregisterWait"));
|
|
}
|
|
}
|
|
|
|
|
|
static void tty_watcher_start() {
|
|
assert(tty_watcher_initialized);
|
|
if (!tty_watcher_active) {
|
|
tty_watcher_active = true;
|
|
uv_ref(uv_default_loop());
|
|
tty_watcher_arm();
|
|
}
|
|
}
|
|
|
|
|
|
static void tty_watcher_stop() {
|
|
if (tty_watcher_active) {
|
|
tty_watcher_active = false;
|
|
uv_unref(uv_default_loop());
|
|
tty_watcher_disarm();
|
|
}
|
|
}
|
|
|
|
|
|
static inline void tty_emit_error(Handle<Value> err) {
|
|
HandleScope scope;
|
|
Handle<Object> global = v8::Context::GetCurrent()->Global();
|
|
Handle<Function> *handler = cb_unwrap(tty_error_callback);
|
|
Handle<Value> argv[1] = { err };
|
|
(*handler)->Call(global, 1, argv);
|
|
}
|
|
|
|
|
|
static void tty_poll(uv_async_t* handle, int status) {
|
|
assert(handle == &tty_avail_notifier);
|
|
assert(status == 0);
|
|
|
|
HandleScope scope;
|
|
TryCatch try_catch;
|
|
Handle<Object> global = v8::Context::GetCurrent()->Global();
|
|
Handle<Function> *callback;
|
|
INPUT_RECORD input;
|
|
KEY_EVENT_RECORD k;
|
|
const char *keyName;
|
|
DWORD i, j, numev, read;
|
|
Handle<Value> argv[2];
|
|
Handle<Object> key;
|
|
|
|
if (!GetNumberOfConsoleInputEvents(tty_handle, &numev)) {
|
|
tty_emit_error(ErrnoException(GetLastError(),
|
|
"GetNumberOfConsoleInputEvents"));
|
|
numev = 0;
|
|
}
|
|
|
|
for (i = numev; i > 0 &&
|
|
tty_watcher_active; i--) {
|
|
if (!ReadConsoleInputW(tty_handle, &input, 1, &read)) {
|
|
tty_emit_error(ErrnoException(GetLastError(), "ReadConsoleInputW"));
|
|
break;
|
|
}
|
|
|
|
switch (input.EventType) {
|
|
case KEY_EVENT:
|
|
// Skip if no callback set
|
|
if (!tty_keypress_callback)
|
|
break;
|
|
|
|
k = input.Event.KeyEvent;
|
|
|
|
// Ignore keyup
|
|
if (!k.bKeyDown)
|
|
break;
|
|
|
|
// Try to find a symbolic name for the key
|
|
keyName = (k.wVirtualKeyCode <= MAX_KEY)
|
|
? scancodes[k.wVirtualKeyCode]
|
|
: 0;
|
|
|
|
// The key must have a symbolic name or a char or both
|
|
if (k.uChar.UnicodeChar == 0 && keyName == 0)
|
|
break;
|
|
|
|
// Set the event name and character
|
|
argv[0] = k.uChar.UnicodeChar
|
|
? String::New(reinterpret_cast<uint16_t*>(&k.uChar.UnicodeChar), 1)
|
|
: Undefined();
|
|
|
|
// Set the key info, if any
|
|
if (keyName) {
|
|
key = Object::New();
|
|
key->Set(name_symbol, String::New(keyName));
|
|
key->Set(shift_symbol, Boolean::New(k.dwControlKeyState &
|
|
SHIFT_PRESSED));
|
|
key->Set(ctrl_symbol, Boolean::New(k.dwControlKeyState &
|
|
(LEFT_CTRL_PRESSED | RIGHT_CTRL_PRESSED)));
|
|
key->Set(meta_symbol, Boolean::New(k.dwControlKeyState &
|
|
(LEFT_ALT_PRESSED | RIGHT_ALT_PRESSED)));
|
|
argv[1] = key;
|
|
} else {
|
|
argv[1] = Undefined();
|
|
}
|
|
|
|
callback = cb_unwrap(tty_keypress_callback);
|
|
j = k.wRepeatCount;
|
|
do {
|
|
(*callback)->Call(global, 2, argv);
|
|
} while (--j > 0 && tty_watcher_active);
|
|
break;
|
|
|
|
case WINDOW_BUFFER_SIZE_EVENT:
|
|
if (!tty_resize_callback)
|
|
break;
|
|
callback = cb_unwrap(tty_resize_callback);
|
|
(*callback)->Call(global, 0, argv);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Rearm the watcher
|
|
if (tty_watcher_active) {
|
|
tty_watcher_arm();
|
|
}
|
|
|
|
// Emit fatal errors and unhandled error events
|
|
if (try_catch.HasCaught()) {
|
|
FatalException(try_catch);
|
|
}
|
|
}
|
|
|
|
|
|
/* StartTTYWatcher(fd, onError, onKeypress, onResize) */
|
|
static Handle<Value> InitTTYWatcher(const Arguments& args) {
|
|
HandleScope scope;
|
|
|
|
if (tty_watcher_initialized)
|
|
THROW_ERROR("TTY watcher already initialized")
|
|
|
|
if (!args[0]->IsNumber())
|
|
THROW_BAD_ARGS;
|
|
tty_handle = (HANDLE)_get_osfhandle(args[0]->IntegerValue());
|
|
|
|
if (!args[1]->IsFunction())
|
|
THROW_BAD_ARGS;
|
|
tty_error_callback = cb_persist(args[1]);
|
|
|
|
tty_keypress_callback = args[2]->IsFunction()
|
|
? cb_persist(args[2])
|
|
: NULL;
|
|
|
|
tty_resize_callback = args[3]->IsFunction()
|
|
? cb_persist(args[3])
|
|
: NULL;
|
|
|
|
tty_watcher_initialized = true;
|
|
tty_wait_handle = NULL;
|
|
|
|
return Undefined();
|
|
}
|
|
|
|
|
|
static Handle<Value> DestroyTTYWatcher(const Arguments& args) {
|
|
if (!tty_watcher_initialized)
|
|
THROW_ERROR("TTY watcher not initialized")
|
|
|
|
tty_watcher_stop();
|
|
|
|
if (tty_error_callback != NULL)
|
|
cb_destroy(cb_unwrap(tty_error_callback));
|
|
if (tty_keypress_callback != NULL)
|
|
cb_destroy(cb_unwrap(tty_keypress_callback));
|
|
if (tty_resize_callback != NULL)
|
|
cb_destroy(cb_unwrap(tty_resize_callback));
|
|
|
|
tty_watcher_initialized = false;
|
|
|
|
return Undefined();
|
|
}
|
|
|
|
|
|
static Handle<Value> StartTTYWatcher(const Arguments& args) {
|
|
if (!tty_watcher_initialized)
|
|
THROW_ERROR("TTY watcher not initialized")
|
|
|
|
tty_watcher_start();
|
|
return Undefined();
|
|
}
|
|
|
|
|
|
static Handle<Value> StopTTYWatcher(const Arguments& args) {
|
|
if (!tty_watcher_initialized)
|
|
THROW_ERROR("TTY watcher not initialized")
|
|
|
|
tty_watcher_stop();
|
|
return Undefined();
|
|
}
|
|
|
|
// This exists to prevent process.stdout from keeping the event loop alive.
|
|
// It is only ever called in src/node.js during the initalization of
|
|
// process.stdout and will fail if called more than once. We do not want to
|
|
// expose uv_ref and uv_unref to javascript in general.
|
|
// This should be removed in the future!
|
|
static bool unref_called = false;
|
|
static Handle<Value> Unref(const Arguments& args) {
|
|
HandleScope scope;
|
|
|
|
assert(unref_called == false);
|
|
|
|
//uv_unref(uv_default_loop());
|
|
unref_called = true;
|
|
|
|
return Null();
|
|
}
|
|
|
|
|
|
void Stdio::Initialize(v8::Handle<v8::Object> target) {
|
|
init_scancode_table();
|
|
|
|
uv_async_init(uv_default_loop(), &tty_avail_notifier, tty_poll);
|
|
uv_unref(uv_default_loop());
|
|
|
|
/* Set stdio streams to binary mode. */
|
|
_setmode(_fileno(stdin), _O_BINARY);
|
|
_setmode(_fileno(stdout), _O_BINARY);
|
|
_setmode(_fileno(stderr), _O_BINARY);
|
|
|
|
name_symbol = NODE_PSYMBOL("name");
|
|
shift_symbol = NODE_PSYMBOL("shift");
|
|
ctrl_symbol = NODE_PSYMBOL("ctrl");
|
|
meta_symbol = NODE_PSYMBOL("meta");
|
|
|
|
target->Set(String::NewSymbol("stdoutFD"), Integer::New(STDOUT_FILENO));
|
|
target->Set(String::NewSymbol("stderrFD"), Integer::New(STDERR_FILENO));
|
|
target->Set(String::NewSymbol("stdinFD"), Integer::New(STDIN_FILENO));
|
|
|
|
NODE_SET_METHOD(target, "writeError", WriteError);
|
|
NODE_SET_METHOD(target, "isatty", IsATTY);
|
|
NODE_SET_METHOD(target, "isStdoutBlocking", IsStdoutBlocking);
|
|
NODE_SET_METHOD(target, "isStdinBlocking", IsStdinBlocking);
|
|
NODE_SET_METHOD(target, "setRawMode", SetRawMode);
|
|
NODE_SET_METHOD(target, "openStdin", OpenStdin);
|
|
NODE_SET_METHOD(target, "writeTTY", WriteTTY);
|
|
NODE_SET_METHOD(target, "closeTTY", CloseTTY);
|
|
NODE_SET_METHOD(target, "moveCursor", SetCursor<true>);
|
|
NODE_SET_METHOD(target, "cursorTo", SetCursor<false>);
|
|
NODE_SET_METHOD(target, "clearLine", ClearLine);
|
|
NODE_SET_METHOD(target, "getWindowSize", GetWindowSize);
|
|
NODE_SET_METHOD(target, "initTTYWatcher", InitTTYWatcher);
|
|
NODE_SET_METHOD(target, "destroyTTYWatcher", DestroyTTYWatcher);
|
|
NODE_SET_METHOD(target, "startTTYWatcher", StartTTYWatcher);
|
|
NODE_SET_METHOD(target, "stopTTYWatcher", StopTTYWatcher);
|
|
NODE_SET_METHOD(target, "unref", Unref);
|
|
}
|
|
|
|
|
|
} // namespace node
|
|
|
|
NODE_MODULE(node_stdio, node::Stdio::Initialize);
|