• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2009 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
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25 
26 #ifndef V8StringResource_h
27 #define V8StringResource_h
28 
29 #include "bindings/core/v8/ExceptionState.h"
30 #include "wtf/Threading.h"
31 #include "wtf/text/AtomicString.h"
32 #include "wtf/text/WTFString.h"
33 #include <v8.h>
34 
35 namespace blink {
36 
37 class ExternalStringVisitor;
38 
39 // WebCoreStringResource is a helper class for v8ExternalString. It is used
40 // to manage the life-cycle of the underlying buffer of the external string.
41 class WebCoreStringResourceBase {
42 public:
WebCoreStringResourceBase(const String & string)43     explicit WebCoreStringResourceBase(const String& string)
44         : m_plainString(string)
45     {
46 #if ENABLE(ASSERT)
47         m_threadId = WTF::currentThread();
48 #endif
49         ASSERT(!string.isNull());
50         v8::Isolate::GetCurrent()->AdjustAmountOfExternalAllocatedMemory(memoryConsumption(string));
51     }
52 
WebCoreStringResourceBase(const AtomicString & string)53     explicit WebCoreStringResourceBase(const AtomicString& string)
54         : m_plainString(string.string())
55         , m_atomicString(string)
56     {
57 #if ENABLE(ASSERT)
58         m_threadId = WTF::currentThread();
59 #endif
60         ASSERT(!string.isNull());
61         v8::Isolate::GetCurrent()->AdjustAmountOfExternalAllocatedMemory(memoryConsumption(string));
62     }
63 
~WebCoreStringResourceBase()64     virtual ~WebCoreStringResourceBase()
65     {
66 #if ENABLE(ASSERT)
67         ASSERT(m_threadId == WTF::currentThread());
68 #endif
69         int reducedExternalMemory = -memoryConsumption(m_plainString);
70         if (m_plainString.impl() != m_atomicString.impl() && !m_atomicString.isNull())
71             reducedExternalMemory -= memoryConsumption(m_atomicString.string());
72         v8::Isolate::GetCurrent()->AdjustAmountOfExternalAllocatedMemory(reducedExternalMemory);
73     }
74 
webcoreString()75     const String& webcoreString() { return m_plainString; }
76 
atomicString()77     const AtomicString& atomicString()
78     {
79 #if ENABLE(ASSERT)
80         ASSERT(m_threadId == WTF::currentThread());
81 #endif
82         if (m_atomicString.isNull()) {
83             m_atomicString = AtomicString(m_plainString);
84             ASSERT(!m_atomicString.isNull());
85             if (m_plainString.impl() != m_atomicString.impl())
86                 v8::Isolate::GetCurrent()->AdjustAmountOfExternalAllocatedMemory(memoryConsumption(m_atomicString.string()));
87         }
88         return m_atomicString;
89     }
90 
91 protected:
92     // A shallow copy of the string. Keeps the string buffer alive until the V8 engine garbage collects it.
93     String m_plainString;
94     // If this string is atomic or has been made atomic earlier the
95     // atomic string is held here. In the case where the string starts
96     // off non-atomic and becomes atomic later it is necessary to keep
97     // the original string alive because v8 may keep derived pointers
98     // into that string.
99     AtomicString m_atomicString;
100 
101 private:
memoryConsumption(const String & string)102     static int memoryConsumption(const String& string)
103     {
104         return string.length() * (string.is8Bit() ? sizeof(LChar) : sizeof(UChar));
105     }
106 #if ENABLE(ASSERT)
107     WTF::ThreadIdentifier m_threadId;
108 #endif
109 };
110 
111 class WebCoreStringResource16 FINAL : public WebCoreStringResourceBase, public v8::String::ExternalStringResource {
112 public:
WebCoreStringResource16(const String & string)113     explicit WebCoreStringResource16(const String& string)
114         : WebCoreStringResourceBase(string)
115     {
116         ASSERT(!string.is8Bit());
117     }
118 
WebCoreStringResource16(const AtomicString & string)119     explicit WebCoreStringResource16(const AtomicString& string)
120         : WebCoreStringResourceBase(string)
121     {
122         ASSERT(!string.is8Bit());
123     }
124 
length()125     virtual size_t length() const OVERRIDE { return m_plainString.impl()->length(); }
data()126     virtual const uint16_t* data() const OVERRIDE
127     {
128         return reinterpret_cast<const uint16_t*>(m_plainString.impl()->characters16());
129     }
130 };
131 
132 class WebCoreStringResource8 FINAL : public WebCoreStringResourceBase, public v8::String::ExternalAsciiStringResource {
133 public:
WebCoreStringResource8(const String & string)134     explicit WebCoreStringResource8(const String& string)
135         : WebCoreStringResourceBase(string)
136     {
137         ASSERT(string.is8Bit());
138     }
139 
WebCoreStringResource8(const AtomicString & string)140     explicit WebCoreStringResource8(const AtomicString& string)
141         : WebCoreStringResourceBase(string)
142     {
143         ASSERT(string.is8Bit());
144     }
145 
length()146     virtual size_t length() const OVERRIDE { return m_plainString.impl()->length(); }
data()147     virtual const char* data() const OVERRIDE
148     {
149         return reinterpret_cast<const char*>(m_plainString.impl()->characters8());
150     }
151 };
152 
153 enum ExternalMode {
154     Externalize,
155     DoNotExternalize
156 };
157 
158 template <typename StringType>
159 StringType v8StringToWebCoreString(v8::Handle<v8::String>, ExternalMode);
160 String int32ToWebCoreString(int value);
161 
162 // V8StringResource is an adapter class that converts V8 values to Strings
163 // or AtomicStrings as appropriate, using multiple typecast operators.
164 enum V8StringResourceMode {
165     DefaultMode,
166     TreatNullAsEmptyString,
167     TreatNullAsNullString,
168     TreatNullAndUndefinedAsNullString
169 };
170 
171 template <V8StringResourceMode Mode = DefaultMode>
172 class V8StringResource {
173 public:
V8StringResource()174     V8StringResource()
175         : m_mode(Externalize)
176     {
177     }
178 
V8StringResource(v8::Handle<v8::Value> object)179     V8StringResource(v8::Handle<v8::Value> object)
180         : m_v8Object(object)
181         , m_mode(Externalize)
182     {
183     }
184 
185     void operator=(v8::Handle<v8::Value> object)
186     {
187         m_v8Object = object;
188     }
189 
190     void operator=(const String& string)
191     {
192         setString(string);
193     }
194 
195     void operator=(std::nullptr_t)
196     {
197         setString(String());
198     }
199 
prepare()200     bool prepare()
201     {
202         if (prepareFast())
203             return true;
204 
205         m_v8Object = m_v8Object->ToString();
206         // Handle the case where an exception is thrown as part of invoking toString on the object.
207         if (m_v8Object.IsEmpty())
208             return false;
209         return true;
210     }
211 
prepare(ExceptionState & exceptionState)212     bool prepare(ExceptionState& exceptionState)
213     {
214         if (prepareFast())
215             return true;
216 
217         v8::TryCatch block;
218         m_v8Object = m_v8Object->ToString();
219         // Handle the case where an exception is thrown as part of invoking toString on the object.
220         if (block.HasCaught()) {
221             exceptionState.rethrowV8Exception(block.Exception());
222             return false;
223         }
224         return true;
225     }
226 
String()227     operator String() const { return toString<String>(); }
AtomicString()228     operator AtomicString() const { return toString<AtomicString>(); }
229 
230 private:
prepareFast()231     bool prepareFast()
232     {
233         if (m_v8Object.IsEmpty())
234             return true;
235 
236         if (!isValid()) {
237             setString(fallbackString());
238             return true;
239         }
240 
241         if (LIKELY(m_v8Object->IsString()))
242             return true;
243 
244         if (LIKELY(m_v8Object->IsInt32())) {
245             setString(int32ToWebCoreString(m_v8Object->Int32Value()));
246             return true;
247         }
248 
249         m_mode = DoNotExternalize;
250         return false;
251     }
252 
253     bool isValid() const;
254     String fallbackString() const;
255 
setString(const String & string)256     void setString(const String& string)
257     {
258         m_string = string;
259         m_v8Object.Clear(); // To signal that String is ready.
260     }
261 
262     template <class StringType>
toString()263     StringType toString() const
264     {
265         if (LIKELY(!m_v8Object.IsEmpty()))
266             return v8StringToWebCoreString<StringType>(const_cast<v8::Handle<v8::Value>*>(&m_v8Object)->As<v8::String>(), m_mode);
267 
268         return StringType(m_string);
269     }
270 
271     v8::Handle<v8::Value> m_v8Object;
272     ExternalMode m_mode;
273     String m_string;
274 };
275 
isValid()276 template<> inline bool V8StringResource<DefaultMode>::isValid() const
277 {
278     return true;
279 }
280 
fallbackString()281 template<> inline String V8StringResource<DefaultMode>::fallbackString() const
282 {
283     ASSERT_NOT_REACHED();
284     return String();
285 }
286 
isValid()287 template<> inline bool V8StringResource<TreatNullAsEmptyString>::isValid() const
288 {
289     return !m_v8Object->IsNull();
290 }
291 
fallbackString()292 template<> inline String V8StringResource<TreatNullAsEmptyString>::fallbackString() const
293 {
294     return emptyString();
295 }
296 
isValid()297 template<> inline bool V8StringResource<TreatNullAsNullString>::isValid() const
298 {
299     return !m_v8Object->IsNull();
300 }
301 
fallbackString()302 template<> inline String V8StringResource<TreatNullAsNullString>::fallbackString() const
303 {
304     return String();
305 }
306 
isValid()307 template<> inline bool V8StringResource<TreatNullAndUndefinedAsNullString>::isValid() const
308 {
309     return !m_v8Object->IsNull() && !m_v8Object->IsUndefined();
310 }
311 
fallbackString()312 template<> inline String V8StringResource<TreatNullAndUndefinedAsNullString>::fallbackString() const
313 {
314     return String();
315 }
316 
317 } // namespace blink
318 
319 #endif // V8StringResource_h
320