Index: runtime/lib/vmservice.cc |
diff --git a/runtime/lib/vmservice.cc b/runtime/lib/vmservice.cc |
index 12aadb6831defc8db65c49e9f90837b32d061727..11380429f39844d6df05c5fad5db8bb49dc618d6 100644 |
--- a/runtime/lib/vmservice.cc |
+++ b/runtime/lib/vmservice.cc |
@@ -3,9 +3,10 @@ |
// BSD-style license that can be found in the LICENSE file. |
#include "vm/bootstrap_natives.h" |
- |
#include "vm/dart_api_impl.h" |
+#include "vm/datastream.h" |
#include "vm/exceptions.h" |
+#include "vm/growable_array.h" |
#include "vm/message.h" |
#include "vm/native_entry.h" |
#include "vm/object.h" |
@@ -151,4 +152,245 @@ DEFINE_NATIVE_ENTRY(VMService_RequestAssets, 0) { |
return Service::RequestAssets(); |
} |
+ |
+// TODO(25041): When reading, this class copies out the filenames and contents |
+// into new buffers. It does this because the lifetime of |bytes| is uncertain. |
+// If |bytes| is pinned in memory, then we could instead load up |
+// |filenames_| and |contents_| with pointers into |bytes| without making |
+// copies. |
+class TarArchive { |
+ public: |
+ TarArchive(uint8_t* bytes, intptr_t bytes_length) |
+ : rs_(bytes, bytes_length) {} |
+ |
+ void Read() { |
+ while (HasNext()) { |
+ char* filename; |
+ uint8_t* data; |
+ intptr_t data_length; |
+ if (Next(&filename, &data, &data_length)) { |
+ filenames_.Add(filename); |
+ contents_.Add(data); |
+ content_lengths_.Add(data_length); |
+ } |
+ } |
+ } |
+ |
+ char* NextFilename() { |
+ return filenames_.RemoveLast(); |
+ } |
+ |
+ uint8_t* NextContent() { |
+ return contents_.RemoveLast(); |
+ } |
+ |
+ intptr_t NextContentLength() { |
+ return content_lengths_.RemoveLast(); |
+ } |
+ |
+ bool HasMore() const { |
+ return filenames_.length() > 0; |
+ } |
+ |
+ intptr_t Length() const { return filenames_.length(); } |
+ |
+ private: |
+ enum TarHeaderFields { |
+ kTarHeaderFilenameOffset = 0, |
+ kTarHeaderFilenameSize = 100, |
+ kTarHeaderSizeOffset = 124, |
+ kTarHeaderSizeSize = 12, |
+ kTarHeaderTypeOffset = 156, |
+ kTarHeaderTypeSize = 1, |
+ kTarHeaderSize = 512, |
+ }; |
+ |
+ enum TarType { |
+ kTarAregType = '\0', |
+ kTarRegType = '0', |
+ kTarLnkType = '1', |
+ kTarSymType = '2', |
+ kTarChrType = '3', |
+ kTarBlkType = '4', |
+ kTarDirType = '5', |
+ kTarFifoType = '6', |
+ kTarContType = '7', |
+ kTarXhdType = 'x', |
+ kTarXglType = 'g', |
+ }; |
+ |
+ bool HasNext() const { |
+ return !EndOfArchive(); |
+ } |
+ |
+ bool Next(char** filename, uint8_t** data, intptr_t* data_length) { |
+ intptr_t startOfBlock = rs_.Position(); |
+ *filename = ReadFilename(); |
+ rs_.SetPosition(startOfBlock + kTarHeaderSizeOffset); |
+ intptr_t size = ReadSize(); |
+ rs_.SetPosition(startOfBlock + kTarHeaderTypeOffset); |
+ TarType type = ReadType(); |
+ SeekToNextBlock(kTarHeaderSize); |
+ if ((type != kTarRegType) && (type != kTarAregType)) { |
+ SkipContents(size); |
+ return false; |
+ } |
+ ReadContents(data, size); |
+ *data_length = size; |
+ return true; |
+ } |
+ |
+ void SeekToNextBlock(intptr_t blockSize) { |
+ intptr_t remainder = blockSize - (rs_.Position() % blockSize); |
+ rs_.Advance(remainder); |
+ } |
+ |
+ uint8_t PeekByte(intptr_t i) const { |
+ return *(rs_.AddressOfCurrentPosition() + i); |
+ } |
+ |
+ bool EndOfArchive() const { |
+ if (rs_.PendingBytes() < (kTarHeaderSize * 2)) { |
+ return true; |
+ } |
+ for (intptr_t i = 0; i < (kTarHeaderSize * 2); i++) { |
+ if (PeekByte(i) != 0) { |
+ return false; |
+ } |
+ } |
+ return true; |
+ } |
+ |
+ TarType ReadType() { |
+ return static_cast<TarType>(ReadStream::Raw<1, uint8_t>::Read(&rs_)); |
+ } |
+ |
+ void SkipContents(intptr_t size) { |
+ rs_.Advance(size); |
+ SeekToNextBlock(kTarHeaderSize); |
+ } |
+ |
+ intptr_t ReadCString(char** s, intptr_t length) { |
+ intptr_t to_read = Utils::Minimum(length, rs_.PendingBytes()); |
+ char* result = new char[to_read + 1]; |
+ strncpy(result, |
+ reinterpret_cast<const char*>(rs_.AddressOfCurrentPosition()), |
+ to_read); |
+ result[to_read] = '\0'; |
+ rs_.SetPosition(rs_.Position() + to_read); |
+ *s = result; |
+ return to_read; |
+ } |
+ |
+ intptr_t ReadSize() { |
+ char* octalSize; |
+ unsigned int size; |
+ |
+ ReadCString(&octalSize, kTarHeaderSizeSize); |
+ int result = sscanf(octalSize, "%o", &size); |
+ delete[] octalSize; |
+ |
+ if (result != 1) { |
+ return 0; |
+ } |
+ return size; |
+ } |
+ |
+ char* ReadFilename() { |
+ char* result; |
+ intptr_t result_length = ReadCString(&result, kTarHeaderFilenameSize); |
+ if (result[0] == '/') { |
+ return result; |
+ } |
+ char* fixed_result = new char[result_length + 2]; // '/' + '\0'. |
+ fixed_result[0] = '/'; |
+ strncpy(&fixed_result[1], result, result_length); |
+ fixed_result[result_length + 1] = '\0'; |
+ delete[] result; |
+ return fixed_result; |
+ } |
+ |
+ void ReadContents(uint8_t** data, intptr_t size) { |
+ uint8_t* result = new uint8_t[size]; |
+ rs_.ReadBytes(result, size); |
+ SeekToNextBlock(kTarHeaderSize); |
+ *data = result; |
+ } |
+ |
+ ReadStream rs_; |
+ GrowableArray<char*> filenames_; |
+ GrowableArray<uint8_t*> contents_; |
+ GrowableArray<intptr_t> content_lengths_; |
+ |
+ DISALLOW_COPY_AND_ASSIGN(TarArchive); |
+}; |
+ |
+ |
+static void ContentsFinalizer(void* isolate_callback_data, |
+ Dart_WeakPersistentHandle handle, |
+ void* peer) { |
+ uint8_t* data = reinterpret_cast<uint8_t*>(peer); |
+ delete[] data; |
+} |
+ |
+ |
+static void FilenameFinalizer(void* peer) { |
+ char* filename = reinterpret_cast<char*>(peer); |
+ delete[] filename; |
+} |
+ |
+ |
+DEFINE_NATIVE_ENTRY(VMService_DecodeAssets, 1) { |
+ GET_NON_NULL_NATIVE_ARGUMENT(TypedData, data, arguments->NativeArgAt(0)); |
+ Api::Scope scope(thread); |
+ |
+ Dart_Handle data_handle = Api::NewHandle(thread->isolate(), data.raw()); |
+ |
+ Dart_TypedData_Type typ; |
+ void* bytes; |
+ intptr_t length; |
+ Dart_Handle err = Dart_TypedDataAcquireData( |
+ data_handle, &typ, &bytes, &length); |
+ ASSERT(!Dart_IsError(err)); |
+ |
+ TarArchive archive(reinterpret_cast<uint8_t*>(bytes), length); |
+ archive.Read(); |
+ |
+ err = Dart_TypedDataReleaseData(data_handle); |
+ ASSERT(!Dart_IsError(err)); |
+ |
+ intptr_t archive_size = archive.Length(); |
+ |
+ const Array& result_list = Array::Handle(thread->zone(), |
+ Array::New(2 * archive_size)); |
+ |
+ intptr_t idx = 0; |
+ while (archive.HasMore()) { |
+ char* filename = archive.NextFilename(); |
+ uint8_t* contents = archive.NextContent(); |
+ intptr_t contents_length = archive.NextContentLength(); |
+ |
+ Dart_Handle dart_filename = Dart_NewExternalLatin1String( |
+ reinterpret_cast<uint8_t*>(filename), |
+ strlen(filename), |
+ filename, |
+ FilenameFinalizer); |
+ ASSERT(!Dart_IsError(dart_filename)); |
+ |
+ Dart_Handle dart_contents = Dart_NewExternalTypedData( |
+ Dart_TypedData_kUint8, contents, contents_length); |
+ ASSERT(!Dart_IsError(dart_contents)); |
+ Dart_NewWeakPersistentHandle( |
+ dart_contents, contents, contents_length, ContentsFinalizer); |
+ |
+ result_list.SetAt(idx, Api::UnwrapStringHandle( |
+ thread->zone(), dart_filename)); |
+ result_list.SetAt(idx + 1, Api::UnwrapExternalTypedDataHandle( |
+ thread->zone(), dart_contents)); |
+ idx += 2; |
+ } |
+ |
+ return result_list.raw(); |
+} |
+ |
} // namespace dart |