JsiRuntime.cpp
上传用户:xqtpzdz
上传日期:2022-05-21
资源大小:1764k
文件大小:6k
源码类别:

xml/soap/webservice

开发平台:

Visual C++

  1. /*****************************************************************************
  2.  *****************************************************************************
  3.  *
  4.  * JsiRuntime, class for managing JavaScript runtime environments
  5.  *
  6.  * The JsiRuntime class represents a JavaScript interpreter runtime
  7.  * environment, which is the overall JavaScript engine execution space
  8.  * (compiler, interpreter, decompiler, garbage collector, etc.) which
  9.  * can hold zero or more JavaScript contexts (JsiContext objects).
  10.  *
  11.  *****************************************************************************
  12.  ****************************************************************************/
  13. /****************License************************************************
  14.  * Vocalocity OpenVXI
  15.  * Copyright (C) 2004-2005 by Vocalocity, Inc. All Rights Reserved.
  16.  * This program is free software; you can redistribute it and/or
  17.  * modify it under the terms of the GNU General Public License
  18.  * as published by the Free Software Foundation; either version 2
  19.  * of the License, or (at your option) any later version.
  20.  *  
  21.  * This program is distributed in the hope that it will be useful,
  22.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  23.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  24.  * GNU General Public License for more details.
  25.  *
  26.  * You should have received a copy of the GNU General Public License
  27.  * along with this program; if not, write to the Free Software
  28.  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
  29.  * Vocalocity, the Vocalocity logo, and VocalOS are trademarks or 
  30.  * registered trademarks of Vocalocity, Inc. 
  31.  * OpenVXI is a trademark of Scansoft, Inc. and used under license 
  32.  * by Vocalocity.
  33.  ***********************************************************************/
  34. // -----1=0-------2=0-------3=0-------4=0-------5=0-------6=0-------7=0-------8
  35. #include "SBjsiInternal.h"
  36. #include "JsiRuntime.hpp"          // Defines this class
  37. #ifndef JS_THREADSAFE
  38. #include "VXItrd.h"                // For VXItrdMutex object and functions
  39. #endif
  40. #include "SBjsiLog.h"              // For logging
  41. #include <jsapi.h>                 // SpiderMonkey JavaScript API
  42. // -----1=0-------2=0-------3=0-------4=0-------5=0-------6=0-------7=0-------8
  43. // Constructor, only does initialization that cannot fail
  44. JsiRuntime::JsiRuntime( ) : SBjsiLogger(MODULE_SBJSI, NULL, 0), runtime(NULL)
  45. #ifndef JS_THREADSAFE
  46.   , mutex(NULL)
  47. #endif
  48. {
  49. }
  50. // Destructor
  51. JsiRuntime::~JsiRuntime( )
  52. {
  53.   // Destroy the runtime
  54.   if ( runtime )
  55.     JS_DestroyRuntime (runtime);
  56. #ifndef JS_THREADSAFE
  57.   // Destroy the mutex
  58.   if ( mutex )
  59.     VXItrdMutexDestroy (&mutex);
  60. #endif
  61. }
  62. // Creation method
  63. VXIjsiResult JsiRuntime::Create (long runtimeSize,
  64.  VXIlogInterface *l,
  65.  VXIunsigned diagTagBase)
  66. {
  67.   if (( runtimeSize < 1 ) || ( ! l ))
  68.     return VXIjsi_RESULT_INVALID_ARGUMENT;
  69.   // Initialize the logging base class
  70.   SetLog (l, diagTagBase);
  71.   VXIjsiResult rc = VXIjsi_RESULT_SUCCESS;
  72. #ifndef JS_THREADSAFE
  73.   // Create the mutex
  74.   if ( VXItrdMutexCreate (&mutex) != VXItrd_RESULT_SUCCESS )
  75.     rc = VXIjsi_RESULT_SYSTEM_ERROR;
  76. #endif
  77.   if ( rc == VXIjsi_RESULT_SUCCESS ) {
  78.     // Create the runtime, must do this within a mutex to ensure
  79.     // this is thread safe due to us using a non-thread safe build
  80.     // of SpiderMonkey for performance reasons
  81.     runtime = JS_NewRuntime (runtimeSize);
  82.     if ( runtime == NULL ) {
  83.       rc = VXIjsi_RESULT_OUT_OF_MEMORY;
  84.     } else {
  85.       // Set this as the runtime's private data
  86.       JS_SetRuntimePrivate (runtime, this);
  87.       // Enable garbage collection tracking
  88.       JS_SetGCCallbackRT (runtime, JsiRuntime::GCCallback);
  89.     }
  90.   }
  91.   
  92.   return rc;
  93. }
  94. // Get a new JavaScript context for the runtime
  95. VXIjsiResult JsiRuntime::NewContext (long contextSize, JSContext **context)
  96. {
  97.   if (( contextSize < 1 ) || ( context == NULL ))
  98.     return VXIjsi_RESULT_INVALID_ARGUMENT;
  99. #ifndef JS_THREADSAFE
  100.   if ( AccessBegin( ) == false )
  101.     return VXIjsi_RESULT_SYSTEM_ERROR;
  102. #endif
  103.   // Create the context
  104.   JSContext *newContext = JS_NewContext (runtime, contextSize);
  105.   if ( newContext == NULL )
  106.     return VXIjsi_RESULT_OUT_OF_MEMORY;
  107. #ifndef JS_THREADSAFE
  108.   if ( AccessEnd( ) == false ) {
  109.     DestroyContext (&newContext);
  110.     return VXIjsi_RESULT_SYSTEM_ERROR;
  111.   }
  112. #endif
  113.   *context = newContext;
  114.   return VXIjsi_RESULT_SUCCESS;
  115. }
  116. // Destroy a JavaScript context for the runtime
  117. VXIjsiResult JsiRuntime::DestroyContext (JSContext **context)
  118. {
  119.   if (( context == NULL ) || ( *context == NULL ))
  120.     return VXIjsi_RESULT_INVALID_ARGUMENT;
  121. #ifndef JS_THREADSAFE
  122.   if ( AccessBegin( ) == false )
  123.     return VXIjsi_RESULT_SYSTEM_ERROR;
  124. #endif
  125.   // Destroy the context
  126.   JS_DestroyContext (*context);
  127. #ifndef JS_THREADSAFE
  128.   if ( AccessEnd( ) == false )
  129.     return VXIjsi_RESULT_SYSTEM_ERROR;
  130. #endif
  131.   *context = NULL;
  132.   return VXIjsi_RESULT_SUCCESS;
  133. }
  134. #ifndef JS_THREADSAFE
  135. // Flag that we are beginning and ending access of a context within
  136. // this runtime, used to ensure thread safety. For simplicity,
  137. // return true on success, false on failure (mutex error).
  138. bool JsiRuntime::AccessBegin( ) const
  139. {
  140.   return (VXItrdMutexLock (mutex) == VXItrd_RESULT_SUCCESS ? true : false);
  141. }
  142. bool JsiRuntime::AccessEnd( ) const
  143. {
  144.   return (VXItrdMutexUnlock (mutex) == VXItrd_RESULT_SUCCESS ? true : false);
  145. }
  146. #endif /* JS_THREADSAFE */
  147. // Static callback for diagnostic logging of garbage collection
  148. JSBool JS_DLL_CALLBACK 
  149. JsiRuntime::GCCallback (JSContext *context, JSGCStatus status)
  150. {
  151.   JSBool rc = JS_TRUE;
  152.   // Get the runtime, then this runtime object
  153.   JSRuntime *rt = JS_GetRuntime (context);
  154.   if ( rt == NULL ) {
  155.     rc = JS_FALSE; // Fatal error
  156.   } else {
  157.     const JsiRuntime *pThis = (const JsiRuntime *) JS_GetRuntimePrivate (rt);
  158.     if ( pThis == NULL ) {
  159.       return JS_FALSE; // Fatal error
  160.     } else {
  161.       // Log the garbage collection state
  162.       switch (status) {
  163.       case JSGC_BEGIN:
  164. pThis->Diag (SBJSI_LOG_GC, NULL, L"begin 0x%p", context);
  165. break;
  166.       case JSGC_END:
  167. pThis->Diag (SBJSI_LOG_GC, NULL, L"end 0x%p", context);
  168. break;
  169.       case JSGC_MARK_END:
  170. // Diag (SBJSI_LOG_GC, NULL, L"mark end 0x%p", context);
  171. break;
  172.       default:
  173. pThis->Diag (SBJSI_LOG_GC, NULL, L"unknown %d 0x%p", status, context);
  174. break;
  175.       }
  176.     }
  177.   }
  178.   return rc;
  179. }