2 * Copyright (C) 2010 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
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.
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.
29 #include "APISerializedScriptValue.h"
30 #include "CallbackID.h"
31 #include "ProcessThrottler.h"
32 #include "ShareableBitmap.h"
33 #include "WKAPICast.h"
34 #include <wtf/Function.h>
35 #include <wtf/HashMap.h>
36 #include <wtf/MainThread.h>
37 #include <wtf/RefCounted.h>
38 #include <wtf/Threading.h>
42 class CallbackBase : public RefCounted<CallbackBase> {
51 virtual ~CallbackBase()
55 CallbackID callbackID() const { return m_callbackID; }
60 if (T::type() == m_type)
61 return static_cast<T*>(this);
66 virtual void invalidate(Error) = 0;
70 typedef const TypeTag* Type;
72 explicit CallbackBase(Type type, const ProcessThrottler::BackgroundActivityToken& activityToken)
74 , m_callbackID(CallbackID::generateID())
75 , m_activityToken(activityToken)
81 CallbackID m_callbackID;
82 ProcessThrottler::BackgroundActivityToken m_activityToken;
85 template<typename... T>
86 class GenericCallback : public CallbackBase {
88 typedef Function<void (T..., Error)> CallbackFunction;
90 static Ref<GenericCallback> create(CallbackFunction&& callback, const ProcessThrottler::BackgroundActivityToken& activityToken = nullptr)
92 return adoptRef(*new GenericCallback(WTFMove(callback), activityToken));
95 virtual ~GenericCallback()
97 ASSERT(m_originThread.ptr() == &Thread::current());
101 void performCallbackWithReturnValue(T... returnValue)
103 ASSERT(m_originThread.ptr() == &Thread::current());
108 auto callback = std::exchange(m_callback, std::nullopt);
109 callback.value()(returnValue..., Error::None);
112 void performCallback()
114 performCallbackWithReturnValue();
117 void invalidate(Error error = Error::Unknown) final
119 ASSERT(m_originThread.ptr() == &Thread::current());
124 auto callback = std::exchange(m_callback, std::nullopt);
125 callback.value()(typename std::remove_reference<T>::type()..., error);
129 GenericCallback(CallbackFunction&& callback, const ProcessThrottler::BackgroundActivityToken& activityToken)
130 : CallbackBase(type(), activityToken)
131 , m_callback(WTFMove(callback))
135 friend class CallbackBase;
142 std::optional<CallbackFunction> m_callback;
145 Ref<Thread> m_originThread { Thread::current() };
149 template<typename APIReturnValueType, typename InternalReturnValueType = typename APITypeInfo<APIReturnValueType>::ImplType*>
150 static typename GenericCallback<InternalReturnValueType>::CallbackFunction toGenericCallbackFunction(void* context, void (*callback)(APIReturnValueType, WKErrorRef, void*))
152 return [context, callback](InternalReturnValueType returnValue, CallbackBase::Error error) {
153 callback(toAPI(returnValue), error != CallbackBase::Error::None ? toAPI(API::Error::create().ptr()) : 0, context);
157 typedef GenericCallback<> VoidCallback;
158 typedef GenericCallback<const Vector<WebCore::IntRect>&, double> ComputedPagesCallback;
159 typedef GenericCallback<const ShareableBitmap::Handle&> ImageCallback;
162 void invalidateCallbackMap(HashMap<uint64_t, T>& callbackMap, CallbackBase::Error error)
164 auto map = WTFMove(callbackMap);
165 for (auto& callback : map.values())
166 callback->invalidate(error);
171 CallbackID put(Ref<CallbackBase>&& callback)
173 RELEASE_ASSERT(RunLoop::isMain());
174 auto callbackID = callback->callbackID();
175 RELEASE_ASSERT(callbackID.isValid());
176 RELEASE_ASSERT(!m_map.contains(callbackID.m_id));
177 m_map.set(callbackID.m_id, WTFMove(callback));
181 template<unsigned I, typename T, typename... U>
182 struct GenericCallbackType {
183 typedef typename GenericCallbackType<I - 1, U..., T>::type type;
186 template<typename... U>
187 struct GenericCallbackType<1, CallbackBase::Error, U...> {
188 typedef GenericCallback<U...> type;
191 template<typename... T>
192 CallbackID put(Function<void(T...)>&& function, const ProcessThrottler::BackgroundActivityToken& activityToken)
194 auto callback = GenericCallbackType<sizeof...(T), T...>::type::create(WTFMove(function), activityToken);
195 return put(WTFMove(callback));
199 RefPtr<T> take(CallbackID callbackID)
201 RELEASE_ASSERT(callbackID.isValid());
202 RELEASE_ASSERT(RunLoop::isMain());
203 auto base = m_map.take(callbackID.m_id);
207 return adoptRef(base.leakRef()->as<T>());
210 void invalidate(CallbackBase::Error error)
212 RELEASE_ASSERT(RunLoop::isMain());
213 invalidateCallbackMap(m_map, error);
217 HashMap<uint64_t, RefPtr<CallbackBase>> m_map;
220 } // namespace WebKit