/*
* Copyright (C) 2016 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_NDEBUG 0
#define LOG_TAG "C2AllocatorGralloc"
#include <utils/Log.h>
#include <android/hardware/graphics/allocator/2.0/IAllocator.h>
#include <android/hardware/graphics/mapper/2.0/IMapper.h>
#include <android/hardware/graphics/allocator/3.0/IAllocator.h>
#include <android/hardware/graphics/mapper/3.0/IMapper.h>
#include <cutils/native_handle.h>
#include <hardware/gralloc.h>
#include <C2AllocatorGralloc.h>
#include <C2Buffer.h>
#include <C2PlatformSupport.h>
namespace android {
namespace /* unnamed */ {
enum : uint64_t {
/**
* Usage mask that is passed through from gralloc to Codec 2.0 usage.
*/
PASSTHROUGH_USAGE_MASK =
~(GRALLOC_USAGE_SW_READ_MASK | GRALLOC_USAGE_SW_WRITE_MASK | GRALLOC_USAGE_PROTECTED)
};
// verify that passthrough mask is within the platform mask
static_assert((~C2MemoryUsage::PLATFORM_MASK & PASSTHROUGH_USAGE_MASK) == 0, "");
} // unnamed
C2MemoryUsage C2AndroidMemoryUsage::FromGrallocUsage(uint64_t usage) {
// gralloc does not support WRITE_PROTECTED
return C2MemoryUsage(
((usage & GRALLOC_USAGE_SW_READ_MASK) ? C2MemoryUsage::CPU_READ : 0) |
((usage & GRALLOC_USAGE_SW_WRITE_MASK) ? C2MemoryUsage::CPU_WRITE : 0) |
((usage & GRALLOC_USAGE_PROTECTED) ? C2MemoryUsage::READ_PROTECTED : 0) |
(usage & PASSTHROUGH_USAGE_MASK));
}
uint64_t C2AndroidMemoryUsage::asGrallocUsage() const {
// gralloc does not support WRITE_PROTECTED
return (((expected & C2MemoryUsage::CPU_READ) ? GRALLOC_USAGE_SW_READ_OFTEN : 0) |
((expected & C2MemoryUsage::CPU_WRITE) ? GRALLOC_USAGE_SW_WRITE_OFTEN : 0) |
((expected & C2MemoryUsage::READ_PROTECTED) ? GRALLOC_USAGE_PROTECTED : 0) |
(expected & PASSTHROUGH_USAGE_MASK));
}
using ::android::hardware::hidl_handle;
using ::android::hardware::hidl_vec;
using ::android::hardware::graphics::common::V1_0::BufferUsage;
using PixelFormat2 = ::android::hardware::graphics::common::V1_0::PixelFormat;
using PixelFormat3 = ::android::hardware::graphics::common::V1_2::PixelFormat;
using IAllocator2 = ::android::hardware::graphics::allocator::V2_0::IAllocator;
using BufferDescriptor2 = ::android::hardware::graphics::mapper::V2_0::BufferDescriptor;
using Error2 = ::android::hardware::graphics::mapper::V2_0::Error;
using IMapper2 = ::android::hardware::graphics::mapper::V2_0::IMapper;
using IAllocator3 = ::android::hardware::graphics::allocator::V3_0::IAllocator;
using BufferDescriptor3 = ::android::hardware::graphics::mapper::V3_0::BufferDescriptor;
using Error3 = ::android::hardware::graphics::mapper::V3_0::Error;
using IMapper3 = ::android::hardware::graphics::mapper::V3_0::IMapper;
namespace /* unnamed */ {
struct BufferDescriptorInfo2 {
IMapper2::BufferDescriptorInfo mapperInfo;
uint32_t stride;
};
struct BufferDescriptorInfo3 {
IMapper3::BufferDescriptorInfo mapperInfo;
uint32_t stride;
};
/* ===================================== GRALLOC ALLOCATION ==================================== */
c2_status_t maperr2error(Error2 maperr) {
switch (maperr) {
case Error2::NONE: return C2_OK;
case Error2::BAD_DESCRIPTOR: return C2_BAD_VALUE;
case Error2::BAD_BUFFER: return C2_BAD_VALUE;
case Error2::BAD_VALUE: return C2_BAD_VALUE;
case Error2::NO_RESOURCES: return C2_NO_MEMORY;
case Error2::UNSUPPORTED: return C2_CANNOT_DO;
}
return C2_CORRUPTED;
}
c2_status_t maperr2error(Error3 maperr) {
switch (maperr) {
case Error3::NONE: return C2_OK;
case Error3::BAD_DESCRIPTOR: return C2_BAD_VALUE;
case Error3::BAD_BUFFER: return C2_BAD_VALUE;
case Error3::BAD_VALUE: return C2_BAD_VALUE;
case Error3::NO_RESOURCES: return C2_NO_MEMORY;
case Error3::UNSUPPORTED: return C2_CANNOT_DO;
}
return C2_CORRUPTED;
}
bool native_handle_is_invalid(const native_handle_t *const handle) {
// perform basic validation of a native handle
if (handle == nullptr) {
// null handle is considered valid
return false;
}
return ((size_t)handle->version != sizeof(native_handle_t) ||
handle->numFds < 0 ||
handle->numInts < 0 ||
// for sanity assume handles must occupy less memory than INT_MAX bytes
handle->numFds > int((INT_MAX - handle->version) / sizeof(int)) - handle->numInts);
}
class C2HandleGralloc : public C2Handle {
private:
struct ExtraData {
uint32_t width;
uint32_t height;
uint32_t format;
uint32_t usage_lo;
uint32_t usage_hi;
uint32_t stride;
uint32_t generation;
uint32_t igbp_id_lo;
uint32_t igbp_id_hi;
uint32_t igbp_slot;
uint32_t magic;
};
enum {
NUM_INTS = sizeof(ExtraData) / sizeof(int),
};
const static uint32_t MAGIC = '\xc2gr\x00';
static
const ExtraData* getExtraData(const C2Handle *const handle) {
if (handle == nullptr
|| native_handle_is_invalid(handle)
|| handle->numInts < NUM_INTS) {
return nullptr;
}
return reinterpret_cast<const ExtraData*>(
&handle->data[handle->numFds + handle->numInts - NUM_INTS]);
}
static
ExtraData *getExtraData(C2Handle *const handle) {
return const_cast<ExtraData *>(getExtraData(const_cast<const C2Handle *const>(handle)));
}
public:
void getIgbpData(uint32_t *generation, uint64_t *igbp_id, uint32_t *igbp_slot) const {
const ExtraData *ed = getExtraData(this);
*generation = ed->generation;
*igbp_id = unsigned(ed->igbp_id_lo) | uint64_t(unsigned(ed->igbp_id_hi)) << 32;
*igbp_slot = ed->igbp_slot;
}
static bool isValid(const C2Handle *const o) {
if (o == nullptr) { // null handle is always valid
return true;
}
const ExtraData *xd = getExtraData(o);
// we cannot validate width/height/format/usage without accessing gralloc driver
return xd != nullptr && xd->magic == MAGIC;
}
static C2HandleGralloc* WrapAndMoveNativeHandle(
const native_handle_t *const handle,
uint32_t width, uint32_t height, uint32_t format, uint64_t usage,
uint32_t stride, uint32_t generation, uint64_t igbp_id = 0, uint32_t igbp_slot = 0) {
//CHECK(handle != nullptr);
if (native_handle_is_invalid(handle) ||
handle->numInts > int((INT_MAX - handle->version) / sizeof(int)) - NUM_INTS - handle->numFds) {
return nullptr;
}
ExtraData xd = {
width, height, format, uint32_t(usage & 0xFFFFFFFF), uint32_t(usage >> 32),
stride, generation, uint32_t(igbp_id & 0xFFFFFFFF), uint32_t(igbp_id >> 32),
igbp_slot, MAGIC
};
native_handle_t *res = native_handle_create(handle->numFds, handle->numInts + NUM_INTS);
if (res != nullptr) {
memcpy(&res->data, &handle->data, sizeof(int) * (handle->numFds + handle->numInts));
*getExtraData(res) = xd;
}
return reinterpret_cast<C2HandleGralloc *>(res);
}
static C2HandleGralloc* WrapNativeHandle(
const native_handle_t *const handle,
uint32_t width, uint32_t height, uint32_t format, uint64_t usage,
uint32_t stride, uint32_t generation, uint64_t igbp_id = 0, uint32_t igbp_slot = 0) {
if (handle == nullptr) {
return nullptr;
}
native_handle_t *clone = native_handle_clone(handle);
if (clone == nullptr) {
return nullptr;
}
C2HandleGralloc *res = WrapAndMoveNativeHandle(
clone, width, height, format, usage, stride, generation, igbp_id, igbp_slot);
if (res == nullptr) {
native_handle_close(clone);
}
native_handle_delete(clone);
return res;
}
static bool MigrateNativeHandle(
native_handle_t *handle,
uint32_t generation, uint64_t igbp_id, uint32_t igbp_slot) {
if (handle == nullptr || !isValid(handle)) {
return false;
}
ExtraData *ed = getExtraData(handle);
if (!ed) return false;
ed->generation = generation;
ed->igbp_id_lo = uint32_t(igbp_id & 0xFFFFFFFF);
ed->igbp_id_hi = uint32_t(igbp_id >> 32);
ed->igbp_slot = igbp_slot;
return true;
}
static native_handle_t* UnwrapNativeHandle(
const C2Handle *const handle) {
const ExtraData *xd = getExtraData(handle);
if (xd == nullptr || xd->magic != MAGIC) {
return nullptr;
}
native_handle_t *res = native_handle_create(handle->numFds, handle->numInts - NUM_INTS);
if (res != nullptr) {
memcpy(&res->data, &handle->data, sizeof(int) * (res->numFds + res->numInts));
}
return res;
}
static const C2HandleGralloc* Import(
const C2Handle *const handle,
uint32_t *width, uint32_t *height, uint32_t *format,
uint64_t *usage, uint32_t *stride,
uint32_t *generation, uint64_t *igbp_id, uint32_t *igbp_slot) {
const ExtraData *xd = getExtraData(handle);
if (xd == nullptr) {
return nullptr;
}
*width = xd->width;
*height = xd->height;
*format = xd->format;
*usage = xd->usage_lo | (uint64_t(xd->usage_hi) << 32);
*stride = xd->stride;
*generation = xd->generation;
*igbp_id = xd->igbp_id_lo | (uint64_t(xd->igbp_id_hi) << 32);
*igbp_slot = xd->igbp_slot;
return reinterpret_cast<const C2HandleGralloc *>(handle);
}
};
} // unnamed namespace
native_handle_t *UnwrapNativeCodec2GrallocHandle(const C2Handle *const handle) {
return C2HandleGralloc::UnwrapNativeHandle(handle);
}
C2Handle *WrapNativeCodec2GrallocHandle(
const native_handle_t *const handle,
uint32_t width, uint32_t height, uint32_t format, uint64_t usage, uint32_t stride,
uint32_t generation, uint64_t igbp_id, uint32_t igbp_slot) {
return C2HandleGralloc::WrapNativeHandle(handle, width, height, format, usage, stride,
generation, igbp_id, igbp_slot);
}
bool MigrateNativeCodec2GrallocHandle(
native_handle_t *handle,
uint32_t generation, uint64_t igbp_id, uint32_t igbp_slot) {
return C2HandleGralloc::MigrateNativeHandle(handle, generation, igbp_id, igbp_slot);
}
class C2AllocationGralloc : public C2GraphicAllocation {
public:
virtual ~C2AllocationGralloc() override;
virtual c2_status_t map(
C2Rect rect, C2MemoryUsage usage, C2Fence *fence,
C2PlanarLayout *layout /* nonnull */, uint8_t **addr /* nonnull */) override;
virtual c2_status_t unmap(
uint8_t **addr /* nonnull */, C2Rect rect, C2Fence *fence /* nullable */) override;
virtual C2Allocator::id_t getAllocatorId() const override { return mAllocatorId; }
virtual const C2Handle *handle() const override { return mLockedHandle ? : mHandle; }
virtual bool equals(const std::shared_ptr<const C2GraphicAllocation> &other) const override;
// internal methods
// |handle| will be moved.
C2AllocationGralloc(
const BufferDescriptorInfo2 &info,
const sp<IMapper2> &mapper,
hidl_handle &hidlHandle,
const C2HandleGralloc *const handle,
C2Allocator::id_t allocatorId);
C2AllocationGralloc(
const BufferDescriptorInfo3 &info,
const sp<IMapper3> &mapper,
hidl_handle &hidlHandle,
const C2HandleGralloc *const handle,
C2Allocator::id_t allocatorId);
int dup() const;
c2_status_t status() const;
private:
const BufferDescriptorInfo2 mInfo2{};
const sp<IMapper2> mMapper2{nullptr};
const BufferDescriptorInfo3 mInfo3{};
const sp<IMapper3> mMapper3{nullptr};
const hidl_handle mHidlHandle;
const C2HandleGralloc *mHandle;
buffer_handle_t mBuffer;
const C2HandleGralloc *mLockedHandle;
bool mLocked;
C2Allocator::id_t mAllocatorId;
std::mutex mMappedLock;
};
C2AllocationGralloc::C2AllocationGralloc(
const BufferDescriptorInfo2 &info,
const sp<IMapper2> &mapper,
hidl_handle &hidlHandle,
const C2HandleGralloc *const handle,
C2Allocator::id_t allocatorId)
: C2GraphicAllocation(info.mapperInfo.width, info.mapperInfo.height),
mInfo2(info),
mMapper2(mapper),
mHidlHandle(std::move(hidlHandle)),
mHandle(handle),
mBuffer(nullptr),
mLockedHandle(nullptr),
mLocked(false),
mAllocatorId(allocatorId) {
}
C2AllocationGralloc::C2AllocationGralloc(
const BufferDescriptorInfo3 &info,
const sp<IMapper3> &mapper,
hidl_handle &hidlHandle,
const C2HandleGralloc *const handle,
C2Allocator::id_t allocatorId)
: C2GraphicAllocation(info.mapperInfo.width, info.mapperInfo.height),
mInfo3(info),
mMapper3(mapper),
mHidlHandle(std::move(hidlHandle)),
mHandle(handle),
mBuffer(nullptr),
mLockedHandle(nullptr),
mLocked(false),
mAllocatorId(allocatorId) {
}
C2AllocationGralloc::~C2AllocationGralloc() {
if (mBuffer && mLocked) {
// implementation ignores addresss and rect
uint8_t* addr[C2PlanarLayout::MAX_NUM_PLANES] = {};
unmap(addr, C2Rect(), nullptr);
}
if (mBuffer) {
if (mMapper2) {
if (!mMapper2->freeBuffer(const_cast<native_handle_t *>(
mBuffer)).isOk()) {
ALOGE("failed transaction: freeBuffer");
}
} else {
if (!mMapper3->freeBuffer(const_cast<native_handle_t *>(
mBuffer)).isOk()) {
ALOGE("failed transaction: freeBuffer");
}
}
}
if (mHandle) {
native_handle_delete(
const_cast<native_handle_t *>(reinterpret_cast<const native_handle_t *>(mHandle)));
}
if (mLockedHandle) {
native_handle_delete(
const_cast<native_handle_t *>(
reinterpret_cast<const native_handle_t *>(mLockedHandle)));
}
}
c2_status_t C2AllocationGralloc::map(
C2Rect rect, C2MemoryUsage usage, C2Fence *fence,
C2PlanarLayout *layout /* nonnull */, uint8_t **addr /* nonnull */) {
uint64_t grallocUsage = static_cast<C2AndroidMemoryUsage>(usage).asGrallocUsage();
ALOGV("mapping buffer with usage %#llx => %#llx",
(long long)usage.expected, (long long)grallocUsage);
// TODO
(void) fence;
std::lock_guard<std::mutex> lock(mMappedLock);
if (mBuffer && mLocked) {
ALOGD("already mapped");
return C2_DUPLICATE;
}
if (!layout || !addr) {
ALOGD("wrong param");
return C2_BAD_VALUE;
}
c2_status_t err = C2_OK;
if (!mBuffer) {
if (mMapper2) {
if (!mMapper2->importBuffer(
mHidlHandle, [&err, this](const auto &maperr, const auto &buffer) {
err = maperr2error(maperr);
if (err == C2_OK) {
mBuffer = static_cast<buffer_handle_t>(buffer);
}
}).isOk()) {
ALOGE("failed transaction: importBuffer");
return C2_CORRUPTED;
}
} else {
if (!mMapper3->importBuffer(
mHidlHandle, [&err, this](const auto &maperr, const auto &buffer) {
err = maperr2error(maperr);
if (err == C2_OK) {
mBuffer = static_cast<buffer_handle_t>(buffer);
}
}).isOk()) {
ALOGE("failed transaction: importBuffer (@3.0)");
return C2_CORRUPTED;
}
}
if (err != C2_OK) {
ALOGD("importBuffer failed: %d", err);
return err;
}
if (mBuffer == nullptr) {
ALOGD("importBuffer returned null buffer");
return C2_CORRUPTED;
}
uint32_t generation = 0;
uint64_t igbp_id = 0;
uint32_t igbp_slot = 0;
if (mHandle) {
mHandle->getIgbpData(&generation, &igbp_id, &igbp_slot);
}
if (mMapper2) {
mLockedHandle = C2HandleGralloc::WrapAndMoveNativeHandle(
mBuffer, mInfo2.mapperInfo.width, mInfo2.mapperInfo.height,
(uint32_t)mInfo2.mapperInfo.format, mInfo2.mapperInfo.usage,
mInfo2.stride, generation, igbp_id, igbp_slot);
} else {
mLockedHandle = C2HandleGralloc::WrapAndMoveNativeHandle(
mBuffer, mInfo3.mapperInfo.width, mInfo3.mapperInfo.height,
(uint32_t)mInfo3.mapperInfo.format, mInfo3.mapperInfo.usage,
mInfo3.stride, generation, igbp_id, igbp_slot);
}
}
PixelFormat3 format = mMapper2 ?
PixelFormat3(mInfo2.mapperInfo.format) :
PixelFormat3(mInfo3.mapperInfo.format);
switch (format) {
case PixelFormat3::RGBA_1010102: {
// TRICKY: this is used for media as YUV444 in the case when it is queued directly to a
// Surface. In all other cases it is RGBA. We don't know which case it is here, so
// default to YUV for now.
void *pointer = nullptr;
if (mMapper2) {
if (!mMapper2->lock(
const_cast<native_handle_t *>(mBuffer),
grallocUsage,
{ (int32_t)rect.left, (int32_t)rect.top,
(int32_t)rect.width, (int32_t)rect.height },
// TODO: fence
hidl_handle(),
[&err, &pointer](const auto &maperr, const auto &mapPointer) {
err = maperr2error(maperr);
if (err == C2_OK) {
pointer = mapPointer;
}
}).isOk()) {
ALOGE("failed transaction: lock(RGBA_1010102)");
return C2_CORRUPTED;
}
} else {
if (!mMapper3->lock(
const_cast<native_handle_t *>(mBuffer),
grallocUsage,
{ (int32_t)rect.left, (int32_t)rect.top,
(int32_t)rect.width, (int32_t)rect.height },
// TODO: fence
hidl_handle(),
[&err, &pointer](const auto &maperr, const auto &mapPointer,
int32_t bytesPerPixel, int32_t bytesPerStride) {
err = maperr2error(maperr);
if (err == C2_OK) {
pointer = mapPointer;
}
(void)bytesPerPixel;
(void)bytesPerStride;
}).isOk()) {
ALOGE("failed transaction: lock(RGBA_1010102) (@3.0)");
return C2_CORRUPTED;
}
}
if (err != C2_OK) {
ALOGD("lock failed: %d", err);
return err;
}
// treat as 32-bit values
addr[C2PlanarLayout::PLANE_Y] = (uint8_t *)pointer;
addr[C2PlanarLayout::PLANE_U] = (uint8_t *)pointer;
addr[C2PlanarLayout::PLANE_V] = (uint8_t *)pointer;
addr[C2PlanarLayout::PLANE_A] = (uint8_t *)pointer;
layout->type = C2PlanarLayout::TYPE_YUVA;
layout->numPlanes = 4;
layout->rootPlanes = 1;
int32_t stride = mMapper2 ?
int32_t(mInfo2.stride) :
int32_t(mInfo3.stride);
layout->planes[C2PlanarLayout::PLANE_Y] = {
C2PlaneInfo::CHANNEL_Y, // channel
4, // colInc
4 * stride, // rowInc
1, // mColSampling
1, // mRowSampling
32, // allocatedDepth
10, // bitDepth
10, // rightShift
C2PlaneInfo::LITTLE_END, // endianness
C2PlanarLayout::PLANE_Y, // rootIx
0, // offset
};
layout->planes[C2PlanarLayout::PLANE_U] = {
C2PlaneInfo::CHANNEL_CB, // channel
4, // colInc
4 * stride, // rowInc
1, // mColSampling
1, // mRowSampling
32, // allocatedDepth
10, // bitDepth
0, // rightShift
C2PlaneInfo::LITTLE_END, // endianness
C2PlanarLayout::PLANE_Y, // rootIx
0, // offset
};
layout->planes[C2PlanarLayout::PLANE_V] = {
C2PlaneInfo::CHANNEL_CR, // channel
4, // colInc
4 * stride, // rowInc
1, // mColSampling
1, // mRowSampling
32, // allocatedDepth
10, // bitDepth
20, // rightShift
C2PlaneInfo::LITTLE_END, // endianness
C2PlanarLayout::PLANE_Y, // rootIx
0, // offset
};
layout->planes[C2PlanarLayout::PLANE_A] = {
C2PlaneInfo::CHANNEL_A, // channel
4, // colInc
4 * stride, // rowInc
1, // mColSampling
1, // mRowSampling
32, // allocatedDepth
2, // bitDepth
30, // rightShift
C2PlaneInfo::LITTLE_END, // endianness
C2PlanarLayout::PLANE_Y, // rootIx
0, // offset
};
break;
}
case PixelFormat3::RGBA_8888:
// TODO: alpha channel
// fall-through
case PixelFormat3::RGBX_8888: {
void *pointer = nullptr;
if (mMapper2) {
if (!mMapper2->lock(
const_cast<native_handle_t *>(mBuffer),
grallocUsage,
{ (int32_t)rect.left, (int32_t)rect.top,
(int32_t)rect.width, (int32_t)rect.height },
// TODO: fence
hidl_handle(),
[&err, &pointer](const auto &maperr, const auto &mapPointer) {
err = maperr2error(maperr);
if (err == C2_OK) {
pointer = mapPointer;
}
}).isOk()) {
ALOGE("failed transaction: lock(RGBA_8888)");
return C2_CORRUPTED;
}
} else {
if (!mMapper3->lock(
const_cast<native_handle_t *>(mBuffer),
grallocUsage,
{ (int32_t)rect.left, (int32_t)rect.top,
(int32_t)rect.width, (int32_t)rect.height },
// TODO: fence
hidl_handle(),
[&err, &pointer](const auto &maperr, const auto &mapPointer,
int32_t bytesPerPixel, int32_t bytesPerStride) {
err = maperr2error(maperr);
if (err == C2_OK) {
pointer = mapPointer;
}
(void)bytesPerPixel;
(void)bytesPerStride;
}).isOk()) {
ALOGE("failed transaction: lock(RGBA_8888) (@3.0)");
return C2_CORRUPTED;
}
}
if (err != C2_OK) {
ALOGD("lock failed: %d", err);
return err;
}
addr[C2PlanarLayout::PLANE_R] = (uint8_t *)pointer;
addr[C2PlanarLayout::PLANE_G] = (uint8_t *)pointer + 1;
addr[C2PlanarLayout::PLANE_B] = (uint8_t *)pointer + 2;
layout->type = C2PlanarLayout::TYPE_RGB;
layout->numPlanes = 3;
layout->rootPlanes = 1;
int32_t stride = mMapper2 ?
int32_t(mInfo2.stride) :
int32_t(mInfo3.stride);
layout->planes[C2PlanarLayout::PLANE_R] = {
C2PlaneInfo::CHANNEL_R, // channel
4, // colInc
4 * stride, // rowInc
1, // mColSampling
1, // mRowSampling
8, // allocatedDepth
8, // bitDepth
0, // rightShift
C2PlaneInfo::NATIVE, // endianness
C2PlanarLayout::PLANE_R, // rootIx
0, // offset
};
layout->planes[C2PlanarLayout::PLANE_G] = {
C2PlaneInfo::CHANNEL_G, // channel
4, // colInc
4 * stride, // rowInc
1, // mColSampling
1, // mRowSampling
8, // allocatedDepth
8, // bitDepth
0, // rightShift
C2PlaneInfo::NATIVE, // endianness
C2PlanarLayout::PLANE_R, // rootIx
1, // offset
};
layout->planes[C2PlanarLayout::PLANE_B] = {
C2PlaneInfo::CHANNEL_B, // channel
4, // colInc
4 * stride, // rowInc
1, // mColSampling
1, // mRowSampling
8, // allocatedDepth
8, // bitDepth
0, // rightShift
C2PlaneInfo::NATIVE, // endianness
C2PlanarLayout::PLANE_R, // rootIx
2, // offset
};
break;
}
case PixelFormat3::YCBCR_420_888:
// fall-through
case PixelFormat3::YV12:
// fall-through
default: {
struct YCbCrLayout {
void* y;
void* cb;
void* cr;
uint32_t yStride;
uint32_t cStride;
uint32_t chromaStep;
};
YCbCrLayout ycbcrLayout;
if (mMapper2) {
if (!mMapper2->lockYCbCr(
const_cast<native_handle_t *>(mBuffer), grallocUsage,
{ (int32_t)rect.left, (int32_t)rect.top,
(int32_t)rect.width, (int32_t)rect.height },
// TODO: fence
hidl_handle(),
[&err, &ycbcrLayout](const auto &maperr, const auto &mapLayout) {
err = maperr2error(maperr);
if (err == C2_OK) {
ycbcrLayout = YCbCrLayout{
mapLayout.y,
mapLayout.cb,
mapLayout.cr,
mapLayout.yStride,
mapLayout.cStride,
mapLayout.chromaStep};
}
}).isOk()) {
ALOGE("failed transaction: lockYCbCr");
return C2_CORRUPTED;
}
} else {
if (!mMapper3->lockYCbCr(
const_cast<native_handle_t *>(mBuffer), grallocUsage,
{ (int32_t)rect.left, (int32_t)rect.top,
(int32_t)rect.width, (int32_t)rect.height },
// TODO: fence
hidl_handle(),
[&err, &ycbcrLayout](const auto &maperr, const auto &mapLayout) {
err = maperr2error(maperr);
if (err == C2_OK) {
ycbcrLayout = YCbCrLayout{
mapLayout.y,
mapLayout.cb,
mapLayout.cr,
mapLayout.yStride,
mapLayout.cStride,
mapLayout.chromaStep};
}
}).isOk()) {
ALOGE("failed transaction: lockYCbCr (@3.0)");
return C2_CORRUPTED;
}
}
if (err != C2_OK) {
ALOGD("lockYCbCr failed: %d", err);
return err;
}
addr[C2PlanarLayout::PLANE_Y] = (uint8_t *)ycbcrLayout.y;
addr[C2PlanarLayout::PLANE_U] = (uint8_t *)ycbcrLayout.cb;
addr[C2PlanarLayout::PLANE_V] = (uint8_t *)ycbcrLayout.cr;
layout->type = C2PlanarLayout::TYPE_YUV;
layout->numPlanes = 3;
layout->rootPlanes = 3;
layout->planes[C2PlanarLayout::PLANE_Y] = {
C2PlaneInfo::CHANNEL_Y, // channel
1, // colInc
(int32_t)ycbcrLayout.yStride, // rowInc
1, // mColSampling
1, // mRowSampling
8, // allocatedDepth
8, // bitDepth
0, // rightShift
C2PlaneInfo::NATIVE, // endianness
C2PlanarLayout::PLANE_Y, // rootIx
0, // offset
};
layout->planes[C2PlanarLayout::PLANE_U] = {
C2PlaneInfo::CHANNEL_CB, // channel
(int32_t)ycbcrLayout.chromaStep, // colInc
(int32_t)ycbcrLayout.cStride, // rowInc
2, // mColSampling
2, // mRowSampling
8, // allocatedDepth
8, // bitDepth
0, // rightShift
C2PlaneInfo::NATIVE, // endianness
C2PlanarLayout::PLANE_U, // rootIx
0, // offset
};
layout->planes[C2PlanarLayout::PLANE_V] = {
C2PlaneInfo::CHANNEL_CR, // channel
(int32_t)ycbcrLayout.chromaStep, // colInc
(int32_t)ycbcrLayout.cStride, // rowInc
2, // mColSampling
2, // mRowSampling
8, // allocatedDepth
8, // bitDepth
0, // rightShift
C2PlaneInfo::NATIVE, // endianness
C2PlanarLayout::PLANE_V, // rootIx
0, // offset
};
// handle interleaved formats
intptr_t uvOffset = addr[C2PlanarLayout::PLANE_V] - addr[C2PlanarLayout::PLANE_U];
if (uvOffset > 0 && uvOffset < (intptr_t)ycbcrLayout.chromaStep) {
layout->rootPlanes = 2;
layout->planes[C2PlanarLayout::PLANE_V].rootIx = C2PlanarLayout::PLANE_U;
layout->planes[C2PlanarLayout::PLANE_V].offset = uvOffset;
} else if (uvOffset < 0 && uvOffset > -(intptr_t)ycbcrLayout.chromaStep) {
layout->rootPlanes = 2;
layout->planes[C2PlanarLayout::PLANE_U].rootIx = C2PlanarLayout::PLANE_V;
layout->planes[C2PlanarLayout::PLANE_U].offset = -uvOffset;
}
break;
}
}
mLocked = true;
return C2_OK;
}
c2_status_t C2AllocationGralloc::unmap(
uint8_t **addr, C2Rect rect, C2Fence *fence /* nullable */) {
// TODO: check addr and size, use fence
(void)addr;
(void)rect;
std::lock_guard<std::mutex> lock(mMappedLock);
c2_status_t err = C2_OK;
if (mMapper2) {
if (!mMapper2->unlock(
const_cast<native_handle_t *>(mBuffer),
[&err, &fence](const auto &maperr, const auto &releaseFence) {
// TODO
(void) fence;
(void) releaseFence;
err = maperr2error(maperr);
if (err == C2_OK) {
// TODO: fence
}
}).isOk()) {
ALOGE("failed transaction: unlock");
return C2_CORRUPTED;
}
} else {
if (!mMapper3->unlock(
const_cast<native_handle_t *>(mBuffer),
[&err, &fence](const auto &maperr, const auto &releaseFence) {
// TODO
(void) fence;
(void) releaseFence;
err = maperr2error(maperr);
if (err == C2_OK) {
// TODO: fence
}
}).isOk()) {
ALOGE("failed transaction: unlock (@3.0)");
return C2_CORRUPTED;
}
}
if (err == C2_OK) {
mLocked = false;
}
return err;
}
bool C2AllocationGralloc::equals(const std::shared_ptr<const C2GraphicAllocation> &other) const {
return other && other->handle() == handle();
}
/* ===================================== GRALLOC ALLOCATOR ==================================== */
class C2AllocatorGralloc::Impl {
public:
Impl(id_t id, bool bufferQueue);
id_t getId() const {
return mTraits->id;
}
C2String getName() const {
return mTraits->name;
}
std::shared_ptr<const C2Allocator::Traits> getTraits() const {
return mTraits;
}
c2_status_t newGraphicAllocation(
uint32_t width, uint32_t height, uint32_t format, const C2MemoryUsage &usage,
std::shared_ptr<C2GraphicAllocation> *allocation);
c2_status_t priorGraphicAllocation(
const C2Handle *handle,
std::shared_ptr<C2GraphicAllocation> *allocation);
c2_status_t status() const { return mInit; }
private:
std::shared_ptr<C2Allocator::Traits> mTraits;
c2_status_t mInit;
sp<IAllocator2> mAllocator2;
sp<IMapper2> mMapper2;
sp<IAllocator3> mAllocator3;
sp<IMapper3> mMapper3;
const bool mBufferQueue;
};
void _UnwrapNativeCodec2GrallocMetadata(
const C2Handle *const handle,
uint32_t *width, uint32_t *height, uint32_t *format,uint64_t *usage, uint32_t *stride,
uint32_t *generation, uint64_t *igbp_id, uint32_t *igbp_slot) {
(void)C2HandleGralloc::Import(handle, width, height, format, usage, stride,
generation, igbp_id, igbp_slot);
}
C2AllocatorGralloc::Impl::Impl(id_t id, bool bufferQueue)
: mInit(C2_OK), mBufferQueue(bufferQueue) {
// TODO: get this from allocator
C2MemoryUsage minUsage = { 0, 0 }, maxUsage = { ~(uint64_t)0, ~(uint64_t)0 };
Traits traits = { "android.allocator.gralloc", id, C2Allocator::GRAPHIC, minUsage, maxUsage };
mTraits = std::make_shared<C2Allocator::Traits>(traits);
// gralloc allocator is a singleton, so all objects share a global service
mAllocator3 = IAllocator3::getService();
mMapper3 = IMapper3::getService();
if (!mAllocator3 || !mMapper3) {
mAllocator3 = nullptr;
mMapper3 = nullptr;
mAllocator2 = IAllocator2::getService();
mMapper2 = IMapper2::getService();
if (!mAllocator2 || !mMapper2) {
mAllocator2 = nullptr;
mMapper2 = nullptr;
mInit = C2_CORRUPTED;
}
}
}
c2_status_t C2AllocatorGralloc::Impl::newGraphicAllocation(
uint32_t width, uint32_t height, uint32_t format, const C2MemoryUsage &usage,
std::shared_ptr<C2GraphicAllocation> *allocation) {
uint64_t grallocUsage = static_cast<C2AndroidMemoryUsage>(usage).asGrallocUsage();
ALOGV("allocating buffer with usage %#llx => %#llx",
(long long)usage.expected, (long long)grallocUsage);
c2_status_t err = C2_OK;
hidl_handle buffer{};
if (mMapper2) {
BufferDescriptorInfo2 info = {
{
width,
height,
1u, // layerCount
PixelFormat2(format),
grallocUsage,
},
0u, // stride placeholder
};
BufferDescriptor2 desc;
if (!mMapper2->createDescriptor(
info.mapperInfo, [&err, &desc](const auto &maperr, const auto &descriptor) {
err = maperr2error(maperr);
if (err == C2_OK) {
desc = descriptor;
}
}).isOk()) {
ALOGE("failed transaction: createDescriptor");
return C2_CORRUPTED;
}
if (err != C2_OK) {
return err;
}
// IAllocator shares IMapper error codes.
if (!mAllocator2->allocate(
desc,
1u,
[&err, &buffer, &info](const auto &maperr, const auto &stride, auto &buffers) {
err = maperr2error(maperr);
if (err != C2_OK) {
return;
}
if (buffers.size() != 1u) {
err = C2_CORRUPTED;
return;
}
info.stride = stride;
buffer = buffers[0];
}).isOk()) {
ALOGE("failed transaction: allocate");
return C2_CORRUPTED;
}
if (err != C2_OK) {
return err;
}
allocation->reset(new C2AllocationGralloc(
info, mMapper2, buffer,
C2HandleGralloc::WrapAndMoveNativeHandle(
buffer.getNativeHandle(),
width, height,
format, grallocUsage, info.stride,
0, 0, mBufferQueue ? ~0 : 0),
mTraits->id));
return C2_OK;
} else {
BufferDescriptorInfo3 info = {
{
width,
height,
1u, // layerCount
PixelFormat3(format),
grallocUsage,
},
0u, // stride placeholder
};
BufferDescriptor3 desc;
if (!mMapper3->createDescriptor(
info.mapperInfo, [&err, &desc](const auto &maperr, const auto &descriptor) {
err = maperr2error(maperr);
if (err == C2_OK) {
desc = descriptor;
}
}).isOk()) {
ALOGE("failed transaction: createDescriptor");
return C2_CORRUPTED;
}
if (err != C2_OK) {
return err;
}
// IAllocator shares IMapper error codes.
if (!mAllocator3->allocate(
desc,
1u,
[&err, &buffer, &info](const auto &maperr, const auto &stride, auto &buffers) {
err = maperr2error(maperr);
if (err != C2_OK) {
return;
}
if (buffers.size() != 1u) {
err = C2_CORRUPTED;
return;
}
info.stride = stride;
buffer = buffers[0];
}).isOk()) {
ALOGE("failed transaction: allocate");
return C2_CORRUPTED;
}
if (err != C2_OK) {
return err;
}
allocation->reset(new C2AllocationGralloc(
info, mMapper3, buffer,
C2HandleGralloc::WrapAndMoveNativeHandle(
buffer.getNativeHandle(),
width, height,
format, grallocUsage, info.stride,
0, 0, mBufferQueue ? ~0 : 0),
mTraits->id));
return C2_OK;
}
}
c2_status_t C2AllocatorGralloc::Impl::priorGraphicAllocation(
const C2Handle *handle,
std::shared_ptr<C2GraphicAllocation> *allocation) {
if (mMapper2) {
BufferDescriptorInfo2 info;
info.mapperInfo.layerCount = 1u;
uint32_t generation;
uint64_t igbp_id;
uint32_t igbp_slot;
const C2HandleGralloc *grallocHandle = C2HandleGralloc::Import(
handle,
&info.mapperInfo.width, &info.mapperInfo.height,
(uint32_t *)&info.mapperInfo.format,
(uint64_t *)&info.mapperInfo.usage,
&info.stride,
&generation, &igbp_id, &igbp_slot);
if (grallocHandle == nullptr) {
return C2_BAD_VALUE;
}
hidl_handle hidlHandle;
hidlHandle.setTo(C2HandleGralloc::UnwrapNativeHandle(grallocHandle), true);
allocation->reset(new C2AllocationGralloc(
info, mMapper2, hidlHandle, grallocHandle, mTraits->id));
return C2_OK;
} else {
BufferDescriptorInfo3 info;
info.mapperInfo.layerCount = 1u;
uint32_t generation;
uint64_t igbp_id;
uint32_t igbp_slot;
const C2HandleGralloc *grallocHandle = C2HandleGralloc::Import(
handle,
&info.mapperInfo.width, &info.mapperInfo.height,
(uint32_t *)&info.mapperInfo.format,
(uint64_t *)&info.mapperInfo.usage,
&info.stride,
&generation, &igbp_id, &igbp_slot);
if (grallocHandle == nullptr) {
return C2_BAD_VALUE;
}
hidl_handle hidlHandle;
hidlHandle.setTo(C2HandleGralloc::UnwrapNativeHandle(grallocHandle), true);
allocation->reset(new C2AllocationGralloc(
info, mMapper3, hidlHandle, grallocHandle, mTraits->id));
return C2_OK;
}
}
C2AllocatorGralloc::C2AllocatorGralloc(id_t id, bool bufferQueue)
: mImpl(new Impl(id, bufferQueue)) {}
C2AllocatorGralloc::~C2AllocatorGralloc() { delete mImpl; }
C2Allocator::id_t C2AllocatorGralloc::getId() const {
return mImpl->getId();
}
C2String C2AllocatorGralloc::getName() const {
return mImpl->getName();
}
std::shared_ptr<const C2Allocator::Traits> C2AllocatorGralloc::getTraits() const {
return mImpl->getTraits();
}
c2_status_t C2AllocatorGralloc::newGraphicAllocation(
uint32_t width, uint32_t height, uint32_t format, C2MemoryUsage usage,
std::shared_ptr<C2GraphicAllocation> *allocation) {
return mImpl->newGraphicAllocation(width, height, format, usage, allocation);
}
c2_status_t C2AllocatorGralloc::priorGraphicAllocation(
const C2Handle *handle,
std::shared_ptr<C2GraphicAllocation> *allocation) {
return mImpl->priorGraphicAllocation(handle, allocation);
}
c2_status_t C2AllocatorGralloc::status() const {
return mImpl->status();
}
bool C2AllocatorGralloc::isValid(const C2Handle* const o) {
return C2HandleGralloc::isValid(o);
}
} // namespace android