将环境变量从makefile导出到shell

问题描述 投票:0回答:1

我想修改并导出“ LD_LIBRARY_PATH”环境变量,以便将libjvm.so与我的代码链接。以下是我的Makefile:

 all: run

 helloWorld.class: helloWorld.java
     javac helloWorld.java

 hello_world: hello_world.c
     gcc -L/usr/lib/jvm/java-8-openjdk-amd64/jre/lib/amd64/server/ -I/usr/lib/jvm/java-8-openjdk-amd64/include -I/usr/lib/jvm/java-8-openjdk-amd64/include/linux/ hello_world.c -o hello_world_c_exec -ljvm

 run: helloWorld.class hello_world 
     export LD_LIBRARY_PATH="/usr/lib/jvm/java-8-openjdk-amd64/jre/lib/amd64/server/"
     ./hello_world_c_exec

 clean:
     rm -f helloWorld.class hello_world

在名为“运行”的目标中,我导出“ LD_LIBRARY_PATH”环境变量。但是它不会导出到我当前的shell中。

我收到以下错误:./hello_world_c_exec:加载共享库时出错:libjvm.so:无法打开共享对象文件:无此类文件或目录

如何使用makefile导出'LD_LIBRARY_PATH'?

makefile gnu-make
1个回答
1
投票

配方的每一行通常在单独的外壳中执行,这意味着您的export行与./hello_world_c_exec行进入不同的外壳。自己看看:

$ cat Makefile
JVM_DIR := /tools/opt/jdk-10.0.2/lib/server

foo: LDLIBS := -ljvm
foo: LDFLAGS := -L$(JVM_DIR)

.PHONY: run
run: foo
        export LD_LIBRARY_PATH=$(JVM_DIR)
        $(<D)/$(<F)

输出:

$ make run -d
...
Must remake target 'run'.
export LD_LIBRARY_PATH=/tools/opt/jdk-10.0.2/lib/server
Putting child 0x81ee20 (run) PID 18709 on the chain.
Live child 0x81ee20 (run) PID 18709
Reaping winning child 0x81ee20 PID 18709
./foo
Live child 0x81ee20 (run) PID 18710
./foo: error while loading shared libraries: libjvm.so: cannot open shared object file: No such file or directory
Reaping losing child 0x81ee20 PID 18710
make: *** [Makefile:9: run] Error 127

请注意,每个单独的行都开始了两个单独的过程(18709和18710)。如果可以通过以下几种方法在单个shell中运行它,则可以使其工作:

  1. 使其成为一个脚本:
$ cat Makefile
JVM_DIR := /tools/opt/jdk-10.0.2/lib/server

foo: LDLIBS := -ljvm
foo: LDFLAGS := -L$(JVM_DIR)

.PHONY: run
run: foo
        export LD_LIBRARY_PATH=$(JVM_DIR); \
        $(<D)/$(<F)

$ make run -d
...
Must remake target 'run'.
export LD_LIBRARY_PATH=/tools/opt/jdk-10.0.2/lib/server; \
./foo
Putting child 0x21aa780 (run) PID 22009 on the chain.
Live child 0x21aa780 (run) PID 22009
Reaping winning child 0x21aa780 PID 22009
Removing child 0x21aa780 PID 22009 from chain.
Successfully remade target file 'run'.
  1. 直接在命令行中设置:
$ cat Makefile
JVM_DIR := /tools/opt/jdk-10.0.2/lib/server

foo: LDLIBS := -ljvm
foo: LDFLAGS := -L$(JVM_DIR)

.PHONY: run
run: foo
        LD_LIBRARY_PATH=$(JVM_DIR) $(<D)/$(<F)

$ make run
LD_LIBRARY_PATH=/tools/opt/jdk-10.0.2/lib/server ./foo
  1. 打开One Shell,以便在单个外壳而不是在每一行的单独外壳中执行配方:
$ cat Makefile
JVM_DIR := /tools/opt/jdk-10.0.2/lib/server

foo: LDLIBS := -ljvm
foo: LDFLAGS := -L$(JVM_DIR)

.ONESHELL:

.PHONY: run
run: foo
        export LD_LIBRARY_PATH=$(JVM_DIR)
        $(<D)/$(<F)

$ make run -d
...
Must remake target 'run'.
export LD_LIBRARY_PATH=/tools/opt/jdk-10.0.2/lib/server
./foo
Putting child 0x25258f0 (run) PID 31367 on the chain.
Live child 0x25258f0 (run) PID 31367
Reaping winning child 0x25258f0 PID 31367
Removing child 0x25258f0 PID 31367 from chain.
Successfully remade target file 'run'.

个人,我会选择其他方法。

首先,设置LD_LIBRARY_PATH很麻烦,并且取决于各种设置,可能会导致其他事情失败,例如,当系统已经将LD_LIBRARY_PATH设置为其他设置时;至少您应该附加到变量,而不是将其设置为严格值。

第二,这仍然留下了在Makefile外部运行可执行文件的问题。即使在构建二进制文件时,通过简单的调用来运行它也不是那么简单:

$ make run
cc   -L/tools/opt/jdk-10.0.2/lib/server  foo.c  -ljvm -o foo
export LD_LIBRARY_PATH=/tools/opt/jdk-10.0.2/lib/server
./foo
$ ./foo
./foo: error while loading shared libraries: libjvm.so: cannot open shared object file: No such file or directory

相反,我会告诉链接器将提示嵌入在哪里可以找到该库,因此根本不需要LD_LIBRARY_PATH

$ cat Makefile
JVM_DIR := /tools/opt/jdk-10.0.2/lib/server

foo: LDLIBS := -ljvm
foo: LDFLAGS := -L$(JVM_DIR) -Wl,-rpath,$(JVM_DIR)

.PHONY: run
run: foo
        $(<D)/$(<F)

$ make run
cc   -L/tools/opt/jdk-10.0.2/lib/server -Wl,-rpath,/tools/opt/jdk-10.0.2/lib/server  foo.c  -ljvm -o foo
./foo

$ ./foo
$ echo $?
0
© www.soinside.com 2019 - 2024. All rights reserved.