1 // Copyright 2015 PDFium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "testing/test_support.h" 6 7 #include <stdio.h> 8 #include <string.h> 9 10 #include <string> 11 12 #include "testing/utils/path_service.h" 13 14 #ifdef PDF_ENABLE_V8 15 #include "v8/include/libplatform/libplatform.h" 16 #endif 17 18 namespace { 19 20 #ifdef PDF_ENABLE_V8 21 #ifdef V8_USE_EXTERNAL_STARTUP_DATA 22 // Returns the full path for an external V8 data file based on either 23 // the currect exectuable path or an explicit override. 24 std::string GetFullPathForSnapshotFile(const std::string& exe_path, 25 const std::string& bin_dir, 26 const std::string& filename) { 27 std::string result; 28 if (!bin_dir.empty()) { 29 result = bin_dir; 30 if (*bin_dir.rbegin() != PATH_SEPARATOR) { 31 result += PATH_SEPARATOR; 32 } 33 } else if (!exe_path.empty()) { 34 size_t last_separator = exe_path.rfind(PATH_SEPARATOR); 35 if (last_separator != std::string::npos) { 36 result = exe_path.substr(0, last_separator + 1); 37 } 38 } 39 result += filename; 40 return result; 41 } 42 43 bool GetExternalData(const std::string& exe_path, 44 const std::string& bin_dir, 45 const std::string& filename, 46 v8::StartupData* result_data) { 47 std::string full_path = 48 GetFullPathForSnapshotFile(exe_path, bin_dir, filename); 49 size_t data_length = 0; 50 std::unique_ptr<char, pdfium::FreeDeleter> data_buffer = 51 GetFileContents(full_path.c_str(), &data_length); 52 if (!data_buffer) 53 return false; 54 55 result_data->data = data_buffer.release(); 56 result_data->raw_size = data_length; 57 return true; 58 } 59 #endif // V8_USE_EXTERNAL_STARTUP_DATA 60 61 void InitializeV8Common(const char* exe_path, v8::Platform** platform) { 62 v8::V8::InitializeICUDefaultLocation(exe_path); 63 64 *platform = v8::platform::CreateDefaultPlatform(); 65 v8::V8::InitializePlatform(*platform); 66 67 // By enabling predictable mode, V8 won't post any background tasks. 68 // By enabling GC, it makes it easier to chase use-after-free. 69 const char v8_flags[] = "--predictable --expose-gc"; 70 v8::V8::SetFlagsFromString(v8_flags, static_cast<int>(strlen(v8_flags))); 71 v8::V8::Initialize(); 72 } 73 #endif // PDF_ENABLE_V8 74 75 } // namespace 76 77 std::unique_ptr<char, pdfium::FreeDeleter> GetFileContents(const char* filename, 78 size_t* retlen) { 79 FILE* file = fopen(filename, "rb"); 80 if (!file) { 81 fprintf(stderr, "Failed to open: %s\n", filename); 82 return nullptr; 83 } 84 (void)fseek(file, 0, SEEK_END); 85 size_t file_length = ftell(file); 86 if (!file_length) { 87 return nullptr; 88 } 89 (void)fseek(file, 0, SEEK_SET); 90 std::unique_ptr<char, pdfium::FreeDeleter> buffer( 91 static_cast<char*>(malloc(file_length))); 92 if (!buffer) { 93 return nullptr; 94 } 95 size_t bytes_read = fread(buffer.get(), 1, file_length, file); 96 (void)fclose(file); 97 if (bytes_read != file_length) { 98 fprintf(stderr, "Failed to read: %s\n", filename); 99 return nullptr; 100 } 101 *retlen = bytes_read; 102 return buffer; 103 } 104 105 std::wstring GetPlatformWString(FPDF_WIDESTRING wstr) { 106 if (!wstr) 107 return nullptr; 108 109 size_t characters = 0; 110 while (wstr[characters]) 111 ++characters; 112 113 std::wstring platform_string(characters, L'\0'); 114 for (size_t i = 0; i < characters + 1; ++i) { 115 const unsigned char* ptr = reinterpret_cast<const unsigned char*>(&wstr[i]); 116 platform_string[i] = ptr[0] + 256 * ptr[1]; 117 } 118 return platform_string; 119 } 120 121 std::vector<std::string> StringSplit(const std::string& str, char delimiter) { 122 std::vector<std::string> result; 123 size_t pos = 0; 124 while (1) { 125 size_t found = str.find(delimiter, pos); 126 if (found == std::string::npos) 127 break; 128 129 result.push_back(str.substr(pos, found - pos)); 130 pos = found + 1; 131 } 132 result.push_back(str.substr(pos)); 133 return result; 134 } 135 136 std::unique_ptr<unsigned short, pdfium::FreeDeleter> GetFPDFWideString( 137 const std::wstring& wstr) { 138 size_t length = sizeof(uint16_t) * (wstr.length() + 1); 139 std::unique_ptr<unsigned short, pdfium::FreeDeleter> result( 140 static_cast<unsigned short*>(malloc(length))); 141 char* ptr = reinterpret_cast<char*>(result.get()); 142 size_t i = 0; 143 for (wchar_t w : wstr) { 144 ptr[i++] = w & 0xff; 145 ptr[i++] = (w >> 8) & 0xff; 146 } 147 ptr[i++] = 0; 148 ptr[i] = 0; 149 return result; 150 } 151 152 #ifdef PDF_ENABLE_V8 153 #ifdef V8_USE_EXTERNAL_STARTUP_DATA 154 bool InitializeV8ForPDFium(const std::string& exe_path, 155 const std::string& bin_dir, 156 v8::StartupData* natives_blob, 157 v8::StartupData* snapshot_blob, 158 v8::Platform** platform) { 159 InitializeV8Common(exe_path.c_str(), platform); 160 if (natives_blob && snapshot_blob) { 161 if (!GetExternalData(exe_path, bin_dir, "natives_blob.bin", natives_blob)) 162 return false; 163 if (!GetExternalData(exe_path, bin_dir, "snapshot_blob.bin", snapshot_blob)) 164 return false; 165 v8::V8::SetNativesDataBlob(natives_blob); 166 v8::V8::SetSnapshotDataBlob(snapshot_blob); 167 } 168 return true; 169 } 170 #else // V8_USE_EXTERNAL_STARTUP_DATA 171 bool InitializeV8ForPDFium(const std::string& exe_path, 172 v8::Platform** platform) { 173 InitializeV8Common(exe_path.c_str(), platform); 174 return true; 175 } 176 #endif // V8_USE_EXTERNAL_STARTUP_DATA 177 #endif // PDF_ENABLE_V8 178 179 TestLoader::TestLoader(const char* pBuf, size_t len) 180 : m_pBuf(pBuf), m_Len(len) { 181 } 182 183 // static 184 int TestLoader::GetBlock(void* param, 185 unsigned long pos, 186 unsigned char* pBuf, 187 unsigned long size) { 188 TestLoader* pLoader = static_cast<TestLoader*>(param); 189 if (pos + size < pos || pos + size > pLoader->m_Len) 190 return 0; 191 192 memcpy(pBuf, pLoader->m_pBuf + pos, size); 193 return 1; 194 } 195 196 TestSaver::TestSaver() { 197 FPDF_FILEWRITE::version = 1; 198 FPDF_FILEWRITE::WriteBlock = WriteBlockCallback; 199 } 200 201 void TestSaver::ClearString() { 202 m_String.clear(); 203 } 204 205 // static 206 int TestSaver::WriteBlockCallback(FPDF_FILEWRITE* pFileWrite, 207 const void* data, 208 unsigned long size) { 209 TestSaver* pThis = static_cast<TestSaver*>(pFileWrite); 210 pThis->m_String.append(static_cast<const char*>(data), size); 211 return 1; 212 } 213