/* * Copyright (C) 2009, 2010 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: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 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. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT * OWNER OR 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 CrossThreadCopier_h #define CrossThreadCopier_h #include "platform/PlatformExport.h" #include "platform/heap/Handle.h" #include "wtf/Assertions.h" #include "wtf/Forward.h" #include "wtf/PassOwnPtr.h" #include "wtf/PassRefPtr.h" #include "wtf/RawPtr.h" #include "wtf/RefPtr.h" #include "wtf/ThreadSafeRefCounted.h" #include "wtf/TypeTraits.h" namespace blink { class IntRect; class IntSize; class KURL; class ResourceError; class ResourceRequest; class ResourceResponse; struct CrossThreadResourceResponseData; struct CrossThreadResourceRequestData; template struct CrossThreadCopierPassThrough { STATIC_ONLY(CrossThreadCopierPassThrough); typedef T Type; static Type copy(const T& parameter) { return parameter; } }; template struct CrossThreadCopierBase; // Integers get passed through without any changes. template struct CrossThreadCopierBase : public CrossThreadCopierPassThrough { STATIC_ONLY(CrossThreadCopierBase); }; // To allow a type to be passed across threads using its copy constructor, add a forward declaration of the type and // a CopyThreadCopierBase : public CrossThreadCopierPassThrough { }; to this file. template<> struct CrossThreadCopierBase : public CrossThreadCopierPassThrough { STATIC_ONLY(CrossThreadCopierBase); }; template<> struct CrossThreadCopierBase : public CrossThreadCopierPassThrough { STATIC_ONLY(CrossThreadCopierBase); }; // Custom copy methods. template struct CrossThreadCopierBase { STATIC_ONLY(CrossThreadCopierBase); typedef typename WTF::RemoveTemplate::Type TypeWithoutRefPtr; typedef typename WTF::RemoveTemplate::Type TypeWithoutPassRefPtr; typedef typename std::remove_pointer::type RefCountedType; // Verify that only one of the above did a change. static_assert((std::is_same, T>::value || std::is_same, T>::value || std::is_same::value), "only one type modification should be allowed"); typedef PassRefPtr Type; static Type copy(const T& refPtr) { return refPtr; } }; template struct CrossThreadCopierBase> { STATIC_ONLY(CrossThreadCopierBase); typedef PassOwnPtr Type; static Type copy(Type ownPtr) { return ownPtr; } }; template struct CrossThreadCopierBase*> { STATIC_ONLY(CrossThreadCopierBase); typedef WeakMember* Type; static Type copy(Type ptr) { return ptr; } }; template<> struct CrossThreadCopierBase { STATIC_ONLY(CrossThreadCopierBase); typedef KURL Type; PLATFORM_EXPORT static Type copy(const KURL&); }; template<> struct CrossThreadCopierBase { STATIC_ONLY(CrossThreadCopierBase); typedef String Type; PLATFORM_EXPORT static Type copy(const String&); }; template<> struct CrossThreadCopierBase { STATIC_ONLY(CrossThreadCopierBase); typedef ResourceError Type; PLATFORM_EXPORT static Type copy(const ResourceError&); }; template<> struct CrossThreadCopierBase { STATIC_ONLY(CrossThreadCopierBase); typedef PassOwnPtr Type; PLATFORM_EXPORT static Type copy(const ResourceRequest&); }; template<> struct CrossThreadCopierBase { STATIC_ONLY(CrossThreadCopierBase); typedef PassOwnPtr Type; PLATFORM_EXPORT static Type copy(const ResourceResponse&); }; template struct CrossThreadCopierBase { STATIC_ONLY(CrossThreadCopierBase); typedef typename std::remove_pointer::type TypeWithoutPointer; typedef RawPtr Type; static Type copy(const T& ptr) { return ptr; } }; template struct CrossThreadCopierBase> { STATIC_ONLY(CrossThreadCopierBase); typedef RawPtr Type; static Type copy(const Type& ptr) { return ptr; } }; template struct CrossThreadCopierBase> { STATIC_ONLY(CrossThreadCopierBase); typedef RawPtr Type; static Type copy(const Member& ptr) { return ptr; } }; template struct CrossThreadCopierBase> { STATIC_ONLY(CrossThreadCopierBase); typedef RawPtr Type; static Type copy(const WeakMember& ptr) { return ptr; } }; template struct CrossThreadCopier : public CrossThreadCopierBase::value, WTF::IsSubclassOfTemplate::Type, ThreadSafeRefCounted>::value || WTF::IsSubclassOfTemplate::type, ThreadSafeRefCounted>::value || WTF::IsSubclassOfTemplate::Type, ThreadSafeRefCounted>::value, WTF::IsSubclassOfTemplate::type, GarbageCollected>::value || WTF::IsSubclassOfTemplate::Type, GarbageCollected>::value || WTF::IsSubclassOfTemplate::Type, GarbageCollected>::value || WTF::IsSubclassOfTemplate::Type, GarbageCollected>::value, T> { STATIC_ONLY(CrossThreadCopier); }; // |T| is |C*| or |const WeakPtr&|. template struct AllowCrossThreadAccessWrapper { STACK_ALLOCATED(); public: explicit AllowCrossThreadAccessWrapper(T value) : m_value(value) { } T value() const { return m_value; } private: // This raw pointer is safe since AllowCrossThreadAccessWrapper is // always stack-allocated. Ideally this should be Member if T is // garbage-collected and T* otherwise, but we don't want to introduce // another template magic just for distinguishing Member from T*. // From the perspective of GC, T* always works correctly. GC_PLUGIN_IGNORE("") T m_value; }; template struct CrossThreadCopierBase> { STATIC_ONLY(CrossThreadCopierBase); typedef T Type; static Type copy(const AllowCrossThreadAccessWrapper& wrapper) { return wrapper.value(); } }; template AllowCrossThreadAccessWrapper AllowCrossThreadAccess(T* value) { return AllowCrossThreadAccessWrapper(value); } template AllowCrossThreadAccessWrapper&> AllowCrossThreadAccess(const WeakPtr& value) { return AllowCrossThreadAccessWrapper&>(value); } } // namespace blink #endif // CrossThreadCopier_h