hxassert.cpp
上传用户:dangjiwu
上传日期:2013-07-19
资源大小:42019k
文件大小:19k
源码类别:

Symbian

开发平台:

Visual C++

  1. /* ***** BEGIN LICENSE BLOCK *****
  2.  * Source last modified: $Id: hxassert.cpp,v 1.14.2.6 2004/07/09 01:46:18 hubbe Exp $
  3.  * 
  4.  * Portions Copyright (c) 1995-2004 RealNetworks, Inc. All Rights Reserved.
  5.  * 
  6.  * The contents of this file, and the files included with this file,
  7.  * are subject to the current version of the RealNetworks Public
  8.  * Source License (the "RPSL") available at
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed
  10.  * the file under the current version of the RealNetworks Community
  11.  * Source License (the "RCSL") available at
  12.  * http://www.helixcommunity.org/content/rcsl, in which case the RCSL
  13.  * will apply. You may also obtain the license terms directly from
  14.  * RealNetworks.  You may not use this file except in compliance with
  15.  * the RPSL or, if you have a valid RCSL with RealNetworks applicable
  16.  * to this file, the RCSL.  Please see the applicable RPSL or RCSL for
  17.  * the rights, obligations and limitations governing use of the
  18.  * contents of the file.
  19.  * 
  20.  * Alternatively, the contents of this file may be used under the
  21.  * terms of the GNU General Public License Version 2 or later (the
  22.  * "GPL") in which case the provisions of the GPL are applicable
  23.  * instead of those above. If you wish to allow use of your version of
  24.  * this file only under the terms of the GPL, and not to allow others
  25.  * to use your version of this file under the terms of either the RPSL
  26.  * or RCSL, indicate your decision by deleting the provisions above
  27.  * and replace them with the notice and other provisions required by
  28.  * the GPL. If you do not delete the provisions above, a recipient may
  29.  * use your version of this file under the terms of any one of the
  30.  * RPSL, the RCSL or the GPL.
  31.  * 
  32.  * This file is part of the Helix DNA Technology. RealNetworks is the
  33.  * developer of the Original Code and owns the copyrights in the
  34.  * portions it created.
  35.  * 
  36.  * This file, and the files included with this file, is distributed
  37.  * and made available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY
  38.  * KIND, EITHER EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS
  39.  * ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES
  40.  * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET
  41.  * ENJOYMENT OR NON-INFRINGEMENT.
  42.  * 
  43.  * Technology Compatibility Kit Test Suite(s) Location:
  44.  *    http://www.helixcommunity.org/content/tck
  45.  * 
  46.  * Contributor(s):
  47.  * 
  48.  * ***** END LICENSE BLOCK ***** */
  49. /////////////////////////////////////////////////////////////////////////////
  50. // HXASSERT.CPP
  51. //
  52. // Debugging support implementation.
  53. //
  54. // HX_ASSERT() - asserts an expression is TRUE. Compiles to no-ops in
  55. // retail builds. Provides message box or other UI when 
  56. // expression fails.
  57. //
  58. // HX_ASSERT_VALID_PTR() - asserts that a pointer is valid. Performs more
  59. // rigid verification specifically appropriate for pointers.
  60. //
  61. // HX_VERIFY() - verifies an expression is TRUE. Expression or code DOES NOT 
  62. // compile away in retail builds, but UI of failure is removed.
  63. // In debug builds provides message box or other UI when 
  64. // expression fails.
  65. //
  66. // HX_TRACE() - Similar to DEBUGPRINTF() but no buffer is required. 
  67. // Compiles to no-ops in retail builds.
  68. //
  69. #include "hxtypes.h"
  70. #include "hlxclib/stdio.h"
  71. #include "hlxclib/stdlib.h"
  72. #if defined(_WIN32) || defined(_WINDOWS)
  73. #include "hlxclib/windows.h"
  74. #if !defined(_WIN32) && !defined(WIN32)
  75. #include <shellapi.h>
  76. #include "string.h"
  77. #endif
  78. #endif
  79. #include "hxassert.h"
  80. #include "hxstrutl.h"
  81. #include "debugout.h"
  82. #include "hxtypes.h"
  83. #include "hxresult.h"
  84. #if defined (DEBUG) || defined (_DEBUG)
  85. #ifdef _MACINTOSH
  86. #include <stdarg.h>
  87. #include <string.h>
  88. #include "platform/mac/hxcrt_assert.h"
  89. #define IDIGNORE 2
  90. #define IDRETRY 1
  91. #endif
  92. #ifdef _UNIX
  93. #include <stdarg.h>
  94. #define IDIGNORE 0
  95. #define IDRETRY 1
  96. #include <errno.h>
  97. #include <string.h>
  98. #include <sys/types.h>
  99. #include <sys/wait.h>
  100. #endif
  101. #if defined(_SYMBIAN)
  102. #include "avkon.rsg"
  103. #include <aknglobalmsgquery.h> 
  104. #endif
  105. #if defined(_SYMBIAN) || defined(_OPENWAVE)
  106. #include "hlxclib/stdarg.h"
  107. #define IDIGNORE 0
  108. #define IDRETRY 1
  109. #endif
  110. #if defined(_WIN32)
  111. #include <tchar.h>
  112. #endif
  113. #include "hxheap.h"
  114. #ifdef DEBUG
  115. #undef HX_THIS_FILE
  116. static const char HX_THIS_FILE[] = __FILE__;
  117. #endif
  118. /////////////////////////////////////////////////////////////////////////////
  119. //
  120. // HXOutputDebugString: 
  121. // Helper function used by DEBUGOUTSTR(). This is better than 
  122. // OutputDebugString, because it will check to see if output
  123. // tracing is turned off in the registry. This prevents the massive
  124. // slew of output messages.
  125. //
  126. void STDMETHODCALLTYPE HXOutputDebugString(const char* pString)
  127. {
  128. if (HXWantTraceMessages())
  129. {
  130. #if _MACINTOSH
  131. Str255 pstr = {0x00};
  132. char *p = (char*)pString;
  133. while (*p && pstr[0] < 253) // interrupt safe c2pstr, and we don't mess with pString
  134. pstr[++pstr[0]] = *p++;
  135. pstr[++pstr[0]] = ';'; // add go command so we don't stay in MacsBug
  136. pstr[++pstr[0]] = 'g';
  137. DebugStr(pstr);
  138. #elif defined( DEBUG ) && (defined( _WIN32 ) || defined( _WINDOWS ))
  139. // This is the Windows/Win 95 version
  140. OutputDebugString(OS_STRING(pString));
  141. #elif defined( DEBUG) && defined(_LINUX)
  142.                 fprintf( stderr, "%s", pString);
  143. #else
  144. // Any other platforms....  Undefine it to be safe.
  145. #endif
  146. } // end if
  147. };
  148. /////////////////////////////////////////////////////////////////////////////
  149. //
  150. // HXDebugOptionEnabled: 
  151. // Determine if the given debug option is enabled.
  152. // A lookup is done to the registry key, and if it's present
  153. // and set to '1', TRUE is returned otherwise FALSE
  154. //
  155. #ifdef _WIN16
  156. BOOL far _cdecl HXDebugOptionEnabled(const char* szOption)
  157. #else
  158. BOOL STDMETHODCALLTYPE HXDebugOptionEnabled(const char* szOption)
  159. #endif
  160. {
  161. #if defined( DEBUG ) && (defined( _WIN32 ) || defined( _WINDOWS ))
  162. HKEY  RootKey = HKEY_CLASSES_ROOT;
  163. TCHAR szBuffer[10] = TEXT(""); /* Flawfinder: ignore */
  164. HKEY  hKey;
  165. BOOL  bEnabled = FALSE;
  166. HX_RESULT hRes;
  167. DWORD bufSize = sizeof(szBuffer);
  168. if( RegOpenKey(RootKey, szOption, &hKey) == ERROR_SUCCESS )
  169. {
  170. hRes = RegQueryValue(hKey, "", (char*)szBuffer, (long *)&bufSize);
  171. if (hRes == ERROR_SUCCESS && bufSize != 0)
  172. {
  173. if (strcmp(szBuffer,"1") == 0)
  174. {
  175. bEnabled = TRUE; 
  176. }
  177. }
  178. RegCloseKey(hKey);  
  179. }
  180. return bEnabled;
  181. #else
  182. return FALSE;
  183. // Any other platforms....  Undefine it to be safe.
  184. #endif
  185. }
  186. #if defined(_SYMBIAN)
  187. int QueryAssertActionL(const char* pAssertText)
  188. {
  189.     // Allocate assertion text
  190.     TPtrC8 ptr8((const TUint8*)pAssertText);
  191.     HBufC* pText = HBufC::NewLC(ptr8.Length());
  192.     pText->Des().Copy(ptr8);
  193.     // Show message
  194.     TRequestStatus status = KRequestPending;
  195.     CAknGlobalMsgQuery * pDlg = CAknGlobalMsgQuery::NewL();
  196.     CleanupStack::PushL(pDlg);
  197.     pDlg->ShowMsgQueryL(status, *pText, R_AVKON_SOFTKEYS_OK_DETAILS,
  198.         _L("Assert failed!"), KNullDesC, 0, -1, CAknQueryDialog::EErrorTone );
  199.     User::WaitForRequest(status);
  200.    
  201.     CleanupStack::PopAndDestroy(2); //pText, pDlg
  202.            
  203.     // Go to debugger if user clicked 'details', otherwise ignore
  204.     int nCode = (EAknSoftkeyDetails == status.Int() ? IDRETRY : IDIGNORE);
  205.     return nCode;
  206. }
  207. #endif
  208. /////////////////////////////////////////////////////////////////////////////
  209. //
  210. // HXWantTraceMessages: 
  211. // Helper function used to determine if the system has asked for trace 
  212. // messages.
  213. //
  214. BOOL STDMETHODCALLTYPE HXWantTraceMessages()
  215. {
  216. #if __MWERKS__
  217. return TRUE;
  218. #elif defined( DEBUG ) && (defined( _WIN32 ) || defined( _WINDOWS ))
  219. HKEY  RootKey = HKEY_CLASSES_ROOT;
  220. TCHAR szBuffer[10] = TEXT(""); /* Flawfinder: ignore */
  221. HKEY  hKey;
  222. BOOL  bWantTrace = FALSE;
  223. HX_RESULT hRes;
  224. DWORD bufSize = sizeof(szBuffer);
  225. if( RegOpenKey(RootKey, "HXDebug", &hKey) == ERROR_SUCCESS )
  226. {
  227. hRes = RegQueryValue(hKey, "", (char*)szBuffer, (long *)&bufSize);
  228. if (hRes == ERROR_SUCCESS && bufSize != 0)
  229. {
  230. if (strcmp(szBuffer,"1") == 0)
  231. {
  232. bWantTrace = TRUE; 
  233. }
  234. }
  235. RegCloseKey(hKey);  
  236. }
  237. return bWantTrace;
  238. #elif defined(_UNIX) && defined( DEBUG)
  239.         const char* debugOutputOpts = getenv("HX_DEBUG");
  240.         if( debugOutputOpts && strlen(debugOutputOpts) != 0)
  241.         {
  242.             return TRUE;
  243.         }
  244.         return FALSE;
  245. #else
  246. return FALSE;
  247. // Any other platforms....  Undefine it to be safe.
  248. #endif
  249. }
  250. #if !defined(HELIX_CONFIG_NOSTATICS)
  251. int g_trace_log_enabled = 0;
  252. #endif
  253. /////////////////////////////////////////////////////////////////////////////
  254. //
  255. // HXTrace: Helper function used by HX_TRACE()
  256. //
  257. void STDMETHODVCALLTYPE HXTrace(const char* pszFormat, ...)
  258. {
  259. #if !defined(HELIX_CONFIG_NOSTATICS)    
  260.     if(!g_trace_log_enabled)
  261. return;
  262.     static char z_szDebugBuffer[MAX_TRACE_OUTPUT]; /* Flawfinder: ignore */
  263.     va_list vaMarker;
  264.     va_start( vaMarker, pszFormat );
  265.     vsnprintf( z_szDebugBuffer, sizeof(z_szDebugBuffer), pszFormat, vaMarker );
  266.     va_end( vaMarker );
  267.     DEBUGOUTSTR( z_szDebugBuffer );
  268. #endif
  269. };
  270. /////////////////////////////////////////////////////////////////////////////
  271. //
  272. // HXAssertFailedLine: Helper function used by HX_ASSERT()
  273. //
  274. #ifdef _WIN16
  275. // see comment in hxassert.h
  276. BOOL far _cdecl HXAssertFailedLine(const char* pszExpression, const char* pszFileName, int nLine)
  277. #else
  278. BOOL STDMETHODCALLTYPE HXAssertFailedLine(const char* pszExpression, const char* pszFileName, int nLine)
  279. #endif
  280. {
  281. #if !defined(HELIX_CONFIG_NOSTATICS)    
  282.     static char z_szAssertMessage[MAX_TRACE_OUTPUT]; /* Flawfinder: ignore */
  283.     static BOOL z_nMultiAssertCount = 0;
  284. #else
  285.     char z_szAssertMessage[MAX_TRACE_OUTPUT]; /* Flawfinder: ignore */
  286.     BOOL z_nMultiAssertCount = 0;
  287. #endif        
  288.     // format message into buffer
  289.     SafeSprintf(z_szAssertMessage, MAX_TRACE_OUTPUT, "(%s)... File %s, Line %d", pszExpression, pszFileName, nLine);
  290.     // assume the debugger or auxiliary port
  291.     // output into MacsBug looks better if it's done in one string,
  292.     // since MacsBug always breaks the line after each output
  293.     HX_TRACE("Assertion Failed: %sn", z_szAssertMessage);
  294.     if (z_nMultiAssertCount > 0)
  295.     {
  296.         // assert within assert (examine call stack to determine first one)
  297.         HXDebugBreak();
  298.         return FALSE;
  299.     }
  300.     z_nMultiAssertCount++;
  301. /////////////////////////////////////////////////////////////////
  302. //
  303. // BEGIN: Platform specific portion of HXAssert(), namely, we 
  304. // need to show some UI to tell us that an assertion has failed. 
  305. // The rest of this function is cross-platform.
  306. //
  307. #if defined(_WIN32) || defined(_WINDOWS)
  308. // active popup window for the current thread
  309. HWND hWndParent = GetActiveWindow();
  310. #if !defined(WIN32_PLATFORM_PSPC)
  311. if (hWndParent != NULL)
  312. {
  313.             hWndParent = GetLastActivePopup(hWndParent);
  314. }
  315. #endif /* !defined(WIN32_PLATFORM_PSPC) */
  316. // display the assert
  317. #if !defined(WIN32_PLATFORM_PSPC)
  318. // we remove WM_QUIT because if it is in the queue then the message box
  319. // won't display
  320. MSG msg;
  321. BOOL bQuit = ::PeekMessage(&msg, NULL, WM_QUIT, WM_QUIT, PM_REMOVE);
  322. int nCode = ::MessageBox
  323.             (
  324.                 hWndParent, 
  325.                 z_szAssertMessage,
  326.                 "Assertion Failed!",
  327.                 MB_TASKMODAL|
  328.                 MB_ICONHAND|MB_ABORTRETRYIGNORE|MB_DEFBUTTON3
  329. #if defined(_WIN32)
  330.                 |MB_SETFOREGROUND
  331. #endif
  332.                 );
  333. if (bQuit)
  334.     PostQuitMessage(msg.wParam);
  335. #else /* !defined(WIN32_PLATFORM_PSPC) */
  336. int nCode = ::MessageBox
  337.             (
  338.                 hWndParent, 
  339.                 OS_STRING(z_szAssertMessage),
  340.                 OS_STRING("Assertion Failed!"),
  341.                 MB_ICONHAND|MB_ABORTRETRYIGNORE
  342.                 );
  343. #endif /* !defined(WIN32_PLATFORM_PSPC) */
  344. #elif defined (_MACINTOSH)
  345.         int nCode = HXCRT_ASSERT(z_szAssertMessage);
  346. #elif defined (_UNIX)
  347.         const char *debugopts;
  348.         int debuglevel = 0;
  349.         int nCode = IDIGNORE;
  350. #if defined(_MAC_UNIX)
  351.         debuglevel = 2; // default to something less extreme on Mac
  352. #endif
  353.         debugopts = getenv("HX_DEBUGLEVEL");
  354.         if (debugopts != NULL)
  355.         {
  356.             debuglevel = atoi(debugopts);
  357.         }
  358.         switch(debuglevel)
  359.         {
  360.            case 1: /* debugger */
  361.            {
  362.                fprintf(stderr, "HX_ASSERT failed: %sn", z_szAssertMessage );
  363.                nCode = IDRETRY;
  364.                break;
  365.            }
  366.            case 2: /* terminate */
  367.            {
  368.                fprintf(stderr, "HX_ASSERT failed: %sn", z_szAssertMessage );
  369.                nCode = IDIGNORE;
  370.                break;
  371.            }
  372.            case 3: /* silent */
  373.            {
  374.                nCode = IDIGNORE;
  375.                break;
  376.            }
  377.            case 4: /* interactive */
  378.            {
  379.                setvbuf(stdin, NULL, _IONBF, 0);
  380.                fprintf(stderr, "HX_ASSERT failed: %sn((d)ebug (i)gnore (a)bort)?  ", z_szAssertMessage );
  381.                char input = 'n';
  382.                while (input == 'n')
  383.                    read(0, &input, 1);
  384.                switch (input)
  385.                {
  386.                   case 'i':             /* ignore */
  387.                       nCode = IDIGNORE;
  388.                       break;
  389.                   case 'a':               /* abort */
  390.                       abort();
  391.                       break;
  392.                   case 'd':
  393.                   default:                /* debug */
  394.                       nCode = IDRETRY;
  395.                       break;
  396.                }
  397.                break;
  398.            }
  399.            case 0: /* terminate */
  400.            default:
  401.            {
  402.                fprintf(stderr, "HX_ASSERT failed: %sn", z_szAssertMessage );
  403.                abort();
  404.            }
  405.         }
  406. #endif
  407. #if defined(_SYMBIAN)
  408.     int nCode = IDRETRY;
  409.     TRAPD(err, nCode = QueryAssertActionL(z_szAssertMessage));
  410. #endif //_SYMBIAN        
  411.        
  412. #if defined(_OPENWAVE)
  413.         int nCode = IDIGNORE;
  414. // XXXSAB Fill this in!!!
  415. # ifdef _OPENWAVE_SIMULATOR
  416.         //What to do on the emulator.
  417. # else
  418.         //What to do on the device.
  419. # endif        
  420. #endif        
  421. //
  422. // END: Platform specific portion of HXAssert(). The rest of 
  423. // this function is cross-platform.
  424. //
  425. /////////////////////////////////////////////////////////////////
  426.             // cleanup
  427.             z_nMultiAssertCount--;
  428.             if (nCode == IDIGNORE)
  429.             {
  430.                 return FALSE;   // ignore
  431.             }
  432.             if (nCode == IDRETRY)
  433.             {
  434. return TRUE;
  435.             }
  436.             HXAbort();     // should not return (but otherwise HXDebugBreak)
  437.             return TRUE;
  438. };
  439. /////////////////////////////////////////////////////////////////////////////
  440. //
  441. // HXAssertValidPointer: Helper function used by HX_ASSERT_VALID_PTR()
  442. //
  443. void STDMETHODCALLTYPE HXAssertValidPointer(const void* pVoid, const char* pszFileName, int nLine)
  444. {
  445. if (pVoid == NULL)
  446. {
  447. if (HXAssertFailedLine("HX_ASSERT_VALID_PTR fails on NULL pointer",pszFileName,nLine))
  448. {
  449. HXDebugBreak();
  450. }
  451. return;     // quick escape
  452. }
  453. if (!HXIsValidAddress(pVoid))
  454. {
  455. if (HXAssertFailedLine("HX_ASSERT_VALID_PTR fails with illegal pointer.",pszFileName,nLine))
  456. {
  457. HXDebugBreak();
  458. }
  459. return;     // quick escape
  460. }
  461. };
  462. /////////////////////////////////////////////////////////////////////////////
  463. //
  464. // HXIsValidAddress: Helper function used by HXAssertValidPointer()
  465. //
  466. #ifdef _WIN16
  467. // see comment in hxassert.h for problem with STDMETHODCALLTYPE in win16
  468. BOOL far _cdecl        HXIsValidAddress(const void* lp, ULONG32 nBytes, BOOL bReadWrite)
  469. #else
  470. BOOL STDMETHODCALLTYPE HXIsValidAddress(const void* lp, ULONG32 nBytes, BOOL bReadWrite)
  471. #endif
  472. {
  473. /////////////////////////////////////////////////////////////////
  474. //
  475. // BEGIN: Platform specific portion of HXIsValidAddress(), namely,
  476. // we need to to check if a pointer is a valid pointer.
  477. //
  478. #if defined(_WIN32) || defined(_WINDOWS)
  479. // simple version using Win APIs for pointer validation.
  480. return (
  481. (lp != NULL)
  482.  &&
  483.  !IsBadReadPtr(lp, (UINT)nBytes)
  484.  &&
  485. (!bReadWrite || !IsBadWritePtr((LPVOID)lp, (UINT)nBytes))
  486. );
  487. #else
  488. #  ifdef __MWERKS__
  489. return TRUE;
  490. #  endif
  491. #  ifdef _UNIX
  492. return lp != NULL;
  493. #  endif
  494. #  ifdef _SYMBIAN
  495. return lp != NULL;
  496. #  endif
  497. #  ifdef _OPENWAVE
  498. return lp != NULL;
  499. #  endif
  500. #endif
  501. //
  502. // END: Platform specific portion of HXIsValidAddress(). The rest 
  503. // of this function is cross-platform.
  504. //
  505. /////////////////////////////////////////////////////////////////
  506. }
  507. #ifdef _WIN16
  508. // see comment in hxassert.h for problem with STDMETHODCALLTYPE in win16
  509. BOOL far _cdecl        HXIsValidString(const char* psz, int nLength)
  510. #else
  511. BOOL STDMETHODCALLTYPE HXIsValidString(const char* psz, int nLength)
  512. #endif
  513. {
  514. if (psz == NULL)
  515. {
  516. return FALSE;
  517. }
  518. /////////////////////////////////////////////////////////////////
  519. //
  520. // BEGIN: Platform specific portion of HXIsValidString(), namely,
  521. // we need to to check if a pointer is a valid string pointer.
  522. //
  523. #if defined(_WIN32) && !defined(WIN32_PLATFORM_PSPC)
  524. // simple version using Win APIs for pointer validation.
  525.         return !IsBadStringPtr(psz, nLength);
  526. #else
  527. return TRUE;
  528. #endif
  529. }
  530. #if defined(_SYMBIAN) && !defined (__WINS__)
  531. #include <e32std.h>
  532. void HXDebugBreak()
  533. {
  534.     User::Invariant();
  535. }
  536. #elif defined(_OPENWAVE) && !defined(_OPENWAVE_SIMULATOR)
  537. void HXDebugBreak()
  538. {
  539.     #error Figure out if HXDebugBreak() makes sense on target device...
  540. }
  541. #elif defined(_UNIX)
  542. void HXDebugBreak() 
  543. {
  544.     static int debuggerpid = 0;
  545.     pid_t pid = getpid();
  546.     const char *pname = getenv("PROCESS_NAME");
  547.     const char *pDebuggerProcess = getenv("HX_DEBUGGER");
  548.     
  549.     if (debuggerpid) 
  550.     {
  551. kill(pid, SIGSTOP);
  552. return;
  553.     }
  554.     // This allows the user to override the debug command.  The command 
  555.     // called will be given process name and pid arguments
  556.     if (pDebuggerProcess)
  557.     {
  558. char pCmdTemplate[1024], pCmd[1024]; /* Flawfinder: ignore */
  559. SafeSprintf(pCmdTemplate, 1024, "%s %%s %%d", pDebuggerProcess);
  560. SafeSprintf(pCmd, 1024, pCmdTemplate, pname? pname : "/dev/null", pid);
  561. system(pCmd);
  562. sleep(3);
  563.         return;
  564.     }
  565.     debuggerpid = fork();
  566.     if (debuggerpid)
  567.     {
  568. sleep(3);
  569.     }
  570.     else
  571.     {
  572.         setsid();
  573.         int kid;
  574.         if (!(kid = fork()))
  575.         {
  576.             char buf[1024]; /* Flawfinder: ignore */
  577.             sprintf(buf, "%d", pid); /* Flawfinder: ignore */
  578.     if (!pname)
  579.     {
  580. fprintf(stderr, "Need to set PROCESS_NAME to enable jit debuggingn");
  581.                 fflush(0);
  582. _exit(0);
  583.     }
  584.             if (-1 == (execlp("xterm", "xterm", "-e", "gdb", "-nw", "-nx", pname, buf, NULL)))
  585.             {
  586.                 fprintf(stderr, "failed to start debugger (%s)n", strerror(errno));
  587.                 abort();
  588.             }
  589.         }
  590.         else
  591.         {
  592.             int dummy;
  593.             waitpid(kid, &dummy, 0);
  594.             fflush(0);
  595.             _exit(0);
  596.         }
  597.     }
  598. }
  599. #endif
  600. #endif