packages/services/Car/cpp/car_binder_lib/largeParcelable/include/LargeParcelable.h

146 lines
5.0 KiB
C++

/*
* Copyright (C) 2021 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.
*/
#ifndef CPP_CAR_BINDER_LIB_LARGEPARCELABLE_INCLUDE_LARGEPARCELABLE_H_
#define CPP_CAR_BINDER_LIB_LARGEPARCELABLE_INCLUDE_LARGEPARCELABLE_H_
#include "LargeParcelableBase.h"
#include "SharedMemory.h"
#include <android/binder_parcel.h>
#include <android/binder_parcel_utils.h>
#include <android/binder_status.h>
#include <memory>
#include <optional>
namespace android {
namespace automotive {
namespace car_binder_lib {
// This class allows a stable AIDL parcelable to be marshalled to a shared memory file if its
// serialized parcel exceeds binder limitation.
template <class T>
class LargeParcelable : public LargeParcelableBase {
public:
LargeParcelable() {}
// Use an existing stable AIDL parcelable to initiate this large parcelable. The input
// parcelable's marshal/unmarshal method would be used to read/write from a shared memory
// file if needed.
//
// T must be a Parcelable.
explicit LargeParcelable(std::unique_ptr<T> parcelable) : mParcelable(std::move(parcelable)) {}
// Get the stable AIDL parcelable object. Caller is supposed to use 'LargeParcelable' class to
// parse data from 'AParcel' and then use 'getParcelable' to get the underlying parsed regular
// parcelable object.
inline const std::optional<const T*> getParcelable() const {
if (!hasDeserializedParcelable() || mParcelable == nullptr) {
return std::nullopt;
}
return mParcelable.get();
}
protected:
// Serialize this parcelable to 'dest'.
binder_status_t serialize(AParcel* dest) const override;
// Serialize a NULL parcelable for 'this' class to 'dest'.
binder_status_t serializeNullPayload(AParcel* dest) const override;
// Read a 'Parcelable' from the given AParcel. The src might be parsed to a NULL parcelable.
binder_status_t deserialize(const AParcel& src) override;
private:
std::unique_ptr<T> mParcelable;
// Serialize a nullable Parcelabel 'payload' to 'dest'.
static binder_status_t serializeNullablePayload(const T* payload, AParcel* dest);
};
template <class T>
binder_status_t LargeParcelable<T>::serializeNullablePayload(const T* payload, AParcel* dest) {
int32_t startPosition;
if (DBG_PAYLOAD) {
startPosition = AParcel_getDataPosition(dest);
}
if (payload == nullptr) {
// Write a null parcelable.
if (binder_status_t status =
::ndk::AParcel_writeNullableParcelable(dest, std::optional<T>(std::nullopt));
status != STATUS_OK) {
ALOGE("failed to write null parcelable to parcel, status: %d", status);
return status;
}
} else {
if (binder_status_t status = ::ndk::AParcel_writeParcelable(dest, *payload);
status != STATUS_OK) {
ALOGE("failed to write parcelable to parcel, status: %d", status);
return status;
}
}
if (DBG_PAYLOAD) {
ALOGD("serialize-payload, start:%d size: %d", startPosition,
(AParcel_getDataPosition(dest) - startPosition));
}
return STATUS_OK;
}
template <class T>
binder_status_t LargeParcelable<T>::serialize(AParcel* dest) const {
return serializeNullablePayload(mParcelable.get(), dest);
}
template <class T>
binder_status_t LargeParcelable<T>::serializeNullPayload(AParcel* dest) const {
return serializeNullablePayload(nullptr, dest);
}
template <class T>
binder_status_t LargeParcelable<T>::deserialize(const AParcel& src) {
int32_t startPosition = AParcel_getDataPosition(&src);
if (DBG_PAYLOAD) {
ALOGD("start position: %d", startPosition);
}
std::optional<T> parcelable;
if (binder_status_t status = ::ndk::AParcel_readNullableParcelable(&src, &parcelable);
status != OK) {
ALOGE("failed to read parcelable from parcel, status: %d", status);
return status;
}
int32_t size = (AParcel_getDataPosition(&src) - startPosition);
if (!parcelable.has_value()) {
if (DBG_PAYLOAD) {
ALOGD("deserialize-payload: null parcelable, start: %d, size: %d", startPosition, size);
}
mParcelable = nullptr;
return STATUS_OK;
}
mParcelable = std::make_unique<T>(std::move(parcelable.value()));
if (DBG_PAYLOAD) {
ALOGD("deserialize-payload, start: %d, size: %d", startPosition, size);
}
return STATUS_OK;
}
} // namespace car_binder_lib
} // namespace automotive
} // namespace android
#endif // CPP_CAR_BINDER_LIB_LARGEPARCELABLE_INCLUDE_LARGEPARCELABLE_H_