Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Loading a dynamic C shared library using JNI which also loads another shared library

Using JNI on Java eclipse (Linux), I am loading a dynamic shared library called first.so. So far all goes well. The problems being, that first.so also loads a dynamic library called second.so.

When running the program I am getting many "undefined symbol" errors regarding symbols located in second.so.

It seems that a library loaded using JNI cannot load other C libraries on run-time because we are in a Java environment. Is my assumption correct? Do I need special compile flags to compile the first.so library or is it special arguments to tell eclipse that it will be trying to load an .so during runtime?

Thanks in advance!

like image 742
Ori Gil Avatar asked Feb 07 '14 02:02

Ori Gil


People also ask

What is a shared library?

A shared library or shared object is a file that is intended to be shared by multiple programs. Symbols used by a program are loaded from shared libraries into memory at load time or runtime.

What is JNI wrapper?

JNI is the Java Native Interface. It defines a way for the bytecode that Android compiles from managed code (written in the Java or Kotlin programming languages) to interact with native code (written in C/C++).


1 Answers

It seems that a library loaded using JNI cannot load other C libraries on run-time because we are in a Java environment. Is my assumption correct?

No.

How is libsecond.so used by libfirst.so? Is it a linked dependency, or is it loaded by dlopen?

I've found that something like:

static {
    System.loadLibrary("second");
    System.loadLibrary("first");
}

in the class that uses the JNI usually works.

EDIT: Now that I know how you are loading libsecond.so this worked for me:

Test.java

public class Test {

    public static void main (String args[]) {
        test();
    }

    private native static void test();

    static {
        System.loadLibrary("first");
    }
} 

first.c -- The sole translation unit of libfirst.so

#include <jni.h>
#include "Test.h"
#include <dlfcn.h>
#define LIBNAME "libsecond.so"

#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     Test
 * Method:    test
 * Signature: ()V
 */
JNIEXPORT void JNICALL
Java_Test_test(JNIEnv *env , jclass cls)
{
  void* h;
  void (*sym)(void);

  h = dlopen(LIBNAME, RTLD_LAZY|RTLD_GLOBAL);
  if (h) {
    printf("dlopen " LIBNAME " worked\n");
    sym = (void (*)(void))dlsym(h,"second");
    sym();
  } else {
    printf("dlopen " LIBNAME " failed\n");
  }
}

#ifdef __cplusplus
}
#endif

second.c -- The sole translation unit of libsecond.so

#include <stdio.h>

void
second(void)
{
  printf("hello from second\n");
}

Makefile

CFLAGS=-fPIC

all : libfirst.so libsecond.so 

libsecond.so : second.o
        $(CC) -shared -Wl,-soname,libsecond.so.0 -o $@ $^ -lc

libfirst.so : first.o
        $(CC) -shared -Wl,-soname,libfirst.so.0 -o $@ $^ -ldl -lc

clean:
        rm -f *.o *.so

Test.h can be produced by javah Test. Note that libfirst.so and libsecond.so are not linked together.

like image 175
ldav1s Avatar answered Oct 17 '22 05:10

ldav1s