Index: mojo/bindings/js/drain_data.cc |
diff --git a/mojo/bindings/js/drain_data.cc b/mojo/bindings/js/drain_data.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..a615cd62ce49a35ea55c8e09df682813c0e83f82 |
--- /dev/null |
+++ b/mojo/bindings/js/drain_data.cc |
@@ -0,0 +1,131 @@ |
+// Copyright 2014 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "mojo/bindings/js/drain_data.h" |
+ |
+#include "gin/array_buffer.h" |
+#include "gin/converter.h" |
+#include "gin/dictionary.h" |
+#include "gin/per_context_data.h" |
+#include "gin/per_isolate_data.h" |
+#include "mojo/public/cpp/environment/environment.h" |
+#include "mojo/public/cpp/system/core.h" |
+ |
+namespace mojo { |
+namespace js { |
+ |
+DrainData::DrainData(v8::Isolate* isolate, mojo::Handle handle) |
+ : isolate_(isolate), |
+ handle_(DataPipeConsumerHandle(handle.value())), |
+ wait_id_(0) { |
+ |
+ v8::Handle<v8::Context> context(isolate_->GetCurrentContext()); |
+ runner_ = gin::PerContextData::From(context)->runner()->GetWeakPtr(); |
+ |
+ WaitForData(); |
+} |
+ |
+v8::Handle<v8::Value> DrainData::GetPromise() { |
+ CHECK(resolver_.IsEmpty()); |
+ v8::Handle<v8::Promise::Resolver> resolver( |
+ v8::Promise::Resolver::New(isolate_)); |
+ resolver_.Reset(isolate_, resolver); |
+ return resolver->GetPromise(); |
+} |
+ |
+DrainData::~DrainData() { |
+ if (wait_id_) |
+ Environment::GetDefaultAsyncWaiter()->CancelWait(wait_id_); |
+ resolver_.Reset(); |
+} |
+ |
+void DrainData::WaitForData() { |
+ wait_id_ = Environment::GetDefaultAsyncWaiter()->AsyncWait( |
+ handle_.get().value(), |
+ MOJO_HANDLE_SIGNAL_READABLE, |
+ MOJO_DEADLINE_INDEFINITE, |
+ &DrainData::WaitCompleted, |
+ this); |
+} |
+ |
+void DrainData::DataReady(MojoResult result) { |
+ wait_id_ = 0; |
+ if (result != MOJO_RESULT_OK) { |
+ DeliverData(result); |
+ return; |
+ } |
+ while (result == MOJO_RESULT_OK) { |
+ result = ReadData(); |
+ if (result == MOJO_RESULT_SHOULD_WAIT) |
+ WaitForData(); |
+ else if (result != MOJO_RESULT_OK) |
+ DeliverData(result); |
+ } |
+} |
+ |
+MojoResult DrainData::ReadData() { |
+ const void* buffer; |
+ uint32_t num_bytes = 0; |
+ MojoResult result = BeginReadDataRaw( |
+ handle_.get(), &buffer, &num_bytes, MOJO_READ_DATA_FLAG_NONE); |
+ if (result != MOJO_RESULT_OK) |
+ return result; |
+ const char* p = static_cast<const char*>(buffer); |
+ DataBuffer* data_buffer = new DataBuffer(p, p + num_bytes); |
+ data_buffers_.push_back(data_buffer); |
+ return EndReadDataRaw(handle_.get(), num_bytes); |
+} |
+ |
+void DrainData::DeliverData(MojoResult result) { |
+ if (!runner_) { |
+ delete this; |
+ return; |
+ } |
+ |
+ size_t total_bytes = 0; |
+ for (unsigned i = 0; i < data_buffers_.size(); i++) |
+ total_bytes += data_buffers_[i]->size(); |
+ |
+ // Create a total_bytes length ArrayBuffer return value. |
+ gin::Runner::Scope scope(runner_.get()); |
+ v8::Handle<v8::ArrayBuffer> array_buffer = |
+ v8::ArrayBuffer::New(isolate_, total_bytes); |
+ gin::ArrayBuffer buffer; |
+ ConvertFromV8(isolate_, array_buffer, &buffer); |
+ CHECK_EQ(total_bytes, buffer.num_bytes()); |
+ |
+ // Copy the data_buffers into the ArrayBuffer. |
+ char* array_buffer_ptr = static_cast<char*>(buffer.bytes()); |
+ size_t offset = 0; |
+ for (size_t i = 0; i < data_buffers_.size(); i++) { |
+ size_t num_bytes = data_buffers_[i]->size(); |
+ if (num_bytes == 0) |
+ continue; |
+ const char* data_buffer_ptr = &((*data_buffers_[i])[0]); |
+ memcpy(array_buffer_ptr + offset, data_buffer_ptr, num_bytes); |
+ offset += num_bytes; |
+ } |
+ |
+ // The "settled" value of the promise always includes all of the data |
+ // that was read before either an error occurred or the remote pipe handle |
+ // was closed. The latter is indicated by MOJO_RESULT_FAILED_PRECONDITION. |
+ |
+ v8::Handle<v8::Promise::Resolver> resolver( |
+ v8::Local<v8::Promise::Resolver>::New(isolate_, resolver_)); |
+ |
+ gin::Dictionary dictionary = gin::Dictionary::CreateEmpty(isolate_); |
+ dictionary.Set("result", result); |
+ dictionary.Set("buffer", array_buffer); |
+ v8::Handle<v8::Value> settled_value(ConvertToV8(isolate_, dictionary)); |
+ |
+ if (result == MOJO_RESULT_FAILED_PRECONDITION) |
+ resolver->Resolve(settled_value); |
+ else |
+ resolver->Reject(settled_value); |
+ |
+ delete this; |
+} |
+ |
+} // namespace js |
+} // namespace mojo |