You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
228 lines
7.4 KiB
228 lines
7.4 KiB
// Copyright (C) 2019 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 "host-common/MediaH264DecoderDefault.h"
|
|
#include "base/System.h"
|
|
#include "host-common/H264PingInfoParser.h"
|
|
#include "host-common/MediaH264DecoderGeneric.h"
|
|
|
|
#include <cstdint>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#define MEDIA_H264_DEBUG 0
|
|
|
|
#if MEDIA_H264_DEBUG
|
|
#define H264_DPRINT(fmt,...) fprintf(stderr, "h264-dec: %s:%d " fmt "\n", __func__, __LINE__, ##__VA_ARGS__);
|
|
#else
|
|
#define H264_DPRINT(fmt,...)
|
|
#endif
|
|
|
|
namespace android {
|
|
namespace emulation {
|
|
|
|
namespace {
|
|
MediaH264DecoderPlugin* makeDecoderPlugin(uint64_t pluginid,
|
|
H264PingInfoParser parser) {
|
|
return new MediaH264DecoderGeneric(pluginid, parser);
|
|
}
|
|
|
|
}; // anon namespace
|
|
|
|
|
|
uint64_t MediaH264DecoderDefault::readId(void* ptr) {
|
|
if (nullptr == ptr)
|
|
return 0;
|
|
uint64_t key = H264PingInfoParser::parseHostDecoderId(ptr);
|
|
return key;
|
|
}
|
|
|
|
MediaH264DecoderPlugin* MediaH264DecoderDefault::getDecoder(uint64_t key) {
|
|
{
|
|
std::lock_guard<std::mutex> g(mMapLock);
|
|
auto iter = mDecoders.find(key);
|
|
if (iter != mDecoders.end()) {
|
|
return iter->second;
|
|
}
|
|
}
|
|
H264_DPRINT("Error: cannot find decoder with key %" PRIx64 "", key);
|
|
return nullptr;
|
|
}
|
|
|
|
uint64_t MediaH264DecoderDefault::createId() {
|
|
std::lock_guard<std::mutex> g(mIdLock);
|
|
return ++mId;
|
|
}
|
|
|
|
void MediaH264DecoderDefault::addDecoder(uint64_t key,
|
|
MediaH264DecoderPlugin* val) {
|
|
{
|
|
std::lock_guard<std::mutex> g(mMapLock);
|
|
if (mDecoders.find(key) == mDecoders.end()) {
|
|
mDecoders[key] = val;
|
|
H264_DPRINT("added decoder key %" PRIx64 " val: %p", key, val);
|
|
return;
|
|
}
|
|
}
|
|
H264_DPRINT("cannot add: already exist");
|
|
}
|
|
|
|
void MediaH264DecoderDefault::updateDecoder(uint64_t key,
|
|
MediaH264DecoderPlugin* val) {
|
|
std::lock_guard<std::mutex> g(mMapLock);
|
|
if (mDecoders.find(key) == mDecoders.end()) {
|
|
H264_DPRINT("error: decoder with key %" PRIx64 " does not exist", key);
|
|
} else {
|
|
mDecoders[key] = val;
|
|
H264_DPRINT("updated key %" PRIx64 " with new decoder %p", key, val);
|
|
}
|
|
}
|
|
|
|
void MediaH264DecoderDefault::removeDecoder(uint64_t key) {
|
|
{
|
|
std::lock_guard<std::mutex> g(mMapLock);
|
|
auto iter = mDecoders.find(key);
|
|
if (iter != mDecoders.end()) {
|
|
H264_DPRINT("removed decoder key %" PRIx64 ", val: %p", key,
|
|
mDecoders[key]);
|
|
mDecoders.erase(iter);
|
|
return;
|
|
}
|
|
}
|
|
H264_DPRINT("error: cannot remove decoder, not in map");
|
|
}
|
|
|
|
static void* getReturnAddress(void* ptr) {
|
|
uint8_t* xptr = (uint8_t*)ptr;
|
|
void* pint = (void*)(xptr + 256);
|
|
return pint;
|
|
}
|
|
|
|
|
|
void MediaH264DecoderDefault::handlePing(MediaCodecType type,
|
|
MediaOperation op,
|
|
void* ptr) {
|
|
using InitContextParam = H264PingInfoParser::InitContextParam;
|
|
using DecodeFrameParam = H264PingInfoParser::DecodeFrameParam;
|
|
using ResetParam = H264PingInfoParser::ResetParam;
|
|
using GetImageParam = H264PingInfoParser::GetImageParam;
|
|
|
|
switch (op) {
|
|
case MediaOperation::InitContext: {
|
|
H264PingInfoParser parser{ptr};
|
|
InitContextParam param{};
|
|
parser.parseInitContextParams(ptr, param);
|
|
H264_DPRINT(
|
|
"handle init decoder context request from guest version %u",
|
|
parser.version());
|
|
uint64_t myid = createId();
|
|
MediaH264DecoderPlugin* mydecoder = makeDecoderPlugin(myid, parser);
|
|
addDecoder(myid, mydecoder);
|
|
mydecoder->initH264Context(ptr);
|
|
*(param.pHostDecoderId) = myid;
|
|
H264_DPRINT("done handling InitContext");
|
|
break;
|
|
}
|
|
case MediaOperation::DestroyContext: {
|
|
H264_DPRINT("handle destroy request from guest %p", ptr);
|
|
MediaH264DecoderPlugin* mydecoder = getDecoder(readId(ptr));
|
|
if (!mydecoder)
|
|
return;
|
|
|
|
delete mydecoder;
|
|
removeDecoder(readId(ptr));
|
|
break;
|
|
}
|
|
case MediaOperation::DecodeImage: {
|
|
H264_DPRINT("handle decodeimage request from guest %p", ptr);
|
|
MediaH264DecoderPlugin* mydecoder = getDecoder(readId(ptr));
|
|
if (nullptr == mydecoder)
|
|
return;
|
|
mydecoder->decodeFrame(ptr);
|
|
break;
|
|
}
|
|
case MediaOperation::Flush: {
|
|
H264_DPRINT("handle flush request from guest %p", ptr);
|
|
MediaH264DecoderPlugin* mydecoder = getDecoder(readId(ptr));
|
|
if (nullptr == mydecoder)
|
|
return;
|
|
mydecoder->flush(ptr);
|
|
break;
|
|
}
|
|
case MediaOperation::GetImage: {
|
|
H264_DPRINT("handle getimage request from guest %p", ptr);
|
|
MediaH264DecoderPlugin* mydecoder = getDecoder(readId(ptr));
|
|
if (nullptr == mydecoder)
|
|
return;
|
|
mydecoder->getImage(ptr);
|
|
break;
|
|
}
|
|
case MediaOperation::Reset: {
|
|
H264_DPRINT("handle reset request from guest %p", ptr);
|
|
uint64_t oldId = readId(ptr);
|
|
MediaH264DecoderPlugin* olddecoder = getDecoder(oldId);
|
|
if (nullptr == olddecoder) {
|
|
H264_DPRINT("error, cannot reset on nullptr");
|
|
return;
|
|
}
|
|
MediaH264DecoderPlugin* mydecoder = olddecoder->clone();
|
|
delete olddecoder;
|
|
mydecoder->reset(ptr);
|
|
updateDecoder(oldId, mydecoder);
|
|
break;
|
|
}
|
|
default:
|
|
H264_DPRINT("Unknown command %u\n", (unsigned int)op);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void MediaH264DecoderDefault::save(base::Stream* stream) const {
|
|
stream->putBe64(mId);
|
|
int size = mDecoders.size();
|
|
stream->putBe32(size);
|
|
for (auto item : mDecoders) {
|
|
stream->putBe64(item.first);
|
|
stream->putBe32(item.second->type());
|
|
item.second->save(stream);
|
|
}
|
|
}
|
|
|
|
bool MediaH264DecoderDefault::load(base::Stream* stream) {
|
|
mId = stream->getBe64();
|
|
int size = stream->getBe32();
|
|
for (int i = 0; i < size; ++i) {
|
|
// this is hacky; but we have to know the plugin type
|
|
uint64_t id = stream->getBe64();
|
|
int type = stream->getBe32();
|
|
if (type == MediaH264DecoderPlugin::PLUGIN_TYPE_GENERIC) {
|
|
MediaH264DecoderGeneric* decoder =
|
|
new MediaH264DecoderGeneric(id, H264PingInfoParser(100));
|
|
decoder->load(stream);
|
|
mDecoders[id] = decoder;
|
|
continue;
|
|
}
|
|
fprintf(stderr, "Error, un-implemented %s %d\n", __func__, __LINE__);
|
|
exit(1);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
} // namespace emulation
|
|
} // namespace android
|
|
|