blob: 80e4fc4432b9fcaa42216f74e0b5b9afbb275a92 [file] [log] [blame]
/*
* Copyright 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.
*/
#include <RenderEngineBench.h>
#include <android/bitmap.h>
#include <android/data_space.h>
#include <android/imagedecoder.h>
#include <log/log.h>
#include <renderengine/ExternalTexture.h>
#include <renderengine/RenderEngine.h>
#include <sys/types.h>
using namespace android;
using namespace android::renderengine;
namespace {
struct DecoderDeleter {
void operator()(AImageDecoder* decoder) { AImageDecoder_delete(decoder); }
};
using AutoDecoderDeleter = std::unique_ptr<AImageDecoder, DecoderDeleter>;
bool ok(int aImageDecoderResult, const char* path, const char* method) {
if (aImageDecoderResult == ANDROID_IMAGE_DECODER_SUCCESS) {
return true;
}
ALOGE("Failed AImageDecoder_%s on '%s' with error '%s'", method, path,
AImageDecoder_resultToString(aImageDecoderResult));
return false;
}
} // namespace
namespace renderenginebench {
void decode(const char* path, const sp<GraphicBuffer>& buffer) {
base::unique_fd fd{open(path, O_RDONLY)};
if (fd.get() < 0) {
ALOGE("Failed to open %s", path);
return;
}
AImageDecoder* decoder{nullptr};
auto result = AImageDecoder_createFromFd(fd.get(), &decoder);
if (!ok(result, path, "createFromFd")) {
return;
}
AutoDecoderDeleter deleter(decoder);
LOG_ALWAYS_FATAL_IF(buffer->getWidth() <= 0 || buffer->getHeight() <= 0,
"Impossible buffer size!");
auto width = static_cast<int32_t>(buffer->getWidth());
auto height = static_cast<int32_t>(buffer->getHeight());
result = AImageDecoder_setTargetSize(decoder, width, height);
if (!ok(result, path, "setTargetSize")) {
return;
}
void* pixels{nullptr};
int32_t stride{0};
if (auto status = buffer->lock(GRALLOC_USAGE_SW_WRITE_OFTEN, &pixels,
nullptr /*outBytesPerPixel*/, &stride);
status < 0) {
ALOGE("Failed to lock pixels!");
return;
}
result = AImageDecoder_decodeImage(decoder, pixels, static_cast<size_t>(stride),
static_cast<size_t>(stride * height));
if (auto status = buffer->unlock(); status < 0) {
ALOGE("Failed to unlock pixels!");
}
// For the side effect of logging.
(void)ok(result, path, "decodeImage");
}
void encodeToJpeg(const char* path, const sp<GraphicBuffer>& buffer) {
base::unique_fd fd{open(path, O_WRONLY | O_CREAT, S_IWUSR)};
if (fd.get() < 0) {
ALOGE("Failed to open %s", path);
return;
}
void* pixels{nullptr};
int32_t stride{0};
if (auto status = buffer->lock(GRALLOC_USAGE_SW_READ_OFTEN, &pixels,
nullptr /*outBytesPerPixel*/, &stride);
status < 0) {
ALOGE("Failed to lock pixels!");
return;
}
AndroidBitmapInfo info{
.width = buffer->getWidth(),
.height = buffer->getHeight(),
.stride = static_cast<uint32_t>(stride),
.format = ANDROID_BITMAP_FORMAT_RGBA_8888,
.flags = ANDROID_BITMAP_FLAGS_ALPHA_OPAQUE,
};
int result = AndroidBitmap_compress(&info, ADATASPACE_SRGB, pixels,
ANDROID_BITMAP_COMPRESS_FORMAT_JPEG, 80, &fd,
[](void* fdPtr, const void* data, size_t size) -> bool {
const ssize_t bytesWritten =
write(reinterpret_cast<base::unique_fd*>(fdPtr)
->get(),
data, size);
return bytesWritten > 0 &&
static_cast<size_t>(bytesWritten) == size;
});
if (result == ANDROID_BITMAP_RESULT_SUCCESS) {
ALOGD("Successfully encoded to '%s'", path);
} else {
ALOGE("Failed to encode to %s with error %d", path, result);
}
if (auto status = buffer->unlock(); status < 0) {
ALOGE("Failed to unlock pixels!");
}
}
} // namespace renderenginebench