blob: 69f7894d07d3c339a64b97beb0e7aa8e1d60ac8d [file] [log] [blame]
/*
* Copyright 2018 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#define LOG_TAG "ITransactionCompletedListener"
//#define LOG_NDEBUG 0
#include <gui/ITransactionCompletedListener.h>
namespace android {
namespace { // Anonymous
enum class Tag : uint32_t {
ON_TRANSACTION_COMPLETED = IBinder::FIRST_CALL_TRANSACTION,
LAST = ON_TRANSACTION_COMPLETED,
};
} // Anonymous namespace
status_t FrameEventHistoryStats::writeToParcel(Parcel* output) const {
status_t err = output->writeUint64(frameNumber);
if (err != NO_ERROR) return err;
if (gpuCompositionDoneFence) {
err = output->writeBool(true);
if (err != NO_ERROR) return err;
err = output->write(*gpuCompositionDoneFence);
} else {
err = output->writeBool(false);
}
if (err != NO_ERROR) return err;
err = output->writeInt64(compositorTiming.deadline);
if (err != NO_ERROR) return err;
err = output->writeInt64(compositorTiming.interval);
if (err != NO_ERROR) return err;
err = output->writeInt64(compositorTiming.presentLatency);
if (err != NO_ERROR) return err;
err = output->writeInt64(refreshStartTime);
if (err != NO_ERROR) return err;
err = output->writeInt64(dequeueReadyTime);
return err;
}
status_t FrameEventHistoryStats::readFromParcel(const Parcel* input) {
status_t err = input->readUint64(&frameNumber);
if (err != NO_ERROR) return err;
bool hasFence = false;
err = input->readBool(&hasFence);
if (err != NO_ERROR) return err;
if (hasFence) {
gpuCompositionDoneFence = new Fence();
err = input->read(*gpuCompositionDoneFence);
if (err != NO_ERROR) return err;
}
err = input->readInt64(&(compositorTiming.deadline));
if (err != NO_ERROR) return err;
err = input->readInt64(&(compositorTiming.interval));
if (err != NO_ERROR) return err;
err = input->readInt64(&(compositorTiming.presentLatency));
if (err != NO_ERROR) return err;
err = input->readInt64(&refreshStartTime);
if (err != NO_ERROR) return err;
err = input->readInt64(&dequeueReadyTime);
return err;
}
status_t SurfaceStats::writeToParcel(Parcel* output) const {
status_t err = output->writeStrongBinder(surfaceControl);
if (err != NO_ERROR) {
return err;
}
err = output->writeInt64(acquireTime);
if (err != NO_ERROR) {
return err;
}
if (previousReleaseFence) {
err = output->writeBool(true);
if (err != NO_ERROR) {
return err;
}
err = output->write(*previousReleaseFence);
} else {
err = output->writeBool(false);
}
err = output->writeUint32(transformHint);
if (err != NO_ERROR) {
return err;
}
err = output->writeParcelable(eventStats);
return err;
}
status_t SurfaceStats::readFromParcel(const Parcel* input) {
status_t err = input->readStrongBinder(&surfaceControl);
if (err != NO_ERROR) {
return err;
}
err = input->readInt64(&acquireTime);
if (err != NO_ERROR) {
return err;
}
bool hasFence = false;
err = input->readBool(&hasFence);
if (err != NO_ERROR) {
return err;
}
if (hasFence) {
previousReleaseFence = new Fence();
err = input->read(*previousReleaseFence);
if (err != NO_ERROR) {
return err;
}
}
err = input->readUint32(&transformHint);
if (err != NO_ERROR) {
return err;
}
err = input->readParcelable(&eventStats);
return err;
}
status_t TransactionStats::writeToParcel(Parcel* output) const {
status_t err = output->writeInt64Vector(callbackIds);
if (err != NO_ERROR) {
return err;
}
err = output->writeInt64(latchTime);
if (err != NO_ERROR) {
return err;
}
if (presentFence) {
err = output->writeBool(true);
if (err != NO_ERROR) {
return err;
}
err = output->write(*presentFence);
} else {
err = output->writeBool(false);
}
if (err != NO_ERROR) {
return err;
}
return output->writeParcelableVector(surfaceStats);
}
status_t TransactionStats::readFromParcel(const Parcel* input) {
status_t err = input->readInt64Vector(&callbackIds);
if (err != NO_ERROR) {
return err;
}
err = input->readInt64(&latchTime);
if (err != NO_ERROR) {
return err;
}
bool hasFence = false;
err = input->readBool(&hasFence);
if (err != NO_ERROR) {
return err;
}
if (hasFence) {
presentFence = new Fence();
err = input->read(*presentFence);
if (err != NO_ERROR) {
return err;
}
}
return input->readParcelableVector(&surfaceStats);
}
status_t ListenerStats::writeToParcel(Parcel* output) const {
status_t err = output->writeInt32(static_cast<int32_t>(transactionStats.size()));
if (err != NO_ERROR) {
return err;
}
for (const auto& stats : transactionStats) {
err = output->writeParcelable(stats);
if (err != NO_ERROR) {
return err;
}
}
return NO_ERROR;
}
status_t ListenerStats::readFromParcel(const Parcel* input) {
int32_t transactionStats_size = input->readInt32();
for (int i = 0; i < transactionStats_size; i++) {
TransactionStats stats;
status_t err = input->readParcelable(&stats);
if (err != NO_ERROR) {
return err;
}
transactionStats.push_back(stats);
}
return NO_ERROR;
}
ListenerStats ListenerStats::createEmpty(const sp<IBinder>& listener,
const std::unordered_set<CallbackId>& callbackIds) {
ListenerStats listenerStats;
listenerStats.listener = listener;
listenerStats.transactionStats.emplace_back(callbackIds);
return listenerStats;
}
class BpTransactionCompletedListener : public SafeBpInterface<ITransactionCompletedListener> {
public:
explicit BpTransactionCompletedListener(const sp<IBinder>& impl)
: SafeBpInterface<ITransactionCompletedListener>(impl, "BpTransactionCompletedListener") {
}
~BpTransactionCompletedListener() override;
void onTransactionCompleted(ListenerStats stats) override {
callRemoteAsync<decltype(&ITransactionCompletedListener::
onTransactionCompleted)>(Tag::ON_TRANSACTION_COMPLETED,
stats);
}
};
// Out-of-line virtual method definitions to trigger vtable emission in this translation unit (see
// clang warning -Wweak-vtables)
BpTransactionCompletedListener::~BpTransactionCompletedListener() = default;
IMPLEMENT_META_INTERFACE(TransactionCompletedListener, "android.gui.ITransactionComposerListener");
status_t BnTransactionCompletedListener::onTransact(uint32_t code, const Parcel& data,
Parcel* reply, uint32_t flags) {
if (code < IBinder::FIRST_CALL_TRANSACTION || code > static_cast<uint32_t>(Tag::LAST)) {
return BBinder::onTransact(code, data, reply, flags);
}
auto tag = static_cast<Tag>(code);
switch (tag) {
case Tag::ON_TRANSACTION_COMPLETED:
return callLocalAsync(data, reply,
&ITransactionCompletedListener::onTransactionCompleted);
}
}
}; // namespace android