• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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