/* * Copyright (C) 2017 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 <inttypes.h> #include <fcntl.h> #include <stdlib.h> #include <string.h> #include <sys/time.h> #include <sys/types.h> #include <sys/stat.h> #include <thread> //#define LOG_NDEBUG 0 #define LOG_TAG "codec2" #include <log/log.h> #include <binder/IServiceManager.h> #include <binder/ProcessState.h> #include <media/DataSource.h> #include <media/ICrypto.h> #include <media/IMediaHTTPService.h> #include <media/MediaExtractor.h> #include <media/MediaSource.h> #include <media/stagefright/foundation/ABuffer.h> #include <media/stagefright/foundation/ALooper.h> #include <media/stagefright/foundation/AMessage.h> #include <media/stagefright/foundation/AUtils.h> #include <media/stagefright/DataSourceFactory.h> #include <media/stagefright/MediaDefs.h> #include <media/stagefright/MediaErrors.h> #include <media/stagefright/MediaExtractorFactory.h> #include <media/stagefright/MetaData.h> #include <media/stagefright/Utils.h> #include <gui/GLConsumer.h> #include <gui/IProducerListener.h> #include <gui/Surface.h> #include <gui/SurfaceComposerClient.h> #include <C2AllocatorGralloc.h> #include <C2Buffer.h> #include <C2BufferPriv.h> #include <C2Component.h> #include <C2Config.h> #include <C2Debug.h> #include <C2PlatformSupport.h> #include <C2Work.h> using namespace android; using namespace std::chrono_literals; namespace { class LinearBuffer : public C2Buffer { public: explicit LinearBuffer(const std::shared_ptr<C2LinearBlock> &block) : C2Buffer({ block->share(block->offset(), block->size(), ::C2Fence()) }) {} }; class Listener; class SimplePlayer { public: SimplePlayer(); ~SimplePlayer(); void onWorkDone(std::weak_ptr<C2Component> component, std::list<std::unique_ptr<C2Work>> workItems); void onTripped(std::weak_ptr<C2Component> component, std::vector<std::shared_ptr<C2SettingResult>> settingResult); void onError(std::weak_ptr<C2Component> component, uint32_t errorCode); void play(const sp<IMediaSource> &source); private: typedef std::unique_lock<std::mutex> ULock; std::shared_ptr<Listener> mListener; std::shared_ptr<C2Component> mComponent; sp<IProducerListener> mProducerListener; std::atomic_int mLinearPoolId; std::shared_ptr<C2Allocator> mAllocIon; std::shared_ptr<C2BlockPool> mLinearPool; std::mutex mQueueLock; std::condition_variable mQueueCondition; std::list<std::unique_ptr<C2Work>> mWorkQueue; std::mutex mProcessedLock; std::condition_variable mProcessedCondition; std::list<std::unique_ptr<C2Work>> mProcessedWork; sp<Surface> mSurface; sp<SurfaceComposerClient> mComposerClient; sp<SurfaceControl> mControl; }; class Listener : public C2Component::Listener { public: explicit Listener(SimplePlayer *thiz) : mThis(thiz) {} virtual ~Listener() = default; virtual void onWorkDone_nb(std::weak_ptr<C2Component> component, std::list<std::unique_ptr<C2Work>> workItems) override { mThis->onWorkDone(component, std::move(workItems)); } virtual void onTripped_nb(std::weak_ptr<C2Component> component, std::vector<std::shared_ptr<C2SettingResult>> settingResult) override { mThis->onTripped(component, settingResult); } virtual void onError_nb(std::weak_ptr<C2Component> component, uint32_t errorCode) override { mThis->onError(component, errorCode); } private: SimplePlayer * const mThis; }; SimplePlayer::SimplePlayer() : mListener(new Listener(this)), mProducerListener(new DummyProducerListener), mLinearPoolId(C2BlockPool::PLATFORM_START), mComposerClient(new SurfaceComposerClient) { CHECK_EQ(mComposerClient->initCheck(), (status_t)OK); std::shared_ptr<C2AllocatorStore> store = GetCodec2PlatformAllocatorStore(); CHECK_EQ(store->fetchAllocator(C2AllocatorStore::DEFAULT_LINEAR, &mAllocIon), C2_OK); mLinearPool = std::make_shared<C2PooledBlockPool>(mAllocIon, mLinearPoolId++); mControl = mComposerClient->createSurface( String8("A Surface"), 1280, 800, HAL_PIXEL_FORMAT_YV12); //PIXEL_FORMAT_RGB_565); CHECK(mControl != NULL); CHECK(mControl->isValid()); SurfaceComposerClient::Transaction{} .setLayer(mControl, INT_MAX) .show(mControl) .apply(); mSurface = mControl->getSurface(); CHECK(mSurface != NULL); mSurface->connect(NATIVE_WINDOW_API_CPU, mProducerListener); } SimplePlayer::~SimplePlayer() { mComposerClient->dispose(); } void SimplePlayer::onWorkDone( std::weak_ptr<C2Component> component, std::list<std::unique_ptr<C2Work>> workItems) { ALOGV("SimplePlayer::onWorkDone"); (void) component; ULock l(mProcessedLock); for (auto & item : workItems) { mProcessedWork.push_back(std::move(item)); } mProcessedCondition.notify_all(); } void SimplePlayer::onTripped( std::weak_ptr<C2Component> component, std::vector<std::shared_ptr<C2SettingResult>> settingResult) { (void) component; (void) settingResult; // TODO } void SimplePlayer::onError(std::weak_ptr<C2Component> component, uint32_t errorCode) { (void) component; (void) errorCode; // TODO } void SimplePlayer::play(const sp<IMediaSource> &source) { ALOGV("SimplePlayer::play"); sp<AMessage> format; (void) convertMetaDataToMessage(source->getFormat(), &format); sp<ABuffer> csd0, csd1; format->findBuffer("csd-0", &csd0); format->findBuffer("csd-1", &csd1); status_t err = source->start(); if (err != OK) { fprintf(stderr, "source returned error %d (0x%08x)\n", err, err); return; } std::shared_ptr<C2ComponentStore> store = GetCodec2PlatformComponentStore(); std::shared_ptr<C2Component> component; (void)store->createComponent("c2.android.avc.decoder", &component); (void)component->setListener_vb(mListener, C2_DONT_BLOCK); std::unique_ptr<C2PortBlockPoolsTuning::output> pools = C2PortBlockPoolsTuning::output::AllocUnique({ (uint64_t)C2BlockPool::BASIC_GRAPHIC }); std::vector<std::unique_ptr<C2SettingResult>> result; (void)component->intf()->config_vb({pools.get()}, C2_DONT_BLOCK, &result); component->start(); for (int i = 0; i < 8; ++i) { mWorkQueue.emplace_back(new C2Work); } std::atomic_bool running(true); std::thread surfaceThread([this, &running]() { const sp<IGraphicBufferProducer> &igbp = mSurface->getIGraphicBufferProducer(); while (running) { std::unique_ptr<C2Work> work; { ULock l(mProcessedLock); if (mProcessedWork.empty()) { mProcessedCondition.wait_for(l, 100ms); if (mProcessedWork.empty()) { continue; } } work.swap(mProcessedWork.front()); mProcessedWork.pop_front(); } int slot; sp<Fence> fence; ALOGV("Render: Frame #%lld", work->worklets.front()->output.ordinal.frameIndex.peekll()); const std::shared_ptr<C2Buffer> &output = work->worklets.front()->output.buffers[0]; if (output) { const C2ConstGraphicBlock block = output->data().graphicBlocks().front(); native_handle_t *grallocHandle = UnwrapNativeCodec2GrallocHandle(block.handle()); sp<GraphicBuffer> buffer(new GraphicBuffer( grallocHandle, GraphicBuffer::CLONE_HANDLE, block.width(), block.height(), HAL_PIXEL_FORMAT_YV12, 1, (uint64_t)GRALLOC_USAGE_SW_READ_OFTEN | GRALLOC_USAGE_SW_WRITE_OFTEN, block.width())); native_handle_delete(grallocHandle); status_t err = igbp->attachBuffer(&slot, buffer); IGraphicBufferProducer::QueueBufferInput qbi( (work->worklets.front()->output.ordinal.timestamp * 1000ll).peekll(), false, HAL_DATASPACE_UNKNOWN, Rect(block.width(), block.height()), NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW, 0, Fence::NO_FENCE, 0); IGraphicBufferProducer::QueueBufferOutput qbo; err = igbp->queueBuffer(slot, qbi, &qbo); } work->input.buffers.clear(); work->worklets.clear(); ULock l(mQueueLock); mWorkQueue.push_back(std::move(work)); mQueueCondition.notify_all(); } ALOGV("render loop finished"); }); long numFrames = 0; mLinearPool.reset(new C2PooledBlockPool(mAllocIon, mLinearPoolId++)); for (;;) { size_t size = 0u; void *data = nullptr; int64_t timestamp = 0u; MediaBufferBase *buffer = nullptr; sp<ABuffer> csd; if (csd0 != nullptr) { csd = csd0; csd0 = nullptr; } else if (csd1 != nullptr) { csd = csd1; csd1 = nullptr; } else { status_t err = source->read(&buffer); if (err != OK) { CHECK(buffer == NULL); if (err == INFO_FORMAT_CHANGED) { continue; } break; } MetaDataBase &meta = buffer->meta_data(); CHECK(meta.findInt64(kKeyTime, ×tamp)); size = buffer->size(); data = buffer->data(); } if (csd != nullptr) { size = csd->size(); data = csd->data(); } // Prepare C2Work std::unique_ptr<C2Work> work; while (!work) { ULock l(mQueueLock); if (!mWorkQueue.empty()) { work.swap(mWorkQueue.front()); mWorkQueue.pop_front(); } else { mQueueCondition.wait_for(l, 100ms); } } work->input.flags = (C2FrameData::flags_t)0; work->input.ordinal.timestamp = timestamp; work->input.ordinal.frameIndex = numFrames; std::shared_ptr<C2LinearBlock> block; mLinearPool->fetchLinearBlock( size, { C2MemoryUsage::CPU_READ, C2MemoryUsage::CPU_WRITE }, &block); C2WriteView view = block->map().get(); if (view.error() != C2_OK) { fprintf(stderr, "C2LinearBlock::map() failed : %d", view.error()); break; } memcpy(view.base(), data, size); work->input.buffers.clear(); work->input.buffers.emplace_back(new LinearBuffer(block)); work->worklets.clear(); work->worklets.emplace_back(new C2Worklet); std::list<std::unique_ptr<C2Work>> items; items.push_back(std::move(work)); ALOGV("Frame #%ld size = %zu", numFrames, size); // DO THE DECODING component->queue_nb(&items); if (buffer) { buffer->release(); buffer = NULL; } ++numFrames; } ALOGV("main loop finished"); source->stop(); running.store(false); surfaceThread.join(); component->release(); printf("\n"); } } // namespace static void usage(const char *me) { fprintf(stderr, "usage: %s [options] [input_filename]\n", me); fprintf(stderr, " -h(elp)\n"); } int main(int argc, char **argv) { android::ProcessState::self()->startThreadPool(); int res; while ((res = getopt(argc, argv, "h")) >= 0) { switch (res) { case 'h': default: { usage(argv[0]); exit(1); break; } } } argc -= optind; argv += optind; if (argc < 1) { fprintf(stderr, "No input file specified\n"); return 1; } status_t err = OK; SimplePlayer player; for (int k = 0; k < argc && err == OK; ++k) { const char *filename = argv[k]; sp<DataSource> dataSource = DataSourceFactory::CreateFromURI(NULL /* httpService */, filename); if (strncasecmp(filename, "sine:", 5) && dataSource == NULL) { fprintf(stderr, "Unable to create data source.\n"); return 1; } Vector<sp<IMediaSource> > mediaSources; sp<IMediaSource> mediaSource; sp<IMediaExtractor> extractor = MediaExtractorFactory::Create(dataSource); if (extractor == NULL) { fprintf(stderr, "could not create extractor.\n"); return -1; } sp<MetaData> meta = extractor->getMetaData(); if (meta != NULL) { const char *mime; if (!meta->findCString(kKeyMIMEType, &mime)) { fprintf(stderr, "extractor did not provide MIME type.\n"); return -1; } } size_t numTracks = extractor->countTracks(); size_t i; for (i = 0; i < numTracks; ++i) { meta = extractor->getTrackMetaData( i, MediaExtractor::kIncludeExtensiveMetaData); if (meta == NULL) { break; } const char *mime; meta->findCString(kKeyMIMEType, &mime); // TODO: allowing AVC only for the time being if (!strncasecmp(mime, "video/avc", 9)) { break; } meta = NULL; } if (meta == NULL) { fprintf(stderr, "No AVC track found.\n"); return -1; } mediaSource = extractor->getTrack(i); if (mediaSource == nullptr) { fprintf(stderr, "skip NULL track %zu, total tracks %zu.\n", i, numTracks); return -1; } player.play(mediaSource); } return 0; }