android_media_MediaExtractor.cpp revision 91befdc0c4710234840cdfd853e7d30e8f9de62c
188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber/*
288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * Copyright 2012, The Android Open Source Project
388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber *
488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * Licensed under the Apache License, Version 2.0 (the "License");
588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * you may not use this file except in compliance with the License.
688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * You may obtain a copy of the License at
788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber *
888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber *     http://www.apache.org/licenses/LICENSE-2.0
988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber *
1088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * Unless required by applicable law or agreed to in writing, software
1188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * distributed under the License is distributed on an "AS IS" BASIS,
1288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * See the License for the specific language governing permissions and
1488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber * limitations under the License.
1588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber */
1688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
1788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber//#define LOG_NDEBUG 0
1888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#define LOG_TAG "MediaExtractor-JNI"
1988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include <utils/Log.h>
2088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
2188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include "android_media_MediaExtractor.h"
2288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
2388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include "android_media_Utils.h"
2488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include "android_runtime/AndroidRuntime.h"
2588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include "jni.h"
2688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include "JNIHelp.h"
2788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
2888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include <media/stagefright/foundation/ABuffer.h>
2988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include <media/stagefright/foundation/ADebug.h>
3088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include <media/stagefright/foundation/AMessage.h>
3188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include <media/stagefright/DataSource.h>
3288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include <media/stagefright/MediaErrors.h>
3391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber#include <media/stagefright/MetaData.h>
3488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber#include <media/stagefright/NuMediaExtractor.h>
3588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
3688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Hubernamespace android {
3788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
3888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstruct fields_t {
3988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    jfieldID context;
4091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
4191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    jmethodID cryptoInfoSetID;
4288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber};
4388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
4488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic fields_t gFields;
4588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
4688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber////////////////////////////////////////////////////////////////////////////////
4788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
4888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas HuberJMediaExtractor::JMediaExtractor(JNIEnv *env, jobject thiz)
4988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    : mClass(NULL),
5088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber      mObject(NULL) {
5188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    jclass clazz = env->GetObjectClass(thiz);
5288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    CHECK(clazz != NULL);
5388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
5488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    mClass = (jclass)env->NewGlobalRef(clazz);
5588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    mObject = env->NewWeakGlobalRef(thiz);
5688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
5788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    mImpl = new NuMediaExtractor;
5888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
5988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
6088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas HuberJMediaExtractor::~JMediaExtractor() {
6188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    JNIEnv *env = AndroidRuntime::getJNIEnv();
6288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
6388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    env->DeleteWeakGlobalRef(mObject);
6488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    mObject = NULL;
6588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    env->DeleteGlobalRef(mClass);
6688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    mClass = NULL;
6788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
6888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
6907ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huberstatus_t JMediaExtractor::setDataSource(
7007ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        const char *path, const KeyedVector<String8, String8> *headers) {
7107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    return mImpl->setDataSource(path, headers);
7207ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber}
7307ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
7407ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huberstatus_t JMediaExtractor::setDataSource(int fd, off64_t offset, off64_t size) {
7507ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    return mImpl->setDataSource(fd, offset, size);
7688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
7788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
7888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Hubersize_t JMediaExtractor::countTracks() const {
7988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return mImpl->countTracks();
8088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
8188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
8288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatus_t JMediaExtractor::getTrackFormat(size_t index, jobject *format) const {
8388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<AMessage> msg;
8488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err;
8588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if ((err = mImpl->getTrackFormat(index, &msg)) != OK) {
8688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return err;
8788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
8888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
8988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    JNIEnv *env = AndroidRuntime::getJNIEnv();
9088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
9188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return ConvertMessageToMap(env, msg, format);
9288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
9388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
9488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatus_t JMediaExtractor::selectTrack(size_t index) {
9588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return mImpl->selectTrack(index);
9688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
9788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
9888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatus_t JMediaExtractor::seekTo(int64_t timeUs) {
9988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return mImpl->seekTo(timeUs);
10088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
10188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
10288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatus_t JMediaExtractor::advance() {
10388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return mImpl->advance();
10488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
10588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
10688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatus_t JMediaExtractor::readSampleData(
10788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jobject byteBuf, size_t offset, size_t *sampleSize) {
10888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    JNIEnv *env = AndroidRuntime::getJNIEnv();
10988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
11088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    void *dst = env->GetDirectBufferAddress(byteBuf);
11188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
112c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber    jlong dstSize;
113c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber    jbyteArray byteArray = NULL;
114c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber
11588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (dst == NULL) {
116c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        jclass byteBufClass = env->FindClass("java/nio/ByteBuffer");
117c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        CHECK(byteBufClass != NULL);
118c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber
119c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        jmethodID arrayID =
120c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber            env->GetMethodID(byteBufClass, "array", "()[B");
121c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        CHECK(arrayID != NULL);
122c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber
123c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        byteArray =
124c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber            (jbyteArray)env->CallObjectMethod(byteBuf, arrayID);
125c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber
126c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        if (byteArray == NULL) {
127c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber            return INVALID_OPERATION;
128c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        }
12988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
130c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        jboolean isCopy;
131c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        dst = env->GetByteArrayElements(byteArray, &isCopy);
132c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber
133c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        dstSize = env->GetArrayLength(byteArray);
134c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber    } else {
135c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        dstSize = env->GetDirectBufferCapacity(byteBuf);
136c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber    }
13788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
13888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (dstSize < offset) {
139c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        if (byteArray != NULL) {
140c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber            env->ReleaseByteArrayElements(byteArray, (jbyte *)dst, 0);
141c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        }
142c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber
14388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return -ERANGE;
14488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
14588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
14688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<ABuffer> buffer = new ABuffer((char *)dst + offset, dstSize - offset);
14788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
14888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err = mImpl->readSampleData(buffer);
14988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
150c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber    if (byteArray != NULL) {
151c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber        env->ReleaseByteArrayElements(byteArray, (jbyte *)dst, 0);
152c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber    }
153c52b980277f08aee7981b1fdbca7a89464cf66d9Andreas Huber
15488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err != OK) {
15588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return err;
15688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
15788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
15888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    *sampleSize = buffer->size();
15988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
16088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return OK;
16188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
16288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
16388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatus_t JMediaExtractor::getSampleTrackIndex(size_t *trackIndex) {
16488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return mImpl->getSampleTrackIndex(trackIndex);
16588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
16688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
16788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatus_t JMediaExtractor::getSampleTime(int64_t *sampleTimeUs) {
16888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return mImpl->getSampleTime(sampleTimeUs);
16988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
17088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
1719b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huberstatus_t JMediaExtractor::getSampleFlags(uint32_t *sampleFlags) {
17291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    *sampleFlags = 0;
17391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
17491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    sp<MetaData> meta;
17591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    status_t err = mImpl->getSampleMeta(&meta);
17691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
17791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (err != OK) {
17891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        return err;
17991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
18091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
18191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    int32_t val;
18291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (meta->findInt32(kKeyIsSyncFrame, &val) && val != 0) {
18391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        (*sampleFlags) |= NuMediaExtractor::SAMPLE_FLAG_SYNC;
18491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
18591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
18691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    uint32_t type;
18791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    const void *data;
18891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    size_t size;
18991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (meta->findData(kKeyEncryptedSizes, &type, &data, &size)) {
19091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        (*sampleFlags) |= NuMediaExtractor::SAMPLE_FLAG_ENCRYPTED;
19191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
19291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
19391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    return OK;
19491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber}
19591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
19691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huberstatus_t JMediaExtractor::getSampleMeta(sp<MetaData> *sampleMeta) {
19791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    return mImpl->getSampleMeta(sampleMeta);
1989b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber}
1999b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber
20088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}  // namespace android
20188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
20288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber////////////////////////////////////////////////////////////////////////////////
20388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
20488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberusing namespace android;
20588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
20688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic sp<JMediaExtractor> setMediaExtractor(
20788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz, const sp<JMediaExtractor> &extractor) {
20888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> old =
20988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        (JMediaExtractor *)env->GetIntField(thiz, gFields.context);
21088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
21188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor != NULL) {
21288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        extractor->incStrong(thiz);
21388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
21488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (old != NULL) {
21588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        old->decStrong(thiz);
21688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
21788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    env->SetIntField(thiz, gFields.context, (int)extractor.get());
21888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
21988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return old;
22088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
22188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
22288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic sp<JMediaExtractor> getMediaExtractor(JNIEnv *env, jobject thiz) {
22388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return (JMediaExtractor *)env->GetIntField(thiz, gFields.context);
22488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
22588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
22688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic void android_media_MediaExtractor_release(JNIEnv *env, jobject thiz) {
22788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    setMediaExtractor(env, thiz, NULL);
22888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
22988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
23088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic jint android_media_MediaExtractor_countTracks(
23188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz) {
23288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
23388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
23488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
23588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
23607ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        return -1;
23788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
23888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
23988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return extractor->countTracks();
24088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
24188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
24288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic jobject android_media_MediaExtractor_getTrackFormat(
24388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz, jint index) {
24488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
24588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
24688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
24788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
24888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return NULL;
24988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
25088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
25188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    jobject format;
25288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err = extractor->getTrackFormat(index, &format);
25388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
25488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err != OK) {
25588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
25688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return NULL;
25788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
25888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
25988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return format;
26088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
26188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
26288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic void android_media_MediaExtractor_selectTrack(
26388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz, jint index) {
26488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
26588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
26688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
26788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
26888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return;
26988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
27088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
27188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err = extractor->selectTrack(index);
27288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
27388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err != OK) {
27488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
27588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return;
27688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
27788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
27888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
27988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic void android_media_MediaExtractor_seekTo(
28088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz, jlong timeUs) {
28188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
28288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
28388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
28488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
28588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return;
28688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
28788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
2882b9d6bd092ef1d0cc142bf16f671648d1c84c307Andreas Huber    extractor->seekTo(timeUs);
28988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
29088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
29188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic jboolean android_media_MediaExtractor_advance(
29288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz) {
29388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
29488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
29588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
29688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
29788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return false;
29888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
29988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
30088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err = extractor->advance();
30188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
30288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err == ERROR_END_OF_STREAM) {
30388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return false;
30488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    } else if (err != OK) {
30588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
30688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return false;
30788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
30888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
30988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return true;
31088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
31188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
31288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic jint android_media_MediaExtractor_readSampleData(
31388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz, jobject byteBuf, jint offset) {
31488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
31588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
31688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
31788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
31888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return -1;
31988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
32088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
32188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    size_t sampleSize;
32288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err = extractor->readSampleData(byteBuf, offset, &sampleSize);
32388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
32488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err == ERROR_END_OF_STREAM) {
32588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return -1;
32688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    } else if (err != OK) {
32788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
32888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return false;
32988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
33088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
33188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return sampleSize;
33288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
33388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
33488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic jint android_media_MediaExtractor_getSampleTrackIndex(
33588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz) {
33688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
33788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
33888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
33988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
34088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return -1;
34188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
34288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
34388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    size_t trackIndex;
34488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err = extractor->getSampleTrackIndex(&trackIndex);
34588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
34688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err == ERROR_END_OF_STREAM) {
34788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return -1;
34888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    } else if (err != OK) {
34988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
35088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return false;
35188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
35288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
35388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return trackIndex;
35488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
35588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
35688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic jlong android_media_MediaExtractor_getSampleTime(
35788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz) {
35888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
35988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
36088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (extractor == NULL) {
36188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
36288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return -1ll;
36388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
36488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
36588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    int64_t sampleTimeUs;
36688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    status_t err = extractor->getSampleTime(&sampleTimeUs);
36788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
36888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err == ERROR_END_OF_STREAM) {
36988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return -1ll;
37088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    } else if (err != OK) {
37188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
37288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return false;
37388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
37488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
37588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return sampleTimeUs;
37688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
37788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
3789b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huberstatic jint android_media_MediaExtractor_getSampleFlags(
3799b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber        JNIEnv *env, jobject thiz) {
3809b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
3819b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber
3829b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    if (extractor == NULL) {
3839b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
3849b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber        return -1ll;
3859b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    }
3869b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber
3879b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    uint32_t sampleFlags;
3889b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    status_t err = extractor->getSampleFlags(&sampleFlags);
3899b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber
3909b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    if (err == ERROR_END_OF_STREAM) {
3919b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber        return -1ll;
3929b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    } else if (err != OK) {
3939b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
3949b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber        return false;
3959b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    }
3969b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber
3979b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    return sampleFlags;
3989b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber}
3999b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber
40091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huberstatic jboolean android_media_MediaExtractor_getSampleCryptoInfo(
40191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        JNIEnv *env, jobject thiz, jobject cryptoInfoObj) {
40291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
40391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
40491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (extractor == NULL) {
40591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
40691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        return -1ll;
40791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
40891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
40991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    sp<MetaData> meta;
41091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    status_t err = extractor->getSampleMeta(&meta);
41191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
41291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (err != OK) {
41391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        return false;
41491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
41591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
41691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    uint32_t type;
41791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    const void *data;
41891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    size_t size;
41991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (!meta->findData(kKeyEncryptedSizes, &type, &data, &size)) {
42091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        return false;
42191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
42291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
42391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    size_t numSubSamples = size / sizeof(size_t);
42491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
42591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (numSubSamples == 0) {
42691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        return false;
42791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
42891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
42991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    jintArray numBytesOfEncryptedDataObj = env->NewIntArray(numSubSamples);
43091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    jboolean isCopy;
43191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    jint *dst = env->GetIntArrayElements(numBytesOfEncryptedDataObj, &isCopy);
43291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    for (size_t i = 0; i < numSubSamples; ++i) {
43391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        dst[i] = ((const size_t *)data)[i];
43491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
43591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    env->ReleaseIntArrayElements(numBytesOfEncryptedDataObj, dst, 0);
43691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    dst = NULL;
43791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
43891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    size_t encSize = size;
43991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    jintArray numBytesOfPlainDataObj = NULL;
44091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (meta->findData(kKeyEncryptedSizes, &type, &data, &size)) {
44191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        if (size != encSize) {
44291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            // The two must be of the same length.
44391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            return false;
44491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        }
44591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
44691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        numBytesOfPlainDataObj = env->NewIntArray(numSubSamples);
44791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        jboolean isCopy;
44891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        jint *dst = env->GetIntArrayElements(numBytesOfPlainDataObj, &isCopy);
44991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        for (size_t i = 0; i < numSubSamples; ++i) {
45091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            dst[i] = ((const size_t *)data)[i];
45191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        }
45291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        env->ReleaseIntArrayElements(numBytesOfPlainDataObj, dst, 0);
45391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        dst = NULL;
45491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
45591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
45691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    jbyteArray keyObj = NULL;
45791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (meta->findData(kKeyCryptoKey, &type, &data, &size)) {
45891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        if (size != 16) {
45991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            // Keys must be 16 bytes in length.
46091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            return false;
46191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        }
46291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
46391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        keyObj = env->NewByteArray(size);
46491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        jboolean isCopy;
46591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        jbyte *dst = env->GetByteArrayElements(keyObj, &isCopy);
46691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        memcpy(dst, data, size);
46791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        env->ReleaseByteArrayElements(keyObj, dst, 0);
46891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        dst = NULL;
46991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
47091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
47191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    jbyteArray ivObj = NULL;
47291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (meta->findData(kKeyCryptoIV, &type, &data, &size)) {
47391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        if (size != 16) {
47491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            // IVs must be 16 bytes in length.
47591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            return false;
47691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        }
47791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
47891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        ivObj = env->NewByteArray(size);
47991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        jboolean isCopy;
48091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        jbyte *dst = env->GetByteArrayElements(ivObj, &isCopy);
48191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        memcpy(dst, data, size);
48291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        env->ReleaseByteArrayElements(ivObj, dst, 0);
48391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        dst = NULL;
48491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
48591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
48691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    int32_t mode;
48791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    if (!meta->findInt32(kKeyCryptoMode, &mode)) {
48891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        mode = 0;
48991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    }
49091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
49191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    env->CallVoidMethod(
49291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            cryptoInfoObj,
49391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            gFields.cryptoInfoSetID,
49491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            numSubSamples,
49591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            numBytesOfPlainDataObj,
49691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            numBytesOfEncryptedDataObj,
49791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            keyObj,
49891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            ivObj,
49991befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber            mode);
50091befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
50191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    return true;
50291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber}
50391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
50488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic void android_media_MediaExtractor_native_init(JNIEnv *env) {
50588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    jclass clazz = env->FindClass("android/media/MediaExtractor");
50688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    CHECK(clazz != NULL);
50788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
50888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    gFields.context = env->GetFieldID(clazz, "mNativeContext", "I");
50988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    CHECK(gFields.context != NULL);
51088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
51191befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    clazz = env->FindClass("android/media/MediaCodec$CryptoInfo");
51291befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    CHECK(clazz != NULL);
51391befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
51491befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    gFields.cryptoInfoSetID =
51591befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        env->GetMethodID(clazz, "set", "(I[I[I[B[BI)V");
51691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
51788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    DataSource::RegisterDefaultSniffers();
51888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
51988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
52088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic void android_media_MediaExtractor_native_setup(
52107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        JNIEnv *env, jobject thiz) {
52288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    sp<JMediaExtractor> extractor = new JMediaExtractor(env, thiz);
52307ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    setMediaExtractor(env,thiz, extractor);
52407ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber}
52588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
52607ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huberstatic void android_media_MediaExtractor_setDataSource(
52707ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        JNIEnv *env, jobject thiz,
52807ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        jstring pathObj, jobjectArray keysArray, jobjectArray valuesArray) {
52907ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
53007ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
53107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    if (extractor == NULL) {
53207ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
53307ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        return;
53407ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    }
53507ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
53607ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    if (pathObj == NULL) {
53788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
53888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return;
53988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
54088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
54107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    KeyedVector<String8, String8> headers;
54207ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    if (!ConvertKeyValueArraysToKeyedVector(
54307ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber                env, keysArray, valuesArray, &headers)) {
54407ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        return;
54507ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    }
54688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
54707ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    const char *path = env->GetStringUTFChars(pathObj, NULL);
54807ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
54907ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    if (path == NULL) {
55088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return;
55188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
55288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
55307ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    status_t err = extractor->setDataSource(path, &headers);
55488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
55507ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    env->ReleaseStringUTFChars(pathObj, path);
55607ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    path = NULL;
55788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
55888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    if (err != OK) {
55988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        jniThrowException(
56088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber                env,
56188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber                "java/io/IOException",
56288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber                "Failed to instantiate extractor.");
56388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        return;
56488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    }
56507ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber}
56688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
56707ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huberstatic void android_media_MediaExtractor_setDataSourceFd(
56807ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        JNIEnv *env, jobject thiz,
56907ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        jobject fileDescObj, jlong offset, jlong length) {
57007ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
57107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
57207ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    if (extractor == NULL) {
57307ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        jniThrowException(env, "java/lang/IllegalStateException", NULL);
57407ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        return;
57507ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    }
57607ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
57707ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    if (fileDescObj == NULL) {
57807ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
57907ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        return;
58007ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    }
58107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
58207ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    int fd = jniGetFDFromFileDescriptor(env, fileDescObj);
58307ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
58407ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    status_t err = extractor->setDataSource(fd, offset, length);
58507ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
58607ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    if (err != OK) {
58707ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        jniThrowException(
58807ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber                env,
58907ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber                "java/io/IOException",
59007ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber                "Failed to instantiate extractor.");
59107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber        return;
59207ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    }
59388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
59488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
59588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic void android_media_MediaExtractor_native_finalize(
59688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        JNIEnv *env, jobject thiz) {
59788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    android_media_MediaExtractor_release(env, thiz);
59888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
59988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
60088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberstatic JNINativeMethod gMethods[] = {
60188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "release", "()V", (void *)android_media_MediaExtractor_release },
60288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
60388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "countTracks", "()I", (void *)android_media_MediaExtractor_countTracks },
60488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
60588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "getTrackFormat", "(I)Ljava/util/Map;",
60688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        (void *)android_media_MediaExtractor_getTrackFormat },
60788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
60888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "selectTrack", "(I)V", (void *)android_media_MediaExtractor_selectTrack },
60988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
61088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "seekTo", "(J)V", (void *)android_media_MediaExtractor_seekTo },
61188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
61288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "advance", "()Z", (void *)android_media_MediaExtractor_advance },
61388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
61488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "readSampleData", "(Ljava/nio/ByteBuffer;I)I",
61588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        (void *)android_media_MediaExtractor_readSampleData },
61688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
61788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "getSampleTrackIndex", "()I",
61888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        (void *)android_media_MediaExtractor_getSampleTrackIndex },
61988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
62088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "getSampleTime", "()J",
62188572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber        (void *)android_media_MediaExtractor_getSampleTime },
62288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
6239b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber    { "getSampleFlags", "()I",
6249b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber        (void *)android_media_MediaExtractor_getSampleFlags },
6259b8e496f4d143280deff137c5f30ca8907bc28dbAndreas Huber
62691befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber    { "getSampleCryptoInfo", "(Landroid/media/MediaCodec$CryptoInfo;)Z",
62791befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber        (void *)android_media_MediaExtractor_getSampleCryptoInfo },
62891befdc0c4710234840cdfd853e7d30e8f9de62cAndreas Huber
62988572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "native_init", "()V", (void *)android_media_MediaExtractor_native_init },
63088572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
63107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    { "native_setup", "()V",
63288572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber      (void *)android_media_MediaExtractor_native_setup },
63388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
63488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    { "native_finalize", "()V",
63588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber      (void *)android_media_MediaExtractor_native_finalize },
63607ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
63707ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    { "setDataSource", "(Ljava/lang/String;[Ljava/lang/String;"
63807ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber                       "[Ljava/lang/String;)V",
63907ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber      (void *)android_media_MediaExtractor_setDataSource },
64007ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber
64107ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber    { "setDataSource", "(Ljava/io/FileDescriptor;JJ)V",
64207ea426e3ae8915ca6bf67135f523f42cd920af0Andreas Huber      (void *)android_media_MediaExtractor_setDataSourceFd },
64388572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber};
64488572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber
64588572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huberint register_android_media_MediaExtractor(JNIEnv *env) {
64688572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber    return AndroidRuntime::registerNativeMethods(env,
64788572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber                "android/media/MediaExtractor", gMethods, NELEM(gMethods));
64888572f7a3e9d7ef85c26865a0150f3c2041561c2Andreas Huber}
649