libvlcjni.c 19.2 KB
Newer Older
Rafaël Carré's avatar
Rafaël Carré committed
1
2
3
/*****************************************************************************
 * libvlcjni.c
 *****************************************************************************
Edward Wang's avatar
Edward Wang committed
4
 * Copyright © 2010-2013 VLC authors and VideoLAN
Rafaël Carré's avatar
Rafaël Carré committed
5
 *
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
6
7
8
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation; either version 2.1 of the License, or
Rafaël Carré's avatar
Rafaël Carré committed
9
10
11
12
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
13
14
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Lesser General Public License for more details.
Rafaël Carré's avatar
Rafaël Carré committed
15
 *
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
16
17
18
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
Rafaël Carré's avatar
Rafaël Carré committed
19
20
 *****************************************************************************/

21
22
#include <dirent.h>
#include <errno.h>
23
#include <string.h>
24
#include <pthread.h>
25
#include <sys/stat.h>
26
#include <sys/types.h>
27
#include <unistd.h>
28

29
#include <vlc/vlc.h>
30
31
#include <vlc_common.h>
#include <vlc_url.h>
32

33
#include <jni.h>
34

35
#include <android/api-level.h>
36

37
#include "libvlcjni.h"
ivoire's avatar
ivoire committed
38
#include "aout.h"
39
#include "vout.h"
40
#include "utils.h"
41

42
43
44
#define VOUT_ANDROID_SURFACE 0
#define VOUT_OPENGLES2       1

45
#define LOG_TAG "VLC/JNI/main"
46
47
#include "log.h"

48
libvlc_media_t *new_media(jlong instance, JNIEnv *env, jobject thiz, jstring fileLocation, bool noOmx, bool noVideo)
49
{
Ludovic Fauvet's avatar
Ludovic Fauvet committed
50
    libvlc_instance_t *libvlc = (libvlc_instance_t*)(intptr_t)instance;
51
    jboolean isCopy;
52
53
54
    const char *psz_location = (*env)->GetStringUTFChars(env, fileLocation, &isCopy);
    libvlc_media_t *p_md = libvlc_media_new_location(libvlc, psz_location);
    (*env)->ReleaseStringUTFChars(env, fileLocation, psz_location);
55
56
57
    if (!p_md)
        return NULL;

58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
    if (!noOmx) {
        jclass cls = (*env)->GetObjectClass(env, thiz);
        jmethodID methodId = (*env)->GetMethodID(env, cls, "useIOMX", "()Z");
        if ((*env)->CallBooleanMethod(env, thiz, methodId)) {
            /*
             * Set higher caching values if using iomx decoding, since some omx
             * decoders have a very high latency, and if the preroll data isn't
             * enough to make the decoder output a frame, the playback timing gets
             * started too soon, and every decoded frame appears to be too late.
             * On Nexus One, the decoder latency seems to be 25 input packets
             * for 320x170 H.264, a few packets less on higher resolutions.
             * On Nexus S, the decoder latency seems to be about 7 packets.
             */
            libvlc_media_add_option(p_md, ":file-caching=1500");
            libvlc_media_add_option(p_md, ":network-caching=1500");
73
            libvlc_media_add_option(p_md, ":codec=mediacodec,iomx,all");
74
        }
75
76
        if (noVideo)
            libvlc_media_add_option(p_md, ":no-video");
77
    }
78
    return p_md;
79
80
}

81
libvlc_media_player_t *getMediaPlayer(JNIEnv *env, jobject thiz)
82
{
Ludovic Fauvet's avatar
Ludovic Fauvet committed
83
    return (libvlc_media_player_t*)(intptr_t)getLong(env, thiz, "mInternalMediaPlayerInstance");
84
85
}

Rafaël Carré's avatar
Rafaël Carré committed
86
87
static void releaseMediaPlayer(JNIEnv *env, jobject thiz)
{
88
89
    libvlc_media_player_t* p_mp = getMediaPlayer(env, thiz);
    if (p_mp)
90
    {
91
92
        libvlc_media_player_stop(p_mp);
        libvlc_media_player_release(p_mp);
93
        setLong(env, thiz, "mInternalMediaPlayerInstance", 0);
94
95
96
    }
}

97
98
99
100
101
/* Pointer to the Java virtual machine
 * Note: It's okay to use a static variable for the VM pointer since there
 * can only be one instance of this shared library in a single VM
 */
JavaVM *myVm;
102

103
static jobject eventHandlerInstance = NULL;
104
105
106
107
108

static void vlc_event_callback(const libvlc_event_t *ev, void *data)
{
    JNIEnv *env;

109
    bool isAttached = false;
110

111
    if (eventHandlerInstance == NULL)
Sébastien Toque's avatar
Sébastien Toque committed
112
        return;
113

Rafaël Carré's avatar
Rafaël Carré committed
114
115
    if ((*myVm)->GetEnv(myVm, (void**) &env, JNI_VERSION_1_2) < 0) {
        if ((*myVm)->AttachCurrentThread(myVm, &env, NULL) < 0)
116
            return;
117
        isAttached = true;
118
119
    }

120
121
122
123
124
125
126
127
128
129
    /* Creating the bundle in C allows us to subscribe to more events
     * and get better flexibility for each event. For example, we can
     * have totally different types of data for each event, instead of,
     * for example, only an integer and/or string.
     */
    jclass clsBundle = (*env)->FindClass(env, "android/os/Bundle");
    jmethodID clsCtor = (*env)->GetMethodID(env, clsBundle, "<init>", "()V" );
    jobject bundle = (*env)->NewObject(env, clsBundle, clsCtor);

    jmethodID putInt = (*env)->GetMethodID(env, clsBundle, "putInt", "(Ljava/lang/String;I)V" );
130
    jmethodID putFloat = (*env)->GetMethodID(env, clsBundle, "putFloat", "(Ljava/lang/String;F)V" );
131
132
    jmethodID putString = (*env)->GetMethodID(env, clsBundle, "putString", "(Ljava/lang/String;Ljava/lang/String;)V" );

133
134
135
136
137
    if (ev->type == libvlc_MediaPlayerPositionChanged) {
            jstring sData = (*env)->NewStringUTF(env, "data");
            (*env)->CallVoidMethod(env, bundle, putFloat, sData, ev->u.media_player_position_changed.new_position);
            (*env)->DeleteLocalRef(env, sData);
    } else if(ev->type == libvlc_MediaPlayerVout) {
138
139
140
141
        /* For determining the vout/ES track change */
        jstring sData = (*env)->NewStringUTF(env, "data");
        (*env)->CallVoidMethod(env, bundle, putInt, sData, ev->u.media_player_vout.new_count);
        (*env)->DeleteLocalRef(env, sData);
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
    } else if(ev->type == libvlc_MediaListItemAdded ||
              ev->type == libvlc_MediaListItemDeleted ) {
        jstring item_uri = (*env)->NewStringUTF(env, "item_uri");
        jstring item_index = (*env)->NewStringUTF(env, "item_index");
        char* mrl = libvlc_media_get_mrl(
            ev->type == libvlc_MediaListItemAdded ?
            ev->u.media_list_item_added.item :
            ev->u.media_list_item_deleted.item
            );
        jstring item_uri_value = (*env)->NewStringUTF(env, mrl);
        jint item_index_value;
        if(ev->type == libvlc_MediaListItemAdded)
            item_index_value = ev->u.media_list_item_added.index;
        else
            item_index_value = ev->u.media_list_item_deleted.index;

        (*env)->CallVoidMethod(env, bundle, putString, item_uri, item_uri_value);
        (*env)->CallVoidMethod(env, bundle, putInt, item_index, item_index_value);

        (*env)->DeleteLocalRef(env, item_uri);
        (*env)->DeleteLocalRef(env, item_uri_value);
        (*env)->DeleteLocalRef(env, item_index);
        free(mrl);
165
166
    }

167
    /* Get the object class */
168
    jclass cls = (*env)->GetObjectClass(env, eventHandlerInstance);
169
    if (!cls) {
170
        LOGE("EventHandler: failed to get class reference");
171
        goto end;
172
173
174
    }

    /* Find the callback ID */
175
    jmethodID methodID = (*env)->GetMethodID(env, cls, "callback", "(ILandroid/os/Bundle;)V");
176
    if (methodID) {
177
        (*env)->CallVoidMethod(env, eventHandlerInstance, methodID, ev->type, bundle);
178
    } else {
179
        LOGE("EventHandler: failed to get the callback method");
180
181
    }

182
end:
Edward Wang's avatar
Edward Wang committed
183
    (*env)->DeleteLocalRef(env, bundle);
184
185
    if (isAttached)
        (*myVm)->DetachCurrentThread(myVm);
186
187
}

188
jint JNI_OnLoad(JavaVM *vm, void *reserved)
189
{
190
191
192
    // Keep a reference on the Java VM.
    myVm = vm;

193
    pthread_mutex_init(&vout_android_lock, NULL);
194
    pthread_cond_init(&vout_android_surf_attached, NULL);
195

196
    LOGD("JNI interface loaded.");
197
    return JNI_VERSION_1_2;
198
199
}

200
201
void JNI_OnUnload(JavaVM* vm, void* reserved) {
    pthread_mutex_destroy(&vout_android_lock);
202
    pthread_cond_destroy(&vout_android_surf_attached);
203
204
}

Rafaël Carré's avatar
Rafaël Carré committed
205
206
// FIXME: use atomics
static bool verbosity;
207

208
void Java_org_videolan_libvlc_LibVLC_nativeInit(JNIEnv *env, jobject thiz)
209
{
210
211
212
213
214
    //only use OpenSLES if java side says we can
    jclass cls = (*env)->GetObjectClass(env, thiz);
    jmethodID methodId = (*env)->GetMethodID(env, cls, "getAout", "()I");
    bool use_opensles = (*env)->CallIntMethod(env, thiz, methodId) == AOUT_OPENSLES;

215
216
217
    methodId = (*env)->GetMethodID(env, cls, "getVout", "()I");
    bool use_opengles2 = (*env)->CallIntMethod(env, thiz, methodId) == VOUT_OPENGLES2;

218
219
220
    methodId = (*env)->GetMethodID(env, cls, "timeStretchingEnabled", "()Z");
    bool enable_time_stretch = (*env)->CallBooleanMethod(env, thiz, methodId);

Edward Wang's avatar
Edward Wang committed
221
222
223
    methodId = (*env)->GetMethodID(env, cls, "frameSkipEnabled", "()Z");
    bool enable_frame_skip = (*env)->CallBooleanMethod(env, thiz, methodId);

Edward Wang's avatar
Edward Wang committed
224
225
226
227
228
    methodId = (*env)->GetMethodID(env, cls, "getDeblocking", "()I");
    int deblocking = (*env)->CallIntMethod(env, thiz, methodId);
    char deblockstr[2] = "3";
    snprintf(deblockstr, 2, "%d", deblocking);
    LOGD("Using deblocking level %d", deblocking);
229

Edward Wang's avatar
Edward Wang committed
230
231
232
233
234
235
236
237
    methodId = (*env)->GetMethodID(env, cls, "getNetworkCaching", "()I");
    int networkCaching = (*env)->CallIntMethod(env, thiz, methodId);
    char networkCachingstr[25] = "0";
    if(networkCaching > 0) {
        snprintf(networkCachingstr, 25, "--network-caching=%d", networkCaching);
        LOGD("Using network caching of %d ms", networkCaching);
    }

238
239
240
241
    methodId = (*env)->GetMethodID(env, cls, "getChroma", "()Ljava/lang/String;");
    jstring chroma = (*env)->CallObjectMethod(env, thiz, methodId);
    const char *chromastr = (*env)->GetStringUTFChars(env, chroma, 0);
    LOGD("Chroma set to \"%s\"", chromastr);
Edward Wang's avatar
Edward Wang committed
242

243
244
245
    methodId = (*env)->GetMethodID(env, cls, "getSubtitlesEncoding", "()Ljava/lang/String;");
    jstring subsencoding = (*env)->CallObjectMethod(env, thiz, methodId);
    const char *subsencodingstr = (*env)->GetStringUTFChars(env, subsencoding, 0);
Edward Wang's avatar
Edward Wang committed
246
    LOGD("Subtitle encoding set to \"%s\"", subsencodingstr);
247

248
249
250
    methodId = (*env)->GetMethodID(env, cls, "isVerboseMode", "()Z");
    verbosity = (*env)->CallBooleanMethod(env, thiz, methodId);

251
    /* Don't add any invalid options, otherwise it causes LibVLC to crash */
252
    const char *argv[] = {
253
        "-I", "dummy",
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
254
255
256
        "--no-osd",
        "--no-video-title-show",
        "--no-stats",
257
258
        "--no-plugins-cache",
        "--no-drop-late-frames",
259
260
261
262
        /* The VLC default is to pick the highest resolution possible
         * (i.e. 1080p). For mobile, pick a more sane default for slow
         * mobile data networks and slower hardware. */
        "--preferred-resolution", "360",
Jean-Baptiste Kempf's avatar
Jean-Baptiste Kempf committed
263
        "--avcodec-fast",
264
        "--avcodec-threads=0",
265
        "--subsdec-encoding", subsencodingstr,
266
        enable_time_stretch ? "--audio-time-stretch" : "--no-audio-time-stretch",
Edward Wang's avatar
Edward Wang committed
267
        "--avcodec-skiploopfilter", deblockstr,
268
269
        "--avcodec-skip-frame", enable_frame_skip ? "2" : "0",
        "--avcodec-skip-idct", enable_frame_skip ? "2" : "0",
Edward Wang's avatar
Edward Wang committed
270
        (networkCaching > 0) ? networkCachingstr : "",
271
        use_opensles ? "--aout=opensles" : "--aout=android_audiotrack",
272
        use_opengles2 ? "--vout=gles2" : "--vout=androidsurface",
273
        "--androidsurface-chroma", chromastr != NULL && chromastr[0] != 0 ? chromastr : "RV32",
274
    };
275
    libvlc_instance_t *instance = libvlc_new(sizeof(argv) / sizeof(*argv), argv);
276

Ludovic Fauvet's avatar
Ludovic Fauvet committed
277
    setLong(env, thiz, "mLibVlcInstance", (jlong)(intptr_t) instance);
278

279
    (*env)->ReleaseStringUTFChars(env, chroma, chromastr);
280
281
    (*env)->ReleaseStringUTFChars(env, subsencoding, subsencodingstr);

282
283
    if (!instance)
    {
284
        jclass exc = (*env)->FindClass(env, "org/videolan/libvlc/LibVlcException");
285
286
287
288
        (*env)->ThrowNew(env, exc, "Unable to instantiate LibVLC");
    }

    LOGI("LibVLC initialized: %p", instance);
289

Rafaël Carré's avatar
Rafaël Carré committed
290
    libvlc_log_set(instance, debug_log, &verbosity);
291
292
}

293
void Java_org_videolan_libvlc_LibVLC_nativeDestroy(JNIEnv *env, jobject thiz)
294
{
295
    releaseMediaPlayer(env, thiz);
Edward Wang's avatar
Edward Wang committed
296
    jlong libVlcInstance = getLong(env, thiz, "mLibVlcInstance");
297
298
299
    if (!libVlcInstance)
        return; // Already destroyed

Ludovic Fauvet's avatar
Ludovic Fauvet committed
300
    libvlc_instance_t *instance = (libvlc_instance_t*)(intptr_t) libVlcInstance;
Rafaël Carré's avatar
Rafaël Carré committed
301
    libvlc_log_unset(instance);
302
303
    libvlc_release(instance);

Edward Wang's avatar
Edward Wang committed
304
    setLong(env, thiz, "mLibVlcInstance", 0);
305
306
}

307
void Java_org_videolan_libvlc_LibVLC_detachEventHandler(JNIEnv *env, jobject thiz)
308
{
309
310
311
    if (eventHandlerInstance != NULL) {
        (*env)->DeleteGlobalRef(env, eventHandlerInstance);
        eventHandlerInstance = NULL;
312
313
314
    }
}

315
void Java_org_videolan_libvlc_LibVLC_setEventHandler(JNIEnv *env, jobject thiz, jobject eventHandler)
316
{
317
318
319
    if (eventHandlerInstance != NULL) {
        (*env)->DeleteGlobalRef(env, eventHandlerInstance);
        eventHandlerInstance = NULL;
320
321
    }

322
    eventHandlerInstance = getEventHandlerReference(env, thiz, eventHandler);
323
324
}

325
326
static void create_player_and_play(JNIEnv* env, jobject thiz,
                                   jlong instance, int position) {
327
328
329
    /* Release previous media player, if any */
    releaseMediaPlayer(env, thiz);

330
    /* Create a media player playing environment */
Ludovic Fauvet's avatar
Ludovic Fauvet committed
331
    libvlc_media_player_t *mp = libvlc_media_player_new((libvlc_instance_t*)(intptr_t)instance);
332
333
    jobject myJavaLibVLC = (*env)->NewGlobalRef(env, thiz);

334
335
336
337
    //if AOUT_AUDIOTRACK_JAVA, we use amem
    jclass cls = (*env)->GetObjectClass(env, thiz);
    jmethodID methodId = (*env)->GetMethodID(env, cls, "getAout", "()I");
    if ( (*env)->CallIntMethod(env, thiz, methodId) == AOUT_AUDIOTRACK_JAVA )
338
    {
339
        libvlc_audio_set_callbacks(mp, aout_play, aout_pause, NULL, NULL, NULL,
Sébastien Toque's avatar
Sébastien Toque committed
340
341
                                   (void*) myJavaLibVLC);
        libvlc_audio_set_format_callbacks(mp, aout_open, aout_close);
342
343
    }

344
345
    /* Connect the event manager */
    libvlc_event_manager_t *ev = libvlc_media_player_event_manager(mp);
346
347
348
349
350
    static const libvlc_event_type_t mp_events[] = {
        libvlc_MediaPlayerPlaying,
        libvlc_MediaPlayerPaused,
        libvlc_MediaPlayerEndReached,
        libvlc_MediaPlayerStopped,
351
        libvlc_MediaPlayerVout,
352
353
        libvlc_MediaPlayerPositionChanged,
        libvlc_MediaPlayerEncounteredError
354
    };
355
    for(int i = 0; i < (sizeof(mp_events) / sizeof(*mp_events)); i++)
356
357
        libvlc_event_attach(ev, mp_events[i], vlc_event_callback, myVm);

358

359
    /* Keep a pointer to this media player */
Ludovic Fauvet's avatar
Ludovic Fauvet committed
360
    setLong(env, thiz, "mInternalMediaPlayerInstance", (jlong)(intptr_t)mp);
361

362
363
364
365
    cls = (*env)->GetObjectClass(env, thiz);
    jmethodID methodID = (*env)->GetMethodID(env, cls, "applyEqualizer", "()V");
    (*env)->CallVoidMethod(env, thiz, methodID);

366
    setInt(env, thiz, "mInternalMediaPlayerIndex", (jint)position);
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393

    jfieldID fid = (*env)->GetFieldID(env, cls, "mMediaList", "Lorg/videolan/libvlc/MediaList;");
    jobject list = (*env)->GetObjectField(env, thiz, fid);
    jclass clsList = (*env)->FindClass(env, "org/videolan/libvlc/MediaList");

    methodID = (*env)->GetMethodID(env, clsList, "getMRL", "(I)Ljava/lang/String;");
    jstring mrl = (jstring)(*env)->CallObjectMethod(env, list, methodID, (jint)position);
    const char* p_mrl = (*env)->GetStringUTFChars(env, mrl, 0);

    libvlc_media_t* p_md = libvlc_media_new_location((libvlc_instance_t*)(intptr_t)instance, p_mrl);
    /* media options */
    methodID = (*env)->GetMethodID(env, clsList, "getMediaOptions", "(I)[Ljava/lang/String;");
    jobjectArray mediaOptions = (jobject)(*env)->CallObjectMethod(env, list, methodID, (jint)position);
    int stringCount = (*env)->GetArrayLength(env, mediaOptions);
    for(int i = 0; i < stringCount; i++) {
        jstring option = (jstring)(*env)->GetObjectArrayElement(env, mediaOptions, i);
        const char* p_st = (*env)->GetStringUTFChars(env, option, 0);
        libvlc_media_add_option(p_md, p_st); // option
        (*env)->ReleaseStringUTFChars(env, option, p_st);
    }

    (*env)->ReleaseStringUTFChars(env, mrl, p_mrl);
    (*env)->DeleteLocalRef(env, mrl);
    (*env)->DeleteLocalRef(env, mediaOptions);
    (*env)->DeleteLocalRef(env, list);
    list = NULL;

394
395
    libvlc_media_player_set_media(mp, p_md);
    libvlc_media_player_play(mp);
396
397
}

398
void Java_org_videolan_libvlc_LibVLC_playIndex(JNIEnv *env, jobject thiz,
399
400
401
402
                                            jlong instance, int position) {
    create_player_and_play(env, thiz, instance, position);
}

403
jfloat Java_org_videolan_libvlc_LibVLC_getRate(JNIEnv *env, jobject thiz) {
Edward Wang's avatar
Edward Wang committed
404
405
406
407
408
409
410
    libvlc_media_player_t* mp = getMediaPlayer(env, thiz);
    if(mp)
        return libvlc_media_player_get_rate(mp);
    else
        return 1.00;
}

411
void Java_org_videolan_libvlc_LibVLC_setRate(JNIEnv *env, jobject thiz, jfloat rate) {
Edward Wang's avatar
Edward Wang committed
412
413
414
415
416
    libvlc_media_player_t* mp = getMediaPlayer(env, thiz);
    if(mp)
        libvlc_media_player_set_rate(mp, rate);
}

417
jboolean Java_org_videolan_libvlc_LibVLC_isPlaying(JNIEnv *env, jobject thiz)
418
{
419
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
420
    if (mp)
421
        return !!libvlc_media_player_is_playing(mp);
422
423
    else
        return 0;
424
425
}

426
jboolean Java_org_videolan_libvlc_LibVLC_isSeekable(JNIEnv *env, jobject thiz)
427
{
428
429
430
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return !!libvlc_media_player_is_seekable(mp);
431
    return 0;
432
433
}

434
void Java_org_videolan_libvlc_LibVLC_play(JNIEnv *env, jobject thiz)
435
{
436
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
437
    if (mp)
438
        libvlc_media_player_play(mp);
439
440
}

441
void Java_org_videolan_libvlc_LibVLC_pause(JNIEnv *env, jobject thiz)
442
{
443
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
444
    if (mp)
445
        libvlc_media_player_pause(mp);
446
447
}

448
void Java_org_videolan_libvlc_LibVLC_stop(JNIEnv *env, jobject thiz)
449
{
450
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
451
    if (mp)
452
        libvlc_media_player_stop(mp);
453
454
}

455
jint Java_org_videolan_libvlc_LibVLC_getVolume(JNIEnv *env, jobject thiz)
456
{
457
458
459
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jint) libvlc_audio_get_volume(mp);
460
461
462
    return -1;
}

463
jint Java_org_videolan_libvlc_LibVLC_setVolume(JNIEnv *env, jobject thiz, jint volume)
464
{
465
466
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
467
        //Returns 0 if the volume was set, -1 if it was out of range or error
468
        return (jint) libvlc_audio_set_volume(mp, (int) volume);
469
470
471
    return -1;
}

472
jlong Java_org_videolan_libvlc_LibVLC_getTime(JNIEnv *env, jobject thiz)
473
{
474
475
476
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return libvlc_media_player_get_time(mp);
477
478
479
    return -1;
}

480
void Java_org_videolan_libvlc_LibVLC_setTime(JNIEnv *env, jobject thiz, jlong time)
481
{
482
483
484
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        libvlc_media_player_set_time(mp, time);
485
486
}

487
jfloat Java_org_videolan_libvlc_LibVLC_getPosition(JNIEnv *env, jobject thiz)
488
{
489
490
491
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jfloat) libvlc_media_player_get_position(mp);
492
493
494
    return -1;
}

495
void Java_org_videolan_libvlc_LibVLC_setPosition(JNIEnv *env, jobject thiz, jfloat pos)
496
{
497
498
499
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        libvlc_media_player_set_position(mp, pos);
500
501
}

502
jlong Java_org_videolan_libvlc_LibVLC_getLength(JNIEnv *env, jobject thiz)
503
{
504
505
506
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jlong) libvlc_media_player_get_length(mp);
507
508
509
    return -1;
}

510
jstring Java_org_videolan_libvlc_LibVLC_version(JNIEnv* env, jobject thiz)
511
{
512
513
    return (*env)->NewStringUTF(env, libvlc_get_version());
}
514

515
jstring Java_org_videolan_libvlc_LibVLC_compiler(JNIEnv* env, jobject thiz)
516
517
518
519
{
    return (*env)->NewStringUTF(env, libvlc_get_compiler());
}

520
jstring Java_org_videolan_libvlc_LibVLC_changeset(JNIEnv* env, jobject thiz)
521
522
523
{
    return (*env)->NewStringUTF(env, libvlc_get_changeset());
}