|
1 /* This Source Code Form is subject to the terms of the Mozilla Public |
|
2 * License, v. 2.0. If a copy of the MPL was not distributed with this |
|
3 * file, You can obtain one at http://mozilla.org/MPL/2.0/. |
|
4 */ |
|
5 |
|
6 #include "StreamFunctions.h" |
|
7 #include "nsZipHeader.h" |
|
8 #include "nsMemory.h" |
|
9 #include "prtime.h" |
|
10 |
|
11 #define ZIP_FILE_HEADER_SIGNATURE 0x04034b50 |
|
12 #define ZIP_FILE_HEADER_SIZE 30 |
|
13 #define ZIP_CDS_HEADER_SIGNATURE 0x02014b50 |
|
14 #define ZIP_CDS_HEADER_SIZE 46 |
|
15 |
|
16 #define FLAGS_IS_UTF8 0x800 |
|
17 |
|
18 #define ZIP_EXTENDED_TIMESTAMP_FIELD 0x5455 |
|
19 #define ZIP_EXTENDED_TIMESTAMP_MODTIME 0x01 |
|
20 |
|
21 /** |
|
22 * nsZipHeader represents an entry from a zip file. |
|
23 */ |
|
24 NS_IMPL_ISUPPORTS(nsZipHeader, nsIZipEntry) |
|
25 |
|
26 /* readonly attribute unsigned short compression; */ |
|
27 NS_IMETHODIMP nsZipHeader::GetCompression(uint16_t *aCompression) |
|
28 { |
|
29 NS_ASSERTION(mInited, "Not initalised"); |
|
30 |
|
31 *aCompression = mMethod; |
|
32 return NS_OK; |
|
33 } |
|
34 |
|
35 /* readonly attribute unsigned long size; */ |
|
36 NS_IMETHODIMP nsZipHeader::GetSize(uint32_t *aSize) |
|
37 { |
|
38 NS_ASSERTION(mInited, "Not initalised"); |
|
39 |
|
40 *aSize = mCSize; |
|
41 return NS_OK; |
|
42 } |
|
43 |
|
44 /* readonly attribute unsigned long realSize; */ |
|
45 NS_IMETHODIMP nsZipHeader::GetRealSize(uint32_t *aRealSize) |
|
46 { |
|
47 NS_ASSERTION(mInited, "Not initalised"); |
|
48 |
|
49 *aRealSize = mUSize; |
|
50 return NS_OK; |
|
51 } |
|
52 |
|
53 /* readonly attribute unsigned long CRC32; */ |
|
54 NS_IMETHODIMP nsZipHeader::GetCRC32(uint32_t *aCRC32) |
|
55 { |
|
56 NS_ASSERTION(mInited, "Not initalised"); |
|
57 |
|
58 *aCRC32 = mCRC; |
|
59 return NS_OK; |
|
60 } |
|
61 |
|
62 /* readonly attribute boolean isDirectory; */ |
|
63 NS_IMETHODIMP nsZipHeader::GetIsDirectory(bool *aIsDirectory) |
|
64 { |
|
65 NS_ASSERTION(mInited, "Not initalised"); |
|
66 |
|
67 if (mName.Last() == '/') |
|
68 *aIsDirectory = true; |
|
69 else |
|
70 *aIsDirectory = false; |
|
71 return NS_OK; |
|
72 } |
|
73 |
|
74 /* readonly attribute PRTime lastModifiedTime; */ |
|
75 NS_IMETHODIMP nsZipHeader::GetLastModifiedTime(PRTime *aLastModifiedTime) |
|
76 { |
|
77 NS_ASSERTION(mInited, "Not initalised"); |
|
78 |
|
79 // Try to read timestamp from extra field |
|
80 uint16_t blocksize; |
|
81 const uint8_t *tsField = GetExtraField(ZIP_EXTENDED_TIMESTAMP_FIELD, false, &blocksize); |
|
82 if (tsField && blocksize >= 5) { |
|
83 uint32_t pos = 4; |
|
84 uint8_t flags; |
|
85 flags = READ8(tsField, &pos); |
|
86 if (flags & ZIP_EXTENDED_TIMESTAMP_MODTIME) { |
|
87 *aLastModifiedTime = (PRTime)(READ32(tsField, &pos)) |
|
88 * PR_USEC_PER_SEC; |
|
89 return NS_OK; |
|
90 } |
|
91 } |
|
92 |
|
93 // Use DOS date/time fields |
|
94 // Note that on DST shift we can't handle correctly the hour that is valid |
|
95 // in both DST zones |
|
96 PRExplodedTime time; |
|
97 |
|
98 time.tm_usec = 0; |
|
99 |
|
100 time.tm_hour = (mTime >> 11) & 0x1F; |
|
101 time.tm_min = (mTime >> 5) & 0x3F; |
|
102 time.tm_sec = (mTime & 0x1F) * 2; |
|
103 |
|
104 time.tm_year = (mDate >> 9) + 1980; |
|
105 time.tm_month = ((mDate >> 5) & 0x0F) - 1; |
|
106 time.tm_mday = mDate & 0x1F; |
|
107 |
|
108 time.tm_params.tp_gmt_offset = 0; |
|
109 time.tm_params.tp_dst_offset = 0; |
|
110 |
|
111 PR_NormalizeTime(&time, PR_GMTParameters); |
|
112 time.tm_params.tp_gmt_offset = PR_LocalTimeParameters(&time).tp_gmt_offset; |
|
113 PR_NormalizeTime(&time, PR_GMTParameters); |
|
114 time.tm_params.tp_dst_offset = PR_LocalTimeParameters(&time).tp_dst_offset; |
|
115 |
|
116 *aLastModifiedTime = PR_ImplodeTime(&time); |
|
117 |
|
118 return NS_OK; |
|
119 } |
|
120 |
|
121 /* readonly attribute boolean isSynthetic; */ |
|
122 NS_IMETHODIMP nsZipHeader::GetIsSynthetic(bool *aIsSynthetic) |
|
123 { |
|
124 NS_ASSERTION(mInited, "Not initalised"); |
|
125 |
|
126 *aIsSynthetic = false; |
|
127 return NS_OK; |
|
128 } |
|
129 |
|
130 /* readonly attribute unsigned long permissions; */ |
|
131 NS_IMETHODIMP nsZipHeader::GetPermissions(uint32_t *aPermissions) |
|
132 { |
|
133 NS_ASSERTION(mInited, "Not initalised"); |
|
134 |
|
135 // Always give user read access at least, this matches nsIZipReader's behaviour |
|
136 *aPermissions = ((mEAttr >> 16) & 0xfff) | 0x100; |
|
137 return NS_OK; |
|
138 } |
|
139 |
|
140 void nsZipHeader::Init(const nsACString & aPath, PRTime aDate, uint32_t aAttr, |
|
141 uint32_t aOffset) |
|
142 { |
|
143 NS_ASSERTION(!mInited, "Already initalised"); |
|
144 |
|
145 PRExplodedTime time; |
|
146 PR_ExplodeTime(aDate, PR_LocalTimeParameters, &time); |
|
147 |
|
148 mTime = time.tm_sec / 2 + (time.tm_min << 5) + (time.tm_hour << 11); |
|
149 mDate = time.tm_mday + ((time.tm_month + 1) << 5) + |
|
150 ((time.tm_year - 1980) << 9); |
|
151 |
|
152 // Store modification timestamp as extra field |
|
153 // First fill CDS extra field |
|
154 mFieldLength = 9; |
|
155 mExtraField = new uint8_t[mFieldLength]; |
|
156 if (!mExtraField) { |
|
157 mFieldLength = 0; |
|
158 } else { |
|
159 uint32_t pos = 0; |
|
160 WRITE16(mExtraField.get(), &pos, ZIP_EXTENDED_TIMESTAMP_FIELD); |
|
161 WRITE16(mExtraField.get(), &pos, 5); |
|
162 WRITE8(mExtraField.get(), &pos, ZIP_EXTENDED_TIMESTAMP_MODTIME); |
|
163 WRITE32(mExtraField.get(), &pos, aDate / PR_USEC_PER_SEC); |
|
164 |
|
165 // Fill local extra field |
|
166 mLocalExtraField = new uint8_t[mFieldLength]; |
|
167 if (mLocalExtraField) { |
|
168 mLocalFieldLength = mFieldLength; |
|
169 memcpy(mLocalExtraField.get(), mExtraField.get(), mLocalFieldLength); |
|
170 } |
|
171 } |
|
172 |
|
173 mEAttr = aAttr; |
|
174 mOffset = aOffset; |
|
175 mName = aPath; |
|
176 mComment = NS_LITERAL_CSTRING(""); |
|
177 // Claim a UTF-8 path in case it needs it. |
|
178 mFlags |= FLAGS_IS_UTF8; |
|
179 mInited = true; |
|
180 } |
|
181 |
|
182 uint32_t nsZipHeader::GetFileHeaderLength() |
|
183 { |
|
184 return ZIP_FILE_HEADER_SIZE + mName.Length() + mLocalFieldLength; |
|
185 } |
|
186 |
|
187 nsresult nsZipHeader::WriteFileHeader(nsIOutputStream *aStream) |
|
188 { |
|
189 NS_ASSERTION(mInited, "Not initalised"); |
|
190 |
|
191 uint8_t buf[ZIP_FILE_HEADER_SIZE]; |
|
192 uint32_t pos = 0; |
|
193 WRITE32(buf, &pos, ZIP_FILE_HEADER_SIGNATURE); |
|
194 WRITE16(buf, &pos, mVersionNeeded); |
|
195 WRITE16(buf, &pos, mFlags); |
|
196 WRITE16(buf, &pos, mMethod); |
|
197 WRITE16(buf, &pos, mTime); |
|
198 WRITE16(buf, &pos, mDate); |
|
199 WRITE32(buf, &pos, mCRC); |
|
200 WRITE32(buf, &pos, mCSize); |
|
201 WRITE32(buf, &pos, mUSize); |
|
202 WRITE16(buf, &pos, mName.Length()); |
|
203 WRITE16(buf, &pos, mLocalFieldLength); |
|
204 |
|
205 nsresult rv = ZW_WriteData(aStream, (const char *)buf, pos); |
|
206 NS_ENSURE_SUCCESS(rv, rv); |
|
207 |
|
208 rv = ZW_WriteData(aStream, mName.get(), mName.Length()); |
|
209 NS_ENSURE_SUCCESS(rv, rv); |
|
210 |
|
211 if (mLocalFieldLength) |
|
212 { |
|
213 rv = ZW_WriteData(aStream, (const char *)mLocalExtraField.get(), mLocalFieldLength); |
|
214 NS_ENSURE_SUCCESS(rv, rv); |
|
215 } |
|
216 |
|
217 return NS_OK; |
|
218 } |
|
219 |
|
220 uint32_t nsZipHeader::GetCDSHeaderLength() |
|
221 { |
|
222 return ZIP_CDS_HEADER_SIZE + mName.Length() + mComment.Length() + |
|
223 mFieldLength; |
|
224 } |
|
225 |
|
226 nsresult nsZipHeader::WriteCDSHeader(nsIOutputStream *aStream) |
|
227 { |
|
228 NS_ASSERTION(mInited, "Not initalised"); |
|
229 |
|
230 uint8_t buf[ZIP_CDS_HEADER_SIZE]; |
|
231 uint32_t pos = 0; |
|
232 WRITE32(buf, &pos, ZIP_CDS_HEADER_SIGNATURE); |
|
233 WRITE16(buf, &pos, mVersionMade); |
|
234 WRITE16(buf, &pos, mVersionNeeded); |
|
235 WRITE16(buf, &pos, mFlags); |
|
236 WRITE16(buf, &pos, mMethod); |
|
237 WRITE16(buf, &pos, mTime); |
|
238 WRITE16(buf, &pos, mDate); |
|
239 WRITE32(buf, &pos, mCRC); |
|
240 WRITE32(buf, &pos, mCSize); |
|
241 WRITE32(buf, &pos, mUSize); |
|
242 WRITE16(buf, &pos, mName.Length()); |
|
243 WRITE16(buf, &pos, mFieldLength); |
|
244 WRITE16(buf, &pos, mComment.Length()); |
|
245 WRITE16(buf, &pos, mDisk); |
|
246 WRITE16(buf, &pos, mIAttr); |
|
247 WRITE32(buf, &pos, mEAttr); |
|
248 WRITE32(buf, &pos, mOffset); |
|
249 |
|
250 nsresult rv = ZW_WriteData(aStream, (const char *)buf, pos); |
|
251 NS_ENSURE_SUCCESS(rv, rv); |
|
252 |
|
253 rv = ZW_WriteData(aStream, mName.get(), mName.Length()); |
|
254 NS_ENSURE_SUCCESS(rv, rv); |
|
255 if (mExtraField) { |
|
256 rv = ZW_WriteData(aStream, (const char *)mExtraField.get(), mFieldLength); |
|
257 NS_ENSURE_SUCCESS(rv, rv); |
|
258 } |
|
259 return ZW_WriteData(aStream, mComment.get(), mComment.Length()); |
|
260 } |
|
261 |
|
262 nsresult nsZipHeader::ReadCDSHeader(nsIInputStream *stream) |
|
263 { |
|
264 NS_ASSERTION(!mInited, "Already initalised"); |
|
265 |
|
266 uint8_t buf[ZIP_CDS_HEADER_SIZE]; |
|
267 |
|
268 nsresult rv = ZW_ReadData(stream, (char *)buf, ZIP_CDS_HEADER_SIZE); |
|
269 NS_ENSURE_SUCCESS(rv, rv); |
|
270 |
|
271 uint32_t pos = 0; |
|
272 uint32_t signature = READ32(buf, &pos); |
|
273 if (signature != ZIP_CDS_HEADER_SIGNATURE) |
|
274 return NS_ERROR_FILE_CORRUPTED; |
|
275 |
|
276 mVersionMade = READ16(buf, &pos); |
|
277 mVersionNeeded = READ16(buf, &pos); |
|
278 mFlags = READ16(buf, &pos); |
|
279 mMethod = READ16(buf, &pos); |
|
280 mTime = READ16(buf, &pos); |
|
281 mDate = READ16(buf, &pos); |
|
282 mCRC = READ32(buf, &pos); |
|
283 mCSize = READ32(buf, &pos); |
|
284 mUSize = READ32(buf, &pos); |
|
285 uint16_t namelength = READ16(buf, &pos); |
|
286 mFieldLength = READ16(buf, &pos); |
|
287 uint16_t commentlength = READ16(buf, &pos); |
|
288 mDisk = READ16(buf, &pos); |
|
289 mIAttr = READ16(buf, &pos); |
|
290 mEAttr = READ32(buf, &pos); |
|
291 mOffset = READ32(buf, &pos); |
|
292 |
|
293 if (namelength > 0) { |
|
294 nsAutoArrayPtr<char> field(new char[namelength]); |
|
295 NS_ENSURE_TRUE(field, NS_ERROR_OUT_OF_MEMORY); |
|
296 rv = ZW_ReadData(stream, field.get(), namelength); |
|
297 NS_ENSURE_SUCCESS(rv, rv); |
|
298 mName.Assign(field, namelength); |
|
299 } |
|
300 else |
|
301 mName = NS_LITERAL_CSTRING(""); |
|
302 |
|
303 if (mFieldLength > 0) { |
|
304 mExtraField = new uint8_t[mFieldLength]; |
|
305 NS_ENSURE_TRUE(mExtraField, NS_ERROR_OUT_OF_MEMORY); |
|
306 rv = ZW_ReadData(stream, (char *)mExtraField.get(), mFieldLength); |
|
307 NS_ENSURE_SUCCESS(rv, rv); |
|
308 } |
|
309 |
|
310 if (commentlength > 0) { |
|
311 nsAutoArrayPtr<char> field(new char[commentlength]); |
|
312 NS_ENSURE_TRUE(field, NS_ERROR_OUT_OF_MEMORY); |
|
313 rv = ZW_ReadData(stream, field.get(), commentlength); |
|
314 NS_ENSURE_SUCCESS(rv, rv); |
|
315 mComment.Assign(field, commentlength); |
|
316 } |
|
317 else |
|
318 mComment = NS_LITERAL_CSTRING(""); |
|
319 |
|
320 mInited = true; |
|
321 return NS_OK; |
|
322 } |
|
323 |
|
324 const uint8_t * nsZipHeader::GetExtraField(uint16_t aTag, bool aLocal, uint16_t *aBlockSize) |
|
325 { |
|
326 const uint8_t *buf = aLocal ? mLocalExtraField : mExtraField; |
|
327 uint32_t buflen = aLocal ? mLocalFieldLength : mFieldLength; |
|
328 uint32_t pos = 0; |
|
329 uint16_t tag, blocksize; |
|
330 |
|
331 while (buf && (pos + 4) <= buflen) { |
|
332 tag = READ16(buf, &pos); |
|
333 blocksize = READ16(buf, &pos); |
|
334 |
|
335 if (aTag == tag && (pos + blocksize) <= buflen) { |
|
336 *aBlockSize = blocksize; |
|
337 return buf + pos - 4; |
|
338 } |
|
339 |
|
340 pos += blocksize; |
|
341 } |
|
342 |
|
343 return nullptr; |
|
344 } |
|
345 |
|
346 /* |
|
347 * Pad extra field to align data starting position to specified size. |
|
348 */ |
|
349 nsresult nsZipHeader::PadExtraField(uint32_t aOffset, uint16_t aAlignSize) |
|
350 { |
|
351 uint32_t pad_size; |
|
352 uint32_t pa_offset; |
|
353 uint32_t pa_end; |
|
354 |
|
355 // Check for range and power of 2. |
|
356 if (aAlignSize < 2 || aAlignSize > 32768 || |
|
357 (aAlignSize & (aAlignSize - 1)) != 0) { |
|
358 return NS_ERROR_INVALID_ARG; |
|
359 } |
|
360 |
|
361 // Point to current starting data position. |
|
362 aOffset += ZIP_FILE_HEADER_SIZE + mName.Length() + mLocalFieldLength; |
|
363 |
|
364 // Calculate aligned offset. |
|
365 pa_offset = aOffset & ~(aAlignSize - 1); |
|
366 pa_end = pa_offset + aAlignSize; |
|
367 pad_size = pa_end - aOffset; |
|
368 if (pad_size == 0) { |
|
369 return NS_OK; |
|
370 } |
|
371 |
|
372 // Leave enough room(at least 4 bytes) for valid values in extra field. |
|
373 while (pad_size < 4) { |
|
374 pad_size += aAlignSize; |
|
375 } |
|
376 // Extra field length is 2 bytes. |
|
377 if (mLocalFieldLength + pad_size > 65535) { |
|
378 return NS_ERROR_FAILURE; |
|
379 } |
|
380 |
|
381 nsAutoArrayPtr<uint8_t> field = mLocalExtraField; |
|
382 uint32_t pos = mLocalFieldLength; |
|
383 |
|
384 mLocalExtraField = new uint8_t[mLocalFieldLength + pad_size]; |
|
385 memcpy(mLocalExtraField.get(), field, mLocalFieldLength); |
|
386 // Use 0xFFFF as tag ID to avoid conflict with other IDs. |
|
387 // For more information, please read "Extensible data fields" section in: |
|
388 // http://www.pkware.com/documents/casestudies/APPNOTE.TXT |
|
389 WRITE16(mLocalExtraField.get(), &pos, 0xFFFF); |
|
390 WRITE16(mLocalExtraField.get(), &pos, pad_size - 4); |
|
391 memset(mLocalExtraField.get() + pos, 0, pad_size - 4); |
|
392 mLocalFieldLength += pad_size; |
|
393 |
|
394 return NS_OK; |
|
395 } |