Home | History | Annotate | Download | only in fxjs
      1 // Copyright 2014 PDFium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
      6 
      7 // FXJS_V8 is a layer that makes it easier to define native objects in V8, but
      8 // has no knowledge of PDF-specific native objects. It could in theory be used
      9 // to implement other sets of native objects.
     10 
     11 // PDFium code should include this file rather than including V8 headers
     12 // directly.
     13 
     14 #ifndef FXJS_FXJS_V8_H_
     15 #define FXJS_FXJS_V8_H_
     16 
     17 #include <v8-util.h>
     18 #include <v8.h>
     19 
     20 #include <map>
     21 #include <memory>
     22 #include <vector>
     23 
     24 #include "core/fxcrt/fx_string.h"
     25 #include "fxjs/cjs_v8.h"
     26 
     27 #ifdef PDF_ENABLE_XFA
     28 // Header for CFXJSE_RuntimeData. FXJS_V8 doesn't interpret this class,
     29 // it is just passed along to XFA.
     30 #include "fxjs/cfxjse_runtimedata.h"
     31 #endif  // PDF_ENABLE_XFA
     32 
     33 class CFXJS_Engine;
     34 class CFXJS_ObjDefinition;
     35 
     36 // FXJS_V8 places no restrictions on this class; it merely passes it
     37 // on to caller-provided methods.
     38 class IJS_EventContext;  // A description of the event that caused JS execution.
     39 
     40 enum FXJSOBJTYPE {
     41   FXJSOBJTYPE_DYNAMIC = 0,  // Created by native method and returned to JS.
     42   FXJSOBJTYPE_STATIC,       // Created by init and hung off of global object.
     43   FXJSOBJTYPE_GLOBAL,       // The global object itself (may only appear once).
     44 };
     45 
     46 struct FXJSErr {
     47   const wchar_t* message;
     48   const wchar_t* srcline;
     49   unsigned linnum;
     50 };
     51 
     52 // Global weak map to save dynamic objects.
     53 class V8TemplateMapTraits : public v8::StdMapTraits<void*, v8::Object> {
     54  public:
     55   typedef v8::GlobalValueMap<void*, v8::Object, V8TemplateMapTraits> MapType;
     56   typedef void WeakCallbackDataType;
     57 
     58   static WeakCallbackDataType*
     59   WeakCallbackParameter(MapType* map, void* key, v8::Local<v8::Object> value) {
     60     return key;
     61   }
     62   static MapType* MapFromWeakCallbackInfo(
     63       const v8::WeakCallbackInfo<WeakCallbackDataType>&);
     64 
     65   static void* KeyFromWeakCallbackInfo(
     66       const v8::WeakCallbackInfo<WeakCallbackDataType>& data) {
     67     return data.GetParameter();
     68   }
     69   static const v8::PersistentContainerCallbackType kCallbackType =
     70       v8::kWeakWithInternalFields;
     71   static void DisposeWeak(
     72       const v8::WeakCallbackInfo<WeakCallbackDataType>& data) {}
     73   static void OnWeakCallback(
     74       const v8::WeakCallbackInfo<WeakCallbackDataType>& data) {}
     75   static void Dispose(v8::Isolate* isolate,
     76                       v8::Global<v8::Object> value,
     77                       void* key);
     78   static void DisposeCallbackData(WeakCallbackDataType* callbackData) {}
     79 };
     80 
     81 class V8TemplateMap {
     82  public:
     83   typedef v8::GlobalValueMap<void*, v8::Object, V8TemplateMapTraits> MapType;
     84 
     85   explicit V8TemplateMap(v8::Isolate* isolate);
     86   ~V8TemplateMap();
     87 
     88   void set(void* key, v8::Local<v8::Object> handle);
     89 
     90   friend class V8TemplateMapTraits;
     91 
     92  private:
     93   MapType m_map;
     94 };
     95 
     96 class FXJS_PerIsolateData {
     97  public:
     98   ~FXJS_PerIsolateData();
     99 
    100   static void SetUp(v8::Isolate* pIsolate);
    101   static FXJS_PerIsolateData* Get(v8::Isolate* pIsolate);
    102 
    103   std::vector<std::unique_ptr<CFXJS_ObjDefinition>> m_ObjectDefnArray;
    104 #ifdef PDF_ENABLE_XFA
    105   std::unique_ptr<CFXJSE_RuntimeData> m_pFXJSERuntimeData;
    106 #endif  // PDF_ENABLE_XFA
    107   std::unique_ptr<V8TemplateMap> m_pDynamicObjsMap;
    108 
    109  protected:
    110   explicit FXJS_PerIsolateData(v8::Isolate* pIsolate);
    111 };
    112 
    113 class FXJS_ArrayBufferAllocator : public v8::ArrayBuffer::Allocator {
    114   static const size_t kMaxAllowedBytes = 0x10000000;
    115   void* Allocate(size_t length) override;
    116   void* AllocateUninitialized(size_t length) override;
    117   void Free(void* data, size_t length) override;
    118 };
    119 
    120 void FXJS_Initialize(unsigned int embedderDataSlot, v8::Isolate* pIsolate);
    121 void FXJS_Release();
    122 
    123 // Gets the global isolate set by FXJS_Initialize(), or makes a new one each
    124 // time if there is no such isolate. Returns true if a new isolate had to be
    125 // created.
    126 bool FXJS_GetIsolate(v8::Isolate** pResultIsolate);
    127 
    128 // Get the global isolate's ref count.
    129 size_t FXJS_GlobalIsolateRefCount();
    130 
    131 class CFXJS_Engine : public CJS_V8 {
    132  public:
    133   explicit CFXJS_Engine(v8::Isolate* pIsolate);
    134   ~CFXJS_Engine() override;
    135 
    136   using Constructor = void (*)(CFXJS_Engine* pEngine,
    137                                v8::Local<v8::Object> obj);
    138   using Destructor = void (*)(CFXJS_Engine* pEngine, v8::Local<v8::Object> obj);
    139 
    140   static CFXJS_Engine* CurrentEngineFromIsolate(v8::Isolate* pIsolate);
    141   static int GetObjDefnID(v8::Local<v8::Object> pObj);
    142 
    143   // Always returns a valid, newly-created objDefnID.
    144   int DefineObj(const char* sObjName,
    145                 FXJSOBJTYPE eObjType,
    146                 Constructor pConstructor,
    147                 Destructor pDestructor);
    148 
    149   void DefineObjMethod(int nObjDefnID,
    150                        const char* sMethodName,
    151                        v8::FunctionCallback pMethodCall);
    152   void DefineObjProperty(int nObjDefnID,
    153                          const char* sPropName,
    154                          v8::AccessorGetterCallback pPropGet,
    155                          v8::AccessorSetterCallback pPropPut);
    156   void DefineObjAllProperties(int nObjDefnID,
    157                               v8::NamedPropertyQueryCallback pPropQurey,
    158                               v8::NamedPropertyGetterCallback pPropGet,
    159                               v8::NamedPropertySetterCallback pPropPut,
    160                               v8::NamedPropertyDeleterCallback pPropDel);
    161   void DefineObjConst(int nObjDefnID,
    162                       const char* sConstName,
    163                       v8::Local<v8::Value> pDefault);
    164   void DefineGlobalMethod(const char* sMethodName,
    165                           v8::FunctionCallback pMethodCall);
    166   void DefineGlobalConst(const wchar_t* sConstName,
    167                          v8::FunctionCallback pConstGetter);
    168 
    169   // Called after FXJS_Define* calls made.
    170   void InitializeEngine();
    171   void ReleaseEngine();
    172 
    173   // Called after FXJS_InitializeEngine call made.
    174   int Execute(const WideString& script, FXJSErr* perror);
    175 
    176   v8::Local<v8::Object> GetThisObj();
    177 
    178   v8::Local<v8::Object> NewFxDynamicObj(int nObjDefnID, bool bStatic = false);
    179 
    180   // Native object binding.
    181   void SetObjectPrivate(v8::Local<v8::Object> pObj, void* p);
    182   void* GetObjectPrivate(v8::Local<v8::Object> pObj);
    183   static void FreeObjectPrivate(void* p);
    184   static void FreeObjectPrivate(v8::Local<v8::Object> pObj);
    185 
    186   void Error(const WideString& message);
    187 
    188  protected:
    189   CFXJS_Engine();
    190 
    191  private:
    192   std::vector<v8::Global<v8::Object>*> m_StaticObjects;
    193 };
    194 
    195 #endif  // FXJS_FXJS_V8_H_
    196