VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/zip/tarvfs.cpp@ 98092

Last change on this file since 98092 was 96407, checked in by vboxsync, 2 years ago

scm copyright and license note update

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 45.8 KB
Line 
1/* $Id: tarvfs.cpp 96407 2022-08-22 17:43:14Z vboxsync $ */
2/** @file
3 * IPRT - TAR Virtual Filesystem, Reader.
4 */
5
6/*
7 * Copyright (C) 2010-2022 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#include "internal/iprt.h"
42#include <iprt/zip.h>
43
44#include <iprt/asm.h>
45#include <iprt/assert.h>
46#include <iprt/ctype.h>
47#include <iprt/err.h>
48#include <iprt/poll.h>
49#include <iprt/file.h>
50#include <iprt/string.h>
51#include <iprt/vfs.h>
52#include <iprt/vfslowlevel.h>
53
54#include "tar.h"
55#include "tarvfsreader.h"
56
57
58/**
59 * Converts a numeric header field to the C native type.
60 *
61 * @returns IPRT status code.
62 *
63 * @param pszField The TAR header field.
64 * @param cchField The length of the field.
65 * @param fOctalOnly Must be octal.
66 * @param pi64 Where to store the value.
67 */
68static int rtZipTarHdrFieldToNum(const char *pszField, size_t cchField, bool fOctalOnly, int64_t *pi64)
69{
70 unsigned char const *puchField = (unsigned char const *)pszField;
71 size_t const cchFieldOrg = cchField;
72 if ( fOctalOnly
73 || !(*puchField & 0x80))
74 {
75 /*
76 * Skip leading spaces. Include zeros to save a few slower loops below.
77 */
78 unsigned char ch;
79 while (cchField > 0 && ((ch = *puchField) == ' '|| ch == '0'))
80 cchField--, puchField++;
81
82 /*
83 * Convert octal digits.
84 */
85 int64_t i64 = 0;
86 while (cchField > 0)
87 {
88 unsigned char uDigit = *puchField - '0';
89 if (uDigit >= 8)
90 break;
91 i64 <<= 3;
92 i64 |= uDigit;
93
94 puchField++;
95 cchField--;
96 }
97 *pi64 = i64;
98
99 /*
100 * Was it terminated correctly?
101 */
102 while (cchField > 0)
103 {
104 ch = *puchField++;
105 if (ch != 0 && ch != ' ')
106 return cchField < cchFieldOrg
107 ? VERR_TAR_BAD_NUM_FIELD_TERM
108 : VERR_TAR_BAD_NUM_FIELD;
109 cchField--;
110 }
111 }
112 else
113 {
114 /*
115 * The first byte has the bit 7 set to indicate base-256, while bit 6
116 * is the signed bit. Bits 5:0 are the most significant value bits.
117 */
118 uint64_t u64;
119 if (!(0x40 & *puchField))
120 {
121 /* Positive or zero value. */
122 u64 = *puchField & 0x3f;
123 cchField--;
124 puchField++;
125
126 while (cchField-- > 0)
127 {
128 if (RT_LIKELY(u64 <= (uint64_t)INT64_MAX / 256))
129 u64 = (u64 << 8) | *puchField++;
130 else
131 return VERR_TAR_NUM_VALUE_TOO_LARGE;
132 }
133 }
134 else
135 {
136 /* Negative value (could be used in timestamp). We do manual sign extending here. */
137 u64 = (UINT64_MAX << 6) | (*puchField & 0x3f);
138 cchField--;
139 puchField++;
140
141 while (cchField-- > 0)
142 {
143 if (RT_LIKELY(u64 >= (uint64_t)(INT64_MIN / 256)))
144 u64 = (u64 << 8) | *puchField++;
145 else
146 return VERR_TAR_NUM_VALUE_TOO_LARGE;
147 }
148 }
149 *pi64 = (int64_t)u64;
150 }
151
152 return VINF_SUCCESS;
153}
154
155
156/**
157 * Validates the TAR header.
158 *
159 * @returns VINF_SUCCESS if valid, VERR_TAR_ZERO_HEADER if all zeros, and
160 * the appropriate VERR_TAR_XXX otherwise.
161 * @param pTar The TAR header.
162 * @param penmType Where to return the type of header on success.
163 */
164static int rtZipTarHdrValidate(PCRTZIPTARHDR pTar, PRTZIPTARTYPE penmType)
165{
166 /*
167 * Calc the checksum first since this enables us to detect zero headers.
168 */
169 int32_t i32ChkSum;
170 int32_t i32ChkSumSignedAlt;
171 if (rtZipTarCalcChkSum(pTar, &i32ChkSum, &i32ChkSumSignedAlt))
172 return VERR_TAR_ZERO_HEADER;
173
174 /*
175 * Read the checksum field and match the checksums.
176 */
177 int64_t i64HdrChkSum;
178 int rc = rtZipTarHdrFieldToNum(pTar->Common.chksum, sizeof(pTar->Common.chksum), true /*fOctalOnly*/, &i64HdrChkSum);
179 if (RT_FAILURE(rc))
180 return VERR_TAR_BAD_CHKSUM_FIELD;
181 if ( i32ChkSum != i64HdrChkSum
182 && i32ChkSumSignedAlt != i64HdrChkSum) /** @todo test this */
183 return VERR_TAR_CHKSUM_MISMATCH;
184
185 /*
186 * Detect the TAR type.
187 */
188 RTZIPTARTYPE enmType;
189 if ( pTar->Common.magic[0] == 'u'
190 && pTar->Common.magic[1] == 's'
191 && pTar->Common.magic[2] == 't'
192 && pTar->Common.magic[3] == 'a'
193 && pTar->Common.magic[4] == 'r')
194 {
195/** @todo detect star headers */
196 if ( pTar->Common.magic[5] == '\0'
197 && pTar->Common.version[0] == '0'
198 && pTar->Common.version[1] == '0')
199 enmType = RTZIPTARTYPE_POSIX;
200 else if ( pTar->Common.magic[5] == ' '
201 && pTar->Common.version[0] == ' '
202 && pTar->Common.version[1] == '\0')
203 enmType = RTZIPTARTYPE_GNU;
204 else if ( pTar->Common.magic[5] == '\0' /* VMWare ambiguity - they probably mean posix but */
205 && pTar->Common.version[0] == ' ' /* got the version wrong. */
206 && pTar->Common.version[1] == '\0')
207 enmType = RTZIPTARTYPE_POSIX;
208 else
209 return VERR_TAR_NOT_USTAR_V00;
210 }
211 else
212 enmType = RTZIPTARTYPE_ANCIENT;
213 *penmType = enmType;
214
215 /*
216 * Perform some basic checks.
217 */
218 switch (enmType)
219 {
220 case RTZIPTARTYPE_POSIX:
221 if ( !RT_C_IS_ALNUM(pTar->Common.typeflag)
222 && pTar->Common.typeflag != '\0')
223 return VERR_TAR_UNKNOWN_TYPE_FLAG;
224 break;
225
226 case RTZIPTARTYPE_GNU:
227 switch (pTar->Common.typeflag)
228 {
229 case RTZIPTAR_TF_OLDNORMAL:
230 case RTZIPTAR_TF_NORMAL:
231 case RTZIPTAR_TF_CONTIG:
232 case RTZIPTAR_TF_DIR:
233 case RTZIPTAR_TF_CHR:
234 case RTZIPTAR_TF_BLK:
235 case RTZIPTAR_TF_LINK:
236 case RTZIPTAR_TF_SYMLINK:
237 case RTZIPTAR_TF_FIFO:
238 break;
239
240 case RTZIPTAR_TF_GNU_LONGLINK:
241 case RTZIPTAR_TF_GNU_LONGNAME:
242 break;
243
244 case RTZIPTAR_TF_GNU_DUMPDIR:
245 case RTZIPTAR_TF_GNU_MULTIVOL:
246 case RTZIPTAR_TF_GNU_SPARSE:
247 case RTZIPTAR_TF_GNU_VOLDHR:
248 /** @todo Implement full GNU TAR support. .*/
249 return VERR_TAR_UNSUPPORTED_GNU_HDR_TYPE;
250
251 default:
252 return VERR_TAR_UNKNOWN_TYPE_FLAG;
253 }
254 break;
255
256 case RTZIPTARTYPE_ANCIENT:
257 switch (pTar->Common.typeflag)
258 {
259 case RTZIPTAR_TF_OLDNORMAL:
260 case RTZIPTAR_TF_NORMAL:
261 case RTZIPTAR_TF_CONTIG:
262 case RTZIPTAR_TF_DIR:
263 case RTZIPTAR_TF_LINK:
264 case RTZIPTAR_TF_SYMLINK:
265 case RTZIPTAR_TF_FIFO:
266 break;
267 default:
268 return VERR_TAR_UNKNOWN_TYPE_FLAG;
269 }
270 break;
271 default: /* shut up gcc */
272 AssertFailedReturn(VERR_INTERNAL_ERROR_3);
273 }
274
275 return VINF_SUCCESS;
276}
277
278
279/**
280 * Parses and validates the first TAR header of a archive/file/dir/whatever.
281 *
282 * @returns IPRT status code.
283 * @param pThis The TAR reader stat.
284 * @param pTar The TAR header that has been read.
285 * @param fFirst Set if this is the first header, otherwise
286 * clear.
287 */
288static int rtZipTarReaderParseNextHeader(PRTZIPTARREADER pThis, PCRTZIPTARHDR pHdr, bool fFirst)
289{
290 int rc;
291
292 /*
293 * Basic header validation and detection first.
294 */
295 RTZIPTARTYPE enmType;
296 rc = rtZipTarHdrValidate(pHdr, &enmType);
297 if (RT_FAILURE_NP(rc))
298 {
299 if (rc == VERR_TAR_ZERO_HEADER)
300 {
301 pThis->cZeroHdrs = 1;
302 pThis->enmState = RTZIPTARREADERSTATE_ZERO;
303 return VINF_SUCCESS;
304 }
305 return rc;
306 }
307 if (fFirst)
308 {
309 pThis->enmType = enmType;
310 if (pThis->enmPrevType == RTZIPTARTYPE_INVALID)
311 pThis->enmPrevType = enmType;
312 }
313
314 /*
315 * Handle the header by type.
316 */
317 switch (pHdr->Common.typeflag)
318 {
319 case RTZIPTAR_TF_OLDNORMAL:
320 case RTZIPTAR_TF_NORMAL:
321 case RTZIPTAR_TF_CONTIG:
322 case RTZIPTAR_TF_LINK:
323 case RTZIPTAR_TF_SYMLINK:
324 case RTZIPTAR_TF_CHR:
325 case RTZIPTAR_TF_BLK:
326 case RTZIPTAR_TF_FIFO:
327 case RTZIPTAR_TF_DIR:
328 /*
329 * Extract the name first.
330 */
331 if (!pHdr->Common.name[0])
332 return VERR_TAR_EMPTY_NAME;
333 if (pThis->enmType == RTZIPTARTYPE_POSIX)
334 {
335 Assert(pThis->offGnuLongCur == 0); Assert(pThis->szName[0] == '\0');
336 pThis->szName[0] = '\0';
337 if (pHdr->Posix.prefix[0])
338 {
339 rc = RTStrCopyEx(pThis->szName, sizeof(pThis->szName), pHdr->Posix.prefix, sizeof(pHdr->Posix.prefix));
340 AssertRC(rc); /* shall not fail */
341 rc = RTStrCat(pThis->szName, sizeof(pThis->szName), "/");
342 AssertRC(rc); /* ditto */
343 }
344 rc = RTStrCatEx(pThis->szName, sizeof(pThis->szName), pHdr->Common.name, sizeof(pHdr->Common.name));
345 AssertRCReturn(rc, rc);
346 }
347 else if (pThis->enmType == RTZIPTARTYPE_GNU)
348 {
349 if (!pThis->szName[0])
350 {
351 rc = RTStrCopyEx(pThis->szName, sizeof(pThis->szName), pHdr->Common.name, sizeof(pHdr->Common.name));
352 AssertRCReturn(rc, rc);
353 }
354 }
355 else
356 {
357 /* Old TAR */
358 Assert(pThis->offGnuLongCur == 0); Assert(pThis->szName[0] == '\0');
359 rc = RTStrCopyEx(pThis->szName, sizeof(pThis->szName), pHdr->Common.name, sizeof(pHdr->Common.name));
360 AssertRCReturn(rc, rc);
361 }
362
363 /*
364 * Extract the link target.
365 */
366 if ( pHdr->Common.typeflag == RTZIPTAR_TF_LINK
367 || pHdr->Common.typeflag == RTZIPTAR_TF_SYMLINK)
368 {
369 if ( pThis->enmType == RTZIPTARTYPE_POSIX
370 || pThis->enmType == RTZIPTARTYPE_ANCIENT
371 || (pThis->enmType == RTZIPTARTYPE_GNU && pThis->szTarget[0] == '\0')
372 )
373 {
374 Assert(pThis->szTarget[0] == '\0');
375 rc = RTStrCopyEx(pThis->szTarget, sizeof(pThis->szTarget),
376 pHdr->Common.linkname, sizeof(pHdr->Common.linkname));
377 AssertRCReturn(rc, rc);
378 }
379 }
380 else
381 pThis->szTarget[0] = '\0';
382
383 pThis->Hdr = *pHdr;
384 break;
385
386 case RTZIPTAR_TF_X_HDR:
387 case RTZIPTAR_TF_X_GLOBAL:
388 /** @todo implement PAX */
389 return VERR_TAR_UNSUPPORTED_PAX_TYPE;
390
391 case RTZIPTAR_TF_SOLARIS_XHDR:
392 /** @todo implement solaris / pax attribute lists. */
393 return VERR_TAR_UNSUPPORTED_SOLARIS_HDR_TYPE;
394
395
396 /*
397 * A GNU long name or long link is a dummy record followed by one or
398 * more 512 byte string blocks holding the long name/link. The name
399 * lenght is encoded in the size field, null terminator included. If
400 * it is a symlink or hard link the long name may be followed by a
401 * long link sequence.
402 */
403 case RTZIPTAR_TF_GNU_LONGNAME:
404 case RTZIPTAR_TF_GNU_LONGLINK:
405 {
406 if (strcmp(pHdr->Gnu.name, "././@LongLink"))
407 return VERR_TAR_MALFORMED_GNU_LONGXXXX;
408
409 int64_t cb64;
410 rc = rtZipTarHdrFieldToNum(pHdr->Gnu.size, sizeof(pHdr->Gnu.size), false /*fOctalOnly*/, &cb64);
411 if (RT_FAILURE(rc) || cb64 < 0 || cb64 > _1M)
412 return VERR_TAR_MALFORMED_GNU_LONGXXXX;
413 uint32_t cb = (uint32_t)cb64;
414 if (cb >= sizeof(pThis->szName))
415 return VERR_TAR_NAME_TOO_LONG;
416
417 pThis->cbGnuLongExpect = cb;
418 pThis->offGnuLongCur = 0;
419 pThis->enmState = pHdr->Common.typeflag == RTZIPTAR_TF_GNU_LONGNAME
420 ? RTZIPTARREADERSTATE_GNU_LONGNAME
421 : RTZIPTARREADERSTATE_GNU_LONGLINK;
422 break;
423 }
424
425 case RTZIPTAR_TF_GNU_DUMPDIR:
426 case RTZIPTAR_TF_GNU_MULTIVOL:
427 case RTZIPTAR_TF_GNU_SPARSE:
428 case RTZIPTAR_TF_GNU_VOLDHR:
429 /** @todo Implement or skip GNU headers */
430 return VERR_TAR_UNSUPPORTED_GNU_HDR_TYPE;
431
432 default:
433 return VERR_TAR_UNKNOWN_TYPE_FLAG;
434 }
435
436 return VINF_SUCCESS;
437}
438
439
440/**
441 * Parses and validates a TAR header.
442 *
443 * @returns IPRT status code.
444 * @param pThis The TAR reader stat.
445 * @param pTar The TAR header that has been read.
446 */
447static int rtZipTarReaderParseHeader(PRTZIPTARREADER pThis, PCRTZIPTARHDR pHdr)
448{
449 switch (pThis->enmState)
450 {
451 /*
452 * The first record for a file/directory/whatever.
453 */
454 case RTZIPTARREADERSTATE_FIRST:
455 pThis->Hdr.Common.typeflag = 0x7f;
456 pThis->enmPrevType = pThis->enmType;
457 pThis->enmType = RTZIPTARTYPE_INVALID;
458 pThis->offGnuLongCur = 0;
459 pThis->cbGnuLongExpect = 0;
460 pThis->szName[0] = '\0';
461 pThis->szTarget[0] = '\0';
462 return rtZipTarReaderParseNextHeader(pThis, pHdr, true /*fFirst*/);
463
464 /*
465 * There should only be so many zero headers at the end of the file as
466 * it is a function of the block size used when writing. Don't go on
467 * reading them forever in case someone points us to /dev/zero.
468 */
469 case RTZIPTARREADERSTATE_ZERO:
470 if (!ASMMemIsZero(pHdr, sizeof(*pHdr)))
471 return VERR_TAR_ZERO_HEADER;
472 pThis->cZeroHdrs++;
473 if (pThis->cZeroHdrs <= _64K / 512 + 2)
474 return VINF_SUCCESS;
475 return VERR_TAR_ZERO_HEADER;
476
477 case RTZIPTARREADERSTATE_GNU_LONGNAME:
478 case RTZIPTARREADERSTATE_GNU_LONGLINK:
479 {
480 size_t cbIncoming = RTStrNLen((const char *)pHdr->ab, sizeof(*pHdr));
481 if (cbIncoming < sizeof(*pHdr))
482 cbIncoming += 1;
483
484 if (cbIncoming + pThis->offGnuLongCur > pThis->cbGnuLongExpect)
485 return VERR_TAR_MALFORMED_GNU_LONGXXXX;
486 if ( cbIncoming < sizeof(*pHdr)
487 && cbIncoming + pThis->offGnuLongCur != pThis->cbGnuLongExpect)
488 return VERR_TAR_MALFORMED_GNU_LONGXXXX;
489
490 char *pszDst = pThis->enmState == RTZIPTARREADERSTATE_GNU_LONGNAME ? pThis->szName : pThis->szTarget;
491 pszDst += pThis->offGnuLongCur;
492 memcpy(pszDst, pHdr->ab, cbIncoming);
493
494 pThis->offGnuLongCur += (uint32_t)cbIncoming;
495 if (pThis->offGnuLongCur == pThis->cbGnuLongExpect)
496 pThis->enmState = RTZIPTARREADERSTATE_GNU_NEXT;
497 return VINF_SUCCESS;
498 }
499
500 case RTZIPTARREADERSTATE_GNU_NEXT:
501 pThis->enmState = RTZIPTARREADERSTATE_FIRST;
502 return rtZipTarReaderParseNextHeader(pThis, pHdr, false /*fFirst*/);
503
504 default:
505 return VERR_INTERNAL_ERROR_5;
506 }
507}
508
509
510/**
511 * Translate a TAR header to an IPRT object info structure with additional UNIX
512 * attributes.
513 *
514 * This completes the validation done by rtZipTarHdrValidate.
515 *
516 * @returns VINF_SUCCESS if valid, appropriate VERR_TAR_XXX if not.
517 * @param pThis The TAR reader instance.
518 * @param pObjInfo The object info structure (output).
519 */
520static int rtZipTarReaderGetFsObjInfo(PRTZIPTARREADER pThis, PRTFSOBJINFO pObjInfo)
521{
522 /*
523 * Zap the whole structure, this takes care of unused space in the union.
524 */
525 RT_ZERO(*pObjInfo);
526
527 /*
528 * Convert the TAR field in RTFSOBJINFO order.
529 */
530 int rc;
531 int64_t i64Tmp;
532#define GET_TAR_NUMERIC_FIELD_RET(a_Var, a_Field) \
533 do { \
534 rc = rtZipTarHdrFieldToNum(a_Field, sizeof(a_Field), false /*fOctalOnly*/, &i64Tmp); \
535 if (RT_FAILURE(rc)) \
536 return rc; \
537 (a_Var) = i64Tmp; \
538 if ((a_Var) != i64Tmp) \
539 return VERR_TAR_NUM_VALUE_TOO_LARGE; \
540 } while (0)
541
542 GET_TAR_NUMERIC_FIELD_RET(pObjInfo->cbObject, pThis->Hdr.Common.size);
543 pObjInfo->cbAllocated = RT_ALIGN_64(pObjInfo->cbObject, 512);
544 int64_t c64SecModTime;
545 GET_TAR_NUMERIC_FIELD_RET(c64SecModTime, pThis->Hdr.Common.mtime);
546 RTTimeSpecSetSeconds(&pObjInfo->ChangeTime, c64SecModTime);
547 RTTimeSpecSetSeconds(&pObjInfo->ModificationTime, c64SecModTime);
548 RTTimeSpecSetSeconds(&pObjInfo->AccessTime, c64SecModTime);
549 RTTimeSpecSetSeconds(&pObjInfo->BirthTime, c64SecModTime);
550 if (c64SecModTime != RTTimeSpecGetSeconds(&pObjInfo->ModificationTime))
551 return VERR_TAR_NUM_VALUE_TOO_LARGE;
552 GET_TAR_NUMERIC_FIELD_RET(pObjInfo->Attr.fMode, pThis->Hdr.Common.mode);
553 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_UNIX;
554 GET_TAR_NUMERIC_FIELD_RET(pObjInfo->Attr.u.Unix.uid, pThis->Hdr.Common.uid);
555 GET_TAR_NUMERIC_FIELD_RET(pObjInfo->Attr.u.Unix.gid, pThis->Hdr.Common.gid);
556 pObjInfo->Attr.u.Unix.cHardlinks = 1;
557 pObjInfo->Attr.u.Unix.INodeIdDevice = 0;
558 pObjInfo->Attr.u.Unix.INodeId = 0;
559 pObjInfo->Attr.u.Unix.fFlags = 0;
560 pObjInfo->Attr.u.Unix.GenerationId = 0;
561 pObjInfo->Attr.u.Unix.Device = 0;
562 switch (pThis->enmType)
563 {
564 case RTZIPTARTYPE_POSIX:
565 case RTZIPTARTYPE_GNU:
566 if ( pThis->Hdr.Common.typeflag == RTZIPTAR_TF_CHR
567 || pThis->Hdr.Common.typeflag == RTZIPTAR_TF_BLK)
568 {
569 uint32_t uMajor, uMinor;
570 GET_TAR_NUMERIC_FIELD_RET(uMajor, pThis->Hdr.Common.devmajor);
571 GET_TAR_NUMERIC_FIELD_RET(uMinor, pThis->Hdr.Common.devminor);
572 pObjInfo->Attr.u.Unix.Device = RTDEV_MAKE(uMajor, uMinor);
573 if ( uMajor != RTDEV_MAJOR(pObjInfo->Attr.u.Unix.Device)
574 || uMinor != RTDEV_MINOR(pObjInfo->Attr.u.Unix.Device))
575 return VERR_TAR_DEV_VALUE_TOO_LARGE;
576 }
577 break;
578
579 default:
580 if ( pThis->Hdr.Common.typeflag == RTZIPTAR_TF_CHR
581 || pThis->Hdr.Common.typeflag == RTZIPTAR_TF_BLK)
582 return VERR_TAR_UNKNOWN_TYPE_FLAG;
583 }
584
585#undef GET_TAR_NUMERIC_FIELD_RET
586
587 /*
588 * Massage the result a little bit.
589 * Also validate some more now that we've got the numbers to work with.
590 */
591 if ( (pObjInfo->Attr.fMode & ~RTFS_UNIX_MASK)
592 && pThis->enmType == RTZIPTARTYPE_POSIX)
593 return VERR_TAR_BAD_MODE_FIELD;
594 pObjInfo->Attr.fMode &= RTFS_UNIX_MASK;
595
596 RTFMODE fModeType = 0;
597 switch (pThis->Hdr.Common.typeflag)
598 {
599 case RTZIPTAR_TF_OLDNORMAL:
600 case RTZIPTAR_TF_NORMAL:
601 case RTZIPTAR_TF_CONTIG:
602 {
603 const char *pszEnd = strchr(pThis->szName, '\0');
604 if (pszEnd == &pThis->szName[0] || pszEnd[-1] != '/')
605 fModeType |= RTFS_TYPE_FILE;
606 else
607 fModeType |= RTFS_TYPE_DIRECTORY;
608 break;
609 }
610
611 case RTZIPTAR_TF_LINK:
612 if (pObjInfo->cbObject != 0)
613#if 0 /* too strict */
614 return VERR_TAR_SIZE_NOT_ZERO;
615#else
616 pObjInfo->cbObject = pObjInfo->cbAllocated = 0;
617#endif
618 fModeType |= RTFS_TYPE_FILE; /* no better idea for now */
619 break;
620
621 case RTZIPTAR_TF_SYMLINK:
622 fModeType |= RTFS_TYPE_SYMLINK;
623 break;
624
625 case RTZIPTAR_TF_CHR:
626 fModeType |= RTFS_TYPE_DEV_CHAR;
627 break;
628
629 case RTZIPTAR_TF_BLK:
630 fModeType |= RTFS_TYPE_DEV_BLOCK;
631 break;
632
633 case RTZIPTAR_TF_DIR:
634 fModeType |= RTFS_TYPE_DIRECTORY;
635 break;
636
637 case RTZIPTAR_TF_FIFO:
638 fModeType |= RTFS_TYPE_FIFO;
639 break;
640
641 case RTZIPTAR_TF_GNU_LONGLINK:
642 case RTZIPTAR_TF_GNU_LONGNAME:
643 /* ASSUMES RTFS_TYPE_XXX uses the same values as GNU stored in the mode field. */
644 fModeType = pObjInfo->Attr.fMode & RTFS_TYPE_MASK;
645 switch (fModeType)
646 {
647 case RTFS_TYPE_FILE:
648 case RTFS_TYPE_DIRECTORY:
649 case RTFS_TYPE_SYMLINK:
650 case RTFS_TYPE_DEV_BLOCK:
651 case RTFS_TYPE_DEV_CHAR:
652 case RTFS_TYPE_FIFO:
653 break;
654
655 default:
656 case 0:
657 return VERR_TAR_UNKNOWN_TYPE_FLAG; /** @todo new status code */
658 }
659 break;
660
661 default:
662 return VERR_TAR_UNKNOWN_TYPE_FLAG; /* Should've been caught in validate. */
663 }
664 if ( (pObjInfo->Attr.fMode & RTFS_TYPE_MASK)
665 && (pObjInfo->Attr.fMode & RTFS_TYPE_MASK) != fModeType)
666 return VERR_TAR_MODE_WITH_TYPE;
667 pObjInfo->Attr.fMode &= ~RTFS_TYPE_MASK;
668 pObjInfo->Attr.fMode |= fModeType;
669
670 switch (pThis->Hdr.Common.typeflag)
671 {
672 case RTZIPTAR_TF_CHR:
673 case RTZIPTAR_TF_BLK:
674 case RTZIPTAR_TF_DIR:
675 case RTZIPTAR_TF_FIFO:
676 pObjInfo->cbObject = 0;
677 pObjInfo->cbAllocated = 0;
678 break;
679 }
680
681 return VINF_SUCCESS;
682}
683
684
685/**
686 * Checks if the reader is expecting more headers.
687 *
688 * @returns true / false.
689 * @param pThis The TAR reader instance.
690 */
691static bool rtZipTarReaderExpectingMoreHeaders(PRTZIPTARREADER pThis)
692{
693 return pThis->enmState != RTZIPTARREADERSTATE_FIRST;
694}
695
696
697/**
698 * Checks if we're at the end of the TAR file.
699 *
700 * @returns true / false.
701 * @param pThis The TAR reader instance.
702 */
703static bool rtZipTarReaderIsAtEnd(PRTZIPTARREADER pThis)
704{
705 /*
706 * In theory there shall always be two zero headers at the end of the
707 * archive, but life isn't that simple. We've been creating archives
708 * without any zero headers at the end ourselves for a long long time
709 * (old tar.cpp).
710 *
711 * So, we're fine if the state is 'FIRST' or 'ZERO' here, but we'll barf
712 * if we're in the middle of a multi-header stream (long GNU names, sparse
713 * files, PAX, etc).
714 */
715 return pThis->enmState == RTZIPTARREADERSTATE_FIRST
716 || pThis->enmState == RTZIPTARREADERSTATE_ZERO;
717}
718
719
720/**
721 * Checks if the current TAR object is a hard link or not.
722 *
723 * @returns true if it is, false if not.
724 * @param pThis The TAR reader instance.
725 */
726static bool rtZipTarReaderIsHardlink(PRTZIPTARREADER pThis)
727{
728 return pThis->Hdr.Common.typeflag == RTZIPTAR_TF_LINK;
729}
730
731
732/**
733 * Checks if the TAR header includes a POSIX or GNU user name field.
734 *
735 * @returns true / false.
736 * @param pThis The TAR reader instance.
737 */
738DECLINLINE(bool) rtZipTarReaderHasUserName(PRTZIPTARREADER pThis)
739{
740 return pThis->Hdr.Common.uname[0] != '\0'
741 && ( pThis->enmType == RTZIPTARTYPE_POSIX
742 || pThis->enmType == RTZIPTARTYPE_GNU);
743}
744
745
746/**
747 * Checks if the TAR header includes a POSIX or GNU group name field.
748 *
749 * @returns true / false.
750 * @param pThis The TAR reader instance.
751 */
752DECLINLINE(bool) rtZipTarReaderHasGroupName(PRTZIPTARREADER pThis)
753{
754 return pThis->Hdr.Common.gname[0] != '\0'
755 && ( pThis->enmType == RTZIPTARTYPE_POSIX
756 || pThis->enmType == RTZIPTARTYPE_GNU);
757}
758
759
760/*
761 *
762 * T h e V F S F i l e s y s t e m S t r e a m B i t s.
763 * T h e V F S F i l e s y s t e m S t r e a m B i t s.
764 * T h e V F S F i l e s y s t e m S t r e a m B i t s.
765 *
766 */
767
768/**
769 * @interface_method_impl{RTVFSOBJOPS,pfnClose}
770 */
771static DECLCALLBACK(int) rtZipTarFssBaseObj_Close(void *pvThis)
772{
773 PRTZIPTARBASEOBJ pThis = (PRTZIPTARBASEOBJ)pvThis;
774
775 /* Currently there is nothing we really have to do here. */
776 pThis->offHdr = -1;
777
778 return VINF_SUCCESS;
779}
780
781
782/**
783 * @interface_method_impl{RTVFSOBJOPS,pfnQueryInfo}
784 */
785static DECLCALLBACK(int) rtZipTarFssBaseObj_QueryInfo(void *pvThis, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
786{
787 PRTZIPTARBASEOBJ pThis = (PRTZIPTARBASEOBJ)pvThis;
788
789 /*
790 * Copy the desired data.
791 */
792 switch (enmAddAttr)
793 {
794 case RTFSOBJATTRADD_NOTHING:
795 case RTFSOBJATTRADD_UNIX:
796 *pObjInfo = pThis->ObjInfo;
797 break;
798
799 case RTFSOBJATTRADD_UNIX_OWNER:
800 *pObjInfo = pThis->ObjInfo;
801 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_UNIX_OWNER;
802 pObjInfo->Attr.u.UnixOwner.uid = pThis->ObjInfo.Attr.u.Unix.uid;
803 pObjInfo->Attr.u.UnixOwner.szName[0] = '\0';
804 if (rtZipTarReaderHasUserName(pThis->pTarReader))
805 RTStrCopy(pObjInfo->Attr.u.UnixOwner.szName, sizeof(pObjInfo->Attr.u.UnixOwner.szName),
806 pThis->pTarReader->Hdr.Common.uname);
807 break;
808
809 case RTFSOBJATTRADD_UNIX_GROUP:
810 *pObjInfo = pThis->ObjInfo;
811 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_UNIX_GROUP;
812 pObjInfo->Attr.u.UnixGroup.gid = pThis->ObjInfo.Attr.u.Unix.gid;
813 pObjInfo->Attr.u.UnixGroup.szName[0] = '\0';
814 if (rtZipTarReaderHasGroupName(pThis->pTarReader))
815 RTStrCopy(pObjInfo->Attr.u.UnixGroup.szName, sizeof(pObjInfo->Attr.u.UnixGroup.szName),
816 pThis->pTarReader->Hdr.Common.gname);
817 break;
818
819 case RTFSOBJATTRADD_EASIZE:
820 *pObjInfo = pThis->ObjInfo;
821 pObjInfo->Attr.enmAdditional = RTFSOBJATTRADD_EASIZE;
822 RT_ZERO(pObjInfo->Attr.u);
823 break;
824
825 default:
826 return VERR_NOT_SUPPORTED;
827 }
828
829 return VINF_SUCCESS;
830}
831
832
833/**
834 * Tar filesystem base object operations.
835 */
836static const RTVFSOBJOPS g_rtZipTarFssBaseObjOps =
837{
838 RTVFSOBJOPS_VERSION,
839 RTVFSOBJTYPE_BASE,
840 "TarFsStream::Obj",
841 rtZipTarFssBaseObj_Close,
842 rtZipTarFssBaseObj_QueryInfo,
843 NULL,
844 RTVFSOBJOPS_VERSION
845};
846
847
848/**
849 * @interface_method_impl{RTVFSOBJOPS,pfnClose}
850 */
851static DECLCALLBACK(int) rtZipTarFssIos_Close(void *pvThis)
852{
853 PRTZIPTARIOSTREAM pThis = (PRTZIPTARIOSTREAM)pvThis;
854
855 RTVfsIoStrmRelease(pThis->hVfsIos);
856 pThis->hVfsIos = NIL_RTVFSIOSTREAM;
857
858 return rtZipTarFssBaseObj_Close(&pThis->BaseObj);
859}
860
861
862/**
863 * @interface_method_impl{RTVFSOBJOPS,pfnQueryInfo}
864 */
865static DECLCALLBACK(int) rtZipTarFssIos_QueryInfo(void *pvThis, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
866{
867 PRTZIPTARIOSTREAM pThis = (PRTZIPTARIOSTREAM)pvThis;
868 return rtZipTarFssBaseObj_QueryInfo(&pThis->BaseObj, pObjInfo, enmAddAttr);
869}
870
871
872/**
873 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnRead}
874 */
875static DECLCALLBACK(int) rtZipTarFssIos_Read(void *pvThis, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbRead)
876{
877 PRTZIPTARIOSTREAM pThis = (PRTZIPTARIOSTREAM)pvThis;
878 Assert(pSgBuf->cSegs == 1);
879
880 /*
881 * Make offset into a real offset so it's possible to do random access
882 * on TAR files that are seekable. Fend of reads beyond the end of the
883 * stream.
884 */
885 if (off < 0)
886 off = pThis->offFile;
887 if (off >= pThis->cbFile)
888 return pcbRead ? VINF_EOF : VERR_EOF;
889
890
891 Assert(pThis->cbFile >= pThis->offFile);
892 uint64_t cbLeft = (uint64_t)(pThis->cbFile - off);
893 size_t cbToRead = pSgBuf->paSegs[0].cbSeg;
894 if (cbToRead > cbLeft)
895 {
896 if (!pcbRead)
897 return VERR_EOF;
898 cbToRead = (size_t)cbLeft;
899 }
900
901 /*
902 * Do the reading.
903 */
904 size_t cbReadStack = 0;
905 if (!pcbRead)
906 pcbRead = &cbReadStack;
907 int rc = RTVfsIoStrmReadAt(pThis->hVfsIos, pThis->offStart + off, pSgBuf->paSegs[0].pvSeg, cbToRead, fBlocking, pcbRead);
908 pThis->offFile = off + *pcbRead;
909 if (pThis->offFile >= pThis->cbFile)
910 {
911 Assert(pThis->offFile == pThis->cbFile);
912 pThis->fEndOfStream = true;
913 RTVfsIoStrmSkip(pThis->hVfsIos, pThis->cbPadding);
914 }
915
916 return rc;
917}
918
919
920/**
921 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnWrite}
922 */
923static DECLCALLBACK(int) rtZipTarFssIos_Write(void *pvThis, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbWritten)
924{
925 /* Cannot write to a read-only I/O stream. */
926 NOREF(pvThis); NOREF(off); NOREF(pSgBuf); NOREF(fBlocking); NOREF(pcbWritten);
927 return VERR_ACCESS_DENIED;
928}
929
930
931/**
932 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnFlush}
933 */
934static DECLCALLBACK(int) rtZipTarFssIos_Flush(void *pvThis)
935{
936 /* It's a read only stream, nothing dirty to flush. */
937 NOREF(pvThis);
938 return VINF_SUCCESS;
939}
940
941
942/**
943 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnPollOne}
944 */
945static DECLCALLBACK(int) rtZipTarFssIos_PollOne(void *pvThis, uint32_t fEvents, RTMSINTERVAL cMillies, bool fIntr,
946 uint32_t *pfRetEvents)
947{
948 PRTZIPTARIOSTREAM pThis = (PRTZIPTARIOSTREAM)pvThis;
949
950 /* When we've reached the end, read will be set to indicate it. */
951 if ( (fEvents & RTPOLL_EVT_READ)
952 && pThis->fEndOfStream)
953 {
954 int rc = RTVfsIoStrmPoll(pThis->hVfsIos, fEvents, 0, fIntr, pfRetEvents);
955 if (RT_SUCCESS(rc))
956 *pfRetEvents |= RTPOLL_EVT_READ;
957 else
958 *pfRetEvents = RTPOLL_EVT_READ;
959 return VINF_SUCCESS;
960 }
961
962 return RTVfsIoStrmPoll(pThis->hVfsIos, fEvents, cMillies, fIntr, pfRetEvents);
963}
964
965
966/**
967 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnTell}
968 */
969static DECLCALLBACK(int) rtZipTarFssIos_Tell(void *pvThis, PRTFOFF poffActual)
970{
971 PRTZIPTARIOSTREAM pThis = (PRTZIPTARIOSTREAM)pvThis;
972 *poffActual = pThis->offFile;
973 return VINF_SUCCESS;
974}
975
976
977/**
978 * Tar I/O stream operations.
979 */
980static const RTVFSIOSTREAMOPS g_rtZipTarFssIosOps =
981{
982 { /* Obj */
983 RTVFSOBJOPS_VERSION,
984 RTVFSOBJTYPE_IO_STREAM,
985 "TarFsStream::IoStream",
986 rtZipTarFssIos_Close,
987 rtZipTarFssIos_QueryInfo,
988 NULL,
989 RTVFSOBJOPS_VERSION
990 },
991 RTVFSIOSTREAMOPS_VERSION,
992 RTVFSIOSTREAMOPS_FEAT_NO_SG,
993 rtZipTarFssIos_Read,
994 rtZipTarFssIos_Write,
995 rtZipTarFssIos_Flush,
996 rtZipTarFssIos_PollOne,
997 rtZipTarFssIos_Tell,
998 NULL /*Skip*/,
999 NULL /*ZeroFill*/,
1000 RTVFSIOSTREAMOPS_VERSION
1001};
1002
1003
1004/**
1005 * @interface_method_impl{RTVFSOBJOPS,pfnClose}
1006 */
1007static DECLCALLBACK(int) rtZipTarFssSym_Close(void *pvThis)
1008{
1009 PRTZIPTARBASEOBJ pThis = (PRTZIPTARBASEOBJ)pvThis;
1010 return rtZipTarFssBaseObj_Close(pThis);
1011}
1012
1013
1014/**
1015 * @interface_method_impl{RTVFSOBJOPS,pfnQueryInfo}
1016 */
1017static DECLCALLBACK(int) rtZipTarFssSym_QueryInfo(void *pvThis, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
1018{
1019 PRTZIPTARBASEOBJ pThis = (PRTZIPTARBASEOBJ)pvThis;
1020 return rtZipTarFssBaseObj_QueryInfo(pThis, pObjInfo, enmAddAttr);
1021}
1022
1023/**
1024 * @interface_method_impl{RTVFSOBJSETOPS,pfnMode}
1025 */
1026static DECLCALLBACK(int) rtZipTarFssSym_SetMode(void *pvThis, RTFMODE fMode, RTFMODE fMask)
1027{
1028 NOREF(pvThis); NOREF(fMode); NOREF(fMask);
1029 return VERR_ACCESS_DENIED;
1030}
1031
1032
1033/**
1034 * @interface_method_impl{RTVFSOBJSETOPS,pfnSetTimes}
1035 */
1036static DECLCALLBACK(int) rtZipTarFssSym_SetTimes(void *pvThis, PCRTTIMESPEC pAccessTime, PCRTTIMESPEC pModificationTime,
1037 PCRTTIMESPEC pChangeTime, PCRTTIMESPEC pBirthTime)
1038{
1039 NOREF(pvThis); NOREF(pAccessTime); NOREF(pModificationTime); NOREF(pChangeTime); NOREF(pBirthTime);
1040 return VERR_ACCESS_DENIED;
1041}
1042
1043
1044/**
1045 * @interface_method_impl{RTVFSOBJSETOPS,pfnSetOwner}
1046 */
1047static DECLCALLBACK(int) rtZipTarFssSym_SetOwner(void *pvThis, RTUID uid, RTGID gid)
1048{
1049 NOREF(pvThis); NOREF(uid); NOREF(gid);
1050 return VERR_ACCESS_DENIED;
1051}
1052
1053
1054/**
1055 * @interface_method_impl{RTVFSSYMLINKOPS,pfnRead}
1056 */
1057static DECLCALLBACK(int) rtZipTarFssSym_Read(void *pvThis, char *pszTarget, size_t cbTarget)
1058{
1059 PRTZIPTARBASEOBJ pThis = (PRTZIPTARBASEOBJ)pvThis;
1060 return RTStrCopy(pszTarget, cbTarget, pThis->pTarReader->szTarget);
1061}
1062
1063
1064/**
1065 * Tar symbolic (and hardlink) operations.
1066 */
1067static const RTVFSSYMLINKOPS g_rtZipTarFssSymOps =
1068{
1069 { /* Obj */
1070 RTVFSOBJOPS_VERSION,
1071 RTVFSOBJTYPE_SYMLINK,
1072 "TarFsStream::Symlink",
1073 rtZipTarFssSym_Close,
1074 rtZipTarFssSym_QueryInfo,
1075 NULL,
1076 RTVFSOBJOPS_VERSION
1077 },
1078 RTVFSSYMLINKOPS_VERSION,
1079 0,
1080 { /* ObjSet */
1081 RTVFSOBJSETOPS_VERSION,
1082 RT_UOFFSETOF(RTVFSSYMLINKOPS, ObjSet) - RT_UOFFSETOF(RTVFSSYMLINKOPS, Obj),
1083 rtZipTarFssSym_SetMode,
1084 rtZipTarFssSym_SetTimes,
1085 rtZipTarFssSym_SetOwner,
1086 RTVFSOBJSETOPS_VERSION
1087 },
1088 rtZipTarFssSym_Read,
1089 RTVFSSYMLINKOPS_VERSION
1090};
1091
1092
1093/**
1094 * @interface_method_impl{RTVFSOBJOPS,pfnClose}
1095 */
1096static DECLCALLBACK(int) rtZipTarFss_Close(void *pvThis)
1097{
1098 PRTZIPTARFSSTREAM pThis = (PRTZIPTARFSSTREAM)pvThis;
1099
1100 RTVfsObjRelease(pThis->hVfsCurObj);
1101 pThis->hVfsCurObj = NIL_RTVFSOBJ;
1102 pThis->pCurIosData = NULL;
1103
1104 RTVfsIoStrmRelease(pThis->hVfsIos);
1105 pThis->hVfsIos = NIL_RTVFSIOSTREAM;
1106
1107 return VINF_SUCCESS;
1108}
1109
1110
1111/**
1112 * @interface_method_impl{RTVFSOBJOPS,pfnQueryInfo}
1113 */
1114static DECLCALLBACK(int) rtZipTarFss_QueryInfo(void *pvThis, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
1115{
1116 PRTZIPTARFSSTREAM pThis = (PRTZIPTARFSSTREAM)pvThis;
1117 /* Take the lazy approach here, with the sideffect of providing some info
1118 that is actually kind of useful. */
1119 return RTVfsIoStrmQueryInfo(pThis->hVfsIos, pObjInfo, enmAddAttr);
1120}
1121
1122
1123/**
1124 * @interface_method_impl{RTVFSFSSTREAMOPS,pfnNext}
1125 */
1126DECL_HIDDEN_CALLBACK(int) rtZipTarFss_Next(void *pvThis, char **ppszName, RTVFSOBJTYPE *penmType, PRTVFSOBJ phVfsObj)
1127{
1128 PRTZIPTARFSSTREAM pThis = (PRTZIPTARFSSTREAM)pvThis;
1129
1130 /*
1131 * Dispense with the current object.
1132 */
1133 if (pThis->hVfsCurObj != NIL_RTVFSOBJ)
1134 {
1135 if (pThis->pCurIosData)
1136 {
1137 pThis->pCurIosData->fEndOfStream = true;
1138 pThis->pCurIosData->offFile = pThis->pCurIosData->cbFile;
1139 pThis->pCurIosData = NULL;
1140 }
1141
1142 RTVfsObjRelease(pThis->hVfsCurObj);
1143 pThis->hVfsCurObj = NIL_RTVFSOBJ;
1144 }
1145
1146 /*
1147 * Check if we've already reached the end in some way.
1148 */
1149 if (pThis->fEndOfStream)
1150 return VERR_EOF;
1151 if (pThis->rcFatal != VINF_SUCCESS)
1152 return pThis->rcFatal;
1153
1154 /*
1155 * Make sure the input stream is in the right place.
1156 */
1157 RTFOFF offHdr = RTVfsIoStrmTell(pThis->hVfsIos);
1158 while ( offHdr >= 0
1159 && offHdr < pThis->offNextHdr)
1160 {
1161 int rc = RTVfsIoStrmSkip(pThis->hVfsIos, pThis->offNextHdr - offHdr);
1162 if (RT_FAILURE(rc))
1163 {
1164 /** @todo Ignore if we're at the end of the stream? */
1165 return pThis->rcFatal = rc;
1166 }
1167
1168 offHdr = RTVfsIoStrmTell(pThis->hVfsIos);
1169 }
1170
1171 if (offHdr < 0)
1172 return pThis->rcFatal = (int)offHdr;
1173 if (offHdr > pThis->offNextHdr)
1174 return pThis->rcFatal = VERR_INTERNAL_ERROR_3;
1175 Assert(pThis->offNextHdr == offHdr);
1176 pThis->offCurHdr = offHdr;
1177
1178 /*
1179 * Consume TAR headers.
1180 */
1181 size_t cbHdrs = 0;
1182 int rc;
1183 do
1184 {
1185 /*
1186 * Read the next header.
1187 */
1188 RTZIPTARHDR Hdr;
1189 size_t cbRead;
1190 rc = RTVfsIoStrmRead(pThis->hVfsIos, &Hdr, sizeof(Hdr), true /*fBlocking*/, &cbRead);
1191 if (RT_FAILURE(rc))
1192 return pThis->rcFatal = rc;
1193 if (rc == VINF_EOF && cbRead == 0)
1194 {
1195 pThis->fEndOfStream = true;
1196 return rtZipTarReaderIsAtEnd(&pThis->TarReader) ? VERR_EOF : VERR_TAR_UNEXPECTED_EOS;
1197 }
1198 if (cbRead != sizeof(Hdr))
1199 return pThis->rcFatal = VERR_TAR_UNEXPECTED_EOS;
1200
1201 cbHdrs += sizeof(Hdr);
1202
1203 /*
1204 * Parse the it.
1205 */
1206 rc = rtZipTarReaderParseHeader(&pThis->TarReader, &Hdr);
1207 if (RT_FAILURE(rc))
1208 return pThis->rcFatal = rc;
1209 } while (rtZipTarReaderExpectingMoreHeaders(&pThis->TarReader));
1210
1211 pThis->offNextHdr = offHdr + cbHdrs;
1212
1213 /*
1214 * Fill an object info structure from the current TAR state.
1215 */
1216 RTFSOBJINFO Info;
1217 rc = rtZipTarReaderGetFsObjInfo(&pThis->TarReader, &Info);
1218 if (RT_FAILURE(rc))
1219 return pThis->rcFatal = rc;
1220
1221 /*
1222 * Create an object of the appropriate type.
1223 */
1224 RTVFSOBJTYPE enmType;
1225 RTVFSOBJ hVfsObj;
1226 RTFMODE fType = Info.Attr.fMode & RTFS_TYPE_MASK;
1227 if (rtZipTarReaderIsHardlink(&pThis->TarReader))
1228 fType = RTFS_TYPE_SYMLINK;
1229 switch (fType)
1230 {
1231 /*
1232 * Files are represented by a VFS I/O stream.
1233 */
1234 case RTFS_TYPE_FILE:
1235 {
1236 RTVFSIOSTREAM hVfsIos;
1237 PRTZIPTARIOSTREAM pIosData;
1238 rc = RTVfsNewIoStream(&g_rtZipTarFssIosOps,
1239 sizeof(*pIosData),
1240 RTFILE_O_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN,
1241 NIL_RTVFS,
1242 NIL_RTVFSLOCK,
1243 &hVfsIos,
1244 (void **)&pIosData);
1245 if (RT_FAILURE(rc))
1246 return pThis->rcFatal = rc;
1247
1248 pIosData->BaseObj.offHdr = offHdr;
1249 pIosData->BaseObj.offNextHdr = pThis->offNextHdr;
1250 pIosData->BaseObj.pTarReader = &pThis->TarReader;
1251 pIosData->BaseObj.ObjInfo = Info;
1252 pIosData->cbFile = Info.cbObject;
1253 pIosData->offFile = 0;
1254 pIosData->offStart = RTVfsIoStrmTell(pThis->hVfsIos);
1255 pIosData->cbPadding = (uint32_t)(Info.cbAllocated - Info.cbObject);
1256 pIosData->fEndOfStream = false;
1257 pIosData->hVfsIos = pThis->hVfsIos;
1258 RTVfsIoStrmRetain(pThis->hVfsIos);
1259
1260 pThis->pCurIosData = pIosData;
1261 pThis->offNextHdr += pIosData->cbFile + pIosData->cbPadding;
1262
1263 enmType = RTVFSOBJTYPE_IO_STREAM;
1264 hVfsObj = RTVfsObjFromIoStream(hVfsIos);
1265 RTVfsIoStrmRelease(hVfsIos);
1266 break;
1267 }
1268
1269 /*
1270 * We represent hard links using a symbolic link object. This fits
1271 * best with the way TAR stores it and there is currently no better
1272 * fitting VFS type alternative.
1273 */
1274 case RTFS_TYPE_SYMLINK:
1275 {
1276 RTVFSSYMLINK hVfsSym;
1277 PRTZIPTARBASEOBJ pBaseObjData;
1278 rc = RTVfsNewSymlink(&g_rtZipTarFssSymOps,
1279 sizeof(*pBaseObjData),
1280 NIL_RTVFS,
1281 NIL_RTVFSLOCK,
1282 &hVfsSym,
1283 (void **)&pBaseObjData);
1284 if (RT_FAILURE(rc))
1285 return pThis->rcFatal = rc;
1286
1287 pBaseObjData->offHdr = offHdr;
1288 pBaseObjData->offNextHdr = pThis->offNextHdr;
1289 pBaseObjData->pTarReader = &pThis->TarReader;
1290 pBaseObjData->ObjInfo = Info;
1291
1292 enmType = RTVFSOBJTYPE_SYMLINK;
1293 hVfsObj = RTVfsObjFromSymlink(hVfsSym);
1294 RTVfsSymlinkRelease(hVfsSym);
1295 break;
1296 }
1297
1298 /*
1299 * All other objects are repesented using a VFS base object since they
1300 * carry no data streams (unless some TAR extension implements extended
1301 * attributes / alternative streams).
1302 */
1303 case RTFS_TYPE_DEV_BLOCK:
1304 case RTFS_TYPE_DEV_CHAR:
1305 case RTFS_TYPE_DIRECTORY:
1306 case RTFS_TYPE_FIFO:
1307 {
1308 PRTZIPTARBASEOBJ pBaseObjData;
1309 rc = RTVfsNewBaseObj(&g_rtZipTarFssBaseObjOps,
1310 sizeof(*pBaseObjData),
1311 NIL_RTVFS,
1312 NIL_RTVFSLOCK,
1313 &hVfsObj,
1314 (void **)&pBaseObjData);
1315 if (RT_FAILURE(rc))
1316 return pThis->rcFatal = rc;
1317
1318 pBaseObjData->offHdr = offHdr;
1319 pBaseObjData->offNextHdr = pThis->offNextHdr;
1320 pBaseObjData->pTarReader = &pThis->TarReader;
1321 pBaseObjData->ObjInfo = Info;
1322
1323 enmType = RTVFSOBJTYPE_BASE;
1324 break;
1325 }
1326
1327 default:
1328 AssertFailed();
1329 return pThis->rcFatal = VERR_INTERNAL_ERROR_5;
1330 }
1331 pThis->hVfsCurObj = hVfsObj;
1332
1333 /*
1334 * Set the return data and we're done.
1335 */
1336 if (ppszName)
1337 {
1338 rc = RTStrDupEx(ppszName, pThis->TarReader.szName);
1339 if (RT_FAILURE(rc))
1340 return rc;
1341 }
1342
1343 if (phVfsObj)
1344 {
1345 RTVfsObjRetain(hVfsObj);
1346 *phVfsObj = hVfsObj;
1347 }
1348
1349 if (penmType)
1350 *penmType = enmType;
1351
1352 return VINF_SUCCESS;
1353}
1354
1355
1356
1357/**
1358 * Tar filesystem stream operations.
1359 */
1360static const RTVFSFSSTREAMOPS rtZipTarFssOps =
1361{
1362 { /* Obj */
1363 RTVFSOBJOPS_VERSION,
1364 RTVFSOBJTYPE_FS_STREAM,
1365 "TarFsStream",
1366 rtZipTarFss_Close,
1367 rtZipTarFss_QueryInfo,
1368 NULL,
1369 RTVFSOBJOPS_VERSION
1370 },
1371 RTVFSFSSTREAMOPS_VERSION,
1372 0,
1373 rtZipTarFss_Next,
1374 NULL,
1375 NULL,
1376 NULL,
1377 RTVFSFSSTREAMOPS_VERSION
1378};
1379
1380
1381/**
1382 * Internal function use both by RTZipTarFsStreamFromIoStream() and by
1383 * RTZipTarFsStreamForFile() in updating mode.
1384 */
1385DECLHIDDEN(void) rtZipTarReaderInit(PRTZIPTARFSSTREAM pThis, RTVFSIOSTREAM hVfsIos, uint64_t offStart)
1386{
1387 pThis->hVfsIos = hVfsIos;
1388 pThis->hVfsCurObj = NIL_RTVFSOBJ;
1389 pThis->pCurIosData = NULL;
1390 pThis->offStart = offStart;
1391 pThis->offNextHdr = offStart;
1392 pThis->fEndOfStream = false;
1393 pThis->rcFatal = VINF_SUCCESS;
1394 pThis->TarReader.enmPrevType= RTZIPTARTYPE_INVALID;
1395 pThis->TarReader.enmType = RTZIPTARTYPE_INVALID;
1396 pThis->TarReader.enmState = RTZIPTARREADERSTATE_FIRST;
1397
1398 /* Don't check if it's a TAR stream here, do that in the
1399 rtZipTarFss_Next. */
1400}
1401
1402
1403RTDECL(int) RTZipTarFsStreamFromIoStream(RTVFSIOSTREAM hVfsIosIn, uint32_t fFlags, PRTVFSFSSTREAM phVfsFss)
1404{
1405 /*
1406 * Input validation.
1407 */
1408 AssertPtrReturn(phVfsFss, VERR_INVALID_HANDLE);
1409 *phVfsFss = NIL_RTVFSFSSTREAM;
1410 AssertPtrReturn(hVfsIosIn, VERR_INVALID_HANDLE);
1411 AssertReturn(!fFlags, VERR_INVALID_PARAMETER);
1412
1413 RTFOFF const offStart = RTVfsIoStrmTell(hVfsIosIn);
1414 AssertReturn(offStart >= 0, (int)offStart);
1415
1416 uint32_t cRefs = RTVfsIoStrmRetain(hVfsIosIn);
1417 AssertReturn(cRefs != UINT32_MAX, VERR_INVALID_HANDLE);
1418
1419 /*
1420 * Retain the input stream and create a new filesystem stream handle.
1421 */
1422 PRTZIPTARFSSTREAM pThis;
1423 RTVFSFSSTREAM hVfsFss;
1424 int rc = RTVfsNewFsStream(&rtZipTarFssOps, sizeof(*pThis), NIL_RTVFS, NIL_RTVFSLOCK, RTFILE_O_READ,
1425 &hVfsFss, (void **)&pThis);
1426 if (RT_SUCCESS(rc))
1427 {
1428 rtZipTarReaderInit(pThis, hVfsIosIn, fFlags);
1429 *phVfsFss = hVfsFss;
1430 return VINF_SUCCESS;
1431 }
1432
1433 RTVfsIoStrmRelease(hVfsIosIn);
1434 return rc;
1435}
1436
1437
1438/**
1439 * Used by RTZipTarFsStreamTruncate to resolve @a hVfsObj.
1440 */
1441DECLHIDDEN(PRTZIPTARBASEOBJ) rtZipTarFsStreamBaseObjToPrivate(PRTZIPTARFSSTREAM pThis, RTVFSOBJ hVfsObj)
1442{
1443 PRTZIPTARBASEOBJ pThisObj;
1444 RTVFSOBJTYPE enmType = RTVfsObjGetType(hVfsObj);
1445 switch (enmType)
1446 {
1447 case RTVFSOBJTYPE_IO_STREAM:
1448 {
1449 RTVFSIOSTREAM hVfsIos = RTVfsObjToIoStream(hVfsObj);
1450 AssertReturn(hVfsIos != NIL_RTVFSIOSTREAM, NULL);
1451 PRTZIPTARIOSTREAM pThisStrm = (PRTZIPTARIOSTREAM)RTVfsIoStreamToPrivate(hVfsIos, &g_rtZipTarFssIosOps);
1452 RTVfsIoStrmRelease(hVfsIos);
1453 pThisObj = &pThisStrm->BaseObj;
1454 break;
1455 }
1456
1457 case RTVFSOBJTYPE_SYMLINK:
1458 {
1459 RTVFSSYMLINK hVfsSymlink = RTVfsObjToSymlink(hVfsObj);
1460 AssertReturn(hVfsSymlink != NIL_RTVFSSYMLINK, NULL);
1461 pThisObj = (PRTZIPTARBASEOBJ)RTVfsSymlinkToPrivate(hVfsSymlink, &g_rtZipTarFssSymOps);
1462 RTVfsSymlinkRelease(hVfsSymlink);
1463 break;
1464 }
1465
1466 case RTVFSOBJTYPE_BASE:
1467 pThisObj = (PRTZIPTARBASEOBJ)RTVfsObjToPrivate(hVfsObj, &g_rtZipTarFssBaseObjOps);
1468 break;
1469
1470 default:
1471 /** @todo implement. */
1472 AssertFailedReturn(NULL);
1473 }
1474
1475 AssertReturn(pThisObj->pTarReader == &pThis->TarReader, NULL);
1476 return pThisObj;
1477}
1478
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette