VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/zip/tarcmd.cpp@ 98392

Last change on this file since 98392 was 98103, checked in by vboxsync, 2 years ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 74.8 KB
Line 
1/* $Id: tarcmd.cpp 98103 2023-01-17 14:15:46Z vboxsync $ */
2/** @file
3 * IPRT - A mini TAR Command.
4 */
5
6/*
7 * Copyright (C) 2010-2023 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 <iprt/zip.h>
42
43#include <iprt/asm.h>
44#include <iprt/buildconfig.h>
45#include <iprt/ctype.h>
46#include <iprt/dir.h>
47#include <iprt/err.h>
48#include <iprt/file.h>
49#include <iprt/getopt.h>
50#include <iprt/initterm.h>
51#include <iprt/mem.h>
52#include <iprt/message.h>
53#include <iprt/param.h>
54#include <iprt/path.h>
55#include <iprt/stream.h>
56#include <iprt/string.h>
57#include <iprt/symlink.h>
58#include <iprt/vfs.h>
59
60
61/*********************************************************************************************************************************
62* Defined Constants And Macros *
63*********************************************************************************************************************************/
64#define RTZIPTARCMD_OPT_DELETE 1000
65#define RTZIPTARCMD_OPT_OWNER 1001
66#define RTZIPTARCMD_OPT_GROUP 1002
67#define RTZIPTARCMD_OPT_UTC 1003
68#define RTZIPTARCMD_OPT_PREFIX 1004
69#define RTZIPTARCMD_OPT_FILE_MODE_AND_MASK 1005
70#define RTZIPTARCMD_OPT_FILE_MODE_OR_MASK 1006
71#define RTZIPTARCMD_OPT_DIR_MODE_AND_MASK 1007
72#define RTZIPTARCMD_OPT_DIR_MODE_OR_MASK 1008
73#define RTZIPTARCMD_OPT_FORMAT 1009
74#define RTZIPTARCMD_OPT_READ_AHEAD 1010
75#define RTZIPTARCMD_OPT_USE_PUSH_FILE 1011
76#define RTZIPTARCMD_OPT_NO_RECURSION 1012
77
78/** File format. */
79typedef enum RTZIPTARCMDFORMAT
80{
81 RTZIPTARCMDFORMAT_INVALID = 0,
82 /** Autodetect if possible, defaulting to TAR. */
83 RTZIPTARCMDFORMAT_AUTO_DEFAULT,
84 /** TAR. */
85 RTZIPTARCMDFORMAT_TAR,
86 /** XAR. */
87 RTZIPTARCMDFORMAT_XAR,
88 /** CPIO. */
89 RTZIPTARCMDFORMAT_CPIO
90} RTZIPTARCMDFORMAT;
91
92
93/*********************************************************************************************************************************
94* Structures and Typedefs *
95*********************************************************************************************************************************/
96/**
97 * IPRT TAR option structure.
98 */
99typedef struct RTZIPTARCMDOPS
100{
101 /** The file format. */
102 RTZIPTARCMDFORMAT enmFormat;
103
104 /** The operation (Acdrtux or RTZIPTARCMD_OPT_DELETE). */
105 int iOperation;
106 /** The long operation option name. */
107 const char *pszOperation;
108
109 /** The directory to change into when packing and unpacking. */
110 const char *pszDirectory;
111 /** The tar file name. */
112 const char *pszFile;
113 /** Whether we're verbose or quiet. */
114 bool fVerbose;
115 /** Whether to preserve the original file owner when restoring. */
116 bool fPreserveOwner;
117 /** Whether to preserve the original file group when restoring. */
118 bool fPreserveGroup;
119 /** Whether to skip restoring the modification time (only time stored by the
120 * traditional TAR format). */
121 bool fNoModTime;
122 /** Whether to add a read ahead thread. */
123 bool fReadAhead;
124 /** Use RTVfsFsStrmPushFile instead of RTVfsFsStrmAdd for files. */
125 bool fUsePushFile;
126 /** Whether to handle directories recursively or not. Defaults to \c true. */
127 bool fRecursive;
128 /** The compressor/decompressor method to employ (0, z or j). */
129 char chZipper;
130
131 /** The owner to set. NULL if not applicable.
132 * Always resolved into uidOwner for extraction. */
133 const char *pszOwner;
134 /** The owner ID to set. NIL_RTUID if not applicable. */
135 RTUID uidOwner;
136 /** The group to set. NULL if not applicable.
137 * Always resolved into gidGroup for extraction. */
138 const char *pszGroup;
139 /** The group ID to set. NIL_RTGUID if not applicable. */
140 RTGID gidGroup;
141 /** Display the modification times in UTC instead of local time. */
142 bool fDisplayUtc;
143 /** File mode AND mask. */
144 RTFMODE fFileModeAndMask;
145 /** File mode OR mask. */
146 RTFMODE fFileModeOrMask;
147 /** Directory mode AND mask. */
148 RTFMODE fDirModeAndMask;
149 /** Directory mode OR mask. */
150 RTFMODE fDirModeOrMask;
151
152 /** What to prefix all names with when creating, adding, whatever. */
153 const char *pszPrefix;
154
155 /** The number of files(, directories or whatever) specified. */
156 uint32_t cFiles;
157 /** Array of files(, directories or whatever).
158 * Terminated by a NULL entry. */
159 const char * const *papszFiles;
160
161 /** The TAR format to create. */
162 RTZIPTARFORMAT enmTarFormat;
163 /** TAR creation flags. */
164 uint32_t fTarCreate;
165
166} RTZIPTARCMDOPS;
167/** Pointer to the IPRT tar options. */
168typedef RTZIPTARCMDOPS *PRTZIPTARCMDOPS;
169
170/** The size of the directory entry buffer we're using. */
171#define RTZIPTARCMD_DIRENTRY_BUF_SIZE (sizeof(RTDIRENTRYEX) + RTPATH_MAX)
172
173/**
174 * Callback used by rtZipTarDoWithMembers
175 *
176 * @returns rcExit or RTEXITCODE_FAILURE.
177 * @param pOpts The tar options.
178 * @param hVfsObj The tar object to display
179 * @param pszName The name.
180 * @param rcExit The current exit code.
181 */
182typedef RTEXITCODE (*PFNDOWITHMEMBER)(PRTZIPTARCMDOPS pOpts, RTVFSOBJ hVfsObj, const char *pszName, RTEXITCODE rcExit);
183
184
185/**
186 * Checks if @a pszName is a member of @a papszNames, optionally returning the
187 * index.
188 *
189 * @returns true if the name is in the list, otherwise false.
190 * @param pszName The name to find.
191 * @param papszNames The array of names.
192 * @param piName Where to optionally return the array index.
193 */
194static bool rtZipTarCmdIsNameInArray(const char *pszName, const char * const *papszNames, uint32_t *piName)
195{
196 for (uint32_t iName = 0; papszNames[iName]; iName++)
197 if (!strcmp(papszNames[iName], pszName))
198 {
199 if (piName)
200 *piName = iName;
201 return true;
202 }
203 return false;
204}
205
206
207/**
208 * Queries information about a VFS object.
209 *
210 * @returns VBox status code.
211 * @param pszSpec VFS object spec to use.
212 * @param paObjInfo Where to store the queried object information.
213 * Must at least provide 3 structs, namely for UNIX, UNIX_OWNER and UNIX_GROUP attributes.
214 * @param cObjInfo Number of objection information structs handed in.
215 */
216static int rtZipTarCmdQueryObjInfo(const char *pszSpec, PRTFSOBJINFO paObjInfo, unsigned cObjInfo)
217{
218 AssertPtrReturn(paObjInfo, VERR_INVALID_POINTER);
219 AssertReturn(cObjInfo >= 3, VERR_INVALID_PARAMETER);
220
221 RTERRINFOSTATIC ErrInfo;
222 uint32_t offError;
223 int rc = RTVfsChainQueryInfo(pszSpec, &paObjInfo[0], RTFSOBJATTRADD_UNIX, RTPATH_F_ON_LINK,
224 &offError, RTErrInfoInitStatic(&ErrInfo));
225 if (RT_SUCCESS(rc))
226 {
227 rc = RTVfsChainQueryInfo(pszSpec, &paObjInfo[1], RTFSOBJATTRADD_UNIX_OWNER, RTPATH_F_ON_LINK,
228 &offError, RTErrInfoInitStatic(&ErrInfo));
229 if (RT_SUCCESS(rc))
230 {
231 rc = RTVfsChainQueryInfo(pszSpec, &paObjInfo[2], RTFSOBJATTRADD_UNIX_GROUP, RTPATH_F_ON_LINK,
232 &offError, RTErrInfoInitStatic(&ErrInfo));
233 if (RT_FAILURE(rc))
234 RT_BZERO(&paObjInfo[2], sizeof(RTFSOBJINFO));
235 }
236 else
237 {
238 RT_BZERO(&paObjInfo[1], sizeof(RTFSOBJINFO));
239 RT_BZERO(&paObjInfo[2], sizeof(RTFSOBJINFO));
240 }
241
242 rc = VINF_SUCCESS; /* aObjInfo[1] + aObjInfo[2] are optional. */
243 }
244 else
245 RTVfsChainMsgError("RTVfsChainQueryInfo", pszSpec, rc, offError, &ErrInfo.Core);
246
247 return rc;
248}
249
250
251/**
252 * Archives a file.
253 *
254 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE + printed message.
255 * @param pOpts The options.
256 * @param hVfsFss The TAR filesystem stream handle.
257 * @param pszSrc The file path or VFS spec.
258 * @param paObjInfo[3] Array of three FS object info structures. The first
259 * one is always filled with RTFSOBJATTRADD_UNIX info.
260 * The next two may contain owner and group names if
261 * available. Buffers can be modified.
262 * @param pszDst The name to archive the file under.
263 * @param pErrInfo Error info buffer (saves stack space).
264 */
265static RTEXITCODE rtZipTarCmdArchiveFile(PRTZIPTARCMDOPS pOpts, RTVFSFSSTREAM hVfsFss, const char *pszSrc,
266 RTFSOBJINFO paObjInfo[3], const char *pszDst, PRTERRINFOSTATIC pErrInfo)
267{
268 if (pOpts->fVerbose)
269 RTPrintf("%s\n", pszDst);
270
271 /* Open the file. */
272 uint32_t offError;
273 RTVFSIOSTREAM hVfsIosSrc;
274 int rc = RTVfsChainOpenIoStream(pszSrc, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE,
275 &hVfsIosSrc, &offError, RTErrInfoInitStatic(pErrInfo));
276 if (RT_FAILURE(rc))
277 return RTVfsChainMsgErrorExitFailure("RTVfsChainOpenIoStream", pszSrc, rc, offError, &pErrInfo->Core);
278
279 /* I/O stream to base object. */
280 RTVFSOBJ hVfsObjSrc = RTVfsObjFromIoStream(hVfsIosSrc);
281 if (hVfsObjSrc != NIL_RTVFSOBJ)
282 {
283 /*
284 * Add it to the stream. Got to variants here so we can test the
285 * RTVfsFsStrmPushFile API too.
286 */
287 if (!pOpts->fUsePushFile)
288 rc = RTVfsFsStrmAdd(hVfsFss, pszDst, hVfsObjSrc, 0 /*fFlags*/);
289 else
290 {
291 uint32_t cObjInfo = 1 + (paObjInfo[1].Attr.enmAdditional == RTFSOBJATTRADD_UNIX_OWNER)
292 + (paObjInfo[2].Attr.enmAdditional == RTFSOBJATTRADD_UNIX_GROUP);
293 RTVFSIOSTREAM hVfsIosDst;
294 rc = RTVfsFsStrmPushFile(hVfsFss, pszDst, paObjInfo[0].cbObject, paObjInfo, cObjInfo, 0 /*fFlags*/, &hVfsIosDst);
295 if (RT_SUCCESS(rc))
296 {
297 rc = RTVfsUtilPumpIoStreams(hVfsIosSrc, hVfsIosDst, 0);
298 RTVfsIoStrmRelease(hVfsIosDst);
299 }
300 }
301 RTVfsIoStrmRelease(hVfsIosSrc);
302 RTVfsObjRelease(hVfsObjSrc);
303
304 if (RT_SUCCESS(rc))
305 {
306 if (rc != VINF_SUCCESS)
307 RTMsgWarning("%Rrc adding '%s'", rc, pszDst);
308 return RTEXITCODE_SUCCESS;
309 }
310 return RTMsgErrorExitFailure("%Rrc adding '%s'", rc, pszDst);
311 }
312 RTVfsIoStrmRelease(hVfsIosSrc);
313 return RTMsgErrorExitFailure("RTVfsObjFromIoStream failed unexpectedly!");
314}
315
316
317/**
318 * Sub-directory helper for creating archives.
319 *
320 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE + printed message.
321 * @param pOpts The options.
322 * @param hVfsFss The TAR filesystem stream handle.
323 * @param pszSrc The directory path or VFS spec. We append to the
324 * buffer as we decend.
325 * @param cchSrc The length of the input.
326 * @param paObjInfo[3] Array of three FS object info structures. The first
327 * one is always filled with RTFSOBJATTRADD_UNIX info.
328 * The next two may contain owner and group names if
329 * available. The three buffers can be reused.
330 * @param pszDst The name to archive it the under. We append to the
331 * buffer as we decend.
332 * @param cchDst The length of the input.
333 * @param pDirEntry Directory entry to use for the directory to handle.
334 * @param pErrInfo Error info buffer (saves stack space).
335 */
336static RTEXITCODE rtZipTarCmdArchiveDirSub(PRTZIPTARCMDOPS pOpts, RTVFSFSSTREAM hVfsFss,
337 char *pszSrc, size_t cchSrc, RTFSOBJINFO paObjInfo[3],
338 char pszDst[RTPATH_MAX], size_t cchDst, PRTDIRENTRYEX pDirEntry,
339 PRTERRINFOSTATIC pErrInfo)
340{
341 if (pOpts->fVerbose)
342 RTPrintf("%s\n", pszDst);
343
344 uint32_t offError;
345 RTVFSDIR hVfsIoDir;
346 int rc = RTVfsChainOpenDir(pszSrc, 0 /*fFlags*/,
347 &hVfsIoDir, &offError, RTErrInfoInitStatic(pErrInfo));
348 if (RT_FAILURE(rc))
349 return RTVfsChainMsgErrorExitFailure("RTVfsChainOpenDir", pszSrc, rc, offError, &pErrInfo->Core);
350
351 /* Make sure we've got some room in the path, to save us extra work further down. */
352 if (cchSrc + 3 >= RTPATH_MAX)
353 return RTMsgErrorExitFailure("Source path too long: '%s'\n", pszSrc);
354
355 /* Ensure we've got a trailing slash (there is space for it see above). */
356 if (!RTPATH_IS_SEP(pszSrc[cchSrc - 1]))
357 {
358 pszSrc[cchSrc++] = RTPATH_SLASH;
359 pszSrc[cchSrc] = '\0';
360 }
361
362 /* Ditto for destination. */
363 if (cchDst + 3 >= RTPATH_MAX)
364 return RTMsgErrorExitFailure("Destination path too long: '%s'\n", pszDst);
365
366 if (!RTPATH_IS_SEP(pszDst[cchDst - 1]))
367 {
368 pszDst[cchDst++] = RTPATH_SLASH;
369 pszDst[cchDst] = '\0';
370 }
371
372 /*
373 * Process the files and subdirs.
374 */
375 for (;;)
376 {
377 size_t cbDirEntry = RTZIPTARCMD_DIRENTRY_BUF_SIZE;
378 rc = RTVfsDirReadEx(hVfsIoDir, pDirEntry, &cbDirEntry, RTFSOBJATTRADD_UNIX);
379 if (RT_FAILURE(rc))
380 break;
381
382 /* Check length. */
383 if (pDirEntry->cbName + cchSrc + 3 >= RTPATH_MAX)
384 {
385 rc = VERR_BUFFER_OVERFLOW;
386 break;
387 }
388
389 switch (pDirEntry->Info.Attr.fMode & RTFS_TYPE_MASK)
390 {
391 case RTFS_TYPE_DIRECTORY:
392 {
393 if (RTDirEntryExIsStdDotLink(pDirEntry))
394 continue;
395
396 if (!pOpts->fRecursive)
397 continue;
398
399 memcpy(&pszSrc[cchSrc], pDirEntry->szName, pDirEntry->cbName + 1);
400 if (RT_SUCCESS(rc))
401 {
402 memcpy(&pszDst[cchDst], pDirEntry->szName, pDirEntry->cbName + 1);
403 rc = rtZipTarCmdArchiveDirSub(pOpts, hVfsFss, pszSrc, cchSrc + pDirEntry->cbName, paObjInfo,
404 pszDst, cchDst + pDirEntry->cbName, pDirEntry, pErrInfo);
405 }
406
407 break;
408 }
409
410 case RTFS_TYPE_FILE:
411 {
412 memcpy(&pszSrc[cchSrc], pDirEntry->szName, pDirEntry->cbName + 1);
413 rc = rtZipTarCmdQueryObjInfo(pszSrc, paObjInfo, 3 /* cObjInfo */);
414 if (RT_SUCCESS(rc))
415 {
416 memcpy(&pszDst[cchDst], pDirEntry->szName, pDirEntry->cbName + 1);
417 rc = rtZipTarCmdArchiveFile(pOpts, hVfsFss, pszSrc, paObjInfo, pszDst, pErrInfo);
418 }
419 break;
420 }
421
422 default:
423 {
424 if (pOpts->fVerbose)
425 RTPrintf("Warning: File system type %#x for '%s' not implemented yet, sorry! Skipping ...\n",
426 pDirEntry->Info.Attr.fMode & RTFS_TYPE_MASK, pDirEntry->szName);
427 break;
428 }
429 }
430 }
431
432 RTVfsDirRelease(hVfsIoDir);
433
434 if (rc != VERR_NO_MORE_FILES)
435 return RTMsgErrorExitFailure("RTVfsDirReadEx failed unexpectedly!");
436
437 return RTEXITCODE_SUCCESS;
438}
439
440
441/**
442 * Archives a directory recursively.
443 *
444 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE + printed message.
445 * @param pOpts The options.
446 * @param hVfsFss The TAR filesystem stream handle.
447 * @param pszSrc The directory path or VFS spec. We append to the
448 * buffer as we decend.
449 * @param cchSrc The length of the input.
450 * @param paObjInfo[3] Array of three FS object info structures. The first
451 * one is always filled with RTFSOBJATTRADD_UNIX info.
452 * The next two may contain owner and group names if
453 * available. The three buffers can be reused.
454 * @param pszDst The name to archive it the under. We append to the
455 * buffer as we decend.
456 * @param cchDst The length of the input.
457 * @param pErrInfo Error info buffer (saves stack space).
458 */
459static RTEXITCODE rtZipTarCmdArchiveDir(PRTZIPTARCMDOPS pOpts, RTVFSFSSTREAM hVfsFss, char pszSrc[RTPATH_MAX], size_t cchSrc,
460 RTFSOBJINFO paObjInfo[3], char pszDst[RTPATH_MAX], size_t cchDst,
461 PRTERRINFOSTATIC pErrInfo)
462{
463 RT_NOREF(cchSrc);
464
465 char szSrcAbs[RTPATH_MAX];
466 int rc = RTPathAbs(pszSrc, szSrcAbs, sizeof(szSrcAbs));
467 if (RT_FAILURE(rc))
468 return RTMsgErrorExitFailure("RTPathAbs failed on '%s': %Rrc\n", pszSrc, rc);
469
470 union
471 {
472 uint8_t abPadding[RTZIPTARCMD_DIRENTRY_BUF_SIZE];
473 RTDIRENTRYEX DirEntry;
474 } uBuf;
475
476 return rtZipTarCmdArchiveDirSub(pOpts, hVfsFss, szSrcAbs, strlen(szSrcAbs), paObjInfo, pszDst, cchDst, &uBuf.DirEntry, pErrInfo);
477}
478
479
480/**
481 * Opens the output archive specified by the options.
482 *
483 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE + printed message.
484 * @param pOpts The options.
485 * @param phVfsFss Where to return the TAR filesystem stream handle.
486 */
487static RTEXITCODE rtZipTarCmdOpenOutputArchive(PRTZIPTARCMDOPS pOpts, PRTVFSFSSTREAM phVfsFss)
488{
489 int rc;
490 *phVfsFss = NIL_RTVFSFSSTREAM;
491
492 /*
493 * Open the output file.
494 */
495 RTVFSIOSTREAM hVfsIos;
496 if ( pOpts->pszFile
497 && strcmp(pOpts->pszFile, "-") != 0)
498 {
499 uint32_t offError = 0;
500 RTERRINFOSTATIC ErrInfo;
501 rc = RTVfsChainOpenIoStream(pOpts->pszFile, RTFILE_O_WRITE | RTFILE_O_DENY_WRITE | RTFILE_O_CREATE_REPLACE,
502 &hVfsIos, &offError, RTErrInfoInitStatic(&ErrInfo));
503 if (RT_FAILURE(rc))
504 return RTVfsChainMsgErrorExitFailure("RTVfsChainOpenIoStream", pOpts->pszFile, rc, offError, &ErrInfo.Core);
505 }
506 else
507 {
508 rc = RTVfsIoStrmFromStdHandle(RTHANDLESTD_OUTPUT,
509 RTFILE_O_WRITE | RTFILE_O_DENY_WRITE | RTFILE_O_OPEN,
510 true /*fLeaveOpen*/,
511 &hVfsIos);
512 if (RT_FAILURE(rc))
513 return RTMsgErrorExitFailure("Failed to prepare standard output for writing: %Rrc", rc);
514 }
515
516 /*
517 * Pass it thru a compressor?
518 */
519 RTVFSIOSTREAM hVfsIosComp = NIL_RTVFSIOSTREAM;
520 switch (pOpts->chZipper)
521 {
522 /* no */
523 case '\0':
524 rc = VINF_SUCCESS;
525 break;
526
527 /* gunzip */
528 case 'z':
529 rc = RTZipGzipCompressIoStream(hVfsIos, 0 /*fFlags*/, 6, &hVfsIosComp);
530 if (RT_FAILURE(rc))
531 RTMsgError("Failed to open gzip decompressor: %Rrc", rc);
532 break;
533
534 /* bunzip2 */
535 case 'j':
536 rc = VERR_NOT_SUPPORTED;
537 RTMsgError("bzip2 is not supported by this build");
538 break;
539
540 /* bug */
541 default:
542 rc = VERR_INTERNAL_ERROR_2;
543 RTMsgError("unknown decompression method '%c'", pOpts->chZipper);
544 break;
545 }
546 if (RT_FAILURE(rc))
547 {
548 RTVfsIoStrmRelease(hVfsIos);
549 return RTEXITCODE_FAILURE;
550 }
551
552 if (hVfsIosComp != NIL_RTVFSIOSTREAM)
553 {
554 RTVfsIoStrmRelease(hVfsIos);
555 hVfsIos = hVfsIosComp;
556 hVfsIosComp = NIL_RTVFSIOSTREAM;
557 }
558
559 /*
560 * Open the filesystem stream creator.
561 */
562 if ( pOpts->enmFormat == RTZIPTARCMDFORMAT_TAR
563 || pOpts->enmFormat == RTZIPTARCMDFORMAT_AUTO_DEFAULT)
564 {
565 RTVFSFSSTREAM hVfsFss;
566 rc = RTZipTarFsStreamToIoStream(hVfsIos, pOpts->enmTarFormat, pOpts->fTarCreate, &hVfsFss);
567 if (RT_SUCCESS(rc))
568 {
569 /*
570 * Set transformation options.
571 */
572 rc = RTZipTarFsStreamSetFileMode(hVfsFss, pOpts->fFileModeAndMask, pOpts->fFileModeOrMask);
573 if (RT_SUCCESS(rc))
574 {
575 rc = RTZipTarFsStreamSetDirMode(hVfsFss, pOpts->fDirModeAndMask, pOpts->fDirModeOrMask);
576 if (RT_FAILURE(rc))
577 RTMsgError("RTZipTarFsStreamSetDirMode(%o,%o) failed: %Rrc", pOpts->fDirModeAndMask, pOpts->fDirModeOrMask, rc);
578 }
579 else
580 RTMsgError("RTZipTarFsStreamSetFileMode(%o,%o) failed: %Rrc", pOpts->fFileModeAndMask, pOpts->fFileModeOrMask, rc);
581 if ((pOpts->pszOwner || pOpts->uidOwner != NIL_RTUID) && RT_SUCCESS(rc))
582 {
583 rc = RTZipTarFsStreamSetOwner(hVfsFss, pOpts->uidOwner, pOpts->pszOwner);
584 if (RT_FAILURE(rc))
585 RTMsgError("RTZipTarFsStreamSetOwner(%d,%s) failed: %Rrc", pOpts->uidOwner, pOpts->pszOwner, rc);
586 }
587 if ((pOpts->pszGroup || pOpts->gidGroup != NIL_RTGID) && RT_SUCCESS(rc))
588 {
589 rc = RTZipTarFsStreamSetGroup(hVfsFss, pOpts->gidGroup, pOpts->pszGroup);
590 if (RT_FAILURE(rc))
591 RTMsgError("RTZipTarFsStreamSetGroup(%d,%s) failed: %Rrc", pOpts->gidGroup, pOpts->pszGroup, rc);
592 }
593 if (RT_SUCCESS(rc))
594 *phVfsFss = hVfsFss;
595 else
596 {
597 RTVfsFsStrmRelease(hVfsFss);
598 *phVfsFss = NIL_RTVFSFSSTREAM;
599 }
600 }
601 else
602 rc = RTMsgErrorExitFailure("Failed to open tar filesystem stream: %Rrc", rc);
603 }
604 else
605 rc = VERR_NOT_SUPPORTED;
606 RTVfsIoStrmRelease(hVfsIos);
607 if (RT_FAILURE(rc))
608 return RTMsgErrorExitFailure("Failed to open tar filesystem stream: %Rrc", rc);
609
610 return RTEXITCODE_SUCCESS;
611}
612
613
614/**
615 * Implements archive creation.
616 *
617 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE + printed message.
618 * @param pOpts The options.
619 */
620static RTEXITCODE rtZipTarCreate(PRTZIPTARCMDOPS pOpts)
621{
622 /*
623 * Refuse to create empty archive.
624 */
625 if (pOpts->cFiles == 0)
626 return RTMsgErrorExitFailure("Nothing to archive - refusing to create empty archive!");
627
628 /*
629 * First open the output file.
630 */
631 RTVFSFSSTREAM hVfsFss;
632 RTEXITCODE rcExit = rtZipTarCmdOpenOutputArchive(pOpts, &hVfsFss);
633 if (rcExit != RTEXITCODE_SUCCESS)
634 return rcExit;
635
636 /*
637 * Process the input files.
638 */
639 for (uint32_t iFile = 0; iFile < pOpts->cFiles; iFile++)
640 {
641 const char *pszFile = pOpts->papszFiles[iFile];
642
643 /*
644 * Construct/copy the source name.
645 */
646 int rc = VINF_SUCCESS;
647 char szSrc[RTPATH_MAX];
648 if ( RTPathStartsWithRoot(pszFile)
649 || RTVfsChainIsSpec(pszFile))
650 rc = RTStrCopy(szSrc, sizeof(szSrc), pszFile);
651 else
652 rc = RTPathJoin(szSrc, sizeof(szSrc), pOpts->pszDirectory ? pOpts->pszDirectory : ".", pOpts->papszFiles[iFile]);
653 if (RT_SUCCESS(rc))
654 {
655 /*
656 * Construct the archived name. We must strip leading root specifier.
657 */
658 char *pszFinalPath = NULL;
659 char szDst[RTPATH_MAX];
660 const char *pszDst = pszFile;
661 if (RTVfsChainIsSpec(pszFile))
662 {
663 uint32_t offError;
664 rc = RTVfsChainQueryFinalPath(pszFile, &pszFinalPath, &offError);
665 if (RT_SUCCESS(rc))
666 pszDst = pszFinalPath;
667 else
668 rcExit = RTVfsChainMsgErrorExitFailure("RTVfsChainQueryFinalPath", pszFile, rc, offError, NULL);
669 }
670 if (RT_SUCCESS(rc))
671 {
672 pszDst = RTPathSkipRootSpec(pszDst);
673 if (*pszDst == '\0')
674 {
675 pszDst = pOpts->pszPrefix ? pOpts->pszPrefix : ".";
676 rc = RTStrCopy(szDst, sizeof(szDst), pszDst);
677 }
678 else
679 {
680 if (pOpts->pszPrefix)
681 rc = RTPathJoin(szDst, sizeof(szDst), pOpts->pszPrefix, pszDst);
682 else
683 rc = RTStrCopy(szDst, sizeof(szDst), pszDst);
684 }
685 if (RT_SUCCESS(rc))
686 {
687 /*
688 * What kind of object is this and what affiliations does it have?
689 */
690 RTFSOBJINFO aObjInfo[3];
691 rc = rtZipTarCmdQueryObjInfo(szSrc, aObjInfo, RT_ELEMENTS(aObjInfo));
692 if (RT_SUCCESS(rc))
693 {
694 RTERRINFOSTATIC ErrInfo;
695
696 /*
697 * Process on an object type basis.
698 */
699 RTEXITCODE rcExit2;
700 if (RTFS_IS_DIRECTORY(aObjInfo[0].Attr.fMode))
701 rcExit2 = rtZipTarCmdArchiveDir(pOpts, hVfsFss, szSrc, strlen(szSrc), aObjInfo,
702 szDst, strlen(szDst), &ErrInfo);
703 else if (RTFS_IS_FILE(aObjInfo[0].Attr.fMode))
704 rcExit2 = rtZipTarCmdArchiveFile(pOpts, hVfsFss, szSrc, aObjInfo, szDst, &ErrInfo);
705 else if (RTFS_IS_SYMLINK(aObjInfo[0].Attr.fMode))
706 rcExit2 = RTMsgErrorExitFailure("Symlink archiving is not implemented");
707 else if (RTFS_IS_FIFO(aObjInfo[0].Attr.fMode))
708 rcExit2 = RTMsgErrorExitFailure("FIFO archiving is not implemented");
709 else if (RTFS_IS_SOCKET(aObjInfo[0].Attr.fMode))
710 rcExit2 = RTMsgErrorExitFailure("Socket archiving is not implemented");
711 else if (RTFS_IS_DEV_CHAR(aObjInfo[0].Attr.fMode) || RTFS_IS_DEV_BLOCK(aObjInfo[0].Attr.fMode))
712 rcExit2 = RTMsgErrorExitFailure("Device archiving is not implemented");
713 else if (RTFS_IS_WHITEOUT(aObjInfo[0].Attr.fMode))
714 rcExit2 = RTEXITCODE_SUCCESS;
715 else
716 rcExit2 = RTMsgErrorExitFailure("Unknown file type: %#x\n", aObjInfo[0].Attr.fMode);
717 if (rcExit2 != RTEXITCODE_SUCCESS)
718 rcExit = rcExit2;
719 }
720 else
721 rcExit = RTMsgErrorExitFailure("querying object information for '%s' failed (%s)", szSrc, pszFile);
722 }
723 else
724 rcExit = RTMsgErrorExitFailure("archived file name is too long, skipping '%s' (%s)", pszDst, pszFile);
725 }
726 }
727 else
728 rcExit = RTMsgErrorExitFailure("input file name is too long, skipping '%s'", pszFile);
729 }
730
731 /*
732 * Finalize the archive.
733 */
734 int rc = RTVfsFsStrmEnd(hVfsFss);
735 if (RT_FAILURE(rc))
736 rcExit = RTMsgErrorExitFailure("RTVfsFsStrmEnd failed: %Rrc", rc);
737
738 RTVfsFsStrmRelease(hVfsFss);
739 return rcExit;
740}
741
742
743/**
744 * Opens the input archive specified by the options.
745 *
746 * @returns RTEXITCODE_SUCCESS or RTEXITCODE_FAILURE + printed message.
747 * @param pOpts The options.
748 * @param phVfsFss Where to return the TAR filesystem stream handle.
749 */
750static RTEXITCODE rtZipTarCmdOpenInputArchive(PRTZIPTARCMDOPS pOpts, PRTVFSFSSTREAM phVfsFss)
751{
752 int rc;
753
754 /*
755 * Open the input file.
756 */
757 RTVFSIOSTREAM hVfsIos;
758 if ( pOpts->pszFile
759 && strcmp(pOpts->pszFile, "-") != 0)
760 {
761 uint32_t offError = 0;
762 RTERRINFOSTATIC ErrInfo;
763 rc = RTVfsChainOpenIoStream(pOpts->pszFile, RTFILE_O_READ | RTFILE_O_DENY_WRITE | RTFILE_O_OPEN,
764 &hVfsIos, &offError, RTErrInfoInitStatic(&ErrInfo));
765 if (RT_FAILURE(rc))
766 return RTVfsChainMsgErrorExitFailure("RTVfsChainOpenIoStream", pOpts->pszFile, rc, offError, &ErrInfo.Core);
767 }
768 else
769 {
770 rc = RTVfsIoStrmFromStdHandle(RTHANDLESTD_INPUT,
771 RTFILE_O_READ | RTFILE_O_DENY_WRITE | RTFILE_O_OPEN,
772 true /*fLeaveOpen*/,
773 &hVfsIos);
774 if (RT_FAILURE(rc))
775 return RTMsgErrorExitFailure("Failed to prepare standard in for reading: %Rrc", rc);
776 }
777
778 /*
779 * Pass it thru a decompressor?
780 */
781 RTVFSIOSTREAM hVfsIosDecomp = NIL_RTVFSIOSTREAM;
782 switch (pOpts->chZipper)
783 {
784 /* no */
785 case '\0':
786 rc = VINF_SUCCESS;
787 break;
788
789 /* gunzip */
790 case 'z':
791 rc = RTZipGzipDecompressIoStream(hVfsIos, 0 /*fFlags*/, &hVfsIosDecomp);
792 if (RT_FAILURE(rc))
793 RTMsgError("Failed to open gzip decompressor: %Rrc", rc);
794 break;
795
796 /* bunzip2 */
797 case 'j':
798 rc = VERR_NOT_SUPPORTED;
799 RTMsgError("bzip2 is not supported by this build");
800 break;
801
802 /* bug */
803 default:
804 rc = VERR_INTERNAL_ERROR_2;
805 RTMsgError("unknown decompression method '%c'", pOpts->chZipper);
806 break;
807 }
808 if (RT_FAILURE(rc))
809 {
810 RTVfsIoStrmRelease(hVfsIos);
811 return RTEXITCODE_FAILURE;
812 }
813
814 if (hVfsIosDecomp != NIL_RTVFSIOSTREAM)
815 {
816 RTVfsIoStrmRelease(hVfsIos);
817 hVfsIos = hVfsIosDecomp;
818 hVfsIosDecomp = NIL_RTVFSIOSTREAM;
819 }
820
821 /*
822 * Open the filesystem stream.
823 */
824 if (pOpts->enmFormat == RTZIPTARCMDFORMAT_TAR)
825 rc = RTZipTarFsStreamFromIoStream(hVfsIos, 0/*fFlags*/, phVfsFss);
826 else if (pOpts->enmFormat == RTZIPTARCMDFORMAT_XAR)
827#ifdef IPRT_WITH_XAR /* Requires C++ and XML, so only in some configruation of IPRT. */
828 rc = RTZipXarFsStreamFromIoStream(hVfsIos, 0/*fFlags*/, phVfsFss);
829#else
830 rc = VERR_NOT_SUPPORTED;
831#endif
832 else if (pOpts->enmFormat == RTZIPTARCMDFORMAT_CPIO)
833 rc = RTZipCpioFsStreamFromIoStream(hVfsIos, 0/*fFlags*/, phVfsFss);
834 else /** @todo make RTZipTarFsStreamFromIoStream fail if not tar file! */
835 rc = RTZipTarFsStreamFromIoStream(hVfsIos, 0/*fFlags*/, phVfsFss);
836 RTVfsIoStrmRelease(hVfsIos);
837 if (RT_FAILURE(rc))
838 return RTMsgErrorExitFailure("Failed to open tar filesystem stream: %Rrc", rc);
839
840 return RTEXITCODE_SUCCESS;
841}
842
843
844/**
845 * Worker for the --list and --extract commands.
846 *
847 * @returns The appropriate exit code.
848 * @param pOpts The tar options.
849 * @param pfnCallback The command specific callback.
850 */
851static RTEXITCODE rtZipTarDoWithMembers(PRTZIPTARCMDOPS pOpts, PFNDOWITHMEMBER pfnCallback)
852{
853 /*
854 * Allocate a bitmap to go with the file list. This will be used to
855 * indicate which files we've processed and which not.
856 */
857 uint32_t *pbmFound = NULL;
858 if (pOpts->cFiles)
859 {
860 pbmFound = (uint32_t *)RTMemAllocZ(((pOpts->cFiles + 31) / 32) * sizeof(uint32_t));
861 if (!pbmFound)
862 return RTMsgErrorExitFailure("Failed to allocate the found-file-bitmap");
863 }
864
865
866 /*
867 * Open the input archive.
868 */
869 RTVFSFSSTREAM hVfsFssIn;
870 RTEXITCODE rcExit = rtZipTarCmdOpenInputArchive(pOpts, &hVfsFssIn);
871 if (rcExit == RTEXITCODE_SUCCESS)
872 {
873 /*
874 * Process the stream.
875 */
876 for (;;)
877 {
878 /*
879 * Retrive the next object.
880 */
881 char *pszName;
882 RTVFSOBJ hVfsObj;
883 int rc = RTVfsFsStrmNext(hVfsFssIn, &pszName, NULL, &hVfsObj);
884 if (RT_FAILURE(rc))
885 {
886 if (rc != VERR_EOF)
887 rcExit = RTMsgErrorExitFailure("RTVfsFsStrmNext returned %Rrc", rc);
888 break;
889 }
890
891 /*
892 * Should we process this entry?
893 */
894 uint32_t iFile = UINT32_MAX;
895 if ( !pOpts->cFiles
896 || rtZipTarCmdIsNameInArray(pszName, pOpts->papszFiles, &iFile) )
897 {
898 if (pbmFound)
899 ASMBitSet(pbmFound, iFile);
900
901 rcExit = pfnCallback(pOpts, hVfsObj, pszName, rcExit);
902 }
903
904 /*
905 * Release the current object and string.
906 */
907 RTVfsObjRelease(hVfsObj);
908 RTStrFree(pszName);
909 }
910
911 /*
912 * Complain about any files we didn't find.
913 */
914 for (uint32_t iFile = 0; iFile < pOpts->cFiles; iFile++)
915 if (!ASMBitTest(pbmFound, iFile))
916 {
917 RTMsgError("%s: Was not found in the archive", pOpts->papszFiles[iFile]);
918 rcExit = RTEXITCODE_FAILURE;
919 }
920
921 RTVfsFsStrmRelease(hVfsFssIn);
922 }
923 RTMemFree(pbmFound);
924 return rcExit;
925}
926
927
928/**
929 * Checks if the name contains any escape sequences.
930 *
931 * An escape sequence would generally be one or more '..' references. On DOS
932 * like system, something that would make up a drive letter reference is also
933 * considered an escape sequence.
934 *
935 * @returns true / false.
936 * @param pszName The name to consider.
937 */
938static bool rtZipTarHasEscapeSequence(const char *pszName)
939{
940#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
941 if (pszName[0] == ':')
942 return true;
943#endif
944 while (*pszName)
945 {
946 while (RTPATH_IS_SEP(*pszName))
947 pszName++;
948 if ( pszName[0] == '.'
949 && pszName[1] == '.'
950 && (pszName[2] == '\0' || RTPATH_IS_SLASH(pszName[2])) )
951 return true;
952 while (*pszName && !RTPATH_IS_SEP(*pszName))
953 pszName++;
954 }
955
956 return false;
957}
958
959
960#if !defined(RT_OS_WINDOWS) && !defined(RT_OS_OS2)
961
962/**
963 * Queries the user ID to use when extracting a member.
964 *
965 * @returns rcExit or RTEXITCODE_FAILURE.
966 * @param pOpts The tar options.
967 * @param pUser The user info.
968 * @param pszName The file name to use when complaining.
969 * @param rcExit The current exit code.
970 * @param pUid Where to return the user ID.
971 */
972static RTEXITCODE rtZipTarQueryExtractOwner(PRTZIPTARCMDOPS pOpts, PCRTFSOBJINFO pOwner, const char *pszName, RTEXITCODE rcExit,
973 PRTUID pUid)
974{
975 if (pOpts->uidOwner != NIL_RTUID)
976 *pUid = pOpts->uidOwner;
977 else if (pOpts->fPreserveGroup)
978 {
979 if (!pOwner->Attr.u.UnixGroup.szName[0])
980 *pUid = pOwner->Attr.u.UnixOwner.uid;
981 else
982 {
983 *pUid = NIL_RTUID;
984 return RTMsgErrorExitFailure("%s: User resolving is not implemented.", pszName);
985 }
986 }
987 else
988 *pUid = NIL_RTUID;
989 return rcExit;
990}
991
992
993/**
994 * Queries the group ID to use when extracting a member.
995 *
996 * @returns rcExit or RTEXITCODE_FAILURE.
997 * @param pOpts The tar options.
998 * @param pGroup The group info.
999 * @param pszName The file name to use when complaining.
1000 * @param rcExit The current exit code.
1001 * @param pGid Where to return the group ID.
1002 */
1003static RTEXITCODE rtZipTarQueryExtractGroup(PRTZIPTARCMDOPS pOpts, PCRTFSOBJINFO pGroup, const char *pszName, RTEXITCODE rcExit,
1004 PRTGID pGid)
1005{
1006 if (pOpts->gidGroup != NIL_RTGID)
1007 *pGid = pOpts->gidGroup;
1008 else if (pOpts->fPreserveGroup)
1009 {
1010 if (!pGroup->Attr.u.UnixGroup.szName[0])
1011 *pGid = pGroup->Attr.u.UnixGroup.gid;
1012 else
1013 {
1014 *pGid = NIL_RTGID;
1015 return RTMsgErrorExitFailure("%s: Group resolving is not implemented.", pszName);
1016 }
1017 }
1018 else
1019 *pGid = NIL_RTGID;
1020 return rcExit;
1021}
1022
1023#endif /* !defined(RT_OS_WINDOWS) && !defined(RT_OS_OS2) */
1024
1025
1026/**
1027 * Corrects the file mode and other attributes.
1028 *
1029 * Common worker for rtZipTarCmdExtractFile and rtZipTarCmdExtractHardlink.
1030 *
1031 * @returns rcExit or RTEXITCODE_FAILURE.
1032 * @param pOpts The tar options.
1033 * @param rcExit The current exit code.
1034 * @param hFile The handle to the destination file.
1035 * @param pszDst The destination path (for error reporting).
1036 * @param pUnixInfo The unix fs object info.
1037 * @param pOwner The owner info.
1038 * @param pGroup The group info.
1039 */
1040static RTEXITCODE rtZipTarCmdExtractSetAttribs(PRTZIPTARCMDOPS pOpts, RTEXITCODE rcExit, RTFILE hFile, const char *pszDst,
1041 PCRTFSOBJINFO pUnixInfo, PCRTFSOBJINFO pOwner, PCRTFSOBJINFO pGroup)
1042{
1043 int rc;
1044
1045 if (!pOpts->fNoModTime)
1046 {
1047 rc = RTFileSetTimes(hFile, NULL, &pUnixInfo->ModificationTime, NULL, NULL);
1048 if (RT_FAILURE(rc))
1049 rcExit = RTMsgErrorExitFailure("%s: Error setting times: %Rrc", pszDst, rc);
1050 }
1051
1052#if !defined(RT_OS_WINDOWS) && !defined(RT_OS_OS2)
1053 if ( pOpts->uidOwner != NIL_RTUID
1054 || pOpts->gidGroup != NIL_RTGID
1055 || pOpts->fPreserveOwner
1056 || pOpts->fPreserveGroup)
1057 {
1058 RTUID uidFile;
1059 rcExit = rtZipTarQueryExtractOwner(pOpts, pOwner, pszDst, rcExit, &uidFile);
1060
1061 RTGID gidFile;
1062 rcExit = rtZipTarQueryExtractGroup(pOpts, pGroup, pszDst, rcExit, &gidFile);
1063 if (uidFile != NIL_RTUID || gidFile != NIL_RTGID)
1064 {
1065 rc = RTFileSetOwner(hFile, uidFile, gidFile);
1066 if (RT_FAILURE(rc))
1067 rcExit = RTMsgErrorExitFailure("%s: Error owner/group: %Rrc", pszDst, rc);
1068 }
1069 }
1070#else
1071 RT_NOREF_PV(pOwner); RT_NOREF_PV(pGroup);
1072#endif
1073
1074 RTFMODE fMode = (pUnixInfo->Attr.fMode & pOpts->fFileModeAndMask) | pOpts->fFileModeOrMask;
1075 rc = RTFileSetMode(hFile, fMode | RTFS_TYPE_FILE);
1076 if (RT_FAILURE(rc))
1077 rcExit = RTMsgErrorExitFailure("%s: Error changing mode: %Rrc", pszDst, rc);
1078
1079 return rcExit;
1080}
1081
1082
1083/**
1084 * Extracts a hard linked file.
1085 *
1086 * @returns rcExit or RTEXITCODE_FAILURE.
1087 * @param pOpts The tar options.
1088 * @param rcExit The current exit code.
1089 * @param pszDst The destination path.
1090 * @param pszTarget The target relative path.
1091 * @param pUnixInfo The unix fs object info.
1092 * @param pOwner The owner info.
1093 * @param pGroup The group info.
1094 */
1095static RTEXITCODE rtZipTarCmdExtractHardlink(PRTZIPTARCMDOPS pOpts, RTEXITCODE rcExit, const char *pszDst,
1096 const char *pszTarget, PCRTFSOBJINFO pUnixInfo, PCRTFSOBJINFO pOwner,
1097 PCRTFSOBJINFO pGroup)
1098{
1099 /*
1100 * Construct full target path and check that it exists.
1101 */
1102 char szFullTarget[RTPATH_MAX];
1103 int rc = RTPathJoin(szFullTarget, sizeof(szFullTarget), pOpts->pszDirectory ? pOpts->pszDirectory : ".", pszTarget);
1104 if (RT_FAILURE(rc))
1105 return RTMsgErrorExitFailure("%s: Failed to construct full hardlink target path for %s: %Rrc",
1106 pszDst, pszTarget, rc);
1107
1108 if (!RTFileExists(szFullTarget))
1109 return RTMsgErrorExitFailure("%s: Hardlink target not found (or not a file): %s", pszDst, szFullTarget);
1110
1111 /*
1112 * Try hardlink the file, falling back on copying.
1113 */
1114 /** @todo actual hardlinking */
1115 if (true)
1116 {
1117 RTMsgWarning("%s: Hardlinking not available, copying '%s' instead.", pszDst, szFullTarget);
1118
1119 RTFILE hSrcFile;
1120 rc = RTFileOpen(&hSrcFile, szFullTarget, RTFILE_O_READ | RTFILE_O_DENY_WRITE | RTFILE_O_OPEN);
1121 if (RT_SUCCESS(rc))
1122 {
1123 uint32_t fOpen = RTFILE_O_READWRITE | RTFILE_O_DENY_WRITE | RTFILE_O_CREATE_REPLACE | RTFILE_O_ACCESS_ATTR_DEFAULT
1124 | ((RTFS_UNIX_IWUSR | RTFS_UNIX_IRUSR) << RTFILE_O_CREATE_MODE_SHIFT);
1125 RTFILE hDstFile;
1126 rc = RTFileOpen(&hDstFile, pszDst, fOpen);
1127 if (RT_SUCCESS(rc))
1128 {
1129 rc = RTFileCopyByHandles(hSrcFile, hDstFile);
1130 if (RT_SUCCESS(rc))
1131 {
1132 rcExit = rtZipTarCmdExtractSetAttribs(pOpts, rcExit, hDstFile, pszDst, pUnixInfo, pOwner, pGroup);
1133 rc = RTFileClose(hDstFile);
1134 if (RT_FAILURE(rc))
1135 {
1136 rcExit = RTMsgErrorExitFailure("%s: Error closing hardlinked file copy: %Rrc", pszDst, rc);
1137 RTFileDelete(pszDst);
1138 }
1139 }
1140 else
1141 {
1142 rcExit = RTMsgErrorExitFailure("%s: Failed copying hardlinked file '%s': %Rrc", pszDst, szFullTarget, rc);
1143 rc = RTFileClose(hDstFile);
1144 RTFileDelete(pszDst);
1145 }
1146 }
1147 else
1148 rcExit = RTMsgErrorExitFailure("%s: Error creating file: %Rrc", pszDst, rc);
1149 RTFileClose(hSrcFile);
1150 }
1151 else
1152 rcExit = RTMsgErrorExitFailure("%s: Error opening file '%s' for reading (hardlink target): %Rrc",
1153 pszDst, szFullTarget, rc);
1154 }
1155
1156 return rcExit;
1157}
1158
1159
1160
1161/**
1162 * Extracts a file.
1163 *
1164 * Since we can restore permissions and attributes more efficiently by working
1165 * directly on the file handle, we have special code path for files.
1166 *
1167 * @returns rcExit or RTEXITCODE_FAILURE.
1168 * @param pOpts The tar options.
1169 * @param hVfsObj The tar object to display
1170 * @param rcExit The current exit code.
1171 * @param pszDst The destination path.
1172 * @param pUnixInfo The unix fs object info.
1173 * @param pOwner The owner info.
1174 * @param pGroup The group info.
1175 */
1176static RTEXITCODE rtZipTarCmdExtractFile(PRTZIPTARCMDOPS pOpts, RTVFSOBJ hVfsObj, RTEXITCODE rcExit,
1177 const char *pszDst, PCRTFSOBJINFO pUnixInfo, PCRTFSOBJINFO pOwner, PCRTFSOBJINFO pGroup)
1178{
1179 /*
1180 * Open the destination file and create a stream object for it.
1181 */
1182 uint32_t fOpen = RTFILE_O_READWRITE | RTFILE_O_DENY_WRITE | RTFILE_O_CREATE_REPLACE | RTFILE_O_ACCESS_ATTR_DEFAULT
1183 | ((RTFS_UNIX_IWUSR | RTFS_UNIX_IRUSR) << RTFILE_O_CREATE_MODE_SHIFT);
1184 RTFILE hFile;
1185 int rc = RTFileOpen(&hFile, pszDst, fOpen);
1186 if (RT_FAILURE(rc))
1187 return RTMsgErrorExitFailure("%s: Error creating file: %Rrc", pszDst, rc);
1188
1189 RTVFSIOSTREAM hVfsIosDst;
1190 rc = RTVfsIoStrmFromRTFile(hFile, fOpen, true /*fLeaveOpen*/, &hVfsIosDst);
1191 if (RT_SUCCESS(rc))
1192 {
1193 /*
1194 * Convert source to a stream and optionally add a read ahead stage.
1195 */
1196 RTVFSIOSTREAM hVfsIosSrc = RTVfsObjToIoStream(hVfsObj);
1197 if (pOpts->fReadAhead)
1198 {
1199 RTVFSIOSTREAM hVfsReadAhead;
1200 rc = RTVfsCreateReadAheadForIoStream(hVfsIosSrc, 0 /*fFlag*/, 16 /*cBuffers*/, _256K /*cbBuffer*/, &hVfsReadAhead);
1201 if (RT_SUCCESS(rc))
1202 {
1203 RTVfsIoStrmRelease(hVfsIosSrc);
1204 hVfsIosSrc = hVfsReadAhead;
1205 }
1206 else
1207 AssertRC(rc); /* can be ignored in release builds. */
1208 }
1209
1210 /*
1211 * Pump the data thru and correct the file attributes.
1212 */
1213 rc = RTVfsUtilPumpIoStreams(hVfsIosSrc, hVfsIosDst, (uint32_t)RT_MIN(pUnixInfo->cbObject, _1M));
1214 if (RT_SUCCESS(rc))
1215 rcExit = rtZipTarCmdExtractSetAttribs(pOpts, rcExit, hFile, pszDst, pUnixInfo, pOwner, pGroup);
1216 else
1217 rcExit = RTMsgErrorExitFailure("%s: Error writing out file: %Rrc", pszDst, rc);
1218 RTVfsIoStrmRelease(hVfsIosSrc);
1219 RTVfsIoStrmRelease(hVfsIosDst);
1220 }
1221 else
1222 rcExit = RTMsgErrorExitFailure("%s: Error creating I/O stream for file: %Rrc", pszDst, rc);
1223 RTFileClose(hFile);
1224 return rcExit;
1225}
1226
1227
1228/**
1229 * @callback_method_impl{PFNDOWITHMEMBER, Implements --extract.}
1230 */
1231static RTEXITCODE rtZipTarCmdExtractCallback(PRTZIPTARCMDOPS pOpts, RTVFSOBJ hVfsObj, const char *pszName, RTEXITCODE rcExit)
1232{
1233 if (pOpts->fVerbose)
1234 RTPrintf("%s\n", pszName);
1235
1236 /*
1237 * Query all the information.
1238 */
1239 RTFSOBJINFO UnixInfo;
1240 int rc = RTVfsObjQueryInfo(hVfsObj, &UnixInfo, RTFSOBJATTRADD_UNIX);
1241 if (RT_FAILURE(rc))
1242 return RTMsgErrorExitFailure("RTVfsObjQueryInfo returned %Rrc on '%s'", rc, pszName);
1243
1244 RTFSOBJINFO Owner;
1245 rc = RTVfsObjQueryInfo(hVfsObj, &Owner, RTFSOBJATTRADD_UNIX_OWNER);
1246 if (RT_FAILURE(rc))
1247 return RTMsgErrorExit(RTEXITCODE_FAILURE,
1248 "RTVfsObjQueryInfo(,,UNIX_OWNER) returned %Rrc on '%s'",
1249 rc, pszName);
1250
1251 RTFSOBJINFO Group;
1252 rc = RTVfsObjQueryInfo(hVfsObj, &Group, RTFSOBJATTRADD_UNIX_GROUP);
1253 if (RT_FAILURE(rc))
1254 return RTMsgErrorExit(RTEXITCODE_FAILURE,
1255 "RTVfsObjQueryInfo(,,UNIX_OWNER) returned %Rrc on '%s'",
1256 rc, pszName);
1257
1258 bool fIsHardLink = false;
1259 char szTarget[RTPATH_MAX];
1260 szTarget[0] = '\0';
1261 RTVFSSYMLINK hVfsSymlink = RTVfsObjToSymlink(hVfsObj);
1262 if (hVfsSymlink != NIL_RTVFSSYMLINK)
1263 {
1264 rc = RTVfsSymlinkRead(hVfsSymlink, szTarget, sizeof(szTarget));
1265 RTVfsSymlinkRelease(hVfsSymlink);
1266 if (RT_FAILURE(rc))
1267 return RTMsgErrorExitFailure("%s: RTVfsSymlinkRead failed: %Rrc", pszName, rc);
1268 if (!szTarget[0])
1269 return RTMsgErrorExitFailure("%s: Link target is empty.", pszName);
1270 if (!RTFS_IS_SYMLINK(UnixInfo.Attr.fMode))
1271 {
1272 fIsHardLink = true;
1273 if (!RTFS_IS_FILE(UnixInfo.Attr.fMode))
1274 return RTMsgErrorExitFailure("%s: Hardlinks are only supported for regular files (target=%s).",
1275 pszName, szTarget);
1276 if (rtZipTarHasEscapeSequence(pszName))
1277 return RTMsgErrorExitFailure("%s: Hardlink target '%s' contains an escape sequence.",
1278 pszName, szTarget);
1279 }
1280 }
1281 else if (RTFS_IS_SYMLINK(UnixInfo.Attr.fMode))
1282 return RTMsgErrorExitFailure("Failed to get symlink object for '%s'", pszName);
1283
1284 if (rtZipTarHasEscapeSequence(pszName))
1285 return RTMsgErrorExitFailure("Name '%s' contains an escape sequence.", pszName);
1286
1287 /*
1288 * Construct the path to the extracted member.
1289 */
1290 char szDst[RTPATH_MAX];
1291 rc = RTPathJoin(szDst, sizeof(szDst), pOpts->pszDirectory ? pOpts->pszDirectory : ".", pszName);
1292 if (RT_FAILURE(rc))
1293 return RTMsgErrorExitFailure("%s: Failed to construct destination path for: %Rrc", pszName, rc);
1294
1295 /*
1296 * Extract according to the type.
1297 */
1298 if (!fIsHardLink)
1299 switch (UnixInfo.Attr.fMode & RTFS_TYPE_MASK)
1300 {
1301 case RTFS_TYPE_FILE:
1302 return rtZipTarCmdExtractFile(pOpts, hVfsObj, rcExit, szDst, &UnixInfo, &Owner, &Group);
1303
1304 case RTFS_TYPE_DIRECTORY:
1305 rc = RTDirCreateFullPath(szDst, UnixInfo.Attr.fMode & RTFS_UNIX_ALL_ACCESS_PERMS);
1306 if (RT_FAILURE(rc))
1307 return RTMsgErrorExitFailure("%s: Error creating directory: %Rrc", szDst, rc);
1308 break;
1309
1310 case RTFS_TYPE_SYMLINK:
1311 rc = RTSymlinkCreate(szDst, szTarget, RTSYMLINKTYPE_UNKNOWN, 0);
1312 if (RT_FAILURE(rc))
1313 return RTMsgErrorExitFailure("%s: Error creating symbolic link: %Rrc", szDst, rc);
1314 break;
1315
1316 case RTFS_TYPE_FIFO:
1317 return RTMsgErrorExitFailure("%s: FIFOs are not supported.", pszName);
1318 case RTFS_TYPE_DEV_CHAR:
1319 return RTMsgErrorExitFailure("%s: FIFOs are not supported.", pszName);
1320 case RTFS_TYPE_DEV_BLOCK:
1321 return RTMsgErrorExitFailure("%s: Block devices are not supported.", pszName);
1322 case RTFS_TYPE_SOCKET:
1323 return RTMsgErrorExitFailure("%s: Sockets are not supported.", pszName);
1324 case RTFS_TYPE_WHITEOUT:
1325 return RTMsgErrorExitFailure("%s: Whiteouts are not support.", pszName);
1326 default:
1327 return RTMsgErrorExitFailure("%s: Unknown file type.", pszName);
1328 }
1329 else
1330 return rtZipTarCmdExtractHardlink(pOpts, rcExit, szDst, szTarget, &UnixInfo, &Owner, &Group);
1331
1332 /*
1333 * Set other attributes as requested.
1334 *
1335 * Note! File extraction does get here.
1336 */
1337 if (!pOpts->fNoModTime)
1338 {
1339 rc = RTPathSetTimesEx(szDst, NULL, &UnixInfo.ModificationTime, NULL, NULL, RTPATH_F_ON_LINK);
1340 if (RT_FAILURE(rc) && rc != VERR_NOT_SUPPORTED && rc != VERR_NS_SYMLINK_SET_TIME)
1341 rcExit = RTMsgErrorExitFailure("%s: Error changing modification time: %Rrc.", pszName, rc);
1342 }
1343
1344#if !defined(RT_OS_WINDOWS) && !defined(RT_OS_OS2)
1345 if ( pOpts->uidOwner != NIL_RTUID
1346 || pOpts->gidGroup != NIL_RTGID
1347 || pOpts->fPreserveOwner
1348 || pOpts->fPreserveGroup)
1349 {
1350 RTUID uidFile;
1351 rcExit = rtZipTarQueryExtractOwner(pOpts, &Owner, szDst, rcExit, &uidFile);
1352
1353 RTGID gidFile;
1354 rcExit = rtZipTarQueryExtractGroup(pOpts, &Group, szDst, rcExit, &gidFile);
1355 if (uidFile != NIL_RTUID || gidFile != NIL_RTGID)
1356 {
1357 rc = RTPathSetOwnerEx(szDst, uidFile, gidFile, RTPATH_F_ON_LINK);
1358 if (RT_FAILURE(rc))
1359 rcExit = RTMsgErrorExitFailure("%s: Error owner/group: %Rrc", szDst, rc);
1360 }
1361 }
1362#endif
1363
1364#if !defined(RT_OS_WINDOWS) /** @todo implement RTPathSetMode on windows... */
1365 if (!RTFS_IS_SYMLINK(UnixInfo.Attr.fMode)) /* RTPathSetMode follows symbolic links atm. */
1366 {
1367 RTFMODE fMode;
1368 if (RTFS_IS_DIRECTORY(UnixInfo.Attr.fMode))
1369 fMode = (UnixInfo.Attr.fMode & (pOpts->fDirModeAndMask | RTFS_TYPE_MASK)) | pOpts->fDirModeOrMask;
1370 else
1371 fMode = (UnixInfo.Attr.fMode & (pOpts->fFileModeAndMask | RTFS_TYPE_MASK)) | pOpts->fFileModeOrMask;
1372 rc = RTPathSetMode(szDst, fMode);
1373 if (RT_FAILURE(rc))
1374 rcExit = RTMsgErrorExitFailure("%s: Error changing mode: %Rrc", szDst, rc);
1375 }
1376#endif
1377
1378 return rcExit;
1379}
1380
1381
1382/**
1383 * @callback_method_impl{PFNDOWITHMEMBER, Implements --list.}
1384 */
1385static RTEXITCODE rtZipTarCmdListCallback(PRTZIPTARCMDOPS pOpts, RTVFSOBJ hVfsObj, const char *pszName, RTEXITCODE rcExit)
1386{
1387 /*
1388 * This is very simple in non-verbose mode.
1389 */
1390 if (!pOpts->fVerbose)
1391 {
1392 RTPrintf("%s\n", pszName);
1393 return rcExit;
1394 }
1395
1396 /*
1397 * Query all the information.
1398 */
1399 RTFSOBJINFO UnixInfo;
1400 int rc = RTVfsObjQueryInfo(hVfsObj, &UnixInfo, RTFSOBJATTRADD_UNIX);
1401 if (RT_FAILURE(rc))
1402 {
1403 rcExit = RTMsgErrorExitFailure("RTVfsObjQueryInfo returned %Rrc on '%s'", rc, pszName);
1404 RT_ZERO(UnixInfo);
1405 }
1406
1407 RTFSOBJINFO Owner;
1408 rc = RTVfsObjQueryInfo(hVfsObj, &Owner, RTFSOBJATTRADD_UNIX_OWNER);
1409 if (RT_FAILURE(rc))
1410 {
1411 rcExit = RTMsgErrorExit(RTEXITCODE_FAILURE,
1412 "RTVfsObjQueryInfo(,,UNIX_OWNER) returned %Rrc on '%s'",
1413 rc, pszName);
1414 RT_ZERO(Owner);
1415 }
1416
1417 RTFSOBJINFO Group;
1418 rc = RTVfsObjQueryInfo(hVfsObj, &Group, RTFSOBJATTRADD_UNIX_GROUP);
1419 if (RT_FAILURE(rc))
1420 {
1421 rcExit = RTMsgErrorExit(RTEXITCODE_FAILURE,
1422 "RTVfsObjQueryInfo(,,UNIX_OWNER) returned %Rrc on '%s'",
1423 rc, pszName);
1424 RT_ZERO(Group);
1425 }
1426
1427 const char *pszLinkType = NULL;
1428 char szTarget[RTPATH_MAX];
1429 szTarget[0] = '\0';
1430 RTVFSSYMLINK hVfsSymlink = RTVfsObjToSymlink(hVfsObj);
1431 if (hVfsSymlink != NIL_RTVFSSYMLINK)
1432 {
1433 rc = RTVfsSymlinkRead(hVfsSymlink, szTarget, sizeof(szTarget));
1434 if (RT_FAILURE(rc))
1435 rcExit = RTMsgErrorExitFailure("RTVfsSymlinkRead returned %Rrc on '%s'", rc, pszName);
1436 RTVfsSymlinkRelease(hVfsSymlink);
1437 pszLinkType = RTFS_IS_SYMLINK(UnixInfo.Attr.fMode) ? "->" : "link to";
1438 }
1439 else if (RTFS_IS_SYMLINK(UnixInfo.Attr.fMode))
1440 rcExit = RTMsgErrorExitFailure("Failed to get symlink object for '%s'", pszName);
1441
1442 /*
1443 * Translate the mode mask.
1444 */
1445 char szMode[16];
1446 switch (UnixInfo.Attr.fMode & RTFS_TYPE_MASK)
1447 {
1448 case RTFS_TYPE_FIFO: szMode[0] = 'f'; break;
1449 case RTFS_TYPE_DEV_CHAR: szMode[0] = 'c'; break;
1450 case RTFS_TYPE_DIRECTORY: szMode[0] = 'd'; break;
1451 case RTFS_TYPE_DEV_BLOCK: szMode[0] = 'b'; break;
1452 case RTFS_TYPE_FILE: szMode[0] = '-'; break;
1453 case RTFS_TYPE_SYMLINK: szMode[0] = 'l'; break;
1454 case RTFS_TYPE_SOCKET: szMode[0] = 's'; break;
1455 case RTFS_TYPE_WHITEOUT: szMode[0] = 'w'; break;
1456 default: szMode[0] = '?'; break;
1457 }
1458 if (pszLinkType && szMode[0] != 's')
1459 szMode[0] = 'h';
1460
1461 szMode[1] = UnixInfo.Attr.fMode & RTFS_UNIX_IRUSR ? 'r' : '-';
1462 szMode[2] = UnixInfo.Attr.fMode & RTFS_UNIX_IWUSR ? 'w' : '-';
1463 szMode[3] = UnixInfo.Attr.fMode & RTFS_UNIX_IXUSR ? 'x' : '-';
1464
1465 szMode[4] = UnixInfo.Attr.fMode & RTFS_UNIX_IRGRP ? 'r' : '-';
1466 szMode[5] = UnixInfo.Attr.fMode & RTFS_UNIX_IWGRP ? 'w' : '-';
1467 szMode[6] = UnixInfo.Attr.fMode & RTFS_UNIX_IXGRP ? 'x' : '-';
1468
1469 szMode[7] = UnixInfo.Attr.fMode & RTFS_UNIX_IROTH ? 'r' : '-';
1470 szMode[8] = UnixInfo.Attr.fMode & RTFS_UNIX_IWOTH ? 'w' : '-';
1471 szMode[9] = UnixInfo.Attr.fMode & RTFS_UNIX_IXOTH ? 'x' : '-';
1472 szMode[10] = '\0';
1473
1474 /** @todo sticky and set-uid/gid bits. */
1475
1476 /*
1477 * Make sure we've got valid owner and group strings.
1478 */
1479 if (!Owner.Attr.u.UnixGroup.szName[0])
1480 RTStrPrintf(Owner.Attr.u.UnixOwner.szName, sizeof(Owner.Attr.u.UnixOwner.szName),
1481 "%u", UnixInfo.Attr.u.Unix.uid);
1482
1483 if (!Group.Attr.u.UnixOwner.szName[0])
1484 RTStrPrintf(Group.Attr.u.UnixGroup.szName, sizeof(Group.Attr.u.UnixGroup.szName),
1485 "%u", UnixInfo.Attr.u.Unix.gid);
1486
1487 /*
1488 * Format the modification time.
1489 */
1490 char szModTime[32];
1491 RTTIME ModTime;
1492 PRTTIME pTime;
1493 if (!pOpts->fDisplayUtc)
1494 pTime = RTTimeLocalExplode(&ModTime, &UnixInfo.ModificationTime);
1495 else
1496 pTime = RTTimeExplode(&ModTime, &UnixInfo.ModificationTime);
1497 if (!pTime)
1498 RT_ZERO(ModTime);
1499 RTStrPrintf(szModTime, sizeof(szModTime), "%04d-%02u-%02u %02u:%02u",
1500 ModTime.i32Year, ModTime.u8Month, ModTime.u8MonthDay, ModTime.u8Hour, ModTime.u8Minute);
1501
1502 /*
1503 * Format the size and figure how much space is needed between the
1504 * user/group and the size.
1505 */
1506 char szSize[64];
1507 size_t cchSize;
1508 switch (UnixInfo.Attr.fMode & RTFS_TYPE_MASK)
1509 {
1510 case RTFS_TYPE_DEV_CHAR:
1511 case RTFS_TYPE_DEV_BLOCK:
1512 cchSize = RTStrPrintf(szSize, sizeof(szSize), "%u,%u",
1513 RTDEV_MAJOR(UnixInfo.Attr.u.Unix.Device), RTDEV_MINOR(UnixInfo.Attr.u.Unix.Device));
1514 break;
1515 default:
1516 cchSize = RTStrPrintf(szSize, sizeof(szSize), "%RU64", UnixInfo.cbObject);
1517 break;
1518 }
1519
1520 size_t cchUserGroup = strlen(Owner.Attr.u.UnixOwner.szName)
1521 + 1
1522 + strlen(Group.Attr.u.UnixGroup.szName);
1523 ssize_t cchPad = cchUserGroup + cchSize + 1 < 19
1524 ? 19 - (cchUserGroup + cchSize + 1)
1525 : 0;
1526
1527 /*
1528 * Go to press.
1529 */
1530 if (pszLinkType)
1531 RTPrintf("%s %s/%s%*s %s %s %s %s %s\n",
1532 szMode,
1533 Owner.Attr.u.UnixOwner.szName, Group.Attr.u.UnixGroup.szName,
1534 cchPad, "",
1535 szSize,
1536 szModTime,
1537 pszName,
1538 pszLinkType,
1539 szTarget);
1540 else
1541 RTPrintf("%s %s/%s%*s %s %s %s\n",
1542 szMode,
1543 Owner.Attr.u.UnixOwner.szName, Group.Attr.u.UnixGroup.szName,
1544 cchPad, "",
1545 szSize,
1546 szModTime,
1547 pszName);
1548
1549 return rcExit;
1550}
1551
1552
1553/**
1554 * Display usage.
1555 *
1556 * @param pszProgName The program name.
1557 */
1558static void rtZipTarUsage(const char *pszProgName)
1559{
1560 /*
1561 * 0 1 2 3 4 5 6 7 8
1562 * 012345678901234567890123456789012345678901234567890123456789012345678901234567890
1563 */
1564 RTPrintf("Usage: %s [options]\n"
1565 "\n",
1566 pszProgName);
1567 RTPrintf("Operations:\n"
1568 " -A, --concatenate, --catenate\n"
1569 " Append the content of one tar archive to another. (not impl)\n"
1570 " -c, --create\n"
1571 " Create a new tar archive. (not impl)\n"
1572 " -d, --diff, --compare\n"
1573 " Compare atar archive with the file system. (not impl)\n"
1574 " -r, --append\n"
1575 " Append more files to the tar archive. (not impl)\n"
1576 " -t, --list\n"
1577 " List the contents of the tar archive.\n"
1578 " -u, --update\n"
1579 " Update the archive, adding files that are newer than the\n"
1580 " ones in the archive. (not impl)\n"
1581 " -x, --extract, --get\n"
1582 " Extract the files from the tar archive.\n"
1583 " --delete\n"
1584 " Delete files from the tar archive.\n"
1585 "\n"
1586 );
1587 RTPrintf("Basic Options:\n"
1588 " -C <dir>, --directory <dir> (-A, -c, -d, -r, -u, -x)\n"
1589 " Sets the base directory for input and output file members.\n"
1590 " This does not apply to --file, even if it preceeds it.\n"
1591 " -f <archive>, --file <archive> (all)\n"
1592 " The tar file to create or process. '-' indicates stdout/stdin,\n"
1593 " which is is the default.\n"
1594 " -v, --verbose (all)\n"
1595 " Verbose operation.\n"
1596 " -p, --preserve-permissions (-x)\n"
1597 " Preserve all permissions when extracting. Must be used\n"
1598 " before the mode mask options as it will change some of these.\n"
1599 " -j, --bzip2 (all)\n"
1600 " Compress/decompress the archive with bzip2.\n"
1601 " -z, --gzip, --gunzip, --ungzip (all)\n"
1602 " Compress/decompress the archive with gzip.\n"
1603 "\n");
1604 RTPrintf("Misc Options:\n"
1605 " --owner <uid/username> (-A, -c, -d, -r, -u, -x)\n"
1606 " Set the owner of extracted and archived files to the user specified.\n"
1607 " --group <uid/username> (-A, -c, -d, -r, -u, -x)\n"
1608 " Set the group of extracted and archived files to the group specified.\n"
1609 " --utc (-t)\n"
1610 " Display timestamps as UTC instead of local time.\n"
1611 " -S, --sparse (-A, -c, -u)\n"
1612 " Detect sparse files and store them (gnu tar extension).\n"
1613 " --format <format> (-A, -c, -u, but also -d, -r, -x)\n"
1614 " The file format:\n"
1615 " auto (gnu tar)\n"
1616 " default (gnu tar)\n"
1617 " tar (gnu tar)"
1618 " gnu (tar v1.13+), "
1619 " ustar (tar POSIX.1-1988), "
1620 " pax (tar POSIX.1-2001),\n"
1621 " xar\n"
1622 " cpio\n"
1623 " Note! Because XAR/TAR/CPIO detection isn't implemented yet, it\n"
1624 " is necessary to specifcy --format=xar when reading a\n"
1625 " XAR file or --format=cpio for a CPIO file.\n"
1626 " Otherwise this option is only for creation.\n"
1627 "\n");
1628 RTPrintf("IPRT Options:\n"
1629 " --prefix <dir-prefix> (-A, -c, -d, -r, -u)\n"
1630 " Directory prefix to give the members added to the archive.\n"
1631 " --file-mode-and-mask <octal-mode> (-A, -c, -d, -r, -u, -x)\n"
1632 " Restrict the access mode of regular and special files.\n"
1633 " --file-mode-or-mask <octal-mode> (-A, -c, -d, -r, -u, -x)\n"
1634 " Include the given access mode for regular and special files.\n"
1635 " --dir-mode-and-mask <octal-mode> (-A, -c, -d, -r, -u, -x)\n"
1636 " Restrict the access mode of directories.\n"
1637 " --dir-mode-or-mask <octal-mode> (-A, -c, -d, -r, -u, -x)\n"
1638 " Include the given access mode for directories.\n"
1639 " --read-ahead (-x)\n"
1640 " Enabled read ahead thread when extracting files.\n"
1641 " --push-file (-A, -c, -u)\n"
1642 " Use RTVfsFsStrmPushFile instead of RTVfsFsStrmAdd.\n"
1643 "\n");
1644 RTPrintf("Standard Options:\n"
1645 " -h, -?, --help\n"
1646 " Display this help text.\n"
1647 " -V, --version\n"
1648 " Display version number.\n");
1649}
1650
1651
1652RTDECL(RTEXITCODE) RTZipTarCmd(unsigned cArgs, char **papszArgs)
1653{
1654 /*
1655 * Parse the command line.
1656 *
1657 * N.B. This is less flexible that your regular tar program in that it
1658 * requires the operation to be specified as an option. On the other
1659 * hand, you can specify it where ever you like in the command line.
1660 */
1661 static const RTGETOPTDEF s_aOptions[] =
1662 {
1663 /* operations */
1664 { "--concatenate", 'A', RTGETOPT_REQ_NOTHING },
1665 { "--catenate", 'A', RTGETOPT_REQ_NOTHING },
1666 { "--create", 'c', RTGETOPT_REQ_NOTHING },
1667 { "--diff", 'd', RTGETOPT_REQ_NOTHING },
1668 { "--compare", 'd', RTGETOPT_REQ_NOTHING },
1669 { "--append", 'r', RTGETOPT_REQ_NOTHING },
1670 { "--list", 't', RTGETOPT_REQ_NOTHING },
1671 { "--update", 'u', RTGETOPT_REQ_NOTHING },
1672 { "--extract", 'x', RTGETOPT_REQ_NOTHING },
1673 { "--get", 'x', RTGETOPT_REQ_NOTHING },
1674 { "--delete", RTZIPTARCMD_OPT_DELETE, RTGETOPT_REQ_NOTHING },
1675
1676 /* basic options */
1677 { "--directory", 'C', RTGETOPT_REQ_STRING },
1678 { "--file", 'f', RTGETOPT_REQ_STRING },
1679 { "--verbose", 'v', RTGETOPT_REQ_NOTHING },
1680 { "--preserve-permissions", 'p', RTGETOPT_REQ_NOTHING },
1681 { "--bzip2", 'j', RTGETOPT_REQ_NOTHING },
1682 { "--gzip", 'z', RTGETOPT_REQ_NOTHING },
1683 { "--gunzip", 'z', RTGETOPT_REQ_NOTHING },
1684 { "--ungzip", 'z', RTGETOPT_REQ_NOTHING },
1685
1686 /* other options. */
1687 { "--owner", RTZIPTARCMD_OPT_OWNER, RTGETOPT_REQ_STRING },
1688 { "--group", RTZIPTARCMD_OPT_GROUP, RTGETOPT_REQ_STRING },
1689 { "--utc", RTZIPTARCMD_OPT_UTC, RTGETOPT_REQ_NOTHING },
1690 { "--sparse", 'S', RTGETOPT_REQ_NOTHING },
1691 { "--format", RTZIPTARCMD_OPT_FORMAT, RTGETOPT_REQ_STRING },
1692 { "--no-recursion", RTZIPTARCMD_OPT_NO_RECURSION, RTGETOPT_REQ_NOTHING },
1693
1694 /* IPRT extensions */
1695 { "--prefix", RTZIPTARCMD_OPT_PREFIX, RTGETOPT_REQ_STRING },
1696 { "--file-mode-and-mask", RTZIPTARCMD_OPT_FILE_MODE_AND_MASK, RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_OCT },
1697 { "--file-mode-or-mask", RTZIPTARCMD_OPT_FILE_MODE_OR_MASK, RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_OCT },
1698 { "--dir-mode-and-mask", RTZIPTARCMD_OPT_DIR_MODE_AND_MASK, RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_OCT },
1699 { "--dir-mode-or-mask", RTZIPTARCMD_OPT_DIR_MODE_OR_MASK, RTGETOPT_REQ_UINT32 | RTGETOPT_FLAG_OCT },
1700 { "--read-ahead", RTZIPTARCMD_OPT_READ_AHEAD, RTGETOPT_REQ_NOTHING },
1701 { "--use-push-file", RTZIPTARCMD_OPT_USE_PUSH_FILE, RTGETOPT_REQ_NOTHING },
1702 };
1703
1704 RTGETOPTSTATE GetState;
1705 int rc = RTGetOptInit(&GetState, cArgs, papszArgs, s_aOptions, RT_ELEMENTS(s_aOptions), 1,
1706 RTGETOPTINIT_FLAGS_OPTS_FIRST);
1707 if (RT_FAILURE(rc))
1708 return RTMsgErrorExitFailure("RTGetOpt failed: %Rrc", rc);
1709
1710 RTZIPTARCMDOPS Opts;
1711 RT_ZERO(Opts);
1712 Opts.enmFormat = RTZIPTARCMDFORMAT_AUTO_DEFAULT;
1713 Opts.uidOwner = NIL_RTUID;
1714 Opts.gidGroup = NIL_RTUID;
1715 Opts.fFileModeAndMask = RTFS_UNIX_ALL_ACCESS_PERMS;
1716 Opts.fDirModeAndMask = RTFS_UNIX_ALL_ACCESS_PERMS;
1717#if 0
1718 if (RTPermIsSuperUser())
1719 {
1720 Opts.fFileModeAndMask = RTFS_UNIX_ALL_PERMS;
1721 Opts.fDirModeAndMask = RTFS_UNIX_ALL_PERMS;
1722 Opts.fPreserveOwner = true;
1723 Opts.fPreserveGroup = true;
1724 }
1725#endif
1726 Opts.enmTarFormat = RTZIPTARFORMAT_DEFAULT;
1727 Opts.fRecursive = true; /* Recursion is implicit unless otherwise specified. */
1728
1729 RTGETOPTUNION ValueUnion;
1730 while ( (rc = RTGetOpt(&GetState, &ValueUnion)) != 0
1731 && rc != VINF_GETOPT_NOT_OPTION)
1732 {
1733 switch (rc)
1734 {
1735 /* operations */
1736 case 'A':
1737 case 'c':
1738 case 'd':
1739 case 'r':
1740 case 't':
1741 case 'u':
1742 case 'x':
1743 case RTZIPTARCMD_OPT_DELETE:
1744 if (Opts.iOperation)
1745 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Conflicting tar operation (%s already set, now %s)",
1746 Opts.pszOperation, ValueUnion.pDef->pszLong);
1747 Opts.iOperation = rc;
1748 Opts.pszOperation = ValueUnion.pDef->pszLong;
1749 break;
1750
1751 /* basic options */
1752 case 'C':
1753 if (Opts.pszDirectory)
1754 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "You may only specify -C/--directory once");
1755 Opts.pszDirectory = ValueUnion.psz;
1756 break;
1757
1758 case 'f':
1759 if (Opts.pszFile)
1760 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "You may only specify -f/--file once");
1761 Opts.pszFile = ValueUnion.psz;
1762 break;
1763
1764 case 'v':
1765 Opts.fVerbose = true;
1766 break;
1767
1768 case 'p':
1769 Opts.fFileModeAndMask = RTFS_UNIX_ALL_PERMS;
1770 Opts.fDirModeAndMask = RTFS_UNIX_ALL_PERMS;
1771 Opts.fPreserveOwner = true;
1772 Opts.fPreserveGroup = true;
1773 break;
1774
1775 case 'j':
1776 case 'z':
1777 if (Opts.chZipper)
1778 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "You may only specify one compressor / decompressor");
1779 Opts.chZipper = rc;
1780 break;
1781
1782 case RTZIPTARCMD_OPT_OWNER:
1783 if (Opts.pszOwner)
1784 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "You may only specify --owner once");
1785 Opts.pszOwner = ValueUnion.psz;
1786
1787 rc = RTStrToUInt32Full(Opts.pszOwner, 0, &ValueUnion.u32);
1788 if (RT_SUCCESS(rc) && rc != VINF_SUCCESS)
1789 return RTMsgErrorExit(RTEXITCODE_SYNTAX,
1790 "Error convering --owner '%s' into a number: %Rrc", Opts.pszOwner, rc);
1791 if (RT_SUCCESS(rc))
1792 {
1793 Opts.uidOwner = ValueUnion.u32;
1794 Opts.pszOwner = NULL;
1795 }
1796 break;
1797
1798 case RTZIPTARCMD_OPT_GROUP:
1799 if (Opts.pszGroup)
1800 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "You may only specify --group once");
1801 Opts.pszGroup = ValueUnion.psz;
1802
1803 rc = RTStrToUInt32Full(Opts.pszGroup, 0, &ValueUnion.u32);
1804 if (RT_SUCCESS(rc) && rc != VINF_SUCCESS)
1805 return RTMsgErrorExit(RTEXITCODE_SYNTAX,
1806 "Error convering --group '%s' into a number: %Rrc", Opts.pszGroup, rc);
1807 if (RT_SUCCESS(rc))
1808 {
1809 Opts.gidGroup = ValueUnion.u32;
1810 Opts.pszGroup = NULL;
1811 }
1812 break;
1813
1814 case RTZIPTARCMD_OPT_UTC:
1815 Opts.fDisplayUtc = true;
1816 break;
1817
1818 case RTZIPTARCMD_OPT_NO_RECURSION:
1819 Opts.fRecursive = false;
1820 break;
1821
1822 /* GNU */
1823 case 'S':
1824 Opts.fTarCreate |= RTZIPTAR_C_SPARSE;
1825 break;
1826
1827 /* iprt extensions */
1828 case RTZIPTARCMD_OPT_PREFIX:
1829 if (Opts.pszPrefix)
1830 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "You may only specify --prefix once");
1831 Opts.pszPrefix = ValueUnion.psz;
1832 break;
1833
1834 case RTZIPTARCMD_OPT_FILE_MODE_AND_MASK:
1835 Opts.fFileModeAndMask = ValueUnion.u32 & RTFS_UNIX_ALL_PERMS;
1836 break;
1837
1838 case RTZIPTARCMD_OPT_FILE_MODE_OR_MASK:
1839 Opts.fFileModeOrMask = ValueUnion.u32 & RTFS_UNIX_ALL_PERMS;
1840 break;
1841
1842 case RTZIPTARCMD_OPT_DIR_MODE_AND_MASK:
1843 Opts.fDirModeAndMask = ValueUnion.u32 & RTFS_UNIX_ALL_PERMS;
1844 break;
1845
1846 case RTZIPTARCMD_OPT_DIR_MODE_OR_MASK:
1847 Opts.fDirModeOrMask = ValueUnion.u32 & RTFS_UNIX_ALL_PERMS;
1848 break;
1849
1850 case RTZIPTARCMD_OPT_FORMAT:
1851 if (!strcmp(ValueUnion.psz, "auto") || !strcmp(ValueUnion.psz, "default"))
1852 {
1853 Opts.enmFormat = RTZIPTARCMDFORMAT_AUTO_DEFAULT;
1854 Opts.enmTarFormat = RTZIPTARFORMAT_DEFAULT;
1855 }
1856 else if (!strcmp(ValueUnion.psz, "tar"))
1857 {
1858 Opts.enmFormat = RTZIPTARCMDFORMAT_TAR;
1859 Opts.enmTarFormat = RTZIPTARFORMAT_DEFAULT;
1860 }
1861 else if (!strcmp(ValueUnion.psz, "gnu"))
1862 {
1863 Opts.enmFormat = RTZIPTARCMDFORMAT_TAR;
1864 Opts.enmTarFormat = RTZIPTARFORMAT_GNU;
1865 }
1866 else if (!strcmp(ValueUnion.psz, "ustar"))
1867 {
1868 Opts.enmFormat = RTZIPTARCMDFORMAT_TAR;
1869 Opts.enmTarFormat = RTZIPTARFORMAT_USTAR;
1870 }
1871 else if ( !strcmp(ValueUnion.psz, "posix")
1872 || !strcmp(ValueUnion.psz, "pax"))
1873 {
1874 Opts.enmFormat = RTZIPTARCMDFORMAT_TAR;
1875 Opts.enmTarFormat = RTZIPTARFORMAT_PAX;
1876 }
1877 else if (!strcmp(ValueUnion.psz, "xar"))
1878 Opts.enmFormat = RTZIPTARCMDFORMAT_XAR;
1879 else if (!strcmp(ValueUnion.psz, "cpio"))
1880 Opts.enmFormat = RTZIPTARCMDFORMAT_CPIO;
1881 else
1882 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Unknown archive format: '%s'", ValueUnion.psz);
1883 break;
1884
1885 case RTZIPTARCMD_OPT_READ_AHEAD:
1886 Opts.fReadAhead = true;
1887 break;
1888
1889 case RTZIPTARCMD_OPT_USE_PUSH_FILE:
1890 Opts.fUsePushFile = true;
1891 break;
1892
1893 /* Standard bits. */
1894 case 'h':
1895 rtZipTarUsage(RTPathFilename(papszArgs[0]));
1896 return RTEXITCODE_SUCCESS;
1897
1898 case 'V':
1899 RTPrintf("%sr%d\n", RTBldCfgVersion(), RTBldCfgRevision());
1900 return RTEXITCODE_SUCCESS;
1901
1902 default:
1903 return RTGetOptPrintError(rc, &ValueUnion);
1904 }
1905 }
1906
1907 if (rc == VINF_GETOPT_NOT_OPTION)
1908 {
1909 /* this is kind of ugly. */
1910 Assert((unsigned)GetState.iNext - 1 <= cArgs);
1911 Opts.papszFiles = (const char * const *)&papszArgs[GetState.iNext - 1];
1912 Opts.cFiles = cArgs - GetState.iNext + 1;
1913 }
1914
1915 if (!Opts.pszFile)
1916 return RTMsgErrorExitFailure("No archive specified");
1917
1918 /*
1919 * Post proceess the options.
1920 */
1921 if (Opts.iOperation == 0)
1922 {
1923 Opts.iOperation = 't';
1924 Opts.pszOperation = "--list";
1925 }
1926
1927 if ( Opts.iOperation == 'x'
1928 && Opts.pszOwner)
1929 return RTMsgErrorExitFailure("The use of --owner with %s has not implemented yet", Opts.pszOperation);
1930
1931 if ( Opts.iOperation == 'x'
1932 && Opts.pszGroup)
1933 return RTMsgErrorExitFailure("The use of --group with %s has not implemented yet", Opts.pszOperation);
1934
1935 /*
1936 * Do the job.
1937 */
1938 switch (Opts.iOperation)
1939 {
1940 case 't':
1941 return rtZipTarDoWithMembers(&Opts, rtZipTarCmdListCallback);
1942
1943 case 'x':
1944 return rtZipTarDoWithMembers(&Opts, rtZipTarCmdExtractCallback);
1945
1946 case 'c':
1947 return rtZipTarCreate(&Opts);
1948
1949 case 'A':
1950 case 'd':
1951 case 'r':
1952 case 'u':
1953 case RTZIPTARCMD_OPT_DELETE:
1954 return RTMsgErrorExitFailure("The operation %s is not implemented yet", Opts.pszOperation);
1955
1956 default:
1957 return RTMsgErrorExitFailure("Internal error");
1958 }
1959}
1960
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