Merge branch 'v0.6' into v0.8
[platform/upstream/nodejs.git] / src / tty_wrap.cc
1 // Copyright Joyent, Inc. and other Node contributors.
2 //
3 // Permission is hereby granted, free of charge, to any person obtaining a
4 // copy of this software and associated documentation files (the
5 // "Software"), to deal in the Software without restriction, including
6 // without limitation the rights to use, copy, modify, merge, publish,
7 // distribute, sublicense, and/or sell copies of the Software, and to permit
8 // persons to whom the Software is furnished to do so, subject to the
9 // following conditions:
10 //
11 // The above copyright notice and this permission notice shall be included
12 // in all copies or substantial portions of the Software.
13 //
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15 // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
17 // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
18 // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
19 // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
20 // USE OR OTHER DEALINGS IN THE SOFTWARE.
21
22 #include "node.h"
23 #include "node_buffer.h"
24 #include "req_wrap.h"
25 #include "handle_wrap.h"
26 #include "stream_wrap.h"
27 #include "tty_wrap.h"
28
29 namespace node {
30
31 using v8::Object;
32 using v8::Handle;
33 using v8::Local;
34 using v8::Persistent;
35 using v8::Value;
36 using v8::HandleScope;
37 using v8::FunctionTemplate;
38 using v8::String;
39 using v8::Function;
40 using v8::TryCatch;
41 using v8::Context;
42 using v8::Arguments;
43 using v8::Integer;
44 using v8::Undefined;
45
46
47 void TTYWrap::Initialize(Handle<Object> target) {
48   StreamWrap::Initialize(target);
49
50   HandleScope scope;
51
52   Local<FunctionTemplate> t = FunctionTemplate::New(New);
53   t->SetClassName(String::NewSymbol("TTY"));
54
55   t->InstanceTemplate()->SetInternalFieldCount(1);
56
57   NODE_SET_PROTOTYPE_METHOD(t, "close", HandleWrap::Close);
58   NODE_SET_PROTOTYPE_METHOD(t, "unref", HandleWrap::Unref);
59
60   NODE_SET_PROTOTYPE_METHOD(t, "readStart", StreamWrap::ReadStart);
61   NODE_SET_PROTOTYPE_METHOD(t, "readStop", StreamWrap::ReadStop);
62
63   NODE_SET_PROTOTYPE_METHOD(t, "writeBuffer", StreamWrap::WriteBuffer);
64   NODE_SET_PROTOTYPE_METHOD(t, "writeAsciiString", StreamWrap::WriteAsciiString);
65   NODE_SET_PROTOTYPE_METHOD(t, "writeUtf8String", StreamWrap::WriteUtf8String);
66   NODE_SET_PROTOTYPE_METHOD(t, "writeUcs2String", StreamWrap::WriteUcs2String);
67
68   NODE_SET_PROTOTYPE_METHOD(t, "getWindowSize", TTYWrap::GetWindowSize);
69   NODE_SET_PROTOTYPE_METHOD(t, "setRawMode", SetRawMode);
70
71   NODE_SET_METHOD(target, "isTTY", IsTTY);
72   NODE_SET_METHOD(target, "guessHandleType", GuessHandleType);
73
74   target->Set(String::NewSymbol("TTY"), t->GetFunction());
75 }
76
77
78 TTYWrap* TTYWrap::Unwrap(Local<Object> obj) {
79   assert(!obj.IsEmpty());
80   assert(obj->InternalFieldCount() > 0);
81   return static_cast<TTYWrap*>(obj->GetPointerFromInternalField(0));
82 }
83
84
85 uv_tty_t* TTYWrap::UVHandle() {
86   return &handle_;
87 }
88
89
90 Handle<Value> TTYWrap::GuessHandleType(const Arguments& args) {
91   HandleScope scope;
92   int fd = args[0]->Int32Value();
93   assert(fd >= 0);
94
95   uv_handle_type t = uv_guess_handle(fd);
96
97   switch (t) {
98     case UV_TTY:
99       return scope.Close(String::New("TTY"));
100
101     case UV_NAMED_PIPE:
102       return scope.Close(String::New("PIPE"));
103
104     case UV_FILE:
105       return scope.Close(String::New("FILE"));
106
107     default:
108       assert(0);
109       return v8::Undefined();
110   }
111 }
112
113
114 Handle<Value> TTYWrap::IsTTY(const Arguments& args) {
115   HandleScope scope;
116   int fd = args[0]->Int32Value();
117   assert(fd >= 0);
118   return uv_guess_handle(fd) == UV_TTY ? v8::True() : v8::False();
119 }
120
121
122 Handle<Value> TTYWrap::GetWindowSize(const Arguments& args) {
123   HandleScope scope;
124
125   UNWRAP(TTYWrap)
126
127   int width, height;
128   int r = uv_tty_get_winsize(&wrap->handle_, &width, &height);
129
130   if (r) {
131     SetErrno(uv_last_error(uv_default_loop()));
132     return v8::Undefined();
133   }
134
135   Local<v8::Array> a = v8::Array::New(2);
136   a->Set(0, Integer::New(width));
137   a->Set(1, Integer::New(height));
138
139   return scope.Close(a);
140 }
141
142
143 Handle<Value> TTYWrap::SetRawMode(const Arguments& args) {
144   HandleScope scope;
145
146   UNWRAP(TTYWrap)
147
148   int r = uv_tty_set_mode(&wrap->handle_, args[0]->IsTrue());
149
150   if (r) {
151     SetErrno(uv_last_error(uv_default_loop()));
152   }
153
154   return scope.Close(Integer::New(r));
155 }
156
157
158 Handle<Value> TTYWrap::New(const Arguments& args) {
159   HandleScope scope;
160
161   // This constructor should not be exposed to public javascript.
162   // Therefore we assert that we are not trying to call this as a
163   // normal function.
164   assert(args.IsConstructCall());
165
166   int fd = args[0]->Int32Value();
167   assert(fd >= 0);
168
169   TTYWrap* wrap = new TTYWrap(args.This(), fd, args[1]->IsTrue());
170   assert(wrap);
171   wrap->UpdateWriteQueueSize();
172
173   return scope.Close(args.This());
174 }
175
176
177 TTYWrap::TTYWrap(Handle<Object> object, int fd, bool readable)
178     : StreamWrap(object, (uv_stream_t*)&handle_) {
179   uv_tty_init(uv_default_loop(), &handle_, fd, readable);
180 }
181
182 }  // namespace node
183
184 NODE_MODULE(node_tty_wrap, node::TTYWrap::Initialize)