1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/debugrpt.cpp 
   3 // Purpose:     wxDebugReport and related classes implementation 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 2005 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 // License:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 #include "wx/wxprec.h" 
  33 #if wxUSE_DEBUGREPORT && wxUSE_XML 
  35 #include "wx/debugrpt.h" 
  38 #include "wx/filename.h" 
  40 #include "wx/dynlib.h" 
  42 #include "wx/xml/xml.h" 
  45     #include "wx/stackwalk.h" 
  49     #include "wx/msw/crashrpt.h" 
  53     #include "wx/wfstream.h" 
  54     #include "wx/zipstrm.h" 
  55 #endif // wxUSE_ZIPSTREAM 
  57 WX_CHECK_BUILD_OPTIONS("wxQA") 
  59 // ---------------------------------------------------------------------------- 
  60 // XmlStackWalker: stack walker specialization which dumps stack in XML 
  61 // ---------------------------------------------------------------------------- 
  65 class XmlStackWalker 
: public wxStackWalker
 
  68     XmlStackWalker(wxXmlNode 
*nodeStack
) 
  71         m_nodeStack 
= nodeStack
; 
  74     bool IsOk() const { return m_isOk
; } 
  77     virtual void OnStackFrame(const wxStackFrame
& frame
); 
  79     wxXmlNode 
*m_nodeStack
; 
  83 #endif // wxUSE_STACKWALKER 
  85 // ---------------------------------------------------------------------------- 
  87 // ---------------------------------------------------------------------------- 
  90 HexProperty(wxXmlNode 
*node
, const wxChar 
*name
, unsigned long value
) 
  92     node
->AddProperty(name
, wxString::Format(_T("%08lx"), value
)); 
  96 NumProperty(wxXmlNode 
*node
, const wxChar 
*name
, unsigned long value
) 
  98     node
->AddProperty(name
, wxString::Format(_T("%lu"), value
)); 
 102 TextElement(wxXmlNode 
*node
, const wxChar 
*name
, const wxString
& value
) 
 104     wxXmlNode 
*nodeChild 
= new wxXmlNode(wxXML_ELEMENT_NODE
, name
); 
 105     node
->AddChild(nodeChild
); 
 106     nodeChild
->AddChild(new wxXmlNode(wxXML_TEXT_NODE
, _T(""), value
)); 
 110 HexElement(wxXmlNode 
*node
, const wxChar 
*name
, unsigned long value
) 
 112     TextElement(node
, name
, wxString::Format(_T("%08lx"), value
)); 
 115 #if wxUSE_STACKWALKER 
 117 // ============================================================================ 
 118 // XmlStackWalker implementation 
 119 // ============================================================================ 
 121 void XmlStackWalker::OnStackFrame(const wxStackFrame
& frame
) 
 125     wxXmlNode 
*nodeFrame 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("frame")); 
 126     m_nodeStack
->AddChild(nodeFrame
); 
 128     NumProperty(nodeFrame
, _T("level"), frame
.GetLevel()); 
 129     wxString func 
= frame
.GetName(); 
 132         nodeFrame
->AddProperty(_T("function"), func
); 
 133         HexProperty(nodeFrame
, _T("offset"), frame
.GetOffset()); 
 136     if ( frame
.HasSourceLocation() ) 
 138         nodeFrame
->AddProperty(_T("file"), frame
.GetFileName()); 
 139         NumProperty(nodeFrame
, _T("line"), frame
.GetLine()); 
 142     const size_t nParams 
= frame
.GetParamCount(); 
 145         wxXmlNode 
*nodeParams 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("parameters")); 
 146         nodeFrame
->AddChild(nodeParams
); 
 148         for ( size_t n 
= 0; n 
< nParams
; n
++ ) 
 151                 nodeParam 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("parameter")); 
 152             nodeParams
->AddChild(nodeParam
); 
 154             NumProperty(nodeParam
, _T("number"), n
); 
 156             wxString type
, name
, value
; 
 157             if ( !frame
.GetParam(n
, &type
, &name
, &value
) ) 
 161                 TextElement(nodeParam
, _T("type"), type
); 
 164                 TextElement(nodeParam
, _T("name"), name
); 
 166             if ( !value
.empty() ) 
 167                 TextElement(nodeParam
, _T("value"), value
); 
 172 #endif // wxUSE_STACKWALKER 
 174 // ============================================================================ 
 175 // wxDebugReport implementation 
 176 // ============================================================================ 
 178 // ---------------------------------------------------------------------------- 
 179 // initialization and cleanup 
 180 // ---------------------------------------------------------------------------- 
 182 wxDebugReport::wxDebugReport() 
 184     // get a temporary directory name 
 185     wxString appname 
= GetReportName(); 
 187     // we can't use CreateTempFileName() because it creates a file, not a 
 188     // directory, so do our best to create a unique name ourselves 
 190     // of course, this doesn't protect us against malicious users... 
 192     fn
.AssignTempFileName(appname
); 
 193     m_dir
.Printf(_T("%s%c%s_dbgrpt-%lu-%s"), 
 194                  fn
.GetPath().c_str(), wxFILE_SEP_PATH
, appname
.c_str(), 
 196                  wxDateTime::Now().Format(_T("%Y%m%dT%H%M%S")).c_str()); 
 198     // as we are going to save the process state there use restrictive 
 200     if ( !wxMkdir(m_dir
, 0700) ) 
 202         wxLogSysError(_("Failed to create directory \"%s\""), m_dir
.c_str()); 
 203         wxLogError(_("Debug report couldn't be created.")); 
 209 wxDebugReport::~wxDebugReport() 
 211     if ( !m_dir
.empty() ) 
 213         // remove all files in this directory 
 216         for ( bool cont 
= dir
.GetFirst(&file
); cont
; cont 
= dir
.GetNext(&file
) ) 
 218             if ( wxRemove(wxFileName(m_dir
, file
).GetFullPath()) != 0 ) 
 220                 wxLogSysError(_("Failed to remove debug report file \"%s\""), 
 228     if ( !m_dir
.empty() ) 
 230         // Temp fix: what should this be? eVC++ doesn't like wxRmDir 
 232         if ( wxRmdir(m_dir
.fn_str()) != 0 ) 
 234         if ( wxRmDir(m_dir
.fn_str()) != 0 ) 
 237             wxLogSysError(_("Failed to clean up debug report directory \"%s\""), 
 243 // ---------------------------------------------------------------------------- 
 245 // ---------------------------------------------------------------------------- 
 247 wxString 
wxDebugReport::GetReportName() const 
 250         return wxTheApp
->GetAppName(); 
 256 wxDebugReport::AddFile(const wxString
& filename
, const wxString
& description
) 
 259     wxFileName 
fn(filename
); 
 260     if ( fn
.IsAbsolute() ) 
 262         // we need to copy the file to the debug report directory: give it the 
 264         name 
= fn
.GetFullName(); 
 265         wxCopyFile(fn
.GetFullPath(), 
 266                    wxFileName(GetDirectory(), name
).GetFullPath()); 
 268     else // file relative to the report directory 
 272         wxASSERT_MSG( wxFileName(GetDirectory(), name
).FileExists(), 
 273                       _T("file should exist in debug report directory") ); 
 277     m_descriptions
.Add(description
); 
 281 wxDebugReport::AddText(const wxString
& filename
, 
 282                        const wxString
& text
, 
 283                        const wxString
& description
) 
 285     wxASSERT_MSG( !wxFileName(filename
).IsAbsolute(), 
 286                   _T("filename should be relative to debug report directory") ); 
 288     wxFileName 
fn(GetDirectory(), filename
); 
 289     wxFFile 
file(fn
.GetFullPath(), _T("w")); 
 290     if ( !file
.IsOpened() || !file
.Write(text
) ) 
 293     AddFile(filename
, description
); 
 298 void wxDebugReport::RemoveFile(const wxString
& name
) 
 300     const int n 
= m_files
.Index(name
); 
 301     wxCHECK_RET( n 
!= wxNOT_FOUND
, _T("No such file in wxDebugReport") ); 
 304     m_descriptions
.RemoveAt(n
); 
 306     wxRemove(wxFileName(GetDirectory(), name
).GetFullPath()); 
 309 bool wxDebugReport::GetFile(size_t n
, wxString 
*name
, wxString 
*desc
) const 
 311     if ( n 
>= m_files
.GetCount() ) 
 317         *desc 
= m_descriptions
[n
]; 
 322 void wxDebugReport::AddAll(Context context
) 
 324 #if wxUSE_STACKWALKER 
 326 #endif // wxUSE_STACKWALKER 
 328 #if wxUSE_CRASHREPORT 
 330 #endif // wxUSE_CRASHREPORT 
 332 #if !wxUSE_STACKWALKER && !wxUSE_CRASHREPORT 
 333     wxUnusedVar(context
); 
 337 // ---------------------------------------------------------------------------- 
 338 // adding basic text information about current context 
 339 // ---------------------------------------------------------------------------- 
 341 #if wxUSE_STACKWALKER 
 343 bool wxDebugReport::DoAddSystemInfo(wxXmlNode 
*nodeSystemInfo
) 
 345     nodeSystemInfo
->AddProperty(_T("description"), wxGetOsDescription()); 
 350 bool wxDebugReport::DoAddLoadedModules(wxXmlNode 
*nodeModules
) 
 352     wxDynamicLibraryDetailsArray 
modules(wxDynamicLibrary::ListLoaded()); 
 353     const size_t count 
= modules
.GetCount(); 
 357     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 359         const wxDynamicLibraryDetails
& info 
= modules
[n
]; 
 361         wxXmlNode 
*nodeModule 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("module")); 
 362         nodeModules
->AddChild(nodeModule
); 
 364         wxString path 
= info
.GetPath(); 
 366             path 
= info
.GetName(); 
 368             nodeModule
->AddProperty(_T("path"), path
); 
 372         if ( info
.GetAddress(&addr
, &len
) ) 
 374             HexProperty(nodeModule
, _T("address"), (unsigned long)addr
); 
 375             HexProperty(nodeModule
, _T("size"), len
); 
 378         wxString ver 
= info
.GetVersion(); 
 381             nodeModule
->AddProperty(_T("version"), ver
); 
 388 bool wxDebugReport::DoAddExceptionInfo(wxXmlNode 
*nodeContext
) 
 390 #if wxUSE_CRASHREPORT 
 395     wxXmlNode 
*nodeExc 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("exception")); 
 396     nodeContext
->AddChild(nodeExc
); 
 398     HexProperty(nodeExc
, _T("code"), c
.code
); 
 399     nodeExc
->AddProperty(_T("name"), c
.GetExceptionString()); 
 400     HexProperty(nodeExc
, _T("address"), (unsigned long)c
.addr
); 
 403     wxXmlNode 
*nodeRegs 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("registers")); 
 404     nodeContext
->AddChild(nodeRegs
); 
 405     HexElement(nodeRegs
, _T("eax"), c
.regs
.eax
); 
 406     HexElement(nodeRegs
, _T("ebx"), c
.regs
.ebx
); 
 407     HexElement(nodeRegs
, _T("ecx"), c
.regs
.edx
); 
 408     HexElement(nodeRegs
, _T("edx"), c
.regs
.edx
); 
 409     HexElement(nodeRegs
, _T("esi"), c
.regs
.esi
); 
 410     HexElement(nodeRegs
, _T("edi"), c
.regs
.edi
); 
 412     HexElement(nodeRegs
, _T("ebp"), c
.regs
.ebp
); 
 413     HexElement(nodeRegs
, _T("esp"), c
.regs
.esp
); 
 414     HexElement(nodeRegs
, _T("eip"), c
.regs
.eip
); 
 416     HexElement(nodeRegs
, _T("cs"), c
.regs
.cs
); 
 417     HexElement(nodeRegs
, _T("ds"), c
.regs
.ds
); 
 418     HexElement(nodeRegs
, _T("es"), c
.regs
.es
); 
 419     HexElement(nodeRegs
, _T("fs"), c
.regs
.fs
); 
 420     HexElement(nodeRegs
, _T("gs"), c
.regs
.gs
); 
 421     HexElement(nodeRegs
, _T("ss"), c
.regs
.ss
); 
 423     HexElement(nodeRegs
, _T("flags"), c
.regs
.flags
); 
 427 #else // !wxUSE_CRASHREPORT 
 428     wxUnusedVar(nodeContext
); 
 431 #endif // wxUSE_CRASHREPORT/!wxUSE_CRASHREPORT 
 434 bool wxDebugReport::AddContext(wxDebugReport::Context ctx
) 
 436     wxCHECK_MSG( IsOk(), false, _T("use IsOk() first") ); 
 438     // create XML dump of current context 
 439     wxXmlDocument xmldoc
; 
 440     wxXmlNode 
*nodeRoot 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("report")); 
 441     xmldoc
.SetRoot(nodeRoot
); 
 442     nodeRoot
->AddProperty(_T("version"), _T("1.0")); 
 443     nodeRoot
->AddProperty(_T("kind"), ctx 
== Context_Current 
? _T("user") 
 446     // add system information 
 447     wxXmlNode 
*nodeSystemInfo 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("system")); 
 448     if ( DoAddSystemInfo(nodeSystemInfo
) ) 
 449         nodeRoot
->AddChild(nodeSystemInfo
); 
 451         delete nodeSystemInfo
; 
 453     // add information about the loaded modules 
 454     wxXmlNode 
*nodeModules 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("modules")); 
 455     if ( DoAddLoadedModules(nodeModules
) ) 
 456         nodeRoot
->AddChild(nodeModules
); 
 460     // add CPU context information: this only makes sense for exceptions as our 
 461     // current context is not very interesting otherwise 
 462     if ( ctx 
== Context_Exception 
) 
 464         wxXmlNode 
*nodeContext 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("context")); 
 465         if ( DoAddExceptionInfo(nodeContext
) ) 
 466             nodeRoot
->AddChild(nodeContext
); 
 471     // add stack traceback 
 472 #if wxUSE_STACKWALKER 
 473     wxXmlNode 
*nodeStack 
= new wxXmlNode(wxXML_ELEMENT_NODE
, _T("stack")); 
 474     XmlStackWalker 
sw(nodeStack
); 
 475     if ( ctx 
== Context_Exception 
) 
 477         sw
.WalkFromException(); 
 479     else // Context_Current 
 485         nodeRoot
->AddChild(nodeStack
); 
 488 #endif // wxUSE_STACKWALKER 
 490     // finally let the user add any extra information he needs 
 491     DoAddCustomContext(nodeRoot
); 
 494     // save the entire context dump in a file 
 495     wxFileName 
fn(m_dir
, GetReportName(), _T("xml")); 
 497     if ( !xmldoc
.Save(fn
.GetFullPath()) ) 
 500     AddFile(fn
.GetFullName(), _("process context description")); 
 505 #endif // wxUSE_STACKWALKER 
 507 // ---------------------------------------------------------------------------- 
 509 // ---------------------------------------------------------------------------- 
 511 #if wxUSE_CRASHREPORT 
 513 bool wxDebugReport::AddDump(Context ctx
) 
 515     wxCHECK_MSG( IsOk(), false, _T("use IsOk() first") ); 
 517     wxFileName 
fn(m_dir
, GetReportName(), _T("dmp")); 
 518     wxCrashReport::SetFileName(fn
.GetFullPath()); 
 520     if ( !(ctx 
== Context_Exception 
? wxCrashReport::Generate() 
 521                                     : wxCrashReport::GenerateNow()) ) 
 524     AddFile(fn
.GetFullName(), _("dump of the process state (binary)")); 
 529 #endif // wxUSE_CRASHREPORT 
 531 // ---------------------------------------------------------------------------- 
 533 // ---------------------------------------------------------------------------- 
 535 bool wxDebugReport::Process() 
 537     if ( !GetFilesCount() ) 
 539         wxLogError(_("Debug report generation has failed.")); 
 546         wxLogError(_("Processing debug report has failed, leaving the files in \"%s\" directory."), 
 547                    GetDirectory().c_str()); 
 557 bool wxDebugReport::DoProcess() 
 559     wxString msg 
= _("*** A debug report has been generated\n"); 
 560     msg 
+= wxString::Format(_("*** It can be found in \"%s\"\n"), 
 561                             GetDirectory().c_str()); 
 562     msg 
+= _("*** And includes the following files:\n"); 
 565     const size_t count 
= GetFilesCount(); 
 566     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 568         GetFile(n
, &name
, &desc
); 
 569         msg 
+= wxString::Format(_("\t%s: %s\n"), name
.c_str(), desc
.c_str()); 
 572     msg 
+= _("\nPlease send this report to the program maintainer, thank you!\n"); 
 574     wxLogMessage(_T("%s"), msg
.c_str()); 
 576     // we have to do this or the report would be deleted, and we don't even 
 577     // have any way to ask the user if he wants to keep it from here 
 583 // ============================================================================ 
 584 // wxDebugReport-derived classes 
 585 // ============================================================================ 
 589 // ---------------------------------------------------------------------------- 
 590 // wxDebugReportCompress 
 591 // ---------------------------------------------------------------------------- 
 593 bool wxDebugReportCompress::DoProcess() 
 595     const size_t count 
= GetFilesCount(); 
 599     // create the streams 
 600     wxFileName 
fn(GetDirectory(), GetReportName(), _T("zip")); 
 601     wxFFileOutputStream 
os(fn
.GetFullPath(), _T("wb")); 
 602     wxZipOutputStream 
zos(os
, 9); 
 604     // add all files to the ZIP one 
 606     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 608         GetFile(n
, &name
, &desc
); 
 610         wxZipEntry 
*ze 
= new wxZipEntry(name
); 
 611         ze
->SetComment(desc
); 
 613         if ( !zos
.PutNextEntry(ze
) ) 
 616         wxFileName 
filename(fn
.GetPath(), name
); 
 617         wxFFileInputStream 
is(filename
.GetFullPath()); 
 618         if ( !is
.IsOk() || !zos
.Write(is
).IsOk() ) 
 625     m_zipfile 
= fn
.GetFullPath(); 
 630 // ---------------------------------------------------------------------------- 
 631 // wxDebugReportUpload 
 632 // ---------------------------------------------------------------------------- 
 634 wxDebugReportUpload::wxDebugReportUpload(const wxString
& url
, 
 635                                          const wxString
& input
, 
 636                                          const wxString
& action
, 
 637                                          const wxString
& curl
) 
 642     if ( m_uploadURL
.Last() != _T('/') ) 
 643         m_uploadURL 
+= _T('/'); 
 644     m_uploadURL 
+= action
; 
 647 bool wxDebugReportUpload::DoProcess() 
 649     if ( !wxDebugReportCompress::DoProcess() ) 
 653     wxArrayString output
, errors
; 
 654     int rc 
= wxExecute(wxString::Format
 
 656                             _T("%s -F %s=@\"%s\" %s"), 
 658                             m_inputField
.c_str(), 
 659                             GetCompressedFileName().c_str(), 
 666         wxLogError(_("Failed to execute curl, please install it in PATH.")); 
 670         const size_t count 
= errors
.GetCount(); 
 673             for ( size_t n 
= 0; n 
< count
; n
++ ) 
 675                 wxLogWarning(_T("%s"), errors
[n
].c_str()); 
 679         wxLogError(_("Failed to upload the debug report (error code %d)."), rc
); 
 683         if ( OnServerReply(output
) ) 
 690 #endif // wxUSE_ZIPSTREAM 
 692 #endif // wxUSE_DEBUGREPORT