gem5  v22.1.0.0
image_file_data.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2002-2004 The Regents of The University of Michigan
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met: redistributions of source code must retain the above copyright
8  * notice, this list of conditions and the following disclaimer;
9  * redistributions in binary form must reproduce the above copyright
10  * notice, this list of conditions and the following disclaimer in the
11  * documentation and/or other materials provided with the distribution;
12  * neither the name of the copyright holders nor the names of its
13  * contributors may be used to endorse or promote products derived from
14  * this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
30 
31 #include <fcntl.h>
32 #include <sys/mman.h>
33 #include <sys/types.h>
34 #include <unistd.h>
35 #include <zlib.h>
36 
37 #include <cstdio>
38 #include <vector>
39 
40 #include "base/logging.hh"
41 
42 namespace gem5
43 {
44 
45 GEM5_DEPRECATED_NAMESPACE(Loader, loader);
46 namespace loader
47 {
48 
49 static bool
51 {
52  uint8_t buf[2] = {0};
53  size_t sz = pread(fd, buf, 2, 0);
54  panic_if(sz != 2, "Couldn't read magic bytes from object file");
55  return ((buf[0] == 0x1f) && (buf[1] == 0x8b));
56 }
57 
58 static int
60 {
61  const size_t blk_sz = 4096;
62 
63  gzFile fdz = gzdopen(fd, "rb");
64  if (!fdz) {
65  return -1;
66  }
67 
68  size_t tmp_len = strlen(P_tmpdir);
69  char *tmpnam = (char*) malloc(tmp_len + 20);
70  strcpy(tmpnam, P_tmpdir);
71  strcpy(tmpnam+tmp_len, "/gem5-gz-obj-XXXXXX"); // 19 chars
72  fd = mkstemp(tmpnam); // repurposing fd variable for output
73  if (fd < 0) {
74  free(tmpnam);
75  gzclose(fdz);
76  return fd;
77  }
78 
79  if (unlink(tmpnam) != 0)
80  warn("couldn't remove temporary file %s\n", tmpnam);
81 
82  free(tmpnam);
83 
84  auto buf = new uint8_t[blk_sz];
85  int r; // size of (r)emaining uncopied data in (buf)fer
86  while ((r = gzread(fdz, buf, blk_sz)) > 0) {
87  auto p = buf; // pointer into buffer
88  while (r > 0) {
89  auto sz = write(fd, p, r);
90  assert(sz <= r);
91  r -= sz;
92  p += sz;
93  }
94  }
95  delete[] buf;
96  gzclose(fdz);
97  if (r < 0) { // error
98  close(fd);
99  return -1;
100  }
101  assert(r == 0); // finished successfully
102  return fd; // return fd to decompressed temporary file for mmap()'ing
103 }
104 
105 ImageFileData::ImageFileData(const std::string &fname)
106 {
107  _filename = fname;
108 
109  // Open the file.
110  int fd = open(fname.c_str(), O_RDONLY);
111  fatal_if(fd < 0, "Failed to open file %s.\n"
112  "This error typically occurs when the file path specified is "
113  "incorrect.\n", fname);
114 
115  // Decompress GZ files.
116  if (hasGzipMagic(fd)) {
117  fd = doGzipLoad(fd);
118  panic_if(fd < 0, "Failed to unzip file %s.\n", fname);
119  }
120 
121  // Find the length of the file by seeking to the end.
122  off_t off = lseek(fd, 0, SEEK_END);
123  fatal_if(off < 0, "Failed to determine size of file %s.\n", fname);
124  _len = static_cast<size_t>(off);
125 
126  // Mmap the whole shebang.
127  _data = (uint8_t *)mmap(NULL, _len, PROT_READ, MAP_SHARED, fd, 0);
128  close(fd);
129 
130  panic_if(_data == MAP_FAILED, "Failed to mmap file %s.\n", fname);
131 }
132 
134 {
135  munmap((void *)_data, _len);
136 }
137 
138 } // namespace loader
139 } // namespace gem5
ImageFileData(const std::string &f_name)
#define fatal_if(cond,...)
Conditional fatal macro that checks the supplied condition and only causes a fatal error if the condi...
Definition: logging.hh:226
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:204
#define warn(...)
Definition: logging.hh:246
Bitfield< 14, 12 > fd
Definition: types.hh:150
Bitfield< 5 > r
Definition: pagetable.hh:60
Bitfield< 54 > p
Definition: pagetable.hh:70
static bool hasGzipMagic(int fd)
static int doGzipLoad(int fd)
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
GEM5_DEPRECATED_NAMESPACE(GuestABI, guest_abi)

Generated on Wed Dec 21 2022 10:22:29 for gem5 by doxygen 1.9.1