VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxService/VBoxServiceToolBox.cpp@ 37423

Last change on this file since 37423 was 37405, checked in by vboxsync, 14 years ago

VBoxService/Toolbox: Update for ls (support symlinks).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 38.1 KB
Line 
1/* $Id: VBoxServiceToolBox.cpp 37405 2011-06-10 10:36:56Z vboxsync $ */
2/** @file
3 * VBoxServiceToolBox - Internal (BusyBox-like) toolbox.
4 */
5
6/*
7 * Copyright (C) 2011 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#include <stdio.h>
23
24#include <iprt/assert.h>
25#include <iprt/buildconfig.h>
26#include <iprt/dir.h>
27#include <iprt/file.h>
28#include <iprt/getopt.h>
29#include <iprt/list.h>
30#include <iprt/mem.h>
31#include <iprt/message.h>
32#include <iprt/path.h>
33#include <iprt/string.h>
34#include <iprt/stream.h>
35
36#ifndef RT_OS_WINDOWS
37#include <sys/stat.h>
38#endif
39
40#include <VBox/VBoxGuestLib.h>
41#include <VBox/version.h>
42#include "VBoxServiceInternal.h"
43#include "VBoxServiceUtils.h"
44
45
46#define CAT_OPT_NO_CONTENT_INDEXED 1000
47
48#define LS_OPT_MACHINE_READABLE 1000
49
50typedef enum VBOXSERVICETOOLBOXLSFLAG
51{
52 VBOXSERVICETOOLBOXLSFLAG_NONE = 0,
53 VBOXSERVICETOOLBOXLSFLAG_RECURSIVE = 0x00000001,
54 VBOXSERVICETOOLBOXLSFLAG_LONG = 0x00000002,
55 VBOXSERVICETOOLBOXLSFLAG_PARSEABLE = 0x00000004,
56 VBOXSERVICETOOLBOXLSFLAG_SYMLINKS = 0x00000008
57} VBOXSERVICETOOLBOXLSFLAG;
58
59/* Enable the following define to be able to debug/invoke the toolbox
60 * commandos directly from command line, e.g. VBoxService vbox_cat [args] */
61#ifdef DEBUG
62//# define VBOXSERVICE_TOOLBOX_DEBUG
63#endif
64
65/**
66 * An file/directory entry. Used to cache
67 * file names/paths for later processing.
68 */
69typedef struct VBOXSERVICETOOLBOXPATHENTRY
70{
71 /** Our node. */
72 RTLISTNODE Node;
73 /** Name of the entry. */
74 char *pszName;
75} VBOXSERVICETOOLBOXPATHENTRY, *PVBOXSERVICETOOLBOXPATHENTRY;
76
77typedef struct VBOXSERVICETOOLBOXDIRENTRY
78{
79 /** Our node. */
80 RTLISTNODE Node;
81 /** The actual entry. */
82 RTDIRENTRYEX dirEntry;
83} VBOXSERVICETOOLBOXDIRENTRY, *PVBOXSERVICETOOLBOXDIRENTRY;
84
85
86/**
87 * Displays a help text to stdout.
88 */
89static void VBoxServiceToolboxShowUsage(void)
90{
91 RTPrintf("Toolbox Usage:\n"
92 "cat [FILE] - Concatenate FILE(s), or standard input, to standard output.\n"
93 "\n"
94 /** @todo Document options! */
95 "ls [OPTION]... FILE... - List information about the FILEs (the current directory by default).\n"
96 "\n"
97 /** @todo Document options! */
98 "mkdir [OPTION]... DIRECTORY... - Create the DIRECTORY(ies), if they do not already exist.\n"
99 "\n"
100 /** @todo Document options! */
101 "stat [OPTION]... FILE... - Display file or file system status.\n"
102 "\n"
103 /** @todo Document options! */
104 "\n");
105}
106
107
108/**
109 * Displays the program's version number.
110 */
111static void VBoxServiceToolboxShowVersion(void)
112{
113 RTPrintf("%sr%d\n", VBOX_VERSION_STRING, RTBldCfgRevision());
114}
115
116
117/**
118 * Displays an error message because of syntax error.
119 *
120 * @return VERR_INVALID_PARAMETER
121 * @param pszFormat
122 */
123static int VBoxServiceToolboxErrorSyntax(const char *pszFormat, ...)
124{
125 va_list args;
126
127 va_start(args, pszFormat);
128 RTPrintf("\n"
129 "Syntax error: %N\n", pszFormat, &args);
130 va_end(args);
131 return VERR_INVALID_PARAMETER;
132}
133
134
135/**
136 * Destroys a path buffer list.
137 *
138 * @return IPRT status code.
139 * @param pList Pointer to list to destroy.
140 */
141static void VBoxServiceToolboxPathBufDestroy(PRTLISTNODE pList)
142{
143 AssertPtr(pList);
144 /** @todo use RTListForEachSafe */
145 PVBOXSERVICETOOLBOXPATHENTRY pNode = RTListGetFirst(pList, VBOXSERVICETOOLBOXPATHENTRY, Node);
146 while (pNode)
147 {
148 PVBOXSERVICETOOLBOXPATHENTRY pNext = RTListNodeIsLast(pList, &pNode->Node)
149 ? NULL
150 : RTListNodeGetNext(&pNode->Node, VBOXSERVICETOOLBOXPATHENTRY, Node);
151 RTListNodeRemove(&pNode->Node);
152
153 RTStrFree(pNode->pszName);
154
155 RTMemFree(pNode);
156 pNode = pNext;
157 }
158}
159
160
161/**
162 * Adds a path entry (file/directory/whatever) to a given path buffer list.
163 *
164 * @return IPRT status code.
165 * @param pList Pointer to list to add entry to.
166 * @param pszName Name of entry to add.
167 */
168static int VBoxServiceToolboxPathBufAddPathEntry(PRTLISTNODE pList, const char *pszName)
169{
170 AssertPtrReturn(pList, VERR_INVALID_PARAMETER);
171
172 int rc = VINF_SUCCESS;
173 PVBOXSERVICETOOLBOXPATHENTRY pNode = (PVBOXSERVICETOOLBOXPATHENTRY)RTMemAlloc(sizeof(VBOXSERVICETOOLBOXPATHENTRY));
174 if (pNode)
175 {
176 pNode->pszName = RTStrDup(pszName);
177 AssertPtr(pNode->pszName);
178
179 /*rc =*/ RTListAppend(pList, &pNode->Node);
180 }
181 else
182 rc = VERR_NO_MEMORY;
183 return rc;
184}
185
186
187/**
188 * Performs the actual output operation of "vbox_cat".
189 *
190 * @return IPRT status code.
191 * @param hInput Handle of input file (if any) to use;
192 * else stdin will be used.
193 * @param hOutput Handle of output file (if any) to use;
194 * else stdout will be used.
195 */
196static int VBoxServiceToolboxCatOutput(RTFILE hInput, RTFILE hOutput)
197{
198 int rc = VINF_SUCCESS;
199 if (hInput == NIL_RTFILE)
200 {
201 rc = RTFileFromNative(&hInput, RTFILE_NATIVE_STDIN);
202 if (RT_FAILURE(rc))
203 RTMsgError("cat: Could not translate input file to native handle, rc=%Rrc\n", rc);
204 }
205
206 if (hOutput == NIL_RTFILE)
207 {
208 rc = RTFileFromNative(&hOutput, RTFILE_NATIVE_STDOUT);
209 if (RT_FAILURE(rc))
210 RTMsgError("cat: Could not translate output file to native handle, rc=%Rrc\n", rc);
211 }
212
213 if (RT_SUCCESS(rc))
214 {
215 uint8_t abBuf[_64K];
216 size_t cbRead;
217 for (;;)
218 {
219 rc = RTFileRead(hInput, abBuf, sizeof(abBuf), &cbRead);
220 if (RT_SUCCESS(rc) && cbRead > 0)
221 {
222 rc = RTFileWrite(hOutput, abBuf, cbRead, NULL /* Try to write all at once! */);
223 cbRead = 0;
224 }
225 else
226 {
227 if (rc == VERR_BROKEN_PIPE)
228 rc = VINF_SUCCESS;
229 else if (RT_FAILURE(rc))
230 RTMsgError("cat: Error while reading input, rc=%Rrc\n", rc);
231 break;
232 }
233 }
234 }
235 return rc;
236}
237
238
239/**
240 * Main function for tool "vbox_cat".
241 *
242 * @return RTEXITCODE.
243 * @param argc Number of arguments.
244 * @param argv Pointer to argument array.
245 */
246static RTEXITCODE VBoxServiceToolboxCat(int argc, char **argv)
247{
248 static const RTGETOPTDEF s_aOptions[] =
249 {
250 /* Sorted by short ops. */
251 { "--show-all", 'a', RTGETOPT_REQ_NOTHING },
252 { "--number-nonblank", 'b', RTGETOPT_REQ_NOTHING},
253 { NULL, 'e', RTGETOPT_REQ_NOTHING},
254 { NULL, 'E', RTGETOPT_REQ_NOTHING},
255 { "--flags", 'f', RTGETOPT_REQ_STRING},
256 { "--no-content-indexed", CAT_OPT_NO_CONTENT_INDEXED, RTGETOPT_REQ_NOTHING},
257 { "--number", 'n', RTGETOPT_REQ_NOTHING},
258 { "--output", 'o', RTGETOPT_REQ_STRING},
259 { "--squeeze-blank", 's', RTGETOPT_REQ_NOTHING},
260 { NULL, 't', RTGETOPT_REQ_NOTHING},
261 { "--show-tabs", 'T', RTGETOPT_REQ_NOTHING},
262 { NULL, 'u', RTGETOPT_REQ_NOTHING},
263 { "--show-noneprinting", 'v', RTGETOPT_REQ_NOTHING}
264 };
265
266 int ch;
267 RTGETOPTUNION ValueUnion;
268 RTGETOPTSTATE GetState;
269
270 RTGetOptInit(&GetState, argc, argv,
271 s_aOptions, RT_ELEMENTS(s_aOptions),
272 /* Index of argv to start with. */
273#ifdef VBOXSERVICE_TOOLBOX_DEBUG
274 2,
275#else
276 1,
277#endif
278 0);
279
280 int rc = VINF_SUCCESS;
281 bool fUsageOK = true;
282
283 char szOutput[RTPATH_MAX] = { 0 };
284 RTFILE hOutput = NIL_RTFILE;
285 uint32_t fFlags = RTFILE_O_CREATE_REPLACE /* Output file flags. */
286 | RTFILE_O_WRITE
287 | RTFILE_O_DENY_WRITE;
288
289 /* Init directory list. */
290 RTLISTNODE inputList;
291 RTListInit(&inputList);
292
293 while ( (ch = RTGetOpt(&GetState, &ValueUnion))
294 && RT_SUCCESS(rc))
295 {
296 /* For options that require an argument, ValueUnion has received the value. */
297 switch (ch)
298 {
299 case 'a':
300 case 'b':
301 case 'e':
302 case 'E':
303 case 'n':
304 case 's':
305 case 't':
306 case 'T':
307 case 'v':
308 RTMsgError("cat: Sorry, option '%s' is not implemented yet!\n",
309 ValueUnion.pDef->pszLong);
310 rc = VERR_INVALID_PARAMETER;
311 break;
312
313 case 'h':
314 VBoxServiceToolboxShowUsage();
315 return RTEXITCODE_SUCCESS;
316
317 case 'o':
318 if (!RTStrPrintf(szOutput, sizeof(szOutput), ValueUnion.psz))
319 rc = VERR_NO_MEMORY;
320 break;
321
322 case 'u':
323 /* Ignored. */
324 break;
325
326 case 'V':
327 VBoxServiceToolboxShowVersion();
328 return RTEXITCODE_SUCCESS;
329
330 case CAT_OPT_NO_CONTENT_INDEXED:
331 fFlags |= RTFILE_O_NOT_CONTENT_INDEXED;
332 break;
333
334 case VINF_GETOPT_NOT_OPTION:
335 {
336 /* Add file(s) to buffer. This enables processing multiple paths
337 * at once.
338 *
339 * Since the non-options (RTGETOPTINIT_FLAGS_OPTS_FIRST) come last when
340 * processing this loop it's safe to immediately exit on syntax errors
341 * or showing the help text (see above). */
342 rc = VBoxServiceToolboxPathBufAddPathEntry(&inputList, ValueUnion.psz);
343 break;
344 }
345
346 default:
347 return RTGetOptPrintError(ch, &ValueUnion);
348 }
349 }
350
351 if (RT_SUCCESS(rc))
352 {
353 if (strlen(szOutput))
354 {
355 rc = RTFileOpen(&hOutput, szOutput, fFlags);
356 if (RT_FAILURE(rc))
357 RTMsgError("cat: Could not create output file '%s', rc=%Rrc\n",
358 szOutput, rc);
359 }
360
361 if (RT_SUCCESS(rc))
362 {
363 /* Process each input file. */
364 PVBOXSERVICETOOLBOXPATHENTRY pNodeIt;
365 RTFILE hInput = NIL_RTFILE;
366 RTListForEach(&inputList, pNodeIt, VBOXSERVICETOOLBOXPATHENTRY, Node)
367 {
368 rc = RTFileOpen(&hInput, pNodeIt->pszName,
369 RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_WRITE);
370 if (RT_SUCCESS(rc))
371 {
372 rc = VBoxServiceToolboxCatOutput(hInput, hOutput);
373 RTFileClose(hInput);
374 }
375 else
376 {
377 PCRTSTATUSMSG pMsg = RTErrGet(rc);
378 if (pMsg)
379 RTMsgError("cat: Could not open input file '%s': %s\n",
380 pNodeIt->pszName, pMsg->pszMsgFull);
381 else
382 RTMsgError("cat: Could not open input file '%s', rc=%Rrc\n", pNodeIt->pszName, rc);
383 }
384
385 if (RT_FAILURE(rc))
386 break;
387 }
388
389 /* If not input files were defined, process stdin. */
390 if (RTListNodeIsFirst(&inputList, &inputList))
391 rc = VBoxServiceToolboxCatOutput(hInput, hOutput);
392 }
393 }
394
395 if (hOutput != NIL_RTFILE)
396 RTFileClose(hOutput);
397 VBoxServiceToolboxPathBufDestroy(&inputList);
398
399 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
400}
401
402/**
403 * Helper routine for ls tool doing the actual parsing and output of
404 * a specified directory.
405 *
406 * @return IPRT status code.
407 * @param pszDir Directory (path) to ouptut.
408 * @param uFlags Flags of type VBOXSERVICETOOLBOXLSFLAG.
409 */
410static int VBoxServiceToolboxLsOutput(const char *pszDir, uint32_t uFlags)
411{
412 AssertPtrReturn(pszDir, VERR_INVALID_PARAMETER);
413
414 if (uFlags & VBOXSERVICETOOLBOXLSFLAG_PARSEABLE)
415 RTPrintf("dname=%s%c", pszDir, 0);
416 else if (uFlags & VBOXSERVICETOOLBOXLSFLAG_RECURSIVE)
417 RTPrintf("%s:\n", pszDir);
418
419 char szPathAbs[RTPATH_MAX + 1];
420 int rc = RTPathAbs(pszDir, szPathAbs, sizeof(szPathAbs));
421 if (RT_FAILURE(rc))
422 {
423 RTMsgError("ls: Failed to retrieve absolute path of '%s', rc=%Rrc\n", pszDir, rc);
424 return rc;
425 }
426
427 PRTDIR pDir;
428 rc = RTDirOpen(&pDir, szPathAbs);
429 if (RT_FAILURE(rc))
430 {
431 RTMsgError("ls: Failed to open directory '%s', rc=%Rrc\n", szPathAbs, rc);
432 return rc;
433 }
434
435 RTLISTNODE dirList;
436 RTListInit(&dirList);
437
438 /* To prevent races we need to read in the directory entries once
439 * and process them afterwards: First loop is displaying the current
440 * directory's content and second loop is diving deeper into
441 * sub directories (if wanted). */
442 for (;RT_SUCCESS(rc);)
443 {
444 RTDIRENTRYEX DirEntry;
445 rc = RTDirReadEx(pDir, &DirEntry, NULL, RTFSOBJATTRADD_UNIX, RTPATH_F_ON_LINK);
446 if (RT_SUCCESS(rc))
447 {
448 PVBOXSERVICETOOLBOXDIRENTRY pNode = (PVBOXSERVICETOOLBOXDIRENTRY)RTMemAlloc(sizeof(VBOXSERVICETOOLBOXDIRENTRY));
449 if (pNode)
450 {
451 memcpy(&pNode->dirEntry, &DirEntry, sizeof(RTDIRENTRYEX));
452 /*rc =*/ RTListAppend(&dirList, &pNode->Node);
453 }
454 else
455 rc = VERR_NO_MEMORY;
456 }
457 }
458
459 if (rc == VERR_NO_MORE_FILES)
460 rc = VINF_SUCCESS;
461
462 int rc2 = RTDirClose(pDir);
463 if (RT_FAILURE(rc2))
464 {
465 RTMsgError("ls: Failed to close dir '%s', rc=%Rrc\n",
466 pszDir, rc2);
467 if (RT_SUCCESS(rc))
468 rc = rc2;
469 }
470
471 if (RT_SUCCESS(rc))
472 {
473 PVBOXSERVICETOOLBOXDIRENTRY pNodeIt;
474 RTListForEach(&dirList, pNodeIt, VBOXSERVICETOOLBOXDIRENTRY, Node)
475 {
476 RTFMODE fMode = pNodeIt->dirEntry.Info.Attr.fMode;
477 char cFileType;
478 switch (fMode & RTFS_TYPE_MASK)
479 {
480 case RTFS_TYPE_FIFO: cFileType = 'f'; break;
481 case RTFS_TYPE_DEV_CHAR: cFileType = 'c'; break;
482 case RTFS_TYPE_DIRECTORY: cFileType = 'd'; break;
483 case RTFS_TYPE_DEV_BLOCK: cFileType = 'b'; break;
484 case RTFS_TYPE_FILE: cFileType = '-'; break;
485 case RTFS_TYPE_SYMLINK: cFileType = 'l'; break;
486 case RTFS_TYPE_SOCKET: cFileType = 's'; break;
487 case RTFS_TYPE_WHITEOUT: cFileType = 'w'; break;
488 default:
489 cFileType = '?';
490 break;
491 }
492 /** @todo sticy bits++ */
493
494 if (!(uFlags & VBOXSERVICETOOLBOXLSFLAG_LONG))
495 {
496 if (uFlags & VBOXSERVICETOOLBOXLSFLAG_PARSEABLE)
497 {
498 /** @todo Skip node_id if not present/available! */
499 RTPrintf("ftype=%c%cnode_id=%RU64%cname_len=%RU16%cname=%s%c",
500 cFileType, 0, (uint64_t)pNodeIt->dirEntry.Info.Attr.u.Unix.INodeId, 0,
501 pNodeIt->dirEntry.cbName, 0, pNodeIt->dirEntry.szName, 0);
502 }
503 else
504 RTPrintf("%c %#18llx %3d %s\n", (uint64_t)pNodeIt->dirEntry.Info.Attr.u.Unix.INodeId,
505 cFileType, pNodeIt->dirEntry.cbName, pNodeIt->dirEntry.szName);
506
507 if (uFlags & VBOXSERVICETOOLBOXLSFLAG_PARSEABLE) /* End of data block. */
508 RTPrintf("%c%c", 0, 0);
509 }
510 else
511 {
512 if (uFlags & VBOXSERVICETOOLBOXLSFLAG_PARSEABLE)
513 {
514 RTPrintf("ftype=%c%c", cFileType, 0);
515 RTPrintf("owner_mask=%c%c%c%c",
516 fMode & RTFS_UNIX_IRUSR ? 'r' : '-',
517 fMode & RTFS_UNIX_IWUSR ? 'w' : '-',
518 fMode & RTFS_UNIX_IXUSR ? 'x' : '-', 0);
519 RTPrintf("group_mask=%c%c%c%c",
520 fMode & RTFS_UNIX_IRGRP ? 'r' : '-',
521 fMode & RTFS_UNIX_IWGRP ? 'w' : '-',
522 fMode & RTFS_UNIX_IXGRP ? 'x' : '-', 0);
523 RTPrintf("other_mask=%c%c%c%c",
524 fMode & RTFS_UNIX_IROTH ? 'r' : '-',
525 fMode & RTFS_UNIX_IWOTH ? 'w' : '-',
526 fMode & RTFS_UNIX_IXOTH ? 'x' : '-', 0);
527 RTPrintf("dos_mask=%c%c%c%c%c%c%c%c%c%c%c%c%c%c%c",
528 fMode & RTFS_DOS_READONLY ? 'R' : '-',
529 fMode & RTFS_DOS_HIDDEN ? 'H' : '-',
530 fMode & RTFS_DOS_SYSTEM ? 'S' : '-',
531 fMode & RTFS_DOS_DIRECTORY ? 'D' : '-',
532 fMode & RTFS_DOS_ARCHIVED ? 'A' : '-',
533 fMode & RTFS_DOS_NT_DEVICE ? 'd' : '-',
534 fMode & RTFS_DOS_NT_NORMAL ? 'N' : '-',
535 fMode & RTFS_DOS_NT_TEMPORARY ? 'T' : '-',
536 fMode & RTFS_DOS_NT_SPARSE_FILE ? 'P' : '-',
537 fMode & RTFS_DOS_NT_REPARSE_POINT ? 'J' : '-',
538 fMode & RTFS_DOS_NT_COMPRESSED ? 'C' : '-',
539 fMode & RTFS_DOS_NT_OFFLINE ? 'O' : '-',
540 fMode & RTFS_DOS_NT_NOT_CONTENT_INDEXED ? 'I' : '-',
541 fMode & RTFS_DOS_NT_ENCRYPTED ? 'E' : '-', 0);
542
543 char szTimeBirth[256];
544 RTTimeSpecToString(&pNodeIt->dirEntry.Info.BirthTime, szTimeBirth, sizeof(szTimeBirth));
545 char szTimeChange[256];
546 RTTimeSpecToString(&pNodeIt->dirEntry.Info.ChangeTime, szTimeChange, sizeof(szTimeChange));
547 char szTimeModification[256];
548 RTTimeSpecToString(&pNodeIt->dirEntry.Info.ModificationTime, szTimeModification, sizeof(szTimeModification));
549 char szTimeAccess[256];
550 RTTimeSpecToString(&pNodeIt->dirEntry.Info.AccessTime, szTimeAccess, sizeof(szTimeAccess));
551
552 RTPrintf("hlinks=%RU32%cuid=%RU32%cgid=%RU32%cst_size=%RI64%calloc=%RI64%c"
553 "st_birthtime=%s%cst_ctime=%s%cst_mtime=%s%cst_atime=%s%c",
554 pNodeIt->dirEntry.Info.Attr.u.Unix.cHardlinks, 0,
555 pNodeIt->dirEntry.Info.Attr.u.Unix.uid, 0,
556 pNodeIt->dirEntry.Info.Attr.u.Unix.gid, 0,
557 pNodeIt->dirEntry.Info.cbObject, 0,
558 pNodeIt->dirEntry.Info.cbAllocated, 0,
559 szTimeBirth, 0,
560 szTimeChange, 0,
561 szTimeModification, 0,
562 szTimeAccess, 0);
563 RTPrintf("cname_len=%RU16%cname=%s%c",
564 pNodeIt->dirEntry.cbName, 0, pNodeIt->dirEntry.szName, 0);
565
566 /* End of data block. */
567 RTPrintf("%c%c", 0, 0);
568 }
569 else
570 {
571 RTPrintf("%c", cFileType);
572 RTPrintf("%c%c%c",
573 fMode & RTFS_UNIX_IRUSR ? 'r' : '-',
574 fMode & RTFS_UNIX_IWUSR ? 'w' : '-',
575 fMode & RTFS_UNIX_IXUSR ? 'x' : '-');
576 RTPrintf("%c%c%c",
577 fMode & RTFS_UNIX_IRGRP ? 'r' : '-',
578 fMode & RTFS_UNIX_IWGRP ? 'w' : '-',
579 fMode & RTFS_UNIX_IXGRP ? 'x' : '-');
580 RTPrintf("%c%c%c",
581 fMode & RTFS_UNIX_IROTH ? 'r' : '-',
582 fMode & RTFS_UNIX_IWOTH ? 'w' : '-',
583 fMode & RTFS_UNIX_IXOTH ? 'x' : '-');
584 RTPrintf(" %c%c%c%c%c%c%c%c%c%c%c%c%c%c",
585 fMode & RTFS_DOS_READONLY ? 'R' : '-',
586 fMode & RTFS_DOS_HIDDEN ? 'H' : '-',
587 fMode & RTFS_DOS_SYSTEM ? 'S' : '-',
588 fMode & RTFS_DOS_DIRECTORY ? 'D' : '-',
589 fMode & RTFS_DOS_ARCHIVED ? 'A' : '-',
590 fMode & RTFS_DOS_NT_DEVICE ? 'd' : '-',
591 fMode & RTFS_DOS_NT_NORMAL ? 'N' : '-',
592 fMode & RTFS_DOS_NT_TEMPORARY ? 'T' : '-',
593 fMode & RTFS_DOS_NT_SPARSE_FILE ? 'P' : '-',
594 fMode & RTFS_DOS_NT_REPARSE_POINT ? 'J' : '-',
595 fMode & RTFS_DOS_NT_COMPRESSED ? 'C' : '-',
596 fMode & RTFS_DOS_NT_OFFLINE ? 'O' : '-',
597 fMode & RTFS_DOS_NT_NOT_CONTENT_INDEXED ? 'I' : '-',
598 fMode & RTFS_DOS_NT_ENCRYPTED ? 'E' : '-');
599 RTPrintf(" %d %4d %4d %10lld %10lld %#llx %#llx %#llx %#llx",
600 pNodeIt->dirEntry.Info.Attr.u.Unix.cHardlinks,
601 pNodeIt->dirEntry.Info.Attr.u.Unix.uid,
602 pNodeIt->dirEntry.Info.Attr.u.Unix.gid,
603 pNodeIt->dirEntry.Info.cbObject,
604 pNodeIt->dirEntry.Info.cbAllocated,
605 pNodeIt->dirEntry.Info.BirthTime,
606 pNodeIt->dirEntry.Info.ChangeTime,
607 pNodeIt->dirEntry.Info.ModificationTime,
608 pNodeIt->dirEntry.Info.AccessTime);
609 RTPrintf(" %2d %s\n", pNodeIt->dirEntry.cbName, pNodeIt->dirEntry.szName);
610 }
611 }
612 if (RT_FAILURE(rc))
613 break;
614 }
615
616 /* If everything went fine we do the second run (if needed) ... */
617 if ( RT_SUCCESS(rc)
618 && (uFlags & VBOXSERVICETOOLBOXLSFLAG_RECURSIVE))
619 {
620 /* Process all sub-directories. */
621 PVBOXSERVICETOOLBOXDIRENTRY pNodeIt;
622 RTListForEach(&dirList, pNodeIt, VBOXSERVICETOOLBOXDIRENTRY, Node)
623 {
624 RTFMODE fMode = pNodeIt->dirEntry.Info.Attr.fMode;
625 switch (fMode & RTFS_TYPE_MASK)
626 {
627 case RTFS_TYPE_SYMLINK:
628 if (!(uFlags & VBOXSERVICETOOLBOXLSFLAG_SYMLINKS))
629 break;
630 /* Fall through is intentional. */
631 case RTFS_TYPE_DIRECTORY:
632 {
633 const char *pszName = pNodeIt->dirEntry.szName;
634 if ( !RTStrICmp(pszName, ".")
635 || !RTStrICmp(pszName, ".."))
636 {
637 /* Skip dot directories. */
638 continue;
639 }
640
641 char szPath[RTPATH_MAX];
642 rc = RTPathJoin(szPath, sizeof(szPath),
643 pszDir, pNodeIt->dirEntry.szName);
644 if (RT_SUCCESS(rc))
645 rc = VBoxServiceToolboxLsOutput(szPath, uFlags);
646 }
647 break;
648
649 default: /* Ignore the rest. */
650 break;
651 }
652 if (RT_FAILURE(rc))
653 break;
654 }
655 }
656 }
657
658 /* Clean up the mess. */
659 PVBOXSERVICETOOLBOXDIRENTRY pNode, pSafe;
660 RTListForEachSafe(&dirList, pNode, pSafe, VBOXSERVICETOOLBOXDIRENTRY, Node)
661 {
662 RTListNodeRemove(&pNode->Node);
663 RTMemFree(pNode);
664 }
665 return rc;
666}
667
668
669/**
670 * Main function for tool "vbox_ls".
671 *
672 * @return RTEXITCODE.
673 * @param argc Number of arguments.
674 * @param argv Pointer to argument array.
675 */
676static RTEXITCODE VBoxServiceToolboxLs(int argc, char **argv)
677{
678 static const RTGETOPTDEF s_aOptions[] =
679 {
680 { "--machinereadable", LS_OPT_MACHINE_READABLE, RTGETOPT_REQ_NOTHING },
681 { "--dereference", 'L', RTGETOPT_REQ_NOTHING },
682 { NULL, 'l', RTGETOPT_REQ_NOTHING },
683 { NULL, 'R', RTGETOPT_REQ_NOTHING },
684 { "--verbose", 'v', RTGETOPT_REQ_NOTHING}
685 };
686
687 int ch;
688 RTGETOPTUNION ValueUnion;
689 RTGETOPTSTATE GetState;
690 RTGetOptInit(&GetState, argc, argv,
691 s_aOptions, RT_ELEMENTS(s_aOptions),
692 /* Index of argv to start with. */
693#ifdef VBOXSERVICE_TOOLBOX_DEBUG
694 2,
695#else
696 1,
697#endif
698 RTGETOPTINIT_FLAGS_OPTS_FIRST);
699
700 int rc = VINF_SUCCESS;
701 bool fVerbose = false;
702 uint32_t fFlags = VBOXSERVICETOOLBOXLSFLAG_NONE;
703
704 /* Init file list. */
705 RTLISTNODE fileList;
706 RTListInit(&fileList);
707
708 while ( (ch = RTGetOpt(&GetState, &ValueUnion))
709 && RT_SUCCESS(rc))
710 {
711 /* For options that require an argument, ValueUnion has received the value. */
712 switch (ch)
713 {
714 case 'h':
715 VBoxServiceToolboxShowUsage();
716 return RTEXITCODE_SUCCESS;
717
718 case 'L': /* Dereference symlinks. */
719 fFlags |= VBOXSERVICETOOLBOXLSFLAG_SYMLINKS;
720 break;
721
722 case 'l': /* Print long format. */
723 fFlags |= VBOXSERVICETOOLBOXLSFLAG_LONG;
724 break;
725
726 case LS_OPT_MACHINE_READABLE:
727 fFlags |= VBOXSERVICETOOLBOXLSFLAG_PARSEABLE;
728 break;
729
730 case 'R': /* Recursive processing. */
731 fFlags |= VBOXSERVICETOOLBOXLSFLAG_RECURSIVE;
732 break;
733
734 case 'v':
735 fVerbose = true;
736 break;
737
738 case 'V':
739 VBoxServiceToolboxShowVersion();
740 return RTEXITCODE_SUCCESS;
741
742 case VINF_GETOPT_NOT_OPTION:
743 {
744 /* Add file(s) to buffer. This enables processing multiple files
745 * at once.
746 *
747 * Since the non-options (RTGETOPTINIT_FLAGS_OPTS_FIRST) come last when
748 * processing this loop it's safe to immediately exit on syntax errors
749 * or showing the help text (see above). */
750 rc = VBoxServiceToolboxPathBufAddPathEntry(&fileList, ValueUnion.psz);
751 break;
752 }
753
754 default:
755 return RTGetOptPrintError(ch, &ValueUnion);
756 }
757 }
758
759 if (RT_SUCCESS(rc))
760 {
761 /* If not files given add current directory to list. */
762 if (RTListIsEmpty(&fileList))
763 {
764 char szDirCur[RTPATH_MAX + 1];
765 rc = RTPathGetCurrent(szDirCur, sizeof(szDirCur));
766 if (RT_SUCCESS(rc))
767 {
768 rc = VBoxServiceToolboxPathBufAddPathEntry(&fileList, szDirCur);
769 if (RT_FAILURE(rc))
770 RTMsgError("ls: Adding current directory failed, rc=%Rrc\n", rc);
771 }
772 else
773 RTMsgError("ls: Getting current directory failed, rc=%Rrc\n", rc);
774 }
775
776 /* Print magic/version. */
777 if (fFlags & VBOXSERVICETOOLBOXLSFLAG_PARSEABLE)
778 RTPrintf("hdr_id=vbt_ls%chdr_ver=%u%c", 0, 1 /* Version 1 */, 0);
779
780 PVBOXSERVICETOOLBOXPATHENTRY pNodeIt;
781 RTListForEach(&fileList, pNodeIt, VBOXSERVICETOOLBOXPATHENTRY, Node)
782 {
783 rc = VBoxServiceToolboxLsOutput(pNodeIt->pszName, fFlags);
784 if (RT_FAILURE(rc))
785 RTMsgError("ls: Failed while enumerating directory '%s', rc=%Rrc\n",
786 pNodeIt->pszName, rc);
787 }
788
789 if (fFlags & VBOXSERVICETOOLBOXLSFLAG_PARSEABLE) /* Output termination. */
790 RTPrintf("%c%c%c%c", 0, 0, 0, 0);
791 }
792 else if (fVerbose)
793 RTMsgError("ls: Failed with rc=%Rrc\n", rc);
794
795 VBoxServiceToolboxPathBufDestroy(&fileList);
796 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
797}
798
799
800/**
801 * Main function for tool "vbox_mkdir".
802 *
803 * @return RTEXITCODE.
804 * @param argc Number of arguments.
805 * @param argv Pointer to argument array.
806 */
807static RTEXITCODE VBoxServiceToolboxMkDir(int argc, char **argv)
808{
809 static const RTGETOPTDEF s_aOptions[] =
810 {
811 { "--mode", 'm', RTGETOPT_REQ_STRING },
812 { "--parents", 'p', RTGETOPT_REQ_NOTHING},
813 { "--verbose", 'v', RTGETOPT_REQ_NOTHING}
814 };
815
816 int ch;
817 RTGETOPTUNION ValueUnion;
818 RTGETOPTSTATE GetState;
819 RTGetOptInit(&GetState, argc, argv,
820 s_aOptions, RT_ELEMENTS(s_aOptions),
821 /* Index of argv to start with. */
822#ifdef VBOXSERVICE_TOOLBOX_DEBUG
823 2,
824#else
825 1,
826#endif
827 RTGETOPTINIT_FLAGS_OPTS_FIRST);
828
829 int rc = VINF_SUCCESS;
830 bool fMakeParentDirs = false;
831 bool fVerbose = false;
832
833 RTFMODE newMode = 0;
834 RTFMODE dirMode = RTFS_UNIX_IRWXU | RTFS_UNIX_IRWXG | RTFS_UNIX_IRWXO;
835
836 /* Init directory list. */
837 RTLISTNODE dirList;
838 RTListInit(&dirList);
839
840 while ( (ch = RTGetOpt(&GetState, &ValueUnion))
841 && RT_SUCCESS(rc))
842 {
843 /* For options that require an argument, ValueUnion has received the value. */
844 switch (ch)
845 {
846 case 'h':
847 VBoxServiceToolboxShowUsage();
848 return RTEXITCODE_SUCCESS;
849
850 case 'p':
851 fMakeParentDirs = true;
852 break;
853
854 case 'm':
855 rc = RTStrToUInt32Ex(ValueUnion.psz, NULL, 8 /* Base */, &newMode);
856 if (RT_FAILURE(rc)) /* Only octet based values supported right now! */
857 {
858 RTMsgError("mkdir: Mode flag strings not implemented yet! Use octal numbers instead.\n");
859 return RTEXITCODE_SYNTAX;
860 }
861 break;
862
863 case 'v':
864 fVerbose = true;
865 break;
866
867 case 'V':
868 VBoxServiceToolboxShowVersion();
869 return RTEXITCODE_SUCCESS;
870
871 case VINF_GETOPT_NOT_OPTION:
872 {
873 /* Add path(s) to buffer. This enables processing multiple paths
874 * at once.
875 *
876 * Since the non-options (RTGETOPTINIT_FLAGS_OPTS_FIRST) come last when
877 * processing this loop it's safe to immediately exit on syntax errors
878 * or showing the help text (see above). */
879 rc = VBoxServiceToolboxPathBufAddPathEntry(&dirList, ValueUnion.psz);
880 break;
881 }
882
883 default:
884 return RTGetOptPrintError(ch, &ValueUnion);
885 }
886 }
887
888 if (RT_SUCCESS(rc))
889 {
890 if (fMakeParentDirs || newMode)
891 {
892#ifndef RT_OS_WINDOWS
893 mode_t umaskMode = umask(0); /* Get current umask. */
894 if (newMode)
895 dirMode = newMode;
896#endif
897 }
898
899 PVBOXSERVICETOOLBOXPATHENTRY pNodeIt;
900 RTListForEach(&dirList, pNodeIt, VBOXSERVICETOOLBOXPATHENTRY, Node)
901 {
902 rc = fMakeParentDirs ?
903 RTDirCreateFullPath(pNodeIt->pszName, dirMode)
904 : RTDirCreate(pNodeIt->pszName, dirMode);
905
906 if (RT_SUCCESS(rc) && fVerbose)
907 RTMsgError("mkdir: Created directory 's', mode %#RTfmode\n", pNodeIt->pszName, dirMode);
908 else if (RT_FAILURE(rc)) /** @todo Add a switch with more helpful error texts! */
909 {
910 PCRTSTATUSMSG pMsg = RTErrGet(rc);
911 if (pMsg)
912 RTMsgError("mkdir: Could not create directory '%s': %s\n",
913 pNodeIt->pszName, pMsg->pszMsgFull);
914 else
915 RTMsgError("mkdir: Could not create directory '%s', rc=%Rrc\n", pNodeIt->pszName, rc);
916 break;
917 }
918 }
919 }
920 else if (fVerbose)
921 RTMsgError("mkdir: Failed with rc=%Rrc\n", rc);
922
923 VBoxServiceToolboxPathBufDestroy(&dirList);
924 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
925}
926
927
928/**
929 * Main function for tool "vbox_stat".
930 *
931 * @return RTEXITCODE.
932 * @param argc Number of arguments.
933 * @param argv Pointer to argument array.
934 */
935static RTEXITCODE VBoxServiceToolboxStat(int argc, char **argv)
936{
937 static const RTGETOPTDEF s_aOptions[] =
938 {
939 { "--file-system", 'f', RTGETOPT_REQ_NOTHING },
940 { "--dereference", 'L', RTGETOPT_REQ_NOTHING},
941 { "--terse", 't', RTGETOPT_REQ_NOTHING},
942 { "--verbose", 'v', RTGETOPT_REQ_NOTHING}
943 };
944
945 int ch;
946 RTGETOPTUNION ValueUnion;
947 RTGETOPTSTATE GetState;
948 RTGetOptInit(&GetState, argc, argv,
949 s_aOptions, RT_ELEMENTS(s_aOptions),
950 /* Index of argv to start with. */
951#ifdef VBOXSERVICE_TOOLBOX_DEBUG
952 2,
953#else
954 1,
955#endif
956 RTGETOPTINIT_FLAGS_OPTS_FIRST);
957
958 int rc = VINF_SUCCESS;
959 bool fVerbose = false;
960
961 /* Init file list. */
962 RTLISTNODE fileList;
963 RTListInit(&fileList);
964
965 while ( (ch = RTGetOpt(&GetState, &ValueUnion))
966 && RT_SUCCESS(rc))
967 {
968 /* For options that require an argument, ValueUnion has received the value. */
969 switch (ch)
970 {
971 case 'h':
972 VBoxServiceToolboxShowUsage();
973 return RTEXITCODE_SUCCESS;
974
975 case 'f':
976 case 'L':
977 RTMsgError("stat: Sorry, option '%s' is not implemented yet!\n",
978 ValueUnion.pDef->pszLong);
979 rc = VERR_INVALID_PARAMETER;
980 break;
981
982 case 'v':
983 fVerbose = true;
984 break;
985
986 case 'V':
987 VBoxServiceToolboxShowVersion();
988 return RTEXITCODE_SUCCESS;
989
990 case VINF_GETOPT_NOT_OPTION:
991 {
992 /* Add file(s) to buffer. This enables processing multiple files
993 * at once.
994 *
995 * Since the non-options (RTGETOPTINIT_FLAGS_OPTS_FIRST) come last when
996 * processing this loop it's safe to immediately exit on syntax errors
997 * or showing the help text (see above). */
998 rc = VBoxServiceToolboxPathBufAddPathEntry(&fileList, ValueUnion.psz);
999 break;
1000 }
1001
1002 default:
1003 return RTGetOptPrintError(ch, &ValueUnion);
1004 }
1005 }
1006
1007 if (RT_SUCCESS(rc))
1008 {
1009 PVBOXSERVICETOOLBOXPATHENTRY pNodeIt;
1010 RTListForEach(&fileList, pNodeIt, VBOXSERVICETOOLBOXPATHENTRY, Node)
1011 {
1012 /* Only check for file existence for now. */
1013 if (RTFileExists(pNodeIt->pszName))
1014 {
1015 /** @todo Do some more work (query size etc.) here later.
1016 * Not needed for now. */
1017 }
1018 else
1019 {
1020 RTMsgError("stat: Cannot stat for '%s': No such file or directory\n",
1021 pNodeIt->pszName);
1022 rc = VERR_FILE_NOT_FOUND;
1023 /* Do not break here -- process every file in the list
1024 * and keep failing rc. */
1025 }
1026 }
1027
1028 if (RTListIsEmpty(&fileList))
1029 RTMsgError("stat: Missing operand\n");
1030 }
1031 else if (fVerbose)
1032 RTMsgError("stat: Failed with rc=%Rrc\n", rc);
1033
1034 VBoxServiceToolboxPathBufDestroy(&fileList);
1035 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1036}
1037
1038
1039/**
1040 * Entry point for internal toolbox.
1041 *
1042 * @return True if an internal tool was handled, false if not.
1043 * @param argc Number of arguments.
1044 * @param argv Pointer to argument array.
1045 * @param prcExit Where to store the exit code when an
1046 * internal toolbox command was handled.
1047 */
1048bool VBoxServiceToolboxMain(int argc, char **argv, RTEXITCODE *prcExit)
1049{
1050 if (argc > 0) /* Do we have at least a main command? */
1051 {
1052 int iCmdIdx = 0;
1053#ifdef VBOXSERVICE_TOOLBOX_DEBUG
1054 iCmdIdx = 1;
1055#endif
1056 if ( !strcmp(argv[iCmdIdx], "cat")
1057 || !strcmp(argv[iCmdIdx], "vbox_cat"))
1058 {
1059 *prcExit = VBoxServiceToolboxCat(argc, argv);
1060 return true;
1061 }
1062
1063 if ( !strcmp(argv[iCmdIdx], "ls")
1064 || !strcmp(argv[iCmdIdx], "vbox_ls"))
1065 {
1066 *prcExit = VBoxServiceToolboxLs(argc, argv);
1067 return true;
1068 }
1069
1070 if ( !strcmp(argv[iCmdIdx], "mkdir")
1071 || !strcmp(argv[iCmdIdx], "vbox_mkdir"))
1072 {
1073 *prcExit = VBoxServiceToolboxMkDir(argc, argv);
1074 return true;
1075 }
1076
1077 if ( !strcmp(argv[iCmdIdx], "stat")
1078 || !strcmp(argv[iCmdIdx], "vbox_stat"))
1079 {
1080 *prcExit = VBoxServiceToolboxStat(argc, argv);
1081 return true;
1082 }
1083 }
1084 return false;
1085}
1086
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