- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我想在 Linux 中的 Java 程序中运行我的 C 插件系统,我已经编写了绑定(bind)并且库在我的 Java 程序中加载得很好。
插件系统提供了一个加载插件的命令,该命令被包装到 Java 中,因此我可以在我的 Java shell 中调用 i 并且它可以工作。
当插件尝试从插件系统运行命令时出现问题,我得到一个 undefined symbol 错误。但是,我已经仔细地将插件使用的所有函数链接到库中,系统是使用 -fPIC
和 -shared
编译的。我还使用 objdump -T
查看了符号表并列出了函数。
我觉得这个过程有点疯狂,我加载 C 代码在 Java 中加载 C 代码......这可能吗?
如果有人曾经遇到过这个问题或想分享有关可能解决方案的想法,我将非常感激。
谢谢。
编辑:在此示例中,我已尽力模仿我的程序。我也遇到错误,但完全不同。
// plugin_system.c
#include <dlfcn.h>
#include <stdio.h>
int load_and_run_plugin() {
void *lib = dlopen("/home/kowa/code/c/test_dir/jni/plug/plugin.so", RTLD_LAZY);
if (lib == NULL) {
fprintf(stderr, "Cannot open library.\n");
return -1;
}
void (*plug_func)(void) = dlsym(lib, "plug_function");
if (plug_func == NULL) {
dlclose(lib);
fprintf(stderr, "Cannot find plugin function.\n");
return -1;
}
plug_func();
return 0;
}
// PluginSystem.c
#include "PluginSystem.h"
#include "plugin_system.h"
/*
* Class: PluginSystem
* Method: cmdPlug
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_PluginSystem_cmdPlug
(JNIEnv *env, jclass this)
{
printf("Calling load_and_run function...\n");
load_and_run_plugin();
printf("End of call to load_and_run function\n");
}
// plugin_user_api.c
#include <stdio.h>
static void verbose_stdout(char *str)
{
printf("%s", str);
}
void (*verbose) (char *str) = verbose_stdout;
// plugin.c
#include "plugin_user_api.h"
int plug_function() {
verbose("I'm a plugin and I use external functions.\n");
return 0;
}
// PluginSystem.java
public class PluginSystem
{
static {
System.load("/home/kowa/code/c/test_dir/jni/plug/plugin_system.so");
}
public static native void cmdPlug();
public static void main(String args[]) {
PluginSystem.cmdPlug();
}
}
// Makefile
JAVA_HOME = /usr/lib/jvm/java-8-openjdk
C_INCLUDE_PATH = $(JAVA_HOME)/include $(JAVA_HOME)/include/linux
INCLUDE = $(foreach i, $(C_INCLUDE_PATH), -I$i)
CFLAGS = -Wall -O2 -std=gnu99
all:
gcc $(CFLAGS) -fPIC -c plugin.c
gcc $(CFLAGS) -Wl,-soname,plugin.so -shared -o plugin.so plugin.o
gcc $(CFLAGS) $(INCLUDE) -fPIC -c plugin_system.c
gcc $(CFLAGS) $(INCLUDE) -fPIC -c plugin_user_api.c
gcc $(CFLAGS) $(INCLUDE) -fPIC -c PluginSystem.c
gcc $(CFLAGS) -Wl,-soname,plugin_system.so -shared -o plugin_system.so plugin_system.o plugin_user_api.o PluginSystem.o
sed -i "s:System.load(.*):System.load(\"$(shell echo `pwd`)/plugin_system.so\"):" PluginSystem.java
javac PluginSystem.java
clean:
rm *.o *.so *.class
输出:
Calling load_and_run function...
#
# A fatal error has been detected by the Java Runtime Environment:
#
# SIGSEGV (0xb) at pc=0x0000000000000000, pid=1653, tid=0x00007f01c76a2700
#
# JRE version: OpenJDK Runtime Environment (8.0_92-b14) (build 1.8.0_92-b14)
# Java VM: OpenJDK 64-Bit Server VM (25.92-b14 mixed mode linux-amd64 compressed oops)
# Problematic frame:
# C 0x0000000000000000
#
# Core dump written. Default location: /home/kowa/code/c/test_dir/jni/plug/core or core.1653
#
# An error report file with more information is saved as:
# /home/kowa/code/c/test_dir/jni/plug/hs_err_pid1653.log
#
# If you would like to submit a bug report, please visit:
# http://bugreport.java.com/bugreport/crash.jsp
# The crash happened outside the Java Virtual Machine in native code.
# See problematic frame for where to report the bug.
#
[2] 1653 abort (core dumped) java PluginSystem
日志文件太长了(654 行!)所以不要怪我没有发布它!
最佳答案
您的示例代码对我有用,但有以下注意事项:
我运行 javah
来生成我自己的 PluginSystem.h
版本,并且我编写了我自己的 plugin_system.h
,因为这些没有发布
我将硬编码的共享库路径更改为适合我的本地系统的路径。
为方便起见,我将 plugin_user_api.c
合并到 plugin.c
中并删除了 plugin_user_api.h
(未提供)无论如何)
我以更惯用的风格重写了您的Makefile
如果我构建 native 库、插件和 Java 类,然后运行它会产生以下结果:
$ /usr/lib/jvm/java-1.8.0/bin/java PluginSystem
Calling load_and_run function...
I'm a plugin and I use external functions.
End of call to load_and_run function
如果我不是将 plugin_user_api.c
合并到 plugin.c
中,而是将其合并到 plugin_system.c
中,我能够复制您的问题>。在这种情况下,我可以实现 C 驱动程序能够加载插件但 Java 无法加载插件的结果。这表明 Java 正在使用 RTLD_LOCAL
标志加载库,因此它的符号不会暴露给随后加载的库,例如插件。
至少有三种可能的解决方案:
plugin_system.so
之间插入另一个加载程序,使您能够控制 plugin_system.so
的符号范围。其中,第一个需要对现有代码进行最少的修改,但最后一个是最安全的,并且最符合 Java 为避免公开库的动态符号所做的有意努力。事实上,您可能会考虑让您的插件系统做同样的事情,以降低插件相互干扰的风险。
这是您的原始代码的变体,它演示了第三种方法。它添加了一个必须在插件加载后调用的每个插件初始化函数;这避免了对任何现有插件函数的签名进行任何更改的需要。这种变体还避免将插件的符号暴露给 Java(或 C)碰巧加载的其他库。 Java 端无需更改:
plugin_system.h
#ifndef PLUGIN_SYSTEM_H
#define PLUGIN_SYSTEM_H
struct plugin_context {
void (*verbose) (char *str);
};
#endif
plugin_system.c
#include <dlfcn.h>
#include <stdio.h>
#include "plugin_system.h"
typedef int (*init_function)(struct plugin_context *);
static void verbose_stdout(char *str)
{
printf("%s", str);
}
static struct plugin_context context = { .verbose = verbose_stdout };
int load_and_run_plugin() {
void *lib = dlopen("/home/kowa/code/c/test_dir/jni/plug/plugin.so",
RTLD_LAZY | RTLD_LOCAL);
if (lib == NULL) {
fprintf(stderr, "Cannot open library.\n");
return -1;
}
init_function plug_init = dlsym(lib, "plug_init");
if (plug_init == NULL) {
dlclose(lib);
fprintf(stderr, "Cannot find plugin initialization function.\n");
return -1;
}
plug_init(&context); // error checking omitted
void (*plug_func)(void) = dlsym(lib, "plug_function");
if (plug_func == NULL) {
dlclose(lib);
fprintf(stderr, "Cannot find plugin function.\n");
return -1;
}
plug_func(); // error checking omitted
return 0;
}
plugin.c
include <stdio.h>
#include "plugin_system.h"
static void (*verbose) (char *str);
int plug_init(struct plugin_context *context) {
verbose = context->verbose;
return 0;
}
int plug_function() {
if (verbose) {
verbose("I'm a plugin and I use external functions.\n");
return 0;
} else {
return -1;
}
}
关于java - 使用 JNI 加载 C 插件系统 : undefined symbol,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37377270/
我们的开发环境是这样配置的,当我们运行代码的调试版本时,它会在崩溃或 ^C 时进入 gdb。随着最近的一些更改,这种情况不再发生(退出程序而不是进入 gdb),我怀疑符号大小的增加导致了这个问题。 有
刚刚浏览了一个教程,想到了我看到的地方 first_name: 还有一个地方 :first_name 这样对吗?有什么区别? 最佳答案 哈希语法在 Ruby 1.9.2 中发生了变化,以更接近 jso
这里是一个相当抽象的问题,因为我不知道从哪里开始我自己的调查。 我有一个用 CMake 构建的 C 包,它生成 librpdb.so;我为同一个库设置了一个 Ruby Gem,它生成 rpdb.bun
我尝试使用 Symbol 创建对象键并用 Symbol.for 找到对应的值,但它不起作用: const sym = Symbol("x"); let obj = { [sym]: "val" }
这可能是一个愚蠢的问题,但我很高兴知道为什么我们使用带有一些标志的短形式的单符号和带有完整标志的双符号? 例子: 1) -h & --help 2) -f & --force 谁能解释一下原因? 最
我们希望能够在删除物理构建区域时删除符号服务器内容,symstore del 命令对事务 ID 起作用。这是未知的。 How to extract the transaction ID based o
我在一个我不太理解的小程序上遇到这个问题(我对节点红色有点陌生),代码是 var profile = msg.user.profile; var cart = profile.cart = pr
我正在尝试创建一种工资单以在控制台中打印,但我从代码中收到以下错误。很多时候它实际上只是一个错误,但我认为我没有足够的 java 知识来自己修复它。 import java.io.*; import
在 C# 项目中,我在 UnhandledException 中创建了小型转储。在我的 Dev 机器中,项目源和 bin 位于路径 K:\projects\*MYPROJECT* 下,如果我设法让它在
我正在尝试针对另一个使用 libcurl 共享库的共享库 (libtheirstuff.so) 交叉编译我自己的共享库 (libmystuff.so),但出现以下错误: libmystuff.so:
我试图遍历一个数组来检查它是否包含任何通过指定函数的项目。我通过向 Array 对象添加一个 .any() 原型(prototype)来做到这一点: Array.prototype.any = (co
除了这个 undefined symbol 错误外,一切正常: bash-3.2$ make g++ -Wall -g solvePlanningProblem.o Position.o AStarN
我 rsync 目录“Promotion”包含两台具有不同目录结构的机器之间的绝对符号链接(symbolic link)。因此绝对符号链接(symbolic link)在两台机器上都不起作用。为了使它
我有以下 JSX - What is your e-mail address? setStateForProperties(e)}
根据 SVG 的 symbol文档,我可以添加 refX/refY属性给它。 如果我理解正确,我可以使用这些属性来定义符号坐标系中的引用点,因此当我使用 引用它时元素,它将相对于该引用点(而不是默认
请解释以下有关“找不到符号”,“无法解析符号”或“找不到符号”错误的信息: 是什么意思? 什么原因可以导致它们? 程序员如何解决它们? 该问题旨在对Java中的这些常见编译错误进行全面的问答。 最佳答
请解释以下有关“找不到符号”,“无法解析符号”或“找不到符号”错误的信息: 是什么意思? 什么原因可以导致它们? 程序员如何解决它们? 该问题旨在对Java中的这些常见编译错误进行全面的问答。 最佳答
请解释以下有关“找不到符号”,“无法解析符号”或“找不到符号”错误的信息: 是什么意思? 什么原因可以导致它们? 程序员如何解决它们? 该问题旨在对Java中的这些常见编译错误进行全面的问答。 最佳答
请解释以下有关“找不到符号”,“无法解析符号”或“找不到符号”错误的信息: 是什么意思? 什么原因可以导致它们? 程序员如何解决它们? 该问题旨在对Java中的这些常见编译错误进行全面的问答。 最佳答
请解释以下有关“找不到符号”,“无法解析符号”或“找不到符号”错误的信息: 是什么意思? 什么原因可以导致它们? 程序员如何解决它们? 该问题旨在对Java中的这些常见编译错误进行全面的问答。 最佳答
我是一名优秀的程序员,十分优秀!