0739fdc2a678d4f4db4bff7d8b75b473c7c4765a
[WebKit-https.git] / Source / WebKit / UIProcess / mac / TiledCoreAnimationDrawingAreaProxy.mm
1 /*
2  * Copyright (C) 2011-2018 Apple 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 #import "config.h"
27 #import "TiledCoreAnimationDrawingAreaProxy.h"
28
29 #if !PLATFORM(IOS_FAMILY)
30
31 #import "ColorSpaceData.h"
32 #import "DrawingAreaMessages.h"
33 #import "DrawingAreaProxyMessages.h"
34 #import "LayerTreeContext.h"
35 #import "WebPageProxy.h"
36 #import "WebProcessProxy.h"
37 #import <pal/spi/cocoa/QuartzCoreSPI.h>
38 #import <wtf/BlockPtr.h>
39 #import <wtf/MachSendRight.h>
40
41 namespace WebKit {
42 using namespace IPC;
43 using namespace WebCore;
44
45 TiledCoreAnimationDrawingAreaProxy::TiledCoreAnimationDrawingAreaProxy(WebPageProxy& webPageProxy, WebProcessProxy& process)
46     : DrawingAreaProxy(DrawingAreaTypeTiledCoreAnimation, webPageProxy, process)
47     , m_isWaitingForDidUpdateGeometry(false)
48 {
49 }
50
51 TiledCoreAnimationDrawingAreaProxy::~TiledCoreAnimationDrawingAreaProxy()
52 {
53     m_callbacks.invalidate(CallbackBase::Error::OwnerWasInvalidated);
54 }
55
56 void TiledCoreAnimationDrawingAreaProxy::deviceScaleFactorDidChange()
57 {
58     send(Messages::DrawingArea::SetDeviceScaleFactor(m_webPageProxy.deviceScaleFactor()));
59 }
60
61 void TiledCoreAnimationDrawingAreaProxy::sizeDidChange()
62 {
63     if (!m_webPageProxy.hasRunningProcess())
64         return;
65
66     // We only want one UpdateGeometry message in flight at once, so if we've already sent one but
67     // haven't yet received the reply we'll just return early here.
68     if (m_isWaitingForDidUpdateGeometry)
69         return;
70
71     sendUpdateGeometry();
72 }
73
74 void TiledCoreAnimationDrawingAreaProxy::colorSpaceDidChange()
75 {
76     send(Messages::DrawingArea::SetColorSpace(m_webPageProxy.colorSpace()));
77 }
78
79 void TiledCoreAnimationDrawingAreaProxy::viewLayoutSizeDidChange()
80 {
81     if (!m_webPageProxy.hasRunningProcess())
82         return;
83
84     // We only want one UpdateGeometry message in flight at once, so if we've already sent one but
85     // haven't yet received the reply we'll just return early here.
86     if (m_isWaitingForDidUpdateGeometry)
87         return;
88
89     sendUpdateGeometry();
90 }
91
92 void TiledCoreAnimationDrawingAreaProxy::enterAcceleratedCompositingMode(uint64_t backingStoreStateID, const LayerTreeContext& layerTreeContext)
93 {
94     m_webPageProxy.enterAcceleratedCompositingMode(layerTreeContext);
95 }
96
97 void TiledCoreAnimationDrawingAreaProxy::exitAcceleratedCompositingMode(uint64_t backingStoreStateID, const UpdateInfo&)
98 {
99     // This should never be called.
100     ASSERT_NOT_REACHED();
101 }
102
103 void TiledCoreAnimationDrawingAreaProxy::updateAcceleratedCompositingMode(uint64_t backingStoreStateID, const LayerTreeContext& layerTreeContext)
104 {
105     m_webPageProxy.updateAcceleratedCompositingMode(layerTreeContext);
106 }
107
108 void TiledCoreAnimationDrawingAreaProxy::didUpdateGeometry()
109 {
110     ASSERT(m_isWaitingForDidUpdateGeometry);
111
112     m_isWaitingForDidUpdateGeometry = false;
113
114     IntSize viewLayoutSize = m_webPageProxy.viewLayoutSize();
115
116     // If the WKView was resized while we were waiting for a DidUpdateGeometry reply from the web process,
117     // we need to resend the new size here.
118     if (m_lastSentSize != m_size || m_lastSentViewLayoutSize != viewLayoutSize)
119         sendUpdateGeometry();
120 }
121
122 void TiledCoreAnimationDrawingAreaProxy::waitForDidUpdateActivityState(ActivityStateChangeID)
123 {
124     Seconds activityStateUpdateTimeout = Seconds::fromMilliseconds(250);
125     process().connection()->waitForAndDispatchImmediately<Messages::WebPageProxy::DidUpdateActivityState>(m_webPageProxy.pageID(), activityStateUpdateTimeout, IPC::WaitForOption::InterruptWaitingIfSyncMessageArrives);
126 }
127
128 void TiledCoreAnimationDrawingAreaProxy::intrinsicContentSizeDidChange(const IntSize& newIntrinsicContentSize)
129 {
130     if (m_webPageProxy.viewLayoutSize().width() > 0)
131         m_webPageProxy.intrinsicContentSizeDidChange(newIntrinsicContentSize);
132 }
133
134 void TiledCoreAnimationDrawingAreaProxy::willSendUpdateGeometry()
135 {
136     m_lastSentViewLayoutSize = m_webPageProxy.viewLayoutSize();
137     m_lastSentSize = m_size;
138     m_isWaitingForDidUpdateGeometry = true;
139 }
140
141 MachSendRight TiledCoreAnimationDrawingAreaProxy::createFence()
142 {
143     if (!m_webPageProxy.hasRunningProcess())
144         return MachSendRight();
145
146     RetainPtr<CAContext> rootLayerContext = [m_webPageProxy.acceleratedCompositingRootLayer() context];
147     if (!rootLayerContext)
148         return MachSendRight();
149
150     // Don't fence if we don't have a connection, because the message
151     // will likely get dropped on the floor (if the Web process is terminated)
152     // or queued up until process launch completes, and there's nothing useful
153     // to synchronize in these cases.
154     if (!process().connection())
155         return MachSendRight();
156
157     // Don't fence if we have incoming synchronous messages, because we may not
158     // be able to reply to the message until the fence times out.
159     if (process().connection()->hasIncomingSyncMessage())
160         return MachSendRight();
161
162     MachSendRight fencePort = MachSendRight::adopt([rootLayerContext createFencePort]);
163
164     // Invalidate the fence if a synchronous message arrives while it's installed,
165     // because we won't be able to reply during the fence-wait.
166     uint64_t callbackID = process().connection()->installIncomingSyncMessageCallback([rootLayerContext] {
167         [rootLayerContext invalidateFences];
168     });
169     RefPtr<WebPageProxy> retainedPage = &m_webPageProxy;
170     [CATransaction addCommitHandler:[callbackID, retainedPage] {
171         if (!retainedPage->hasRunningProcess())
172             return;
173         if (Connection* connection = retainedPage->process().connection())
174             connection->uninstallIncomingSyncMessageCallback(callbackID);
175     } forPhase:kCATransactionPhasePostCommit];
176
177     return fencePort;
178 }
179
180 void TiledCoreAnimationDrawingAreaProxy::sendUpdateGeometry()
181 {
182     ASSERT(!m_isWaitingForDidUpdateGeometry);
183
184     willSendUpdateGeometry();
185     send(Messages::DrawingArea::UpdateGeometry(m_size, true /* flushSynchronously */, createFence()));
186 }
187
188 void TiledCoreAnimationDrawingAreaProxy::adjustTransientZoom(double scale, FloatPoint origin)
189 {
190     send(Messages::DrawingArea::AdjustTransientZoom(scale, origin));
191 }
192
193 void TiledCoreAnimationDrawingAreaProxy::commitTransientZoom(double scale, FloatPoint origin)
194 {
195     send(Messages::DrawingArea::CommitTransientZoom(scale, origin));
196 }
197
198 void TiledCoreAnimationDrawingAreaProxy::dispatchAfterEnsuringDrawing(WTF::Function<void (CallbackBase::Error)>&& callback)
199 {
200     if (!m_webPageProxy.hasRunningProcess()) {
201         callback(CallbackBase::Error::OwnerWasInvalidated);
202         return;
203     }
204
205     send(Messages::DrawingArea::AddTransactionCallbackID(m_callbacks.put(WTFMove(callback), nullptr)));
206 }
207
208 void TiledCoreAnimationDrawingAreaProxy::dispatchPresentationCallbacksAfterFlushingLayers(const Vector<CallbackID>& callbackIDs)
209 {
210     for (auto& callbackID : callbackIDs) {
211         if (auto callback = m_callbacks.take<VoidCallback>(callbackID))
212             callback->performCallback();
213     }
214 }
215
216 } // namespace WebKit
217
218 #endif // !PLATFORM(IOS_FAMILY)