libvlcjni.c 19.5 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
#include "native_crash_handler.h"
42

43
44
45
#define VOUT_ANDROID_SURFACE 0
#define VOUT_OPENGLES2       1

46
47
48
49
#define HW_ACCELERATION_DISABLED 0
#define HW_ACCELERATION_DECODING 1
#define HW_ACCELERATION_FULL     2

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

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

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

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

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

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

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

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

115
    bool isAttached = false;
116

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        /* 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
278
        (networkCaching > 0) ? networkCachingstr : "",
279
280

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

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

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

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

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

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

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

    init_native_crash_handler(env, thiz);
307
308
}

309
void Java_org_videolan_libvlc_LibVLC_nativeDestroy(JNIEnv *env, jobject thiz)
310
{
311
312
    destroy_native_crash_handler(env);

313
    releaseMediaPlayer(env, thiz);
Edward Wang's avatar
Edward Wang committed
314
    jlong libVlcInstance = getLong(env, thiz, "mLibVlcInstance");
315
316
317
    if (!libVlcInstance)
        return; // Already destroyed

Ludovic Fauvet's avatar
Ludovic Fauvet committed
318
    libvlc_instance_t *instance = (libvlc_instance_t*)(intptr_t) libVlcInstance;
Rafaël Carré's avatar
Rafaël Carré committed
319
    libvlc_log_unset(instance);
320
321
    libvlc_release(instance);

Edward Wang's avatar
Edward Wang committed
322
    setLong(env, thiz, "mLibVlcInstance", 0);
323
324
}

325
void Java_org_videolan_libvlc_LibVLC_detachEventHandler(JNIEnv *env, jobject thiz)
326
{
327
328
329
    if (eventHandlerInstance != NULL) {
        (*env)->DeleteGlobalRef(env, eventHandlerInstance);
        eventHandlerInstance = NULL;
330
331
332
    }
}

333
void Java_org_videolan_libvlc_LibVLC_setEventHandler(JNIEnv *env, jobject thiz, jobject eventHandler)
334
{
335
336
337
    if (eventHandlerInstance != NULL) {
        (*env)->DeleteGlobalRef(env, eventHandlerInstance);
        eventHandlerInstance = NULL;
338
339
    }

340
    eventHandlerInstance = getEventHandlerReference(env, thiz, eventHandler);
341
342
}

Edward Wang's avatar
Edward Wang committed
343
void Java_org_videolan_libvlc_LibVLC_playMRL(JNIEnv *env, jobject thiz, jlong instance,
344
345
                                             jstring mrl, jobjectArray mediaOptions)
{
346
347
348
    /* Release previous media player, if any */
    releaseMediaPlayer(env, thiz);

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

354
355
356
357
    //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 )
358
    {
359
        libvlc_audio_set_callbacks(mp, aout_play, aout_pause, NULL, NULL, NULL,
Sébastien Toque's avatar
Sébastien Toque committed
360
361
                                   (void*) myJavaLibVLC);
        libvlc_audio_set_format_callbacks(mp, aout_open, aout_close);
362
363
    }

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

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

381
382
383
384
    cls = (*env)->GetObjectClass(env, thiz);
    jmethodID methodID = (*env)->GetMethodID(env, cls, "applyEqualizer", "()V");
    (*env)->CallVoidMethod(env, thiz, methodID);

385
386
387
388
    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 */
389
390
391
392
393
394
395
396
397
398
    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);
        }
399
    }
400

401
402
    (*env)->ReleaseStringUTFChars(env, mrl, p_mrl);

403
404
405
406
407
408
409
410
    /* 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);

411
412
    libvlc_media_player_set_media(mp, p_md);
    libvlc_media_player_play(mp);
413
414
}

415
jfloat Java_org_videolan_libvlc_LibVLC_getRate(JNIEnv *env, jobject thiz) {
Edward Wang's avatar
Edward Wang committed
416
417
418
419
420
421
422
    libvlc_media_player_t* mp = getMediaPlayer(env, thiz);
    if(mp)
        return libvlc_media_player_get_rate(mp);
    else
        return 1.00;
}

423
void Java_org_videolan_libvlc_LibVLC_setRate(JNIEnv *env, jobject thiz, jfloat rate) {
Edward Wang's avatar
Edward Wang committed
424
425
426
427
428
    libvlc_media_player_t* mp = getMediaPlayer(env, thiz);
    if(mp)
        libvlc_media_player_set_rate(mp, rate);
}

429
jboolean Java_org_videolan_libvlc_LibVLC_isPlaying(JNIEnv *env, jobject thiz)
430
{
431
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
432
    if (mp)
433
        return !!libvlc_media_player_is_playing(mp);
434
435
    else
        return 0;
436
437
}

438
jboolean Java_org_videolan_libvlc_LibVLC_isSeekable(JNIEnv *env, jobject thiz)
439
{
440
441
442
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return !!libvlc_media_player_is_seekable(mp);
443
    return 0;
444
445
}

446
void Java_org_videolan_libvlc_LibVLC_play(JNIEnv *env, jobject thiz)
447
{
448
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
449
    if (mp)
450
        libvlc_media_player_play(mp);
451
452
}

453
void Java_org_videolan_libvlc_LibVLC_pause(JNIEnv *env, jobject thiz)
454
{
455
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
456
    if (mp)
457
        libvlc_media_player_pause(mp);
458
459
}

460
void Java_org_videolan_libvlc_LibVLC_stop(JNIEnv *env, jobject thiz)
461
{
462
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
463
    if (mp)
464
        libvlc_media_player_stop(mp);
465
466
}

467
jint Java_org_videolan_libvlc_LibVLC_getVolume(JNIEnv *env, jobject thiz)
468
{
469
470
471
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jint) libvlc_audio_get_volume(mp);
472
473
474
    return -1;
}

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

484
jlong Java_org_videolan_libvlc_LibVLC_getTime(JNIEnv *env, jobject thiz)
485
{
486
487
488
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return libvlc_media_player_get_time(mp);
489
490
491
    return -1;
}

492
void Java_org_videolan_libvlc_LibVLC_setTime(JNIEnv *env, jobject thiz, jlong time)
493
{
494
495
496
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        libvlc_media_player_set_time(mp, time);
497
498
}

499
jfloat Java_org_videolan_libvlc_LibVLC_getPosition(JNIEnv *env, jobject thiz)
500
{
501
502
503
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jfloat) libvlc_media_player_get_position(mp);
504
505
506
    return -1;
}

507
void Java_org_videolan_libvlc_LibVLC_setPosition(JNIEnv *env, jobject thiz, jfloat pos)
508
{
509
510
511
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        libvlc_media_player_set_position(mp, pos);
512
513
}

514
jlong Java_org_videolan_libvlc_LibVLC_getLength(JNIEnv *env, jobject thiz)
515
{
516
517
518
    libvlc_media_player_t *mp = getMediaPlayer(env, thiz);
    if (mp)
        return (jlong) libvlc_media_player_get_length(mp);
519
520
521
    return -1;
}

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

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

532
jstring Java_org_videolan_libvlc_LibVLC_changeset(JNIEnv* env, jobject thiz)
533
534
535
{
    return (*env)->NewStringUTF(env, libvlc_get_changeset());
}