1 /* 2 * Copyright (C) 2010 Google Inc. All rights reserved. 3 * 4 * Redistribution and use in source and binary forms, with or without 5 * modification, are permitted provided that the following conditions are 6 * met: 7 * 8 * * Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * * Redistributions in binary form must reproduce the above 11 * copyright notice, this list of conditions and the following disclaimer 12 * in the documentation and/or other materials provided with the 13 * distribution. 14 * * Neither the name of Google Inc. nor the names of its 15 * contributors may be used to endorse or promote products derived from 16 * this software without specific prior written permission. 17 * 18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 19 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 20 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 21 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 22 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 23 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 24 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 29 */ 30 31 #ifndef ScriptWrappable_h 32 #define ScriptWrappable_h 33 34 #include "bindings/v8/UnsafePersistent.h" 35 #include "bindings/v8/V8Utilities.h" 36 #include "bindings/v8/WrapperTypeInfo.h" 37 #include <v8.h> 38 39 // Helper to call webCoreInitializeScriptWrappableForInterface in the global namespace. 40 template <class C> inline void initializeScriptWrappableHelper(C* object) 41 { 42 void webCoreInitializeScriptWrappableForInterface(C*); 43 webCoreInitializeScriptWrappableForInterface(object); 44 } 45 46 namespace WebCore { 47 48 class ScriptWrappable { 49 public: 50 ScriptWrappable() : m_wrapperOrTypeInfo(0) { } 51 52 // Wrappables need to be initialized with their most derrived type for which 53 // bindings exist, in much the same way that certain other types need to be 54 // adopted and so forth. The overloaded initializeScriptWrappableForInterface() 55 // functions are implemented by the generated V8 bindings code. Declaring the 56 // extern function in the template avoids making a centralized header of all 57 // the bindings in the universe. C++11's extern template feature may provide 58 // a cleaner solution someday. 59 template <class C> static void init(C* object) 60 { 61 initializeScriptWrappableHelper(object); 62 } 63 64 void setWrapper(v8::Handle<v8::Object> wrapper, v8::Isolate* isolate, const WrapperConfiguration& configuration) 65 { 66 ASSERT(!containsWrapper()); 67 if (!*wrapper) { 68 m_wrapperOrTypeInfo = 0; 69 return; 70 } 71 v8::Persistent<v8::Object> persistent(isolate, wrapper); 72 configuration.configureWrapper(&persistent); 73 persistent.SetWeak(this, &setWeakCallback); 74 m_wrapperOrTypeInfo = reinterpret_cast<uintptr_t>(persistent.ClearAndLeak()) | 1; 75 ASSERT(containsWrapper()); 76 } 77 78 v8::Local<v8::Object> newLocalWrapper(v8::Isolate* isolate) const 79 { 80 return unsafePersistent().newLocal(isolate); 81 } 82 83 const WrapperTypeInfo* typeInfo() 84 { 85 if (containsTypeInfo()) 86 return reinterpret_cast<const WrapperTypeInfo*>(m_wrapperOrTypeInfo); 87 88 if (containsWrapper()) 89 return toWrapperTypeInfo(*(unsafePersistent().persistent())); 90 91 return 0; 92 } 93 94 void setTypeInfo(const WrapperTypeInfo* info) 95 { 96 m_wrapperOrTypeInfo = reinterpret_cast<uintptr_t>(info); 97 ASSERT(containsTypeInfo()); 98 } 99 100 static bool wrapperCanBeStoredInObject(const void*) { return false; } 101 static bool wrapperCanBeStoredInObject(const ScriptWrappable*) { return true; } 102 103 static void setWrapperInObject(void*, v8::Handle<v8::Object>, v8::Isolate*, const WrapperConfiguration&) 104 { 105 ASSERT_NOT_REACHED(); 106 } 107 108 static void setWrapperInObject(ScriptWrappable* object, v8::Handle<v8::Object> wrapper, v8::Isolate* isolate, const WrapperConfiguration& configuration) 109 { 110 object->setWrapper(wrapper, isolate, configuration); 111 } 112 113 static const WrapperTypeInfo* getTypeInfoFromObject(void* object) 114 { 115 ASSERT_NOT_REACHED(); 116 return 0; 117 } 118 119 static const WrapperTypeInfo* getTypeInfoFromObject(ScriptWrappable* object) 120 { 121 return object->typeInfo(); 122 } 123 124 static void setTypeInfoInObject(void* object, const WrapperTypeInfo* info) 125 { 126 ASSERT_NOT_REACHED(); 127 } 128 129 static void setTypeInfoInObject(ScriptWrappable* object, const WrapperTypeInfo* info) 130 { 131 object->setTypeInfo(info); 132 } 133 134 template<typename V8T, typename T> 135 static bool setReturnValueWithSecurityCheck(v8::ReturnValue<v8::Value> returnValue, T* object) 136 { 137 return ScriptWrappable::getUnsafeWrapperFromObject(object).template setReturnValueWithSecurityCheck<V8T>(returnValue, object); 138 } 139 140 template<typename T> 141 static bool setReturnValue(v8::ReturnValue<v8::Value> returnValue, T* object) 142 { 143 return ScriptWrappable::getUnsafeWrapperFromObject(object).setReturnValue(returnValue); 144 } 145 146 protected: 147 ~ScriptWrappable() 148 { 149 ASSERT(m_wrapperOrTypeInfo); // Assert initialization via init() even if not subsequently wrapped. 150 m_wrapperOrTypeInfo = 0; // Break UAF attempts to wrap. 151 } 152 153 private: 154 // For calling unsafePersistent and getWrapperFromObject. 155 friend class MinorGCWrapperVisitor; 156 friend class DOMDataStore; 157 158 UnsafePersistent<v8::Object> unsafePersistent() const 159 { 160 v8::Object* object = containsWrapper() ? reinterpret_cast<v8::Object*>(m_wrapperOrTypeInfo & ~1) : 0; 161 return UnsafePersistent<v8::Object>(object); 162 } 163 164 static UnsafePersistent<v8::Object> getUnsafeWrapperFromObject(void*) 165 { 166 ASSERT_NOT_REACHED(); 167 return UnsafePersistent<v8::Object>(); 168 } 169 170 static UnsafePersistent<v8::Object> getUnsafeWrapperFromObject(ScriptWrappable* object) 171 { 172 return object->unsafePersistent(); 173 } 174 175 inline bool containsWrapper() const { return (m_wrapperOrTypeInfo & 1) == 1; } 176 inline bool containsTypeInfo() const { return m_wrapperOrTypeInfo && (m_wrapperOrTypeInfo & 1) == 0; } 177 178 inline void disposeWrapper(v8::Local<v8::Object> value, const WrapperTypeInfo* info) 179 { 180 ASSERT(containsWrapper()); 181 ASSERT(value == *unsafePersistent().persistent()); 182 unsafePersistent().dispose(); 183 setTypeInfo(info); 184 } 185 186 // If zero, then this contains nothing, otherwise: 187 // If the bottom bit it set, then this contains a pointer to a wrapper object in the remainging bits. 188 // If the bottom bit is clear, then this contains a pointer to the wrapper type info in the remaining bits. 189 uintptr_t m_wrapperOrTypeInfo; 190 191 static void setWeakCallback(const v8::WeakCallbackData<v8::Object, ScriptWrappable>& data) 192 { 193 ASSERT(*data.GetParameter()->unsafePersistent().persistent() == data.GetValue()); 194 195 // Note: |object| might not be equal to |data|.GetParameter(), e.g., if ScriptWrappable isn't a left-most base class. 196 void* object = toNative(data.GetValue()); 197 const WrapperTypeInfo* info = toWrapperTypeInfo(data.GetValue()); 198 ASSERT(info->derefObjectFunction); 199 200 data.GetParameter()->disposeWrapper(data.GetValue(), info); 201 // FIXME: I noticed that 50%~ of minor GC cycle times can be consumed 202 // inside data.GetParameter()->deref(), which causes Node destructions. We should 203 // make Node destructions incremental. 204 info->derefObject(object); 205 } 206 }; 207 208 } // namespace WebCore 209 210 #endif // ScriptWrappable_h 211