Dart: Merge message pipe query and read into one native call

The native call uses a persistent message buffer per isolate.

This is a relanding of:

https://codereview.chromium.org/1405103002/ and
https://codereview.chromium.org/1410053002/

without using isolate data.

R=johnmccutchan@google.com

Review URL: https://codereview.chromium.org/1415253008 .
diff --git a/mojo/public/platform/dart/mojo_natives.cc b/mojo/public/platform/dart/mojo_natives.cc
index 8739db4..05bfc30 100644
--- a/mojo/public/platform/dart/mojo_natives.cc
+++ b/mojo/public/platform/dart/mojo_natives.cc
@@ -37,6 +37,7 @@
   V(MojoMessagePipe_Create, 1)             \
   V(MojoMessagePipe_Write, 5)              \
   V(MojoMessagePipe_Read, 5)               \
+  V(MojoMessagePipe_QueryAndRead, 3)       \
   V(Mojo_GetTimeTicksNow, 0)               \
   V(MojoHandle_Close, 1)                   \
   V(MojoHandle_Wait, 3)                    \
@@ -718,6 +719,127 @@
   Dart_SetReturnValue(arguments, list);
 }
 
+void ByteArrayFinalizer(void* isolate_callback_data,
+                        Dart_WeakPersistentHandle handle,
+                        void* peer) {
+  uint8_t* byte_array = reinterpret_cast<uint8_t*>(peer);
+  delete[] byte_array;
+}
+
+void HandleArrayFinalizer(void* isolate_callback_data,
+                          Dart_WeakPersistentHandle handle,
+                          void* peer) {
+  uint32_t* handle_array = reinterpret_cast<uint32_t*>(peer);
+  delete[] handle_array;
+}
+
+void MojoMessagePipe_QueryAndRead(Dart_NativeArguments arguments) {
+  Dart_Handle err;
+  int64_t dart_handle;
+  int64_t flags = 0;
+  CHECK_INTEGER_ARGUMENT(arguments, 0, &dart_handle, Null);
+  CHECK_INTEGER_ARGUMENT(arguments, 1, &flags, Null);
+  Dart_Handle result = Dart_GetNativeArgument(arguments, 2);
+
+  Dart_Handle data = Dart_ListGetAt(result, 1);
+  Dart_Handle handles = Dart_ListGetAt(result, 2);
+
+  // Query the number of bytes and handles available.
+  uint32_t blen = 0;
+  uint32_t hlen = 0;
+  MojoResult res =
+      MojoReadMessage(static_cast<MojoHandle>(dart_handle), nullptr, &blen,
+                      nullptr, &hlen, static_cast<MojoReadMessageFlags>(flags));
+
+  if ((res != MOJO_RESULT_OK) && (res != MOJO_RESULT_RESOURCE_EXHAUSTED)) {
+    Dart_ListSetAt(result, 0, Dart_NewInteger(res));
+    Dart_ListSetAt(result, 1, data);
+    Dart_ListSetAt(result, 2, handles);
+    Dart_ListSetAt(result, 3, Dart_NewInteger(0));
+    Dart_ListSetAt(result, 4, Dart_NewInteger(0));
+    return;
+  }
+
+  Dart_TypedData_Type typ;
+  void* bytes = nullptr;
+  intptr_t bytes_len = 0;
+  if ((blen > 0) && Dart_IsNull(data)) {
+    uint8_t* new_byte_data = new uint8_t[blen];
+    data = Dart_NewExternalTypedData(Dart_TypedData_kByteData, new_byte_data,
+                                     blen);
+    MOJO_DCHECK(!Dart_IsError(data));
+    Dart_NewWeakPersistentHandle(data, new_byte_data, blen, ByteArrayFinalizer);
+  } else if (blen > 0) {
+    err = Dart_TypedDataAcquireData(data, &typ, &bytes, &bytes_len);
+    MOJO_DCHECK(!Dart_IsError(err));
+    err = Dart_TypedDataReleaseData(data);
+    MOJO_DCHECK(!Dart_IsError(err));
+    if (static_cast<uintptr_t>(bytes_len) < blen) {
+      uint8_t* new_byte_data = new uint8_t[blen];
+      data = Dart_NewExternalTypedData(Dart_TypedData_kByteData, new_byte_data,
+                                       blen);
+      MOJO_DCHECK(!Dart_IsError(data));
+      Dart_NewWeakPersistentHandle(data, new_byte_data, blen,
+                                   ByteArrayFinalizer);
+    }
+  }
+
+  void* handle_bytes = nullptr;
+  intptr_t handles_len = 0;
+  if ((hlen > 0) && Dart_IsNull(handles)) {
+    uint32_t* new_handle_data = new uint32_t[hlen];
+    handles = Dart_NewExternalTypedData(Dart_TypedData_kUint32, new_handle_data,
+                                        hlen);
+    MOJO_DCHECK(!Dart_IsError(handles));
+    Dart_NewWeakPersistentHandle(handles, new_handle_data,
+                                 hlen * sizeof(uint32_t), HandleArrayFinalizer);
+  } else if (hlen > 0) {
+    err = Dart_TypedDataAcquireData(handles, &typ, &handle_bytes, &handles_len);
+    MOJO_DCHECK(!Dart_IsError(err));
+    err = Dart_TypedDataReleaseData(handles);
+    MOJO_DCHECK(!Dart_IsError(err));
+    if (static_cast<uintptr_t>(handles_len) < hlen) {
+      uint32_t* new_handle_data = new uint32_t[hlen];
+      handles = Dart_NewExternalTypedData(Dart_TypedData_kUint32,
+                                          new_handle_data, hlen);
+      MOJO_DCHECK(!Dart_IsError(handles));
+      Dart_NewWeakPersistentHandle(handles, new_handle_data,
+                                   hlen * sizeof(uint32_t),
+                                   HandleArrayFinalizer);
+    }
+  }
+
+  if (blen > 0) {
+    err = Dart_TypedDataAcquireData(data, &typ, &bytes, &bytes_len);
+    MOJO_DCHECK(!Dart_IsError(err));
+  }
+
+  if (hlen > 0) {
+    err = Dart_TypedDataAcquireData(handles, &typ, &handle_bytes, &handles_len);
+    MOJO_DCHECK(!Dart_IsError(err));
+  }
+
+  res = MojoReadMessage(static_cast<MojoHandle>(dart_handle), bytes, &blen,
+                        reinterpret_cast<MojoHandle*>(handle_bytes), &hlen,
+                        static_cast<MojoReadMessageFlags>(flags));
+
+  if (blen > 0) {
+    err = Dart_TypedDataReleaseData(data);
+    MOJO_DCHECK(!Dart_IsError(err));
+  }
+
+  if (hlen > 0) {
+    err = Dart_TypedDataReleaseData(handles);
+    MOJO_DCHECK(!Dart_IsError(err));
+  }
+
+  Dart_ListSetAt(result, 0, Dart_NewInteger(res));
+  Dart_ListSetAt(result, 1, data);
+  Dart_ListSetAt(result, 2, handles);
+  Dart_ListSetAt(result, 3, Dart_NewInteger(blen));
+  Dart_ListSetAt(result, 4, Dart_NewInteger(hlen));
+}
+
 struct MojoWaitManyState {
   MojoWaitManyState() {}