blob: 21e267870153b6b0c7e8d31559dbf8b52de8e1c5 [file] [log] [blame]
Marissa Walle2ffb422018-10-12 11:33:52 -07001/*
2 * Copyright 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#pragma once
18
19#include <condition_variable>
20#include <deque>
21#include <mutex>
22#include <thread>
23#include <unordered_map>
24
25#include <android-base/thread_annotations.h>
26
27#include <binder/IBinder.h>
28#include <gui/ITransactionCompletedListener.h>
Marissa Wallfda30bb2018-10-12 11:34:28 -070029#include <ui/Fence.h>
Marissa Walle2ffb422018-10-12 11:33:52 -070030
31namespace android {
32
Marissa Walld600d572019-03-26 15:38:50 -070033struct CallbackIdsHash {
34 // CallbackId vectors have several properties that let us get away with this simple hash.
35 // 1) CallbackIds are never 0 so if something has gone wrong and our CallbackId vector is
36 // empty we can still hash 0.
37 // 2) CallbackId vectors for the same listener either are identical or contain none of the
38 // same members. It is sufficient to just check the first CallbackId in the vectors. If
39 // they match, they are the same. If they do not match, they are not the same.
Greg Kaisera9e843a2019-04-01 06:23:09 -070040 std::size_t operator()(const std::vector<CallbackId>& callbackIds) const {
Marissa Walld600d572019-03-26 15:38:50 -070041 return std::hash<CallbackId>{}((callbackIds.empty()) ? 0 : callbackIds.front());
42 }
43};
44
Marissa Walle2ffb422018-10-12 11:33:52 -070045class CallbackHandle : public RefBase {
46public:
47 CallbackHandle(const sp<ITransactionCompletedListener>& transactionListener,
48 const std::vector<CallbackId>& ids, const sp<IBinder>& sc);
49
50 sp<ITransactionCompletedListener> listener;
51 std::vector<CallbackId> callbackIds;
52 sp<IBinder> surfaceControl;
Marissa Wallfda30bb2018-10-12 11:34:28 -070053
54 bool releasePreviousBuffer = false;
Marissa Wall5a68a772018-12-22 17:43:42 -080055 sp<Fence> previousReleaseFence;
Marissa Wallfda30bb2018-10-12 11:34:28 -070056 nsecs_t acquireTime = -1;
Marissa Wall5a68a772018-12-22 17:43:42 -080057 nsecs_t latchTime = -1;
Marissa Walle2ffb422018-10-12 11:33:52 -070058};
59
60class TransactionCompletedThread {
61public:
62 ~TransactionCompletedThread();
63
64 void run();
65
Marissa Walld600d572019-03-26 15:38:50 -070066 // Adds listener and callbackIds in case there are no SurfaceControls that are supposed
67 // to be included in the callback. This functions should be call before attempting to add any
68 // callback handles.
69 status_t addCallback(const sp<ITransactionCompletedListener>& transactionListener,
70 const std::vector<CallbackId>& callbackIds);
71
Marissa Walle2ffb422018-10-12 11:33:52 -070072 // Informs the TransactionCompletedThread that there is a Transaction with a CallbackHandle
73 // that needs to be latched and presented this frame. This function should be called once the
74 // layer has received the CallbackHandle so the TransactionCompletedThread knows not to send
75 // a callback for that Listener/Transaction pair until that CallbackHandle has been latched and
76 // presented.
Marissa Walld600d572019-03-26 15:38:50 -070077 status_t registerPendingCallbackHandle(const sp<CallbackHandle>& handle);
Marissa Wall5a68a772018-12-22 17:43:42 -080078 // Notifies the TransactionCompletedThread that a pending CallbackHandle has been presented.
Marissa Walld600d572019-03-26 15:38:50 -070079 status_t addPresentedCallbackHandles(const std::deque<sp<CallbackHandle>>& handles);
Marissa Walle2ffb422018-10-12 11:33:52 -070080
81 // Adds the Transaction CallbackHandle from a layer that does not need to be relatched and
82 // presented this frame.
Marissa Walld600d572019-03-26 15:38:50 -070083 status_t addUnpresentedCallbackHandle(const sp<CallbackHandle>& handle);
Marissa Wall3dad52d2019-03-22 14:03:19 -070084
Marissa Wallfda30bb2018-10-12 11:34:28 -070085 void addPresentFence(const sp<Fence>& presentFence);
86
Marissa Walle2ffb422018-10-12 11:33:52 -070087 void sendCallbacks();
88
89private:
90 void threadMain();
91
Marissa Walld600d572019-03-26 15:38:50 -070092 status_t findTransactionStats(const sp<ITransactionCompletedListener>& listener,
93 const std::vector<CallbackId>& callbackIds,
94 TransactionStats** outTransactionStats) REQUIRES(mMutex);
95
Marissa Wall3dad52d2019-03-22 14:03:19 -070096 status_t addCallbackHandle(const sp<CallbackHandle>& handle) REQUIRES(mMutex);
Marissa Walle2ffb422018-10-12 11:33:52 -070097
98 class ThreadDeathRecipient : public IBinder::DeathRecipient {
99 public:
100 // This function is a no-op. isBinderAlive needs a linked DeathRecipient to work.
101 // Death recipients needs a binderDied function.
102 //
103 // (isBinderAlive checks if BpBinder's mAlive is 0. mAlive is only set to 0 in sendObituary.
104 // sendObituary is only called if linkToDeath was called with a DeathRecipient.)
105 void binderDied(const wp<IBinder>& /*who*/) override {}
106 };
107 sp<ThreadDeathRecipient> mDeathRecipient;
108
Marissa Walld600d572019-03-26 15:38:50 -0700109 struct ITransactionCompletedListenerHash {
110 std::size_t operator()(const sp<ITransactionCompletedListener>& listener) const {
111 return std::hash<IBinder*>{}((listener) ? IInterface::asBinder(listener).get()
112 : nullptr);
Marissa Walle2ffb422018-10-12 11:33:52 -0700113 }
114 };
115
Marissa Wall99343ba2018-11-13 10:39:08 -0800116 // Protects the creation and destruction of mThread
117 std::mutex mThreadMutex;
118
119 std::thread mThread GUARDED_BY(mThreadMutex);
Marissa Walle2ffb422018-10-12 11:33:52 -0700120
121 std::mutex mMutex;
122 std::condition_variable_any mConditionVariable;
123
124 std::unordered_map<
Marissa Walld600d572019-03-26 15:38:50 -0700125 sp<ITransactionCompletedListener>,
Marissa Walle2ffb422018-10-12 11:33:52 -0700126 std::unordered_map<std::vector<CallbackId>, uint32_t /*count*/, CallbackIdsHash>,
Marissa Walld600d572019-03-26 15:38:50 -0700127 ITransactionCompletedListenerHash>
Marissa Walle2ffb422018-10-12 11:33:52 -0700128 mPendingTransactions GUARDED_BY(mMutex);
Marissa Walld600d572019-03-26 15:38:50 -0700129 std::unordered_map<sp<ITransactionCompletedListener>, std::deque<TransactionStats>,
130 ITransactionCompletedListenerHash>
131 mCompletedTransactions GUARDED_BY(mMutex);
Marissa Walle2ffb422018-10-12 11:33:52 -0700132
133 bool mRunning GUARDED_BY(mMutex) = false;
134 bool mKeepRunning GUARDED_BY(mMutex) = true;
Marissa Wallfda30bb2018-10-12 11:34:28 -0700135
136 sp<Fence> mPresentFence GUARDED_BY(mMutex);
Marissa Walle2ffb422018-10-12 11:33:52 -0700137};
138
139} // namespace android