blob: a74c4a3c3d54c727387dd8a52bed48433785209d [file] [log] [blame]
/*
* Copyright (C) 2009 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef V8ValueCache_h
#define V8ValueCache_h
#include <v8.h>
#include <wtf/HashMap.h>
#include <wtf/RefPtr.h>
#include <wtf/text/AtomicString.h>
#include <wtf/text/WTFString.h>
namespace WebCore {
class ExternalStringVisitor;
class StringCache {
public:
StringCache() { }
v8::Local<v8::String> v8ExternalString(StringImpl* stringImpl, v8::Isolate* isolate)
{
if (m_lastStringImpl.get() == stringImpl && m_lastV8String.IsWeak())
return v8::Local<v8::String>::New(m_lastV8String);
return v8ExternalStringSlow(stringImpl, isolate);
}
void clearOnGC()
{
m_lastStringImpl = 0;
m_lastV8String.Clear();
}
void remove(StringImpl*);
void reportMemoryUsage(MemoryObjectInfo*) const;
private:
v8::Local<v8::String> v8ExternalStringSlow(StringImpl*, v8::Isolate*);
HashMap<StringImpl*, v8::String*> m_stringCache;
v8::Persistent<v8::String> m_lastV8String;
// Note: RefPtr is a must as we cache by StringImpl* equality, not identity
// hence lastStringImpl might be not a key of the cache (in sense of identity)
// and hence it's not refed on addition.
RefPtr<StringImpl> m_lastStringImpl;
};
const int numberOfCachedSmallIntegers = 64;
// WebCoreStringResource is a helper class for v8ExternalString. It is used
// to manage the life-cycle of the underlying buffer of the external string.
class WebCoreStringResource : public v8::String::ExternalStringResource {
public:
explicit WebCoreStringResource(const String& string)
: m_plainString(string)
{
#ifndef NDEBUG
m_threadId = WTF::currentThread();
#endif
ASSERT(!string.isNull());
v8::V8::AdjustAmountOfExternalAllocatedMemory(2 * string.length());
}
explicit WebCoreStringResource(const AtomicString& string)
: m_plainString(string.string())
, m_atomicString(string)
{
#ifndef NDEBUG
m_threadId = WTF::currentThread();
#endif
ASSERT(!string.isNull());
v8::V8::AdjustAmountOfExternalAllocatedMemory(2 * string.length());
}
virtual ~WebCoreStringResource()
{
#ifndef NDEBUG
ASSERT(m_threadId == WTF::currentThread());
#endif
int reducedExternalMemory = -2 * m_plainString.length();
if (m_plainString.impl() != m_atomicString.impl() && !m_atomicString.isNull())
reducedExternalMemory *= 2;
v8::V8::AdjustAmountOfExternalAllocatedMemory(reducedExternalMemory);
}
virtual const uint16_t* data() const
{
return reinterpret_cast<const uint16_t*>(m_plainString.impl()->characters());
}
virtual size_t length() const { return m_plainString.impl()->length(); }
String webcoreString() { return m_plainString; }
AtomicString atomicString()
{
#ifndef NDEBUG
ASSERT(m_threadId == WTF::currentThread());
#endif
if (m_atomicString.isNull()) {
m_atomicString = AtomicString(m_plainString);
ASSERT(!m_atomicString.isNull());
if (m_plainString.impl() != m_atomicString.impl())
v8::V8::AdjustAmountOfExternalAllocatedMemory(2 * m_atomicString.length());
}
return m_atomicString;
}
void visitStrings(ExternalStringVisitor*);
static WebCoreStringResource* toStringResource(v8::Handle<v8::String> v8String)
{
return static_cast<WebCoreStringResource*>(v8String->GetExternalStringResource());
}
private:
// A shallow copy of the string. Keeps the string buffer alive until the V8 engine garbage collects it.
String m_plainString;
// If this string is atomic or has been made atomic earlier the
// atomic string is held here. In the case where the string starts
// off non-atomic and becomes atomic later it is necessary to keep
// the original string alive because v8 may keep derived pointers
// into that string.
AtomicString m_atomicString;
#ifndef NDEBUG
WTF::ThreadIdentifier m_threadId;
#endif
};
class IntegerCache {
public:
IntegerCache() : m_initialized(false) { };
~IntegerCache();
v8::Handle<v8::Integer> v8Integer(int value)
{
if (!m_initialized)
createSmallIntegers();
if (0 <= value && value < numberOfCachedSmallIntegers)
return m_smallIntegers[value];
return v8::Integer::New(value);
}
v8::Handle<v8::Integer> v8UnsignedInteger(unsigned value)
{
if (!m_initialized)
createSmallIntegers();
if (value < static_cast<unsigned>(numberOfCachedSmallIntegers))
return m_smallIntegers[value];
return v8::Integer::NewFromUnsigned(value);
}
private:
void createSmallIntegers();
v8::Persistent<v8::Integer> m_smallIntegers[numberOfCachedSmallIntegers];
bool m_initialized;
};
} // namespace WebCore
#endif // V8ValueCache_h