libvlcjni.c 19.4 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
46
47
48
#define HW_ACCELERATION_DISABLED 0
#define HW_ACCELERATION_DECODING 1
#define HW_ACCELERATION_FULL     2

49
#define LOG_TAG "VLC/JNI/main"
50
51
#include "log.h"

52
libvlc_media_t *new_media(jlong instance, JNIEnv *env, jobject thiz, jstring fileLocation, bool noOmx, bool noVideo)
53
{
Ludovic Fauvet's avatar
Ludovic Fauvet committed
54
    libvlc_instance_t *libvlc = (libvlc_instance_t*)(intptr_t)instance;
55
    jboolean isCopy;
56
57
58
    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);
59
60
61
    if (!p_md)
        return NULL;

62
63
    if (!noOmx) {
        jclass cls = (*env)->GetObjectClass(env, thiz);
64
65
66
        jmethodID methodId = (*env)->GetMethodID(env, cls, "getHardwareAcceleration", "()I");
        int hardwareAcceleration = (*env)->CallIntMethod(env, thiz, methodId);
        if (hardwareAcceleration == HW_ACCELERATION_DECODING || hardwareAcceleration == HW_ACCELERATION_FULL) {
67
68
69
70
71
72
73
74
75
76
77
            /*
             * 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");
78
            libvlc_media_add_option(p_md, ":codec=mediacodec,iomx,all");
79
        }
80
81
        if (noVideo)
            libvlc_media_add_option(p_md, ":no-video");
82
    }
83
    return p_md;
84
85
}

86
libvlc_media_player_t *getMediaPlayer(JNIEnv *env, jobject thiz)
87
{
Ludovic Fauvet's avatar
Ludovic Fauvet committed
88
    return (libvlc_media_player_t*)(intptr_t)getLong(env, thiz, "mInternalMediaPlayerInstance");
89
90
}

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

102
103
104
105
106
/* 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;
107

108
static jobject eventHandlerInstance = NULL;
109
110
111
112
113

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

114
    bool isAttached = false;
115

116
    if (eventHandlerInstance == NULL)
Sébastien Toque's avatar
Sébastien Toque committed
117
        return;
118

Rafaël Carré's avatar
Rafaël Carré committed
119
120
    if ((*myVm)->GetEnv(myVm, (void**) &env, JNI_VERSION_1_2) < 0) {
        if ((*myVm)->AttachCurrentThread(myVm, &env, NULL) < 0)
121
            return;
122
        isAttached = true;
123
124
    }

125
126
127
128
129
130
131
132
133
134
    /* 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" );
135
    jmethodID putFloat = (*env)->GetMethodID(env, clsBundle, "putFloat", "(Ljava/lang/String;F)V" );
136
137
    jmethodID putString = (*env)->GetMethodID(env, clsBundle, "putString", "(Ljava/lang/String;Ljava/lang/String;)V" );

138
139
140
141
142
    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) {
143
144
145
146
        /* 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);
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
    } 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);
170
171
    }

172
    /* Get the object class */
173
    jclass cls = (*env)->GetObjectClass(env, eventHandlerInstance);
174
    if (!cls) {
175
        LOGE("EventHandler: failed to get class reference");
176
        goto end;
177
178
179
    }

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

187
end:
Edward Wang's avatar
Edward Wang committed
188
    (*env)->DeleteLocalRef(env, bundle);
189
190
    if (isAttached)
        (*myVm)->DetachCurrentThread(myVm);
191
192
}

193
jint JNI_OnLoad(JavaVM *vm, void *reserved)
194
{
195
196
197
    // Keep a reference on the Java VM.
    myVm = vm;

198
    pthread_mutex_init(&vout_android_lock, NULL);
199
    pthread_cond_init(&vout_android_surf_attached, NULL);
200

201
    LOGD("JNI interface loaded.");
202
    return JNI_VERSION_1_2;
203
204
}

205
206
void JNI_OnUnload(JavaVM* vm, void* reserved) {
    pthread_mutex_destroy(&vout_android_lock);
207
    pthread_cond_destroy(&vout_android_surf_attached);
208
209
}

Rafaël Carré's avatar
Rafaël Carré committed
210
211
// FIXME: use atomics
static bool verbosity;
212

213
void Java_org_videolan_libvlc_LibVLC_nativeInit(JNIEnv *env, jobject thiz)
214
{
215
216
217
218
219
    //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;

220
221
222
    methodId = (*env)->GetMethodID(env, cls, "getVout", "()I");
    bool use_opengles2 = (*env)->CallIntMethod(env, thiz, methodId) == VOUT_OPENGLES2;

223
224
225
    methodId = (*env)->GetMethodID(env, cls, "timeStretchingEnabled", "()Z");
    bool enable_time_stretch = (*env)->CallBooleanMethod(env, thiz, methodId);

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

Edward Wang's avatar
Edward Wang committed
229
230
    methodId = (*env)->GetMethodID(env, cls, "getDeblocking", "()I");
    int deblocking = (*env)->CallIntMethod(env, thiz, methodId);
Rafaël Carré's avatar
Rafaël Carré committed
231
232
    char deblockstr[2];
    snprintf(deblockstr, sizeof(deblockstr), "%d", deblocking);
Edward Wang's avatar
Edward Wang committed
233
    LOGD("Using deblocking level %d", deblocking);
234

Edward Wang's avatar
Edward Wang committed
235
236
    methodId = (*env)->GetMethodID(env, cls, "getNetworkCaching", "()I");
    int networkCaching = (*env)->CallIntMethod(env, thiz, methodId);
Rafaël Carré's avatar
Rafaël Carré committed
237
    char networkCachingstr[25];
Edward Wang's avatar
Edward Wang committed
238
    if(networkCaching > 0) {
Rafaël Carré's avatar
Rafaël Carré committed
239
        snprintf(networkCachingstr, sizeof(networkCachingstr), "--network-caching=%d", networkCaching);
Edward Wang's avatar
Edward Wang committed
240
241
242
        LOGD("Using network caching of %d ms", networkCaching);
    }

243
244
245
246
    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
247

248
249
250
    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
251
    LOGD("Subtitle encoding set to \"%s\"", subsencodingstr);
252

253
254
255
    methodId = (*env)->GetMethodID(env, cls, "isVerboseMode", "()Z");
    verbosity = (*env)->CallBooleanMethod(env, thiz, methodId);

256
257
    methodId = (*env)->GetMethodID(env, cls, "getHardwareAcceleration", "()I");
    int hardwareAcceleration = (*env)->CallIntMethod(env, thiz, methodId);
258
259
260
    /* With the MediaCodec opaque mode we cannot use the OpenGL ES vout. */
    if (hardwareAcceleration == HW_ACCELERATION_FULL)
        use_opengles2 = false;
261

262
    /* Don't add any invalid options, otherwise it causes LibVLC to crash */
263
    const char *argv[] = {
264
        /* CPU intensive plugin, setting for slow devices */
265
        enable_time_stretch ? "--audio-time-stretch" : "--no-audio-time-stretch",
266
267
268

        /* avcodec speed settings for slow devices */
        "--avcodec-fast", // non-spec-compliant speedup tricks
Edward Wang's avatar
Edward Wang committed
269
        "--avcodec-skiploopfilter", deblockstr,
270
271
        "--avcodec-skip-frame", enable_frame_skip ? "2" : "0",
        "--avcodec-skip-idct", enable_frame_skip ? "2" : "0",
272
273
274
275
276

        /* Remove me when UTF-8 is enforced by law */
        "--subsdec-encoding", subsencodingstr,

        /* XXX: why can't the default be fine ? #7792 */
Edward Wang's avatar
Edward Wang committed
277
        (networkCaching > 0) ? networkCachingstr : "",
278
279

        /* Android audio API is a mess */
280
        use_opensles ? "--aout=opensles" : "--aout=android_audiotrack",
281
282

        /* Android video API is a mess */
283
        use_opengles2 ? "--vout=gles2" : "--vout=androidsurface",
284
        "--androidsurface-chroma", chromastr != NULL && chromastr[0] != 0 ? chromastr : "RV32",
285
        /* XXX: we can't recover from direct rendering failure */
286
        (hardwareAcceleration == HW_ACCELERATION_FULL) ? "" : "--no-mediacodec-dr",
287
    };
288
    libvlc_instance_t *instance = libvlc_new(sizeof(argv) / sizeof(*argv), argv);
289

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

292
    (*env)->ReleaseStringUTFChars(env, chroma, chromastr);
293
294
    (*env)->ReleaseStringUTFChars(env, subsencoding, subsencodingstr);

295
296
    if (!instance)
    {
297
        jclass exc = (*env)->FindClass(env, "org/videolan/libvlc/LibVlcException");
298
299
300
301
        (*env)->ThrowNew(env, exc, "Unable to instantiate LibVLC");
    }

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

Rafaël Carré's avatar
Rafaël Carré committed
303
    libvlc_log_set(instance, debug_log, &verbosity);
304
305
}

306
void Java_org_videolan_libvlc_LibVLC_nativeDestroy(JNIEnv *env, jobject thiz)
307
{
308
    releaseMediaPlayer(env, thiz);
Edward Wang's avatar
Edward Wang committed
309
    jlong libVlcInstance = getLong(env, thiz, "mLibVlcInstance");
310
311
312
    if (!libVlcInstance)
        return; // Already destroyed

Ludovic Fauvet's avatar
Ludovic Fauvet committed
313
    libvlc_instance_t *instance = (libvlc_instance_t*)(intptr_t) libVlcInstance;
Rafaël Carré's avatar
Rafaël Carré committed
314
    libvlc_log_unset(instance);
315
316
    libvlc_release(instance);

Edward Wang's avatar
Edward Wang committed
317
    setLong(env, thiz, "mLibVlcInstance", 0);
318
319
}

320
void Java_org_videolan_libvlc_LibVLC_detachEventHandler(JNIEnv *env, jobject thiz)
321
{
322
323
324
    if (eventHandlerInstance != NULL) {
        (*env)->DeleteGlobalRef(env, eventHandlerInstance);
        eventHandlerInstance = NULL;
325
326
327
    }
}

328
void Java_org_videolan_libvlc_LibVLC_setEventHandler(JNIEnv *env, jobject thiz, jobject eventHandler)
329
{
330
331
332
    if (eventHandlerInstance != NULL) {
        (*env)->DeleteGlobalRef(env, eventHandlerInstance);
        eventHandlerInstance = NULL;
333
334
    }

335
    eventHandlerInstance = getEventHandlerReference(env, thiz, eventHandler);
336
337
}

Edward Wang's avatar
Edward Wang committed
338
void Java_org_videolan_libvlc_LibVLC_playMRL(JNIEnv *env, jobject thiz, jlong instance,
339
340
                                             jstring mrl, jobjectArray mediaOptions)
{
341
342
343
    /* Release previous media player, if any */
    releaseMediaPlayer(env, thiz);

344
    /* Create a media player playing environment */
Ludovic Fauvet's avatar
Ludovic Fauvet committed
345
    libvlc_media_player_t *mp = libvlc_media_player_new((libvlc_instance_t*)(intptr_t)instance);
Rafaël Carré's avatar
Rafaël Carré committed
346
    libvlc_media_player_set_video_title_display(mp, libvlc_position_disable, 0);
347
348
    jobject myJavaLibVLC = (*env)->NewGlobalRef(env, thiz);

349
350
351
352
    //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 )
353
    {
354
        libvlc_audio_set_callbacks(mp, aout_play, aout_pause, NULL, NULL, NULL,
Sébastien Toque's avatar
Sébastien Toque committed
355
356
                                   (void*) myJavaLibVLC);
        libvlc_audio_set_format_callbacks(mp, aout_open, aout_close);
357
358
    }

359
360
    /* Connect the event manager */
    libvlc_event_manager_t *ev = libvlc_media_player_event_manager(mp);
361
362
363
364
365
    static const libvlc_event_type_t mp_events[] = {
        libvlc_MediaPlayerPlaying,
        libvlc_MediaPlayerPaused,
        libvlc_MediaPlayerEndReached,
        libvlc_MediaPlayerStopped,
366
        libvlc_MediaPlayerVout,
367
368
        libvlc_MediaPlayerPositionChanged,
        libvlc_MediaPlayerEncounteredError
369
    };
370
    for(int i = 0; i < (sizeof(mp_events) / sizeof(*mp_events)); i++)
371
372
        libvlc_event_attach(ev, mp_events[i], vlc_event_callback, myVm);

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

376
377
378
379
    cls = (*env)->GetObjectClass(env, thiz);
    jmethodID methodID = (*env)->GetMethodID(env, cls, "applyEqualizer", "()V");
    (*env)->CallVoidMethod(env, thiz, methodID);

380
381
382
383
    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 */
384
385
386
387
388
389
390
391
392
393
    if (mediaOptions != NULL)
    {
        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);
        }
394
    }
395

396
397
    (*env)->ReleaseStringUTFChars(env, mrl, p_mrl);

398
399
400
401
402
403
404
405
    /* Connect the media event manager. */
    libvlc_event_manager_t *ev_media = libvlc_media_event_manager(p_md);
    static const libvlc_event_type_t mp_media_events[] = {
        libvlc_MediaParsedChanged
    };
    for(int i = 0; i < (sizeof(mp_media_events) / sizeof(*mp_media_events)); i++)
        libvlc_event_attach(ev_media, mp_media_events[i], vlc_event_callback, myVm);

406
407
    libvlc_media_player_set_media(mp, p_md);
    libvlc_media_player_play(mp);
408
409
}

410
jfloat Java_org_videolan_libvlc_LibVLC_getRate(JNIEnv *env, jobject thiz) {
Edward Wang's avatar
Edward Wang committed
411
412
413
414
415
416
417
    libvlc_media_player_t* mp = getMediaPlayer(env, thiz);
    if(mp)
        return libvlc_media_player_get_rate(mp);
    else
        return 1.00;
}

418
void Java_org_videolan_libvlc_LibVLC_setRate(JNIEnv *env, jobject thiz, jfloat rate) {
Edward Wang's avatar
Edward Wang committed
419
420
421
422
423
    libvlc_media_player_t* mp = getMediaPlayer(env, thiz);
    if(mp)
        libvlc_media_player_set_rate(mp, rate);
}

424
jboolean Java_org_videolan_libvlc_LibVLC_isPlaying(JNIEnv *env, jobject thiz)
425
{
426
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
427
    if (mp)
428
        return !!libvlc_media_player_is_playing(mp);
429
430
    else
        return 0;
431
432
}

433
jboolean Java_org_videolan_libvlc_LibVLC_isSeekable(JNIEnv *env, jobject thiz)
434
{
435
436
437
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return !!libvlc_media_player_is_seekable(mp);
438
    return 0;
439
440
}

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

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

455
void Java_org_videolan_libvlc_LibVLC_stop(JNIEnv *env, jobject thiz)
456
{
457
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
458
    if (mp)
459
        libvlc_media_player_stop(mp);
460
461
}

462
jint Java_org_videolan_libvlc_LibVLC_getVolume(JNIEnv *env, jobject thiz)
463
{
464
465
466
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jint) libvlc_audio_get_volume(mp);
467
468
469
    return -1;
}

470
jint Java_org_videolan_libvlc_LibVLC_setVolume(JNIEnv *env, jobject thiz, jint volume)
471
{
472
473
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
474
        //Returns 0 if the volume was set, -1 if it was out of range or error
475
        return (jint) libvlc_audio_set_volume(mp, (int) volume);
476
477
478
    return -1;
}

479
jlong Java_org_videolan_libvlc_LibVLC_getTime(JNIEnv *env, jobject thiz)
480
{
481
482
483
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return libvlc_media_player_get_time(mp);
484
485
486
    return -1;
}

487
void Java_org_videolan_libvlc_LibVLC_setTime(JNIEnv *env, jobject thiz, jlong time)
488
{
489
490
491
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        libvlc_media_player_set_time(mp, time);
492
493
}

494
jfloat Java_org_videolan_libvlc_LibVLC_getPosition(JNIEnv *env, jobject thiz)
495
{
496
497
498
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jfloat) libvlc_media_player_get_position(mp);
499
500
501
    return -1;
}

502
void Java_org_videolan_libvlc_LibVLC_setPosition(JNIEnv *env, jobject thiz, jfloat pos)
503
{
504
505
506
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        libvlc_media_player_set_position(mp, pos);
507
508
}

509
jlong Java_org_videolan_libvlc_LibVLC_getLength(JNIEnv *env, jobject thiz)
510
{
511
512
513
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jlong) libvlc_media_player_get_length(mp);
514
515
516
    return -1;
}

517
jstring Java_org_videolan_libvlc_LibVLC_version(JNIEnv* env, jobject thiz)
518
{
519
520
    return (*env)->NewStringUTF(env, libvlc_get_version());
}
521

522
jstring Java_org_videolan_libvlc_LibVLC_compiler(JNIEnv* env, jobject thiz)
523
524
525
526
{
    return (*env)->NewStringUTF(env, libvlc_get_compiler());
}

527
jstring Java_org_videolan_libvlc_LibVLC_changeset(JNIEnv* env, jobject thiz)
528
529
530
{
    return (*env)->NewStringUTF(env, libvlc_get_changeset());
}