/* * Copyright (C) 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_NDEBUG 0 #define LOG_TAG "Codec2-Component" #include #include #include #include #include #include #include #include namespace hardware { namespace google { namespace media { namespace c2 { namespace V1_0 { namespace utils { using namespace ::android; namespace /* unnamed */ { // Implementation of ConfigurableC2Intf based on C2ComponentInterface struct CompIntf : public ConfigurableC2Intf { CompIntf(const std::shared_ptr& intf) : ConfigurableC2Intf(intf->getName()), mIntf(intf) { } virtual c2_status_t config( const std::vector& params, c2_blocking_t mayBlock, std::vector>* const failures ) override { ALOGV("config"); return mIntf->config_vb(params, mayBlock, failures); } virtual c2_status_t query( const std::vector& indices, c2_blocking_t mayBlock, std::vector>* const params ) const override { ALOGV("query"); return mIntf->query_vb({}, indices, mayBlock, params); } virtual c2_status_t querySupportedParams( std::vector>* const params ) const override { ALOGV("querySupportedParams"); return mIntf->querySupportedParams_nb(params); } virtual c2_status_t querySupportedValues( std::vector& fields, c2_blocking_t mayBlock) const override { ALOGV("querySupportedValues"); return mIntf->querySupportedValues_vb(fields, mayBlock); } protected: std::shared_ptr mIntf; }; } // unnamed namespace // ComponentInterface ComponentInterface::ComponentInterface( const std::shared_ptr& intf, const sp& store) : Configurable(new CachedConfigurable(std::make_unique(intf))), mInterface(intf) { mInit = init(store.get()); } c2_status_t ComponentInterface::status() const { return mInit; } // ComponentListener wrapper struct Component::Listener : public C2Component::Listener { Listener(const sp& component) : mComponent(component), mListener(component->mListener) { } virtual void onError_nb( std::weak_ptr /* c2component */, uint32_t errorCode) override { ALOGV("onError"); sp listener = mListener.promote(); if (listener) { Return transStatus = listener->onError(Status::OK, errorCode); if (!transStatus.isOk()) { ALOGE("onError -- transaction failed."); } } } virtual void onTripped_nb( std::weak_ptr /* c2component */, std::vector> c2settingResult ) override { ALOGV("onTripped"); sp listener = mListener.promote(); if (listener) { hidl_vec settingResults(c2settingResult.size()); size_t ix = 0; for (const std::shared_ptr &c2result : c2settingResult) { if (c2result) { if (objcpy(&settingResults[ix++], *c2result) != Status::OK) { break; } } } settingResults.resize(ix); Return transStatus = listener->onTripped(settingResults); if (!transStatus.isOk()) { ALOGE("onTripped -- transaction failed."); } } } virtual void onWorkDone_nb( std::weak_ptr /* c2component */, std::list> c2workItems) override { ALOGV("onWorkDone"); sp listener = mListener.promote(); if (listener) { WorkBundle workBundle; sp strongComponent = mComponent.promote(); if (objcpy(&workBundle, c2workItems, strongComponent ? &strongComponent->mBufferPoolSender : nullptr) != Status::OK) { ALOGE("onWorkDone() received corrupted work items."); return; } Return transStatus = listener->onWorkDone(workBundle); if (!transStatus.isOk()) { ALOGE("onWorkDone -- transaction failed."); return; } yieldBufferQueueBlocks(c2workItems, true); } } protected: wp mComponent; wp mListener; }; // Component Component::Component( const std::shared_ptr& component, const sp& listener, const sp& store, const sp<::android::hardware::media::bufferpool::V1_0:: IClientManager>& clientPoolManager) : Configurable(new CachedConfigurable( std::make_unique(component->intf()))), mComponent(component), mInterface(component->intf()), mListener(listener), mStore(store), mBufferPoolSender(clientPoolManager) { // Retrieve supported parameters from store // TODO: We could cache this per component/interface type mInit = init(store.get()); } c2_status_t Component::status() const { return mInit; } // Methods from ::android::hardware::media::c2::V1_0::IComponent Return Component::queue(const WorkBundle& workBundle) { ALOGV("queue -- converting input"); std::list> c2works; // TODO: Connect with bufferpool API for buffer transfers if (objcpy(&c2works, workBundle) != C2_OK) { ALOGV("queue -- corrupted"); return Status::CORRUPTED; } ALOGV("queue -- calling"); return static_cast(mComponent->queue_nb(&c2works)); } Return Component::flush(flush_cb _hidl_cb) { std::list> c2flushedWorks; ALOGV("flush -- calling"); c2_status_t c2res = mComponent->flush_sm( C2Component::FLUSH_COMPONENT, &c2flushedWorks); WorkBundle flushedWorkBundle; Status res = static_cast(c2res); if (c2res == C2_OK) { ALOGV("flush -- converting output"); res = objcpy(&flushedWorkBundle, c2flushedWorks, &mBufferPoolSender); } _hidl_cb(res, flushedWorkBundle); yieldBufferQueueBlocks(c2flushedWorks, true); return Void(); } Return Component::drain(bool withEos) { ALOGV("drain"); return static_cast(mComponent->drain_nb(withEos ? C2Component::DRAIN_COMPONENT_WITH_EOS : C2Component::DRAIN_COMPONENT_NO_EOS)); } Return Component::setOutputSurface( uint64_t blockPoolId, const sp& surface) { std::shared_ptr pool; GetCodec2BlockPool(blockPoolId, mComponent, &pool); if (pool && pool->getAllocatorId() == C2PlatformAllocatorStore::BUFFERQUEUE) { std::shared_ptr bqPool = std::static_pointer_cast(pool); C2BufferQueueBlockPool::OnRenderCallback cb = [this](uint64_t producer, int32_t slot, int64_t nsecs) { // TODO: batch this hidl_vec rendered; rendered.resize(1); rendered[0] = { producer, slot, nsecs }; mListener->onFramesRendered(rendered); }; if (bqPool) { bqPool->setRenderCallback(cb); bqPool->configureProducer(surface); } } return Status::OK; } Return Component::connectToOmxInputSurface( const sp& producer, const sp<::android::hardware::media::omx::V1_0:: IGraphicBufferSource>& source) { // TODO implement (void)producer; (void)source; return Status::OMITTED; } Return Component::disconnectFromInputSurface() { // TODO implement return Status::OK; } namespace /* unnamed */ { struct BlockPoolIntf : public ConfigurableC2Intf { BlockPoolIntf(const std::shared_ptr& pool) : ConfigurableC2Intf("C2BlockPool:" + std::to_string(pool->getLocalId())), mPool(pool) { } virtual c2_status_t config( const std::vector& params, c2_blocking_t mayBlock, std::vector>* const failures ) override { (void)params; (void)mayBlock; (void)failures; return C2_OK; } virtual c2_status_t query( const std::vector& indices, c2_blocking_t mayBlock, std::vector>* const params ) const override { (void)indices; (void)mayBlock; (void)params; return C2_OK; } virtual c2_status_t querySupportedParams( std::vector>* const params ) const override { (void)params; return C2_OK; } virtual c2_status_t querySupportedValues( std::vector& fields, c2_blocking_t mayBlock) const override { (void)fields; (void)mayBlock; return C2_OK; } protected: std::shared_ptr mPool; }; } // unnamed namespace Return Component::createBlockPool( uint32_t allocatorId, createBlockPool_cb _hidl_cb) { std::shared_ptr blockPool; c2_status_t status = CreateCodec2BlockPool( static_cast(allocatorId), mComponent, &blockPool); if (status != C2_OK) { blockPool = nullptr; } if (blockPool) { mBlockPoolsMutex.lock(); mBlockPools.emplace(blockPool->getLocalId(), blockPool); mBlockPoolsMutex.unlock(); } else if (status == C2_OK) { status = C2_CORRUPTED; } _hidl_cb(static_cast(status), blockPool ? blockPool->getLocalId() : 0, new CachedConfigurable( std::make_unique(blockPool))); return Void(); } Return Component::destroyBlockPool(uint64_t blockPoolId) { std::lock_guard lock(mBlockPoolsMutex); return mBlockPools.erase(blockPoolId) == 1 ? Status::OK : Status::CORRUPTED; } Return Component::start() { ALOGV("start"); return static_cast(mComponent->start()); } Return Component::stop() { ALOGV("stop"); return static_cast(mComponent->stop()); } Return Component::reset() { ALOGV("reset"); Status status = static_cast(mComponent->reset()); { std::lock_guard lock(mBlockPoolsMutex); mBlockPools.clear(); } return status; } Return Component::release() { ALOGV("release"); Status status = static_cast(mComponent->release()); { std::lock_guard lock(mBlockPoolsMutex); mBlockPools.clear(); } return status; } void Component::setLocalId(const Component::LocalId& localId) { mLocalId = localId; } void Component::initListener(const sp& self) { std::shared_ptr c2listener = std::make_shared(self); c2_status_t res = mComponent->setListener_vb(c2listener, C2_DONT_BLOCK); if (res != C2_OK) { mInit = res; } } Component::~Component() { mStore->reportComponentDeath(mLocalId); } Component::InterfaceKey::InterfaceKey(const sp& component) { isRemote = component->isRemote(); if (isRemote) { remote = ::android::hardware::toBinder(component); } else { local = component; } } } // namespace utils } // namespace V1_0 } // namespace c2 } // namespace media } // namespace google } // namespace hardware