blob: cd5c2f39be30fe4e444dfff00f8ba9c7964c6648 [file] [log] [blame]
Damien Martin-Guillerez08441122015-05-28 11:12:31 +00001// Copyright 2015 Google Inc. All rights reserved.
2//
3// Author: Alan Donovan <adonovan@google.com>
4//
5// Licensed under the Apache License, Version 2.0 (the "License");
6// you may not use this file except in compliance with the License.
7// You may obtain a copy of the License at
8//
9// http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing, software
12// distributed under the License is distributed on an "AS IS" BASIS,
13// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14// See the License for the specific language governing permissions and
15// limitations under the License.
16
17//
18// Zip / Unzip file using ijar zip implementation.
19//
20// Note that this Zip implementation intentionally don't compute CRC-32
21// because it is useless computation for jar because Java doesn't care.
22// CRC-32 of all files in the zip file will be set to 0.
23//
24
25#include <stdio.h>
26#include <string.h>
27#include <stdlib.h>
28#include <limits.h>
29#include <fcntl.h>
30#include <unistd.h>
31#include <sys/mman.h>
32#include <errno.h>
33#include <memory>
34
35#include "third_party/ijar/zip.h"
36
37namespace devtools_ijar {
38
39#define SYSCALL(expr) do { \
40 if ((expr) < 0) { \
41 perror(#expr); \
42 abort(); \
43 } \
44 } while (0)
45
46//
47// A ZipExtractorProcessor that extract all files in the ZIP file.
48//
49class UnzipProcessor : public ZipExtractorProcessor {
50 public:
51 // Create a processor who will extract the files into output_root
52 // if "extract" is set to true and will print the list of files and
53 // their unix modes if "verbose" is set to true.
54 UnzipProcessor(const char *output_root, bool verbose, bool extract)
55 : output_root_(output_root), verbose_(verbose), extract_(extract) {}
56 virtual ~UnzipProcessor() {}
57
58 virtual void Process(const char* filename, const u4 attr,
59 const u1* data, const size_t size);
60 virtual bool Accept(const char* filename, const u4 attr) {
61 return true;
62 }
63
64 private:
65 const char *output_root_;
66 const bool verbose_;
67 const bool extract_;
68};
69
70// Concatene 2 path, path1 and path2, using / as a directory separator and
71// puting the result in "out". "size" specify the size of the output buffer
72void concat_path(char* out, const size_t size,
73 const char *path1, const char *path2) {
74 int len1 = strlen(path1);
Damien Martin-Guillerez08441122015-05-28 11:12:31 +000075 int l = len1;
76 strncpy(out, path1, size-1);
77 out[size-1] = 0;
78 if (l < size - 1 && path1[len1] != '/' && path2[0] != '/') {
79 out[l] = '/';
80 l++;
81 out[l] = 0;
82 }
83 if (l < size - 1) {
84 strncat(out, path2, size - 1 - l);
85 }
86}
87
88// Do a recursive mkdir of all folders of path except the last path
89// segment (if path ends with a / then the last path segment is empty).
90// All folders are created using "mode" for creation mode.
91void mkdirs(const char *path, mode_t mode) {
92 char path_[PATH_MAX];
93 struct stat statst;
94 strncpy(path_, path, PATH_MAX);
95 path_[PATH_MAX-1] = 0;
96 char *pointer = path_;
97 while ((pointer = strchr(pointer, '/')) != NULL) {
98 if (path_ != pointer) { // skip leading slash
99 *pointer = 0;
100 if (stat(path_, &statst) != 0) {
101 if (mkdir(path_, mode) < 0) {
102 fprintf(stderr, "Cannot create folder %s: %s\n",
103 path_, strerror(errno));
104 abort();
105 }
106 }
107 *pointer = '/';
108 }
109 pointer++;
110 }
111}
112
113void UnzipProcessor::Process(const char* filename, const u4 attr,
114 const u1* data, const size_t size) {
115 mode_t mode = zipattr_to_mode(attr);
116 mode_t perm = mode & 0777;
117 bool isdir = (mode & S_IFDIR) != 0;
Damien Martin-Guillerez0cd6dfb2015-06-29 11:06:50 +0000118 if (attr == 0) {
119 // Fallback when the external attribute is not set.
120 isdir = filename[strlen(filename)-1] == '/';
121 perm = 0777;
122 }
Damien Martin-Guillerez08441122015-05-28 11:12:31 +0000123 if (verbose_) {
Damien Martin-Guillereza1c73f92015-05-29 12:27:25 +0000124 printf("%c %o %s\n", isdir ? 'd' : 'f', perm, filename);
Damien Martin-Guillerez08441122015-05-28 11:12:31 +0000125 }
126 if (extract_) {
127 char path[PATH_MAX];
128 int fd;
129 concat_path(path, PATH_MAX, output_root_, filename);
130 mkdirs(path, perm);
131 if (!isdir) {
132 fd = open(path, O_CREAT | O_WRONLY, perm);
133 if (fd < 0) {
134 fprintf(stderr, "Cannot open file %s for writing: %s\n",
135 path, strerror(errno));
136 abort();
137 }
138 SYSCALL(write(fd, data, size));
139 SYSCALL(close(fd));
140 }
141 }
142}
143
144// Get the basename of path and store it in output. output_size
145// is the size of the output buffer.
146void basename(const char *path, char *output, size_t output_size) {
147 const char *pointer = strrchr(path, '/');
148 if (pointer == NULL) {
149 pointer = path;
Damien Martin-Guillereza1c73f92015-05-29 12:27:25 +0000150 } else {
151 pointer++; // Skip the leading slash.
Damien Martin-Guillerez08441122015-05-28 11:12:31 +0000152 }
153 strncpy(output, pointer, output_size);
154 output[output_size-1] = 0;
155}
156
157
158// Execute the extraction (or just listing if just v is provided)
159int extract(char *zipfile, bool verbose, bool extract) {
160 char output_root[PATH_MAX];
161 getcwd(output_root, PATH_MAX);
162
163 UnzipProcessor processor(output_root, verbose, extract);
164 std::unique_ptr<ZipExtractor> extractor(ZipExtractor::Create(zipfile,
165 &processor));
166 if (extractor.get() == NULL) {
167 fprintf(stderr, "Unable to open zip file %s: %s.\n", zipfile,
168 strerror(errno));
169 return -1;
170 }
171
172 if (extractor->ProcessAll() < 0) {
173 fprintf(stderr, "%s.\n", extractor->GetError());
174 return -1;
175 }
176 return 0;
177}
178
179// Execute the create operation
180int create(char *zipfile, char **files, bool flatten, bool verbose) {
181 struct stat statst;
182 u8 size = ZipBuilder::EstimateSize(files);
183 if (size == 0) {
184 return -1;
185 }
186 std::unique_ptr<ZipBuilder> builder(ZipBuilder::Create(zipfile, size));
187 if (builder.get() == NULL) {
188 fprintf(stderr, "Unable to create zip file %s: %s.\n",
189 zipfile, strerror(errno));
190 return -1;
191 }
192 for (int i = 0; files[i] != NULL; i++) {
193 stat(files[i], &statst);
194 char path[PATH_MAX];
195 bool isdir = (statst.st_mode & S_IFDIR) != 0;
196
197 if (flatten && isdir) {
198 continue;
199 }
200
201 // Compute the path, flattening it if requested
202 if (flatten) {
203 basename(files[i], path, PATH_MAX);
204 } else {
205 strncpy(path, files[i], PATH_MAX);
206 path[PATH_MAX-1] = 0;
207 size_t len = strlen(path);
208 if (isdir && len < PATH_MAX - 1) {
209 // Add the trailing slash for folders
210 path[len] = '/';
211 path[len+1] = 0;
212 }
213 }
214
215 if (verbose) {
216 mode_t perm = statst.st_mode & 0777;
Damien Martin-Guillereza1c73f92015-05-29 12:27:25 +0000217 printf("%c %o %s\n", isdir ? 'd' : 'f', perm, path);
Damien Martin-Guillerez08441122015-05-28 11:12:31 +0000218 }
219
220 u1 *buffer = builder->NewFile(path, mode_to_zipattr(statst.st_mode));
221 if (isdir) {
222 builder->FinishFile(0);
223 } else {
224 // mmap the input file and memcpy
225 int fd = open(files[i], O_RDONLY);
226 if (fd < 0) {
227 fprintf(stderr, "Can't open file %s for reading: %s.\n",
228 files[i], strerror(errno));
229 return -1;
230 }
231 void *data = mmap(NULL, statst.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
232 if (data == MAP_FAILED) {
233 fprintf(stderr, "Can't mmap file %s for reading: %s.\n",
234 files[i], strerror(errno));
235 return -1;
236 }
237 memcpy(buffer, data, statst.st_size);
238 munmap(data, statst.st_size);
239 builder->FinishFile(statst.st_size);
240 }
241 }
242 if (builder->Finish() < 0) {
243 fprintf(stderr, "%s\n", builder->GetError());
244 return -1;
245 }
246 return 0;
247}
248
249} // namespace devtools_ijar
250
251//
252// main method
253//
254static void usage(char *progname) {
255 fprintf(stderr, "Usage: %s [vxc[f]] x.zip [file1...filen]\n", progname);
256 fprintf(stderr, " v verbose - list all file in x.zip\n");
257 fprintf(stderr, " x extract - extract file in x.zip in current directory\n");
258 fprintf(stderr, " c create - add files to x.zip\n");
259 fprintf(stderr, " f flatten - flatten files to use with create operation\n");
260 fprintf(stderr, "x and c cannot be used in the same command-line.\n");
261 exit(1);
262}
263
264int main(int argc, char **argv) {
265 bool extract = false;
266 bool verbose = false;
267 bool create = false;
268 bool flatten = false;
269
270 if (argc < 3) {
271 usage(argv[0]);
272 }
273
274 for (int i = 0; argv[1][i] != 0; i++) {
275 switch (argv[1][i]) {
276 case 'x':
277 extract = true;
278 break;
279 case 'v':
280 verbose = true;
281 break;
282 case 'c':
283 create = true;
284 break;
285 case 'f':
286 flatten = true;
287 break;
288 default:
289 usage(argv[0]);
290 }
291 }
292 if (create) {
293 if (extract) {
294 usage(argv[0]);
295 }
296 // Create a zip
297 return devtools_ijar::create(argv[2], argv+3, flatten, verbose);
298 } else {
299 if (flatten) {
300 usage(argv[0]);
301 }
302 // Extraction / list mode
303 return devtools_ijar::extract(argv[2], verbose, extract);
304 }
305}