NuMediaExtractor.cpp revision 918c7652b9a38c02e26c0c46541cea82070c0e43
15778822d86b0337407514b9372562b86edfa91cdAndreas Huber/*
25778822d86b0337407514b9372562b86edfa91cdAndreas Huber * Copyright 2012, The Android Open Source Project
35778822d86b0337407514b9372562b86edfa91cdAndreas Huber *
45778822d86b0337407514b9372562b86edfa91cdAndreas Huber * Licensed under the Apache License, Version 2.0 (the "License");
55778822d86b0337407514b9372562b86edfa91cdAndreas Huber * you may not use this file except in compliance with the License.
65778822d86b0337407514b9372562b86edfa91cdAndreas Huber * You may obtain a copy of the License at
75778822d86b0337407514b9372562b86edfa91cdAndreas Huber *
85778822d86b0337407514b9372562b86edfa91cdAndreas Huber *     http://www.apache.org/licenses/LICENSE-2.0
95778822d86b0337407514b9372562b86edfa91cdAndreas Huber *
105778822d86b0337407514b9372562b86edfa91cdAndreas Huber * Unless required by applicable law or agreed to in writing, software
115778822d86b0337407514b9372562b86edfa91cdAndreas Huber * distributed under the License is distributed on an "AS IS" BASIS,
125778822d86b0337407514b9372562b86edfa91cdAndreas Huber * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
135778822d86b0337407514b9372562b86edfa91cdAndreas Huber * See the License for the specific language governing permissions and
145778822d86b0337407514b9372562b86edfa91cdAndreas Huber * limitations under the License.
155778822d86b0337407514b9372562b86edfa91cdAndreas Huber */
165778822d86b0337407514b9372562b86edfa91cdAndreas Huber
175778822d86b0337407514b9372562b86edfa91cdAndreas Huber//#define LOG_NDEBUG 0
185778822d86b0337407514b9372562b86edfa91cdAndreas Huber#define LOG_TAG "NuMediaExtractor"
195778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <utils/Log.h>
205778822d86b0337407514b9372562b86edfa91cdAndreas Huber
215778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/NuMediaExtractor.h>
225778822d86b0337407514b9372562b86edfa91cdAndreas Huber
235778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include "include/ESDS.h"
245778822d86b0337407514b9372562b86edfa91cdAndreas Huber
255778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/foundation/ABuffer.h>
265778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/foundation/ADebug.h>
275778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/foundation/AMessage.h>
285778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/DataSource.h>
29918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber#include <media/stagefright/FileSource.h>
305778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/MediaBuffer.h>
315778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/MediaDefs.h>
325778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/MediaErrors.h>
335778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/MediaExtractor.h>
345778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/MediaSource.h>
355778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/MetaData.h>
365778822d86b0337407514b9372562b86edfa91cdAndreas Huber#include <media/stagefright/Utils.h>
375778822d86b0337407514b9372562b86edfa91cdAndreas Huber
385778822d86b0337407514b9372562b86edfa91cdAndreas Hubernamespace android {
395778822d86b0337407514b9372562b86edfa91cdAndreas Huber
405778822d86b0337407514b9372562b86edfa91cdAndreas HuberNuMediaExtractor::NuMediaExtractor() {
415778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
425778822d86b0337407514b9372562b86edfa91cdAndreas Huber
435778822d86b0337407514b9372562b86edfa91cdAndreas HuberNuMediaExtractor::~NuMediaExtractor() {
445778822d86b0337407514b9372562b86edfa91cdAndreas Huber    releaseTrackSamples();
455778822d86b0337407514b9372562b86edfa91cdAndreas Huber
465778822d86b0337407514b9372562b86edfa91cdAndreas Huber    for (size_t i = 0; i < mSelectedTracks.size(); ++i) {
475778822d86b0337407514b9372562b86edfa91cdAndreas Huber        TrackInfo *info = &mSelectedTracks.editItemAt(i);
485778822d86b0337407514b9372562b86edfa91cdAndreas Huber
495778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK_EQ((status_t)OK, info->mSource->stop());
505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
515778822d86b0337407514b9372562b86edfa91cdAndreas Huber
525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mSelectedTracks.clear();
535778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
545778822d86b0337407514b9372562b86edfa91cdAndreas Huber
55918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huberstatus_t NuMediaExtractor::setDataSource(
56918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber        const char *path, const KeyedVector<String8, String8> *headers) {
57918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    if (mImpl != NULL) {
58918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber        return -EINVAL;
59918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    }
60918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
61918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    sp<DataSource> dataSource = DataSource::CreateFromURI(path, headers);
625778822d86b0337407514b9372562b86edfa91cdAndreas Huber
635778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (dataSource == NULL) {
645778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return -ENOENT;
655778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
665778822d86b0337407514b9372562b86edfa91cdAndreas Huber
675778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mImpl = MediaExtractor::Create(dataSource);
685778822d86b0337407514b9372562b86edfa91cdAndreas Huber
695778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (mImpl == NULL) {
705778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return ERROR_UNSUPPORTED;
715778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
725778822d86b0337407514b9372562b86edfa91cdAndreas Huber
735778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
745778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
755778822d86b0337407514b9372562b86edfa91cdAndreas Huber
76918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huberstatus_t NuMediaExtractor::setDataSource(int fd, off64_t offset, off64_t size) {
77918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    if (mImpl != NULL) {
78918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber        return -EINVAL;
79918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    }
80918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
81918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    sp<FileSource> fileSource = new FileSource(dup(fd), offset, size);
82918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
83918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    status_t err = fileSource->initCheck();
84918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    if (err != OK) {
85918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber        return err;
86918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    }
87918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
88918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    mImpl = MediaExtractor::Create(fileSource);
89918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
90918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    if (mImpl == NULL) {
91918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber        return ERROR_UNSUPPORTED;
92918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    }
93918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
94918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    return OK;
95918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber}
96918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
975778822d86b0337407514b9372562b86edfa91cdAndreas Hubersize_t NuMediaExtractor::countTracks() const {
985778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return mImpl == NULL ? 0 : mImpl->countTracks();
995778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
1005778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1015778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t NuMediaExtractor::getTrackFormat(
1025778822d86b0337407514b9372562b86edfa91cdAndreas Huber        size_t index, sp<AMessage> *format) const {
1035778822d86b0337407514b9372562b86edfa91cdAndreas Huber    *format = NULL;
1045778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1055778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (mImpl == NULL) {
1065778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return -EINVAL;
1075778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
1085778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1095778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (index >= mImpl->countTracks()) {
1105778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return -ERANGE;
1115778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
1125778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1135778822d86b0337407514b9372562b86edfa91cdAndreas Huber    sp<MetaData> meta = mImpl->getTrackMetaData(index);
1145778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1155778822d86b0337407514b9372562b86edfa91cdAndreas Huber    const char *mime;
1165778822d86b0337407514b9372562b86edfa91cdAndreas Huber    CHECK(meta->findCString(kKeyMIMEType, &mime));
1175778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1185778822d86b0337407514b9372562b86edfa91cdAndreas Huber    sp<AMessage> msg = new AMessage;
1195778822d86b0337407514b9372562b86edfa91cdAndreas Huber    msg->setString("mime", mime);
1205778822d86b0337407514b9372562b86edfa91cdAndreas Huber
121918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    int64_t durationUs;
122918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    if (meta->findInt64(kKeyDuration, &durationUs)) {
123918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber        msg->setInt64("durationUs", durationUs);
124918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber    }
125918c7652b9a38c02e26c0c46541cea82070c0e43Andreas Huber
1265778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!strncasecmp("video/", mime, 6)) {
1275778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t width, height;
1285778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(meta->findInt32(kKeyWidth, &width));
1295778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(meta->findInt32(kKeyHeight, &height));
1305778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1315778822d86b0337407514b9372562b86edfa91cdAndreas Huber        msg->setInt32("width", width);
1325778822d86b0337407514b9372562b86edfa91cdAndreas Huber        msg->setInt32("height", height);
1335778822d86b0337407514b9372562b86edfa91cdAndreas Huber    } else {
1345778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(!strncasecmp("audio/", mime, 6));
1355778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1365778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t numChannels, sampleRate;
1375778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(meta->findInt32(kKeyChannelCount, &numChannels));
1385778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(meta->findInt32(kKeySampleRate, &sampleRate));
1395778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1405778822d86b0337407514b9372562b86edfa91cdAndreas Huber        msg->setInt32("channel-count", numChannels);
1415778822d86b0337407514b9372562b86edfa91cdAndreas Huber        msg->setInt32("sample-rate", sampleRate);
142ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
143ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        int32_t isADTS;
144ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        if (meta->findInt32(kKeyIsADTS, &isADTS)) {
145ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber            msg->setInt32("is-adts", true);
146ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        }
1475778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
1485778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1495778822d86b0337407514b9372562b86edfa91cdAndreas Huber    int32_t maxInputSize;
1505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (meta->findInt32(kKeyMaxInputSize, &maxInputSize)) {
1515778822d86b0337407514b9372562b86edfa91cdAndreas Huber        msg->setInt32("max-input-size", maxInputSize);
1525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
1535778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1545778822d86b0337407514b9372562b86edfa91cdAndreas Huber    uint32_t type;
1555778822d86b0337407514b9372562b86edfa91cdAndreas Huber    const void *data;
1565778822d86b0337407514b9372562b86edfa91cdAndreas Huber    size_t size;
1575778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (meta->findData(kKeyAVCC, &type, &data, &size)) {
1585778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // Parse the AVCDecoderConfigurationRecord
1595778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1605778822d86b0337407514b9372562b86edfa91cdAndreas Huber        const uint8_t *ptr = (const uint8_t *)data;
1615778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1625778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(size >= 7);
1635778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK_EQ((unsigned)ptr[0], 1u);  // configurationVersion == 1
1645778822d86b0337407514b9372562b86edfa91cdAndreas Huber        uint8_t profile = ptr[1];
1655778822d86b0337407514b9372562b86edfa91cdAndreas Huber        uint8_t level = ptr[3];
1665778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1675778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // There is decodable content out there that fails the following
1685778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // assertion, let's be lenient for now...
1695778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // CHECK((ptr[4] >> 2) == 0x3f);  // reserved
1705778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1715778822d86b0337407514b9372562b86edfa91cdAndreas Huber        size_t lengthSize = 1 + (ptr[4] & 3);
1725778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1735778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // commented out check below as H264_QVGA_500_NO_AUDIO.3gp
1745778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // violates it...
1755778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // CHECK((ptr[5] >> 5) == 7);  // reserved
1765778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1775778822d86b0337407514b9372562b86edfa91cdAndreas Huber        size_t numSeqParameterSets = ptr[5] & 31;
1785778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1795778822d86b0337407514b9372562b86edfa91cdAndreas Huber        ptr += 6;
1805778822d86b0337407514b9372562b86edfa91cdAndreas Huber        size -= 6;
1815778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1825778822d86b0337407514b9372562b86edfa91cdAndreas Huber        sp<ABuffer> buffer = new ABuffer(1024);
1835778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->setRange(0, 0);
1845778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1855778822d86b0337407514b9372562b86edfa91cdAndreas Huber        for (size_t i = 0; i < numSeqParameterSets; ++i) {
1865778822d86b0337407514b9372562b86edfa91cdAndreas Huber            CHECK(size >= 2);
1875778822d86b0337407514b9372562b86edfa91cdAndreas Huber            size_t length = U16_AT(ptr);
1885778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1895778822d86b0337407514b9372562b86edfa91cdAndreas Huber            ptr += 2;
1905778822d86b0337407514b9372562b86edfa91cdAndreas Huber            size -= 2;
1915778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1925778822d86b0337407514b9372562b86edfa91cdAndreas Huber            CHECK(size >= length);
1935778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1945778822d86b0337407514b9372562b86edfa91cdAndreas Huber            memcpy(buffer->data() + buffer->size(), "\x00\x00\x00\x01", 4);
1955778822d86b0337407514b9372562b86edfa91cdAndreas Huber            memcpy(buffer->data() + buffer->size() + 4, ptr, length);
1965778822d86b0337407514b9372562b86edfa91cdAndreas Huber            buffer->setRange(0, buffer->size() + 4 + length);
1975778822d86b0337407514b9372562b86edfa91cdAndreas Huber
1985778822d86b0337407514b9372562b86edfa91cdAndreas Huber            ptr += length;
1995778822d86b0337407514b9372562b86edfa91cdAndreas Huber            size -= length;
2005778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
2015778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2025778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt32("csd", true);
2035778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt64("timeUs", 0);
2045778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2052d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber        msg->setBuffer("csd-0", buffer);
2065778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2075778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer = new ABuffer(1024);
2085778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->setRange(0, 0);
2095778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2105778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK(size >= 1);
2115778822d86b0337407514b9372562b86edfa91cdAndreas Huber        size_t numPictureParameterSets = *ptr;
2125778822d86b0337407514b9372562b86edfa91cdAndreas Huber        ++ptr;
2135778822d86b0337407514b9372562b86edfa91cdAndreas Huber        --size;
2145778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2155778822d86b0337407514b9372562b86edfa91cdAndreas Huber        for (size_t i = 0; i < numPictureParameterSets; ++i) {
2165778822d86b0337407514b9372562b86edfa91cdAndreas Huber            CHECK(size >= 2);
2175778822d86b0337407514b9372562b86edfa91cdAndreas Huber            size_t length = U16_AT(ptr);
2185778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2195778822d86b0337407514b9372562b86edfa91cdAndreas Huber            ptr += 2;
2205778822d86b0337407514b9372562b86edfa91cdAndreas Huber            size -= 2;
2215778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2225778822d86b0337407514b9372562b86edfa91cdAndreas Huber            CHECK(size >= length);
2235778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2245778822d86b0337407514b9372562b86edfa91cdAndreas Huber            memcpy(buffer->data() + buffer->size(), "\x00\x00\x00\x01", 4);
2255778822d86b0337407514b9372562b86edfa91cdAndreas Huber            memcpy(buffer->data() + buffer->size() + 4, ptr, length);
2265778822d86b0337407514b9372562b86edfa91cdAndreas Huber            buffer->setRange(0, buffer->size() + 4 + length);
2275778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2285778822d86b0337407514b9372562b86edfa91cdAndreas Huber            ptr += length;
2295778822d86b0337407514b9372562b86edfa91cdAndreas Huber            size -= length;
2305778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
2315778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2325778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt32("csd", true);
2335778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt64("timeUs", 0);
2342d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber        msg->setBuffer("csd-1", buffer);
2355778822d86b0337407514b9372562b86edfa91cdAndreas Huber    } else if (meta->findData(kKeyESDS, &type, &data, &size)) {
2365778822d86b0337407514b9372562b86edfa91cdAndreas Huber        ESDS esds((const char *)data, size);
2375778822d86b0337407514b9372562b86edfa91cdAndreas Huber        CHECK_EQ(esds.InitCheck(), (status_t)OK);
2385778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2395778822d86b0337407514b9372562b86edfa91cdAndreas Huber        const void *codec_specific_data;
2405778822d86b0337407514b9372562b86edfa91cdAndreas Huber        size_t codec_specific_data_size;
2415778822d86b0337407514b9372562b86edfa91cdAndreas Huber        esds.getCodecSpecificInfo(
2425778822d86b0337407514b9372562b86edfa91cdAndreas Huber                &codec_specific_data, &codec_specific_data_size);
2435778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2445778822d86b0337407514b9372562b86edfa91cdAndreas Huber        sp<ABuffer> buffer = new ABuffer(codec_specific_data_size);
2455778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2465778822d86b0337407514b9372562b86edfa91cdAndreas Huber        memcpy(buffer->data(), codec_specific_data,
2475778822d86b0337407514b9372562b86edfa91cdAndreas Huber               codec_specific_data_size);
2485778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2495778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt32("csd", true);
2505778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt64("timeUs", 0);
2512d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber        msg->setBuffer("csd-0", buffer);
2525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    } else if (meta->findData(kKeyVorbisInfo, &type, &data, &size)) {
2535778822d86b0337407514b9372562b86edfa91cdAndreas Huber        sp<ABuffer> buffer = new ABuffer(size);
2545778822d86b0337407514b9372562b86edfa91cdAndreas Huber        memcpy(buffer->data(), data, size);
2555778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2565778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt32("csd", true);
2575778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt64("timeUs", 0);
2582d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber        msg->setBuffer("csd-0", buffer);
2595778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2605778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!meta->findData(kKeyVorbisBooks, &type, &data, &size)) {
2615778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return -EINVAL;
2625778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
2635778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2645778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer = new ABuffer(size);
2655778822d86b0337407514b9372562b86edfa91cdAndreas Huber        memcpy(buffer->data(), data, size);
2665778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2675778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt32("csd", true);
2685778822d86b0337407514b9372562b86edfa91cdAndreas Huber        buffer->meta()->setInt64("timeUs", 0);
2692d8bedd05437b6fccdbc6bf70f673ffd86744d59Andreas Huber        msg->setBuffer("csd-1", buffer);
2705778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
2715778822d86b0337407514b9372562b86edfa91cdAndreas Huber
272ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    if (meta->findData(kKeyEMM, &type, &data, &size)) {
273ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        sp<ABuffer> emm = new ABuffer(size);
274ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        memcpy(emm->data(), data, size);
275ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
276ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        msg->setBuffer("emm", emm);
277ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    }
278ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
279ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    if (meta->findData(kKeyECM, &type, &data, &size)) {
280ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        sp<ABuffer> ecm = new ABuffer(size);
281ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        memcpy(ecm->data(), data, size);
282ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
283ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        msg->setBuffer("ecm", ecm);
284ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    }
285ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
2865778822d86b0337407514b9372562b86edfa91cdAndreas Huber    *format = msg;
2875778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2885778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
2895778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
2905778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2915778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t NuMediaExtractor::selectTrack(size_t index) {
2925778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (mImpl == NULL) {
2935778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return -EINVAL;
2945778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
2955778822d86b0337407514b9372562b86edfa91cdAndreas Huber
2965778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (index >= mImpl->countTracks()) {
2975778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return -ERANGE;
2985778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
2995778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3005778822d86b0337407514b9372562b86edfa91cdAndreas Huber    for (size_t i = 0; i < mSelectedTracks.size(); ++i) {
3015778822d86b0337407514b9372562b86edfa91cdAndreas Huber        TrackInfo *info = &mSelectedTracks.editItemAt(i);
3025778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3035778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (info->mTrackIndex == index) {
3045778822d86b0337407514b9372562b86edfa91cdAndreas Huber            // This track has already been selected.
3055778822d86b0337407514b9372562b86edfa91cdAndreas Huber            return OK;
3065778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
3075778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
3085778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3095778822d86b0337407514b9372562b86edfa91cdAndreas Huber    sp<MediaSource> source = mImpl->getTrack(index);
3105778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3115778822d86b0337407514b9372562b86edfa91cdAndreas Huber    CHECK_EQ((status_t)OK, source->start());
3125778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3135778822d86b0337407514b9372562b86edfa91cdAndreas Huber    mSelectedTracks.push();
3145778822d86b0337407514b9372562b86edfa91cdAndreas Huber    TrackInfo *info = &mSelectedTracks.editItemAt(mSelectedTracks.size() - 1);
3155778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3165778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mSource = source;
3175778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mTrackIndex = index;
3185778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mFinalResult = OK;
3195778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mSample = NULL;
3205778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mSampleTimeUs = -1ll;
321ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    info->mSampleFlags = 0;
322ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    info->mTrackFlags = 0;
3235778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3245778822d86b0337407514b9372562b86edfa91cdAndreas Huber    const char *mime;
3255778822d86b0337407514b9372562b86edfa91cdAndreas Huber    CHECK(source->getFormat()->findCString(kKeyMIMEType, &mime));
3265778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3275778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (!strcasecmp(mime, MEDIA_MIMETYPE_AUDIO_VORBIS)) {
328ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        info->mTrackFlags |= kIsVorbis;
3295778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
3305778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3315778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
3325778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
3335778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3345778822d86b0337407514b9372562b86edfa91cdAndreas Hubervoid NuMediaExtractor::releaseTrackSamples() {
3355778822d86b0337407514b9372562b86edfa91cdAndreas Huber    for (size_t i = 0; i < mSelectedTracks.size(); ++i) {
3365778822d86b0337407514b9372562b86edfa91cdAndreas Huber        TrackInfo *info = &mSelectedTracks.editItemAt(i);
3375778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3385778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (info->mSample != NULL) {
3395778822d86b0337407514b9372562b86edfa91cdAndreas Huber            info->mSample->release();
3405778822d86b0337407514b9372562b86edfa91cdAndreas Huber            info->mSample = NULL;
3415778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3425778822d86b0337407514b9372562b86edfa91cdAndreas Huber            info->mSampleTimeUs = -1ll;
343ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber            info->mSampleFlags = 0;
3445778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
3455778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
3465778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
3475778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3485778822d86b0337407514b9372562b86edfa91cdAndreas Huberssize_t NuMediaExtractor::fetchTrackSamples(int64_t seekTimeUs) {
3495778822d86b0337407514b9372562b86edfa91cdAndreas Huber    TrackInfo *minInfo = NULL;
3505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ssize_t minIndex = -1;
3515778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    for (size_t i = 0; i < mSelectedTracks.size(); ++i) {
3535778822d86b0337407514b9372562b86edfa91cdAndreas Huber        TrackInfo *info = &mSelectedTracks.editItemAt(i);
3545778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3555778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (seekTimeUs >= 0ll) {
3565778822d86b0337407514b9372562b86edfa91cdAndreas Huber            info->mFinalResult = OK;
3575778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3585778822d86b0337407514b9372562b86edfa91cdAndreas Huber            if (info->mSample != NULL) {
3595778822d86b0337407514b9372562b86edfa91cdAndreas Huber                info->mSample->release();
3605778822d86b0337407514b9372562b86edfa91cdAndreas Huber                info->mSample = NULL;
3615778822d86b0337407514b9372562b86edfa91cdAndreas Huber                info->mSampleTimeUs = -1ll;
362ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                info->mSampleFlags = 0;
3635778822d86b0337407514b9372562b86edfa91cdAndreas Huber            }
3645778822d86b0337407514b9372562b86edfa91cdAndreas Huber        } else if (info->mFinalResult != OK) {
3655778822d86b0337407514b9372562b86edfa91cdAndreas Huber            continue;
3665778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
3675778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3685778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (info->mSample == NULL) {
3695778822d86b0337407514b9372562b86edfa91cdAndreas Huber            MediaSource::ReadOptions options;
3705778822d86b0337407514b9372562b86edfa91cdAndreas Huber            if (seekTimeUs >= 0ll) {
3715778822d86b0337407514b9372562b86edfa91cdAndreas Huber                options.setSeekTo(seekTimeUs);
3725778822d86b0337407514b9372562b86edfa91cdAndreas Huber            }
3735778822d86b0337407514b9372562b86edfa91cdAndreas Huber            status_t err = info->mSource->read(&info->mSample, &options);
3745778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3755778822d86b0337407514b9372562b86edfa91cdAndreas Huber            if (err != OK) {
3765778822d86b0337407514b9372562b86edfa91cdAndreas Huber                CHECK(info->mSample == NULL);
3775778822d86b0337407514b9372562b86edfa91cdAndreas Huber
3785778822d86b0337407514b9372562b86edfa91cdAndreas Huber                info->mFinalResult = err;
3795778822d86b0337407514b9372562b86edfa91cdAndreas Huber                info->mSampleTimeUs = -1ll;
380ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                info->mSampleFlags = 0;
3815778822d86b0337407514b9372562b86edfa91cdAndreas Huber                continue;
3825778822d86b0337407514b9372562b86edfa91cdAndreas Huber            } else {
3835778822d86b0337407514b9372562b86edfa91cdAndreas Huber                CHECK(info->mSample != NULL);
3845778822d86b0337407514b9372562b86edfa91cdAndreas Huber                CHECK(info->mSample->meta_data()->findInt64(
3855778822d86b0337407514b9372562b86edfa91cdAndreas Huber                            kKeyTime, &info->mSampleTimeUs));
386ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
387ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                info->mSampleFlags = 0;
388ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
389ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                int32_t val;
390ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                if (info->mSample->meta_data()->findInt32(
391ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                            kKeyIsSyncFrame, &val) && val != 0) {
392ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                    info->mSampleFlags |= SAMPLE_FLAG_SYNC;
393ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                }
394ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
395ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                if (info->mSample->meta_data()->findInt32(
396ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                            kKeyScrambling, &val) && val != 0) {
397ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                    info->mSampleFlags |= SAMPLE_FLAG_ENCRYPTED;
398ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber                }
3995778822d86b0337407514b9372562b86edfa91cdAndreas Huber            }
4005778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
4015778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4025778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (minInfo == NULL  || info->mSampleTimeUs < minInfo->mSampleTimeUs) {
4035778822d86b0337407514b9372562b86edfa91cdAndreas Huber            minInfo = info;
4045778822d86b0337407514b9372562b86edfa91cdAndreas Huber            minIndex = i;
4055778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
4065778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4075778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4085778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return minIndex;
4095778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
4105778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4115778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t NuMediaExtractor::seekTo(int64_t timeUs) {
4125778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return fetchTrackSamples(timeUs);
4135778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
4145778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4155778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t NuMediaExtractor::advance() {
4165778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ssize_t minIndex = fetchTrackSamples();
4175778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4185778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (minIndex < 0) {
4195778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return ERROR_END_OF_STREAM;
4205778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4215778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4225778822d86b0337407514b9372562b86edfa91cdAndreas Huber    TrackInfo *info = &mSelectedTracks.editItemAt(minIndex);
4235778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4245778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mSample->release();
4255778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mSample = NULL;
4265778822d86b0337407514b9372562b86edfa91cdAndreas Huber    info->mSampleTimeUs = -1ll;
4275778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4285778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
4295778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
4305778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4315778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t NuMediaExtractor::readSampleData(const sp<ABuffer> &buffer) {
4325778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ssize_t minIndex = fetchTrackSamples();
4335778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4345778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (minIndex < 0) {
4355778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return ERROR_END_OF_STREAM;
4365778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4375778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4385778822d86b0337407514b9372562b86edfa91cdAndreas Huber    TrackInfo *info = &mSelectedTracks.editItemAt(minIndex);
4395778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4405778822d86b0337407514b9372562b86edfa91cdAndreas Huber    size_t sampleSize = info->mSample->range_length();
4415778822d86b0337407514b9372562b86edfa91cdAndreas Huber
442ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    if (info->mTrackFlags & kIsVorbis) {
4435778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // Each sample's data is suffixed by the number of page samples
4445778822d86b0337407514b9372562b86edfa91cdAndreas Huber        // or -1 if not available.
4455778822d86b0337407514b9372562b86edfa91cdAndreas Huber        sampleSize += sizeof(int32_t);
4465778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4475778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4485778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (buffer->capacity() < sampleSize) {
4495778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return -ENOMEM;
4505778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4515778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4525778822d86b0337407514b9372562b86edfa91cdAndreas Huber    const uint8_t *src =
4535778822d86b0337407514b9372562b86edfa91cdAndreas Huber        (const uint8_t *)info->mSample->data()
4545778822d86b0337407514b9372562b86edfa91cdAndreas Huber            + info->mSample->range_offset();
4555778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4565778822d86b0337407514b9372562b86edfa91cdAndreas Huber    memcpy((uint8_t *)buffer->data(), src, info->mSample->range_length());
4575778822d86b0337407514b9372562b86edfa91cdAndreas Huber
458ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    if (info->mTrackFlags & kIsVorbis) {
4595778822d86b0337407514b9372562b86edfa91cdAndreas Huber        int32_t numPageSamples;
4605778822d86b0337407514b9372562b86edfa91cdAndreas Huber        if (!info->mSample->meta_data()->findInt32(
4615778822d86b0337407514b9372562b86edfa91cdAndreas Huber                    kKeyValidSamples, &numPageSamples)) {
4625778822d86b0337407514b9372562b86edfa91cdAndreas Huber            numPageSamples = -1;
4635778822d86b0337407514b9372562b86edfa91cdAndreas Huber        }
4645778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4655778822d86b0337407514b9372562b86edfa91cdAndreas Huber        memcpy((uint8_t *)buffer->data() + info->mSample->range_length(),
4665778822d86b0337407514b9372562b86edfa91cdAndreas Huber               &numPageSamples,
4675778822d86b0337407514b9372562b86edfa91cdAndreas Huber               sizeof(numPageSamples));
4685778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4695778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4705778822d86b0337407514b9372562b86edfa91cdAndreas Huber    buffer->setRange(0, sampleSize);
4715778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4725778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
4735778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
4745778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4755778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t NuMediaExtractor::getSampleTrackIndex(size_t *trackIndex) {
4765778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ssize_t minIndex = fetchTrackSamples();
4775778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4785778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (minIndex < 0) {
4795778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return ERROR_END_OF_STREAM;
4805778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4815778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4825778822d86b0337407514b9372562b86edfa91cdAndreas Huber    TrackInfo *info = &mSelectedTracks.editItemAt(minIndex);
4835778822d86b0337407514b9372562b86edfa91cdAndreas Huber    *trackIndex = info->mTrackIndex;
4845778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4855778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
4865778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
4875778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4885778822d86b0337407514b9372562b86edfa91cdAndreas Huberstatus_t NuMediaExtractor::getSampleTime(int64_t *sampleTimeUs) {
4895778822d86b0337407514b9372562b86edfa91cdAndreas Huber    ssize_t minIndex = fetchTrackSamples();
4905778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4915778822d86b0337407514b9372562b86edfa91cdAndreas Huber    if (minIndex < 0) {
4925778822d86b0337407514b9372562b86edfa91cdAndreas Huber        return ERROR_END_OF_STREAM;
4935778822d86b0337407514b9372562b86edfa91cdAndreas Huber    }
4945778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4955778822d86b0337407514b9372562b86edfa91cdAndreas Huber    TrackInfo *info = &mSelectedTracks.editItemAt(minIndex);
4965778822d86b0337407514b9372562b86edfa91cdAndreas Huber    *sampleTimeUs = info->mSampleTimeUs;
4975778822d86b0337407514b9372562b86edfa91cdAndreas Huber
4985778822d86b0337407514b9372562b86edfa91cdAndreas Huber    return OK;
4995778822d86b0337407514b9372562b86edfa91cdAndreas Huber}
5005778822d86b0337407514b9372562b86edfa91cdAndreas Huber
501ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huberstatus_t NuMediaExtractor::getSampleFlags(uint32_t *sampleFlags) {
502ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    ssize_t minIndex = fetchTrackSamples();
503ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
504ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    if (minIndex < 0) {
505ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber        return ERROR_END_OF_STREAM;
506ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    }
507ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
508ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    TrackInfo *info = &mSelectedTracks.editItemAt(minIndex);
509ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    *sampleFlags = info->mSampleFlags;
510ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
511ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber    return OK;
512ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber}
513ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
5145778822d86b0337407514b9372562b86edfa91cdAndreas Huber}  // namespace android
515