VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxBugReport/VBoxBugReport.cpp@ 60818

Last change on this file since 60818 was 60613, checked in by vboxsync, 9 years ago

VBoxBugReport: use IVirtualBoxClient for getting IVirtualBox reference, it's making things simpler in the future

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 21.1 KB
Line 
1/* $Id: VBoxBugReport.cpp 60613 2016-04-20 18:09:54Z vboxsync $ */
2/** @file
3 * VBoxBugReport - VirtualBox command-line diagnostics tool, main file.
4 */
5
6/*
7 * Copyright (C) 2006-2016 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#include <VBox/com/com.h>
20#include <VBox/com/string.h>
21#include <VBox/com/array.h>
22//#include <VBox/com/Guid.h>
23#include <VBox/com/ErrorInfo.h>
24#include <VBox/com/errorprint.h>
25#include <VBox/com/VirtualBox.h>
26
27#include <VBox/version.h>
28
29#include <iprt/buildconfig.h>
30#include <iprt/env.h>
31#include <iprt/file.h>
32#include <iprt/getopt.h>
33#include <iprt/initterm.h>
34#include <iprt/path.h>
35#include <iprt/process.h>
36#include <iprt/zip.h>
37#include <iprt/cpp/exception.h>
38
39#include <list>
40
41#include "VBoxBugReport.h"
42
43/* Implementation - Base */
44
45#ifndef RT_OS_WINDOWS
46/* @todo Replace with platform-specific implementations. */
47void createBugReportOsSpecific(BugReport* report, const char *pszHome)
48{
49}
50#endif /* !RT_OS_WINDOWS */
51
52
53/* Globals */
54
55static char *g_pszVBoxManage = NULL;
56
57static const RTGETOPTDEF g_aOptions[] =
58{
59 { "-all", 'A', RTGETOPT_REQ_NOTHING },
60 { "--all", 'A', RTGETOPT_REQ_NOTHING },
61 { "-output", 'o', RTGETOPT_REQ_STRING },
62 { "--output", 'o', RTGETOPT_REQ_STRING },
63 { "-text", 't', RTGETOPT_REQ_NOTHING },
64 { "--text", 't', RTGETOPT_REQ_NOTHING }
65};
66
67static const char g_szUsage[] =
68 "Usage: %s [-h|-?|--help] [-A|--all|<vmname>...] [-o <file>|--output=<file>]\n"
69 " Several VM names can be specified at once to be included into single report.\n"
70 " If none is given then no machines will be included. Specifying -A overrides\n"
71 " any VM names provided and included all registered machines.\n"
72 "Options:\n"
73 " -h, -help, --help Print usage information\n"
74 " -A, -all, --all Include all registered machines\n"
75 " -o, -output, --output Specifies the name of the output file\n"
76 " -t, -text, --text Produce a single text file instead of compressed TAR\n"
77 " -V, -version, --version Print version number and exit\n"
78 "\n";
79
80
81/*
82 * This class stores machine-specific file paths that are obtained via
83 * VirtualBox API. In case API is not functioning properly these paths
84 * will be deduced on the best effort basis.
85 */
86class MachineInfo
87{
88public:
89 MachineInfo(const char *name, const char *logFolder, const char *settingsFile);
90 ~MachineInfo();
91 const char *getName() const { return m_name; };
92 const char *getLogPath() const { return m_logpath; };
93 const char *getSettingsFile() const { return m_settings; };
94private:
95 char *m_name;
96 char *m_logpath;
97 char *m_settings;
98};
99
100MachineInfo::MachineInfo(const char *name, const char *logFolder, const char *settingsFile)
101{
102 m_name = RTStrDup(name);
103 m_logpath = RTStrDup(logFolder);
104 m_settings = RTStrDup(settingsFile);
105}
106
107MachineInfo::~MachineInfo()
108{
109 RTStrFree(m_logpath);
110 RTStrFree(m_name);
111 RTStrFree(m_settings);
112 m_logpath = m_name = m_settings = 0;
113}
114
115typedef std::list<MachineInfo*> MachineInfoList;
116
117
118/*
119 * An abstract class serving as the root of the bug report item tree.
120 */
121BugReportItem::BugReportItem(const char *pszTitle)
122{
123 m_pszTitle = RTStrDup(pszTitle);
124}
125
126BugReportItem::~BugReportItem()
127{
128 RTStrFree(m_pszTitle);
129}
130
131const char * BugReportItem::getTitle(void)
132{
133 return m_pszTitle;
134}
135
136
137BugReport::BugReport(const char *pszFileName)
138{
139 m_pszFileName = RTStrDup(pszFileName);
140}
141
142BugReport::~BugReport()
143{
144 for (unsigned i = 0; i < m_Items.size(); ++i)
145 {
146 delete m_Items[i];
147 }
148 RTStrFree(m_pszFileName);
149}
150
151int BugReport::getItemCount(void)
152{
153 return (int)m_Items.size();
154}
155
156void BugReport::addItem(BugReportItem* item)
157{
158 if (item)
159 m_Items.append(item);
160}
161
162void BugReport::process(void)
163{
164 for (unsigned i = 0; i < m_Items.size(); ++i)
165 {
166 BugReportItem *pItem = m_Items[i];
167 RTPrintf("%3u%% - collecting %s...\n", i * 100 / m_Items.size(), pItem->getTitle());
168 processItem(pItem);
169 }
170 RTPrintf("100%% - compressing...\n\n");
171}
172
173
174BugReportStream::BugReportStream(const char *pszTitle) : BugReportItem(pszTitle)
175{
176 handleRtError(RTPathTemp(m_szFileName, RTPATH_MAX),
177 "Failed to obtain path to temporary folder");
178 handleRtError(RTPathAppend(m_szFileName, RTPATH_MAX, "BugRepXXXXX.tmp"),
179 "Failed to append path");
180 handleRtError(RTFileCreateTemp(m_szFileName, 0600),
181 "Failed to create temporary file '%s'", m_szFileName);
182 handleRtError(RTStrmOpen(m_szFileName, "w", &m_Strm),
183 "Failed to open '%s'", m_szFileName);
184}
185
186BugReportStream::~BugReportStream()
187{
188 if (m_Strm)
189 RTStrmClose(m_Strm);
190 RTFileDelete(m_szFileName);
191}
192
193int BugReportStream::printf(const char *pszFmt, ...)
194{
195 va_list va;
196 va_start(va, pszFmt);
197 int cb = RTStrmPrintfV(m_Strm, pszFmt, va);
198 va_end(va);
199 return cb;
200}
201
202int BugReportStream::putStr(const char *pszString)
203{
204 return RTStrmPutStr(m_Strm, pszString);
205}
206
207PRTSTREAM BugReportStream::getStream(void)
208{
209 RTStrmClose(m_Strm);
210 handleRtError(RTStrmOpen(m_szFileName, "r", &m_Strm),
211 "Failed to open '%s'", m_szFileName);
212 return m_Strm;
213}
214
215
216/* Implementation - Generic */
217
218BugReportFile::BugReportFile(const char *pszPath, const char *pszShortName) : BugReportItem(pszShortName)
219{
220 m_Strm = 0;
221 m_pszPath = RTStrDup(pszPath);
222}
223
224BugReportFile::~BugReportFile()
225{
226 if (m_Strm)
227 RTStrmClose(m_Strm);
228 if (m_pszPath)
229 RTStrFree(m_pszPath);
230}
231
232PRTSTREAM BugReportFile::getStream(void)
233{
234 handleRtError(RTStrmOpen(m_pszPath, "rb", &m_Strm),
235 "Failed to open '%s'", m_pszPath);
236 return m_Strm;
237}
238
239
240BugReportCommand::BugReportCommand(const char *pszTitle, const char *pszExec, ...)
241 : BugReportItem(pszTitle), m_Strm(NULL)
242{
243 unsigned cArgs = 0;
244 m_papszArgs[cArgs++] = RTStrDup(pszExec);
245
246 const char *pszArg;
247 va_list va;
248 va_start(va, pszExec);
249 do
250 {
251 if (cArgs >= RT_ELEMENTS(m_papszArgs))
252 throw RTCError(com::Utf8StrFmt("Too many arguments (%u > %u)\n", cArgs+1, RT_ELEMENTS(m_papszArgs)));
253 pszArg = va_arg(va, const char *);
254 m_papszArgs[cArgs++] = pszArg ? RTStrDup(pszArg) : NULL;
255 } while (pszArg);
256 va_end(va);
257}
258
259BugReportCommand::~BugReportCommand()
260{
261 if (m_Strm)
262 RTStrmClose(m_Strm);
263 RTFileDelete(m_szFileName);
264 for (size_t i = 0; i < RT_ELEMENTS(m_papszArgs) && m_papszArgs[i]; ++i)
265 RTStrFree(m_papszArgs[i]);
266}
267
268PRTSTREAM BugReportCommand::getStream(void)
269{
270 handleRtError(RTPathTemp(m_szFileName, RTPATH_MAX),
271 "Failed to obtain path to temporary folder");
272 handleRtError(RTPathAppend(m_szFileName, RTPATH_MAX, "BugRepXXXXX.tmp"),
273 "Failed to append path");
274 handleRtError(RTFileCreateTemp(m_szFileName, 0600),
275 "Failed to create temporary file '%s'", m_szFileName);
276
277 RTHANDLE hStdOutErr;
278 hStdOutErr.enmType = RTHANDLETYPE_FILE;
279 handleRtError(RTFileOpen(&hStdOutErr.u.hFile, m_szFileName,
280 RTFILE_O_WRITE | RTFILE_O_CREATE_REPLACE | RTFILE_O_DENY_WRITE),
281 "Failed to open temporary file '%s'", m_szFileName);
282
283 RTPROCESS hProcess;
284 handleRtError(RTProcCreateEx(m_papszArgs[0], m_papszArgs, RTENV_DEFAULT, 0,
285 NULL, &hStdOutErr, &hStdOutErr,
286 NULL, NULL, &hProcess),
287 "Failed to create process '%s'", m_papszArgs[0]);
288 RTPROCSTATUS status;
289 handleRtError(RTProcWait(hProcess, RTPROCWAIT_FLAGS_BLOCK, &status),
290 "Process wait failed");
291 //if (status.enmReason == RTPROCEXITREASON_NORMAL) {}
292 RTFileClose(hStdOutErr.u.hFile);
293
294 handleRtError(RTStrmOpen(m_szFileName, "r", &m_Strm),
295 "Failed to open '%s'", m_szFileName);
296 return m_Strm;
297}
298
299
300BugReportText::BugReportText(const char *pszFileName) : BugReport(pszFileName)
301{
302 handleRtError(RTStrmOpen(pszFileName, "w", &m_StrmTxt),
303 "Failed to open '%s'", pszFileName);
304}
305
306BugReportText::~BugReportText()
307{
308 if (m_StrmTxt)
309 RTStrmClose(m_StrmTxt);
310}
311
312void BugReportText::processItem(BugReportItem* item)
313{
314 int cb = RTStrmPrintf(m_StrmTxt, "[ %s ] -------------------------------------------\n", item->getTitle());
315 if (!cb)
316 throw RTCError(com::Utf8StrFmt("Write failure (cb=%d)\n", cb));
317
318 PRTSTREAM strmIn = NULL;
319 try
320 {
321 strmIn = item->getStream();
322 }
323 catch (RTCError &e)
324 {
325 strmIn = NULL;
326 RTStrmPutStr(m_StrmTxt, e.what());
327 }
328
329 int rc = VINF_SUCCESS;
330
331 if (strmIn)
332 {
333 char buf[64*1024];
334 size_t cbRead, cbWritten;
335 cbRead = cbWritten = 0;
336 while (RT_SUCCESS(rc = RTStrmReadEx(strmIn, buf, sizeof(buf), &cbRead)) && cbRead)
337 {
338 rc = RTStrmWriteEx(m_StrmTxt, buf, cbRead, &cbWritten);
339 if (RT_FAILURE(rc) || cbRead != cbWritten)
340 throw RTCError(com::Utf8StrFmt("Write failure (rc=%d, cbRead=%lu, cbWritten=%lu)\n",
341 rc, cbRead, cbWritten));
342 }
343 }
344
345 handleRtError(RTStrmPutCh(m_StrmTxt, '\n'), "Write failure");
346}
347
348
349BugReportTarGzip::BugReportTarGzip(const char *pszFileName)
350 : BugReport(pszFileName), m_hTar(NIL_RTTAR), m_hTarFile(NIL_RTTARFILE)
351{
352 VfsIoStreamHandle hVfsOut;
353 handleRtError(RTVfsIoStrmOpenNormal(pszFileName, RTFILE_O_WRITE | RTFILE_O_CREATE | RTFILE_O_DENY_WRITE,
354 hVfsOut.getPtr()),
355 "Failed to create output file '%s'", pszFileName);
356 handleRtError(RTZipGzipCompressIoStream(hVfsOut.get(), 0, 6, m_hVfsGzip.getPtr()),
357 "Failed to create compressed stream for '%s'", pszFileName);
358
359 handleRtError(RTPathTemp(m_szTarName, RTPATH_MAX),
360 "Failed to obtain path to temporary folder");
361 handleRtError(RTPathAppend(m_szTarName, RTPATH_MAX, "BugRepXXXXX.tar"),
362 "Failed to append path");
363 handleRtError(RTFileCreateTemp(m_szTarName, 0600),
364 "Failed to create temporary file '%s'", m_szTarName);
365 handleRtError(RTFileDelete(m_szTarName),
366 "Failed to delete temporary file '%s'", m_szTarName);
367 handleRtError(RTTarOpen(&m_hTar, m_szTarName, RTFILE_O_CREATE | RTFILE_O_WRITE | RTFILE_O_DENY_ALL),
368 "Failed to create TAR file '%s'", m_szTarName);
369
370}
371
372BugReportTarGzip::~BugReportTarGzip()
373{
374 if (m_hTarFile != NIL_RTTARFILE)
375 RTTarFileClose(m_hTarFile);
376 if (m_hTar != NIL_RTTAR)
377 RTTarClose(m_hTar);
378}
379
380void BugReportTarGzip::processItem(BugReportItem* item)
381{
382 /*
383 * @todo Our TAR implementation does not support names larger than 100 characters.
384 * We truncate the title to make sure it will fit into 100-character field of TAR header.
385 */
386 RTCString strTarFile = RTCString(item->getTitle()).substr(0, RTStrNLen(item->getTitle(), 99));
387 handleRtError(RTTarFileOpen(m_hTar, &m_hTarFile, strTarFile.c_str(),
388 RTFILE_O_CREATE | RTFILE_O_WRITE | RTFILE_O_DENY_NONE),
389 "Failed to open '%s' in TAR", strTarFile.c_str());
390
391 PRTSTREAM strmIn = NULL;
392 try
393 {
394 strmIn = item->getStream();
395 }
396 catch (RTCError &e)
397 {
398 strmIn = NULL;
399 handleRtError(RTTarFileWriteAt(m_hTarFile, 0, e.what(), RTStrNLen(e.what(), 1024), NULL),
400 "Failed to write %u bytes to TAR", RTStrNLen(e.what(), 1024));
401 }
402
403 int rc = VINF_SUCCESS;
404
405 if (strmIn)
406 {
407 char buf[64*1024];
408 size_t cbRead = 0;
409 for (uint64_t offset = 0;
410 RT_SUCCESS(rc = RTStrmReadEx(strmIn, buf, sizeof(buf), &cbRead)) && cbRead;
411 offset += cbRead)
412 {
413 handleRtError(RTTarFileWriteAt(m_hTarFile, offset, buf, cbRead, NULL),
414 "Failed to write %u bytes to TAR", cbRead);
415 }
416 }
417
418 if (m_hTarFile)
419 {
420 handleRtError(RTTarFileClose(m_hTarFile), "Failed to close '%s' in TAR", strTarFile.c_str());
421 m_hTarFile = NIL_RTTARFILE;
422 }
423}
424
425void BugReportTarGzip::complete(void)
426{
427 if (m_hTarFile != NIL_RTTARFILE)
428 {
429 RTTarFileClose(m_hTarFile);
430 m_hTarFile = NIL_RTTARFILE;
431 }
432 if (m_hTar != NIL_RTTAR)
433 {
434 RTTarClose(m_hTar);
435 m_hTar = NIL_RTTAR;
436 }
437
438 VfsIoStreamHandle hVfsIn;
439 handleRtError(RTVfsIoStrmOpenNormal(m_szTarName, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE,
440 hVfsIn.getPtr()),
441 "Failed to open TAR file '%s'", m_szTarName);
442
443 int rc;
444 char buf[_64K];
445 size_t cbRead = 0;
446 while (RT_SUCCESS(rc = RTVfsIoStrmRead(hVfsIn.get(), buf, sizeof(buf), true, &cbRead)) && cbRead)
447 handleRtError(RTVfsIoStrmWrite(m_hVfsGzip.get(), buf, cbRead, true, NULL),
448 "Failed to write into compressed stream");
449 handleRtError(rc, "Failed to read from TAR stream");
450 handleRtError(RTVfsIoStrmFlush(m_hVfsGzip.get()), "Failed to flush output stream");
451 m_hVfsGzip.release();
452}
453
454
455/* Implementation - Main */
456
457void createBugReport(BugReport* report, const char *pszHome, MachineInfoList& machines)
458{
459 report->addItem(new BugReportFile(PathJoin(pszHome, "VBoxSVC.log"), "VBoxSVC.log"));
460 report->addItem(new BugReportFile(PathJoin(pszHome, "VBoxSVC.log.1"), "VBoxSVC.log.1"));
461 report->addItem(new BugReportFile(PathJoin(pszHome, "VirtualBox.xml"), "VirtualBox.xml"));
462 report->addItem(new BugReportCommand("HostUsbDevices", g_pszVBoxManage, "list", "usbhost", NULL));
463 report->addItem(new BugReportCommand("HostUsbFilters", g_pszVBoxManage, "list", "usbfilters", NULL));
464 for (MachineInfoList::iterator it = machines.begin(); it != machines.end(); ++it)
465 {
466 report->addItem(new BugReportFile(PathJoin((*it)->getLogPath(), "VBox.log"),
467 PathJoin((*it)->getName(), "VBox.log")));
468 report->addItem(new BugReportFile((*it)->getSettingsFile(),
469 PathJoin((*it)->getName(), RTPathFilename((*it)->getSettingsFile()))));
470 report->addItem(new BugReportCommand(PathJoin((*it)->getName(), "GuestProperties"),
471 g_pszVBoxManage, "guestproperty", "enumerate",
472 (*it)->getName(), NULL));
473 }
474
475 createBugReportOsSpecific(report, pszHome);
476}
477
478void addMachine(MachineInfoList& list, ComPtr<IMachine> machine)
479{
480 com::Bstr name, logFolder, settingsFile;
481 handleComError(machine->COMGETTER(Name)(name.asOutParam()),
482 "Failed to get VM name");
483 handleComError(machine->COMGETTER(LogFolder)(logFolder.asOutParam()),
484 "Failed to get VM log folder");
485 handleComError(machine->COMGETTER(SettingsFilePath)(settingsFile.asOutParam()),
486 "Failed to get VM settings file path");
487 list.push_back(new MachineInfo(com::Utf8Str(name).c_str(),
488 com::Utf8Str(logFolder).c_str(),
489 com::Utf8Str(settingsFile).c_str()));
490}
491
492
493static void printHeader(void)
494{
495 RTStrmPrintf(g_pStdErr, VBOX_PRODUCT " Bug Report Tool " VBOX_VERSION_STRING "\n"
496 "(C) " VBOX_C_YEAR " " VBOX_VENDOR "\n"
497 "All rights reserved.\n\n");
498}
499
500int main(int argc, char *argv[])
501{
502 /*
503 * Initialize the VBox runtime without loading
504 * the support driver.
505 */
506 RTR3InitExe(argc, &argv, 0);
507
508 bool fAllMachines = false;
509 bool fTextOutput = false;
510 const char *pszOutputFile = NULL;
511 std::list<const char *> nameList;
512 RTGETOPTUNION ValueUnion;
513 RTGETOPTSTATE GetState;
514 int ret = RTGetOptInit(&GetState, argc, argv,
515 g_aOptions, RT_ELEMENTS(g_aOptions),
516 1 /* First */, 0 /*fFlags*/);
517 if (RT_FAILURE(ret))
518 return ret;
519 int ch;
520 while ((ch = RTGetOpt(&GetState, &ValueUnion)))
521 {
522 switch(ch)
523 {
524 case 'h':
525 printHeader();
526 RTStrmPrintf(g_pStdErr, g_szUsage, argv[0]);
527 return 0;
528 case 'A':
529 fAllMachines = true;
530 break;
531 case 'o':
532 pszOutputFile = ValueUnion.psz;
533 break;
534 case 't':
535 fTextOutput = true;
536 break;
537 case 'V':
538 RTPrintf("%sr%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr());
539 return 0;
540 case VINF_GETOPT_NOT_OPTION:
541 nameList.push_back(ValueUnion.psz);
542 break;
543 default:
544 return RTGetOptPrintError(ch, &ValueUnion);
545 }
546 }
547
548 printHeader();
549
550 HRESULT hr = S_OK;
551 char homeDir[RTPATH_MAX];
552 com::GetVBoxUserHomeDirectory(homeDir, sizeof(homeDir));
553
554 try
555 {
556 /* Figure out full path to VBoxManage */
557 char *pszVBoxBin = RTStrDup(argv[0]);
558 if (!pszVBoxBin)
559 throw RTCError("Out of memory\n");
560 RTPathStripFilename(pszVBoxBin);
561 g_pszVBoxManage = RTPathJoinA(pszVBoxBin, VBOXMANAGE);
562 if (!g_pszVBoxManage)
563 throw RTCError("Out of memory\n");
564 RTStrFree(pszVBoxBin);
565
566 handleComError(com::Initialize(), "Failed to initialize COM");
567
568 MachineInfoList list;
569
570 do
571 {
572 ComPtr<IVirtualBoxClient> virtualBoxClient;
573 ComPtr<IVirtualBox> virtualBox;
574 ComPtr<ISession> session;
575
576 hr = virtualBoxClient.createLocalObject(CLSID_VirtualBoxClient);
577 if (SUCCEEDED(hr))
578 hr = virtualBoxClient->COMGETTER(VirtualBox)(virtualBox.asOutParam());
579 if (FAILED(hr))
580 RTStrmPrintf(g_pStdErr, "WARNING: Failed to create the VirtualBox object (hr=0x%x)\n", hr);
581 else
582 {
583 hr = session.createInprocObject(CLSID_Session);
584 if (FAILED(hr))
585 RTStrmPrintf(g_pStdErr, "WARNING: Failed to create a session object (hr=0x%x)\n", hr);
586 }
587
588 if (SUCCEEDED(hr))
589 {
590 if (fAllMachines)
591 {
592 com::SafeIfaceArray<IMachine> machines;
593 hr = virtualBox->COMGETTER(Machines)(ComSafeArrayAsOutParam(machines));
594 if (SUCCEEDED(hr))
595 {
596 for (size_t i = 0; i < machines.size(); ++i)
597 {
598 if (machines[i])
599 addMachine(list, machines[i]);
600 }
601 }
602 }
603 else
604 {
605 for ( std::list<const char *>::iterator it = nameList.begin(); it != nameList.end(); ++it)
606 {
607 ComPtr<IMachine> machine;
608 handleComError(virtualBox->FindMachine(com::Bstr(*it).raw(), machine.asOutParam()),
609 "No such machine '%s'", *it);
610 addMachine(list, machine);
611 }
612 }
613 }
614
615 }
616 while(0);
617
618 RTTIMESPEC TimeSpec;
619 RTTIME Time;
620 RTTimeExplode(&Time, RTTimeNow(&TimeSpec));
621 RTCStringFmt strOutFile("%04d-%02d-%02d-%02d-%02d-%02d-bugreport.%s",
622 Time.i32Year, Time.u8Month, Time.u8MonthDay,
623 Time.u8Hour, Time.u8Minute, Time.u8Second,
624 fTextOutput ? "txt" : "tgz");
625 RTCString strFallbackOutFile;
626 if (!pszOutputFile)
627 {
628 RTFILE tmp;
629 pszOutputFile = strOutFile.c_str();
630 int rc = RTFileOpen(&tmp, pszOutputFile, RTFILE_O_WRITE | RTFILE_O_CREATE | RTFILE_O_DENY_WRITE);
631 if (rc == VERR_ACCESS_DENIED)
632 {
633 char szUserHome[RTPATH_MAX];
634 handleRtError(RTPathUserHome(szUserHome, sizeof(szUserHome)), "Failed to obtain home directory");
635 strFallbackOutFile.printf("%s/%s", szUserHome, strOutFile.c_str());
636 pszOutputFile = strFallbackOutFile.c_str();
637 }
638 else if (RT_SUCCESS(rc))
639 {
640 RTFileClose(tmp);
641 RTFileDelete(pszOutputFile);
642 }
643 }
644 BugReport *pReport;
645 if (fTextOutput)
646 pReport = new BugReportText(pszOutputFile);
647 else
648 pReport = new BugReportTarGzip(pszOutputFile);
649 createBugReport(pReport, homeDir, list);
650 pReport->process();
651 pReport->complete();
652 RTPrintf("Report was written to '%s'\n", pszOutputFile);
653 delete pReport;
654 }
655 catch (RTCError &e)
656 {
657 RTStrmPrintf(g_pStdErr, "ERROR: %s\n", e.what());
658 }
659
660 com::Shutdown();
661
662 if (g_pszVBoxManage)
663 RTStrFree(g_pszVBoxManage);
664
665 return SUCCEEDED(hr) ? 0 : 1;
666}
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