1 /*
2 * Copyright (c) 2021-2022 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "file-inl.h"
17 #include "file_items.h"
18 #include "file_item_container.h"
19 #include "utils/string_helpers.h"
20 #include "zip_archive.h"
21 #include "file.h"
22
23 #include "assembly-emitter.h"
24 #include "assembly-parser.h"
25
26 #include <cstdint>
27 #ifdef PANDA_TARGET_MOBILE
28 #include <unistd.h>
29 #endif
30
31 #include <vector>
32
33 #include <gtest/gtest.h>
34
35 namespace panda::panda_file::test {
36
37 static constexpr const char *ABC_FILE = "test_file.abc";
38
GetPandaFile(std::vector<uint8_t> * data)39 static std::unique_ptr<const File> GetPandaFile(std::vector<uint8_t> *data)
40 {
41 os::mem::ConstBytePtr ptr(reinterpret_cast<std::byte *>(data->data()), data->size(),
42 [](std::byte *, size_t) noexcept {});
43 return File::OpenFromMemory(std::move(ptr));
44 }
45
GetEmptyPandaFileBytes()46 static std::vector<uint8_t> GetEmptyPandaFileBytes()
47 {
48 pandasm::Parser p;
49
50 auto source = R"()";
51
52 std::string src_filename = "src.pa";
53 auto res = p.Parse(source, src_filename);
54 ASSERT(p.ShowError().err == pandasm::Error::ErrorType::ERR_NONE);
55
56 auto pf = pandasm::AsmEmitter::Emit(res.Value());
57 ASSERT(pf != nullptr);
58
59 std::vector<uint8_t> data {};
60 const auto header_ptr = reinterpret_cast<const uint8_t *>(pf->GetHeader());
61 data.assign(header_ptr, header_ptr + sizeof(File::Header));
62
63 ASSERT(data.size() == sizeof(File::Header));
64
65 return data;
66 }
67
CreateOrAddZipPandaFile(std::vector<uint8_t> * data,const char * zip_archive_name,const char * filename,int append,int level)68 int CreateOrAddZipPandaFile(std::vector<uint8_t> *data, const char *zip_archive_name, const char *filename, int append,
69 int level)
70 {
71 return CreateOrAddFileIntoZip(zip_archive_name, filename, (*data).data(), (*data).size(), append, level);
72 }
73
CheckAnonMemoryName(const char * zip_archive_name)74 bool CheckAnonMemoryName([[maybe_unused]] const char *zip_archive_name)
75 {
76 // check if [annon:panda-classes.abc extracted in memory from /xx/__OpenPandaFileFromZip__.zip]
77 #ifdef PANDA_TARGET_MOBILE
78 bool result = false;
79 const char *prefix = "[anon:panda-";
80 int pid = getpid();
81 std::stringstream ss;
82 ss << "/proc/" << pid << "/maps";
83 std::ifstream f;
84 f.open(ss.str(), std::ios::in);
85 EXPECT_TRUE(f.is_open());
86 for (std::string line; std::getline(f, line);) {
87 if (line.find(prefix) != std::string::npos && line.find(zip_archive_name) != std::string::npos) {
88 result = true;
89 }
90 }
91 f.close();
92 return result;
93 #else
94 return true;
95 #endif
96 }
97
98 HWTEST(File, GetClassByName, testing::ext::TestSize.Level0)
99 {
100 ItemContainer container;
101
102 std::vector<std::string> names = {"C", "B", "A"};
103 std::vector<ClassItem *> classes;
104
105 for (auto &name : names) {
106 classes.push_back(container.GetOrCreateClassItem(name));
107 }
108
109 MemoryWriter mem_writer;
110
111 ASSERT_TRUE(container.Write(&mem_writer));
112
113 // Read panda file from memory
114
115 auto data = mem_writer.GetData();
116 auto panda_file = GetPandaFile(&data);
117 ASSERT_NE(panda_file, nullptr);
118
119 for (size_t i = 0; i < names.size(); i++) {
120 EXPECT_EQ(panda_file->GetClassId(reinterpret_cast<const uint8_t *>(names[i].c_str())).GetOffset(),
121 classes[i]->GetOffset());
122 }
123 }
124
125 HWTEST(File, OpenPandaFile, testing::ext::TestSize.Level0)
126 {
127 // Create ZIP
128 auto data = GetEmptyPandaFileBytes();
129 int ret;
130 const char *zip_filename = "__OpenPandaFile__.zip";
131 const char *filename1 = ARCHIVE_FILENAME;
132 const char *filename2 = "classses2.abc"; // just for testing.
133 ret = CreateOrAddZipPandaFile(&data, zip_filename, filename1, APPEND_STATUS_CREATE, Z_BEST_COMPRESSION);
134 ASSERT_EQ(ret, 0);
135 ret = CreateOrAddZipPandaFile(&data, zip_filename, filename2, APPEND_STATUS_ADDINZIP, Z_BEST_COMPRESSION);
136 ASSERT_EQ(ret, 0);
137
138 // Open from ZIP
139 auto pf = OpenPandaFile(zip_filename);
140 EXPECT_NE(pf, nullptr);
141 EXPECT_STREQ((pf->GetFilename()).c_str(), zip_filename);
142
143 // Open from ZIP with archive_filename
144 const char *filename3 = "classses3.abc"; // just for testing.
145 pf = OpenPandaFile(zip_filename, filename3);
146 EXPECT_NE(pf, nullptr);
147
148 remove(zip_filename);
149
150 ret = CreateOrAddZipPandaFile(&data, zip_filename, filename2, APPEND_STATUS_CREATE, Z_BEST_COMPRESSION);
151 ASSERT_EQ(ret, 0);
152
153 // Open from ZIP without default archive_filename
154 pf = OpenPandaFile(zip_filename);
155 EXPECT_EQ(pf, nullptr);
156
157 remove(zip_filename);
158 }
159
160 HWTEST(File, OpenPandaFileFromMemory, testing::ext::TestSize.Level0)
161 {
162 auto pf = OpenPandaFileFromMemory(nullptr, -1);
163 EXPECT_EQ(pf, nullptr);
164
165 pf = OpenPandaFileFromMemory(nullptr, 1U);
166 EXPECT_EQ(pf, nullptr);
167 }
168
169 HWTEST(File, OpenPandaFileFromZipNameAnonMem, testing::ext::TestSize.Level0)
170 {
171 // Create ZIP
172 auto data = GetEmptyPandaFileBytes();
173 int ret;
174 const char *zip_filename = "__OpenPandaFileFromZipNameAnonMem__.zip";
175 const char *filename1 = ARCHIVE_FILENAME;
176 ret = CreateOrAddZipPandaFile(&data, zip_filename, filename1, APPEND_STATUS_CREATE, Z_BEST_COMPRESSION);
177 ASSERT_EQ(ret, 0);
178
179 // Open from ZIP
180 auto pf = OpenPandaFile(zip_filename);
181 EXPECT_NE(pf, nullptr);
182 EXPECT_STREQ((pf->GetFilename()).c_str(), zip_filename);
183 ASSERT_TRUE(CheckAnonMemoryName(zip_filename));
184
185 remove(zip_filename);
186 }
187
188 HWTEST(File, OpenPandaFileOrZip, testing::ext::TestSize.Level0)
189 {
190 // Create ZIP
191 auto data = GetEmptyPandaFileBytes();
192 int ret;
193 const char *zip_filename = "__OpenPandaFileOrZip__.zip";
194 const char *filename1 = ARCHIVE_FILENAME;
195 const char *filename2 = "classes2.abc"; // just for testing.
196 ret = CreateOrAddZipPandaFile(&data, zip_filename, filename1, APPEND_STATUS_CREATE, Z_BEST_COMPRESSION);
197 ASSERT_EQ(ret, 0);
198 ret = CreateOrAddZipPandaFile(&data, zip_filename, filename2, APPEND_STATUS_ADDINZIP, Z_BEST_COMPRESSION);
199 ASSERT_EQ(ret, 0);
200
201 // Open from ZIP
202 auto pf = OpenPandaFileOrZip(zip_filename);
203 EXPECT_NE(pf, nullptr);
204 EXPECT_STREQ((pf->GetFilename()).c_str(), zip_filename);
205
206 const char *zip_filename_with_entry = "__OpenPandaFileOrZip__.zip!/classes.abc";
207 auto pf_new = OpenPandaFileOrZip(zip_filename_with_entry);
208 EXPECT_NE(pf_new, nullptr);
209
210 remove(zip_filename);
211 }
212
213 HWTEST(File, OpenPandaFileFromZipErrorHandler, testing::ext::TestSize.Level0)
214 {
215 const char *file_name = "test_file_empty.panda";
216 {
217 auto writer = FileWriter(file_name);
218 ASSERT_TRUE(writer);
219 }
220 auto pf = OpenPandaFile(file_name);
221 EXPECT_EQ(pf, nullptr);
222
223 const char *file_name_zip = "test_file_empty.zip";
224 {
225 auto writer = FileWriter(file_name);
226 ASSERT_TRUE(writer);
227 const std::vector<uint8_t> magic = {'P', 'K'};
228 ASSERT_TRUE(writer.WriteBytes(magic));
229 }
230 pf = OpenPandaFile(file_name_zip);
231 EXPECT_EQ(pf, nullptr);
232
233 // Create ZIP
234 const char *file_name_zip_with_entry = "test_file_with_entry.zip";
235 std::vector<uint8_t> data = {};
236 int ret = CreateOrAddZipPandaFile(&data, file_name_zip_with_entry, ARCHIVE_FILENAME,
237 APPEND_STATUS_CREATE, Z_BEST_COMPRESSION);
238 ASSERT_EQ(ret, 0);
239 pf = OpenPandaFile(file_name_zip_with_entry);
240 EXPECT_EQ(pf, nullptr);
241
242 auto fp = fopen(file_name_zip_with_entry, "a");
243 EXPECT_NE(fp, nullptr);
244 const char *append_str = "error";
245 EXPECT_EQ(fwrite(append_str, sizeof(append_str), 1U, fp), 1U);
246 fclose(fp);
247 pf = OpenPandaFile(file_name_zip_with_entry);
248 EXPECT_EQ(pf, nullptr);
249
250 remove(file_name);
251 remove(file_name_zip);
252 remove(file_name_zip_with_entry);
253 }
254
255 HWTEST(File, HandleArchive, testing::ext::TestSize.Level0)
256 {
257 {
258 ItemContainer container;
259 auto writer = FileWriter(ARCHIVE_FILENAME);
260 ASSERT_TRUE(container.Write(&writer));
261 ASSERT_TRUE(writer.Align(4U)); // to 4 bytes align
262 }
263
264 std::vector<uint8_t> data;
265 {
266 std::ifstream in(ARCHIVE_FILENAME, std::ios::binary);
267 data.insert(data.end(), (std::istreambuf_iterator<char>(in)), std::istreambuf_iterator<char>());
268 ASSERT_TRUE(data.size() > 0U && data.size() % 4U == 0U);
269 }
270
271 // Create ZIP
272 const char *zip_filename = "__HandleArchive__.zip";
273 int ret = CreateOrAddZipPandaFile(&data, zip_filename, ARCHIVE_FILENAME, APPEND_STATUS_CREATE, Z_NO_COMPRESSION);
274 ASSERT_EQ(ret, 0);
275 auto pf = OpenPandaFile(zip_filename);
276 EXPECT_NE(pf, nullptr);
277
278 remove(ARCHIVE_FILENAME);
279 remove(zip_filename);
280 }
281
282 HWTEST(File, CheckHeader, testing::ext::TestSize.Level0)
283 {
284 // Write panda file to disk
285 ItemContainer container;
286
287 auto writer = FileWriter(ABC_FILE);
288 ASSERT_TRUE(container.Write(&writer));
289
290 // Read panda file from disk
291 auto fp = fopen(ABC_FILE, "rb");
292 EXPECT_NE(fp, nullptr);
293
294 os::mem::ConstBytePtr ptr = os::mem::MapFile(os::file::File(fileno(fp)), os::mem::MMAP_PROT_READ,
295 os::mem::MMAP_FLAG_PRIVATE, writer.GetOffset()).ToConst();
296 EXPECT_NE(ptr.Get(), nullptr);
297 EXPECT_TRUE(CheckHeader(ptr, ABC_FILE));
298 fclose(fp);
299
300 remove(ABC_FILE);
301 }
302
303 HWTEST(File, GetMode, testing::ext::TestSize.Level0)
304 {
305 // Write panda file to disk
306 ItemContainer container;
307
308 auto writer = FileWriter(ABC_FILE);
309 ASSERT_TRUE(container.Write(&writer));
310
311 // Read panda file from disk
312 EXPECT_NE(File::Open(ABC_FILE), nullptr);
313 EXPECT_EQ(File::Open(ABC_FILE, File::OpenMode::WRITE_ONLY), nullptr);
314
315 remove(ABC_FILE);
316 }
317
318 HWTEST(File, Open, testing::ext::TestSize.Level0)
319 {
320 EXPECT_EQ(File::Open(ABC_FILE), nullptr);
321
322 auto fp = fopen(ABC_FILE, "w");
323 EXPECT_NE(fp, nullptr);
324 const char *write_str = "error";
325 EXPECT_EQ(fwrite(write_str, sizeof(write_str), 1U, fp), 1U);
326 fclose(fp);
327 EXPECT_EQ(File::Open(ABC_FILE), nullptr);
328 EXPECT_EQ(File::Open(ABC_FILE, File::OpenMode::WRITE_ONLY), nullptr);
329
330 remove(ABC_FILE);
331 }
332
333 HWTEST(File, OpenUncompressedArchive, testing::ext::TestSize.Level0)
334 {
335 // Invalid FD
336 EXPECT_EQ(File::OpenUncompressedArchive(-1, ABC_FILE, 0U, 0U), nullptr);
337
338 // Invalid Size
339 EXPECT_EQ(File::OpenUncompressedArchive(1, ABC_FILE, 0U, 0U), nullptr);
340
341 // Invalid Max Size
342 EXPECT_EQ(File::OpenUncompressedArchive(1, ABC_FILE, -1, 0U), nullptr);
343
344 // Invalid ABC File
345 auto data = GetEmptyPandaFileBytes();
346 auto fp = fopen(ARCHIVE_FILENAME, "w+");
347 EXPECT_NE(fp, nullptr);
348 data[0] = 0U;
349 EXPECT_EQ(fwrite(data.data(), sizeof(uint8_t), data.size(), fp), data.size());
350 (void)fseek(fp, 0, SEEK_SET);
351 EXPECT_EQ(File::OpenUncompressedArchive(fileno(fp), ARCHIVE_FILENAME, sizeof(File::Header), 0U), nullptr);
352 fclose(fp);
353
354 remove(ABC_FILE);
355 }
356
357 } // namespace panda::panda_file::test
358