0

我目前正在为客户开发一个 C 库,为此我需要使用其他库,例如:glib2.0、libxml2、lib、openssl 和 gmp。在我完成开发并创建 .so 后,我尝试用它编译一个测试程序,发现我的库不是独立的,我需要将程序与我上面提到的所有其他库链接。

客户坚持他想要一个 dll 或 .so 产品,并且它应该是自包含的,只有一个文件,以便他的应用程序可以使用它,就像他使用其他库一样。

我不知道该怎么做,有没有办法产生这个自包含的.so?如果不可能,我还有什么其他选择?我在 Solaris 11 中开发,目标系统是 Solaris 服务器。

非常感谢。

生成文件代码:

# C compiler 
CC = gcc 

# C flags
CFLAGS = -fPIC -O3 -g 

# linking flags
LDFLAGS = -shared 
LDLIBS = $(shell xml2-config --cflags --libs) $(shell pkg-config --cflags --libs glib-2.0) -lssl -lcrypto -lm -lgmp -lz 

# target lib
TARGET_LIB = libservices.so
SRCS = lib1.c lib2.c lib3.c lib4.c # source files

OBJS = $(SRCS:.c=.o)
# Compilation 
.PHONY: all
all: ${TARGET_LIB}

$(TARGET_LIB): $(OBJS) 
    ${CC} ${LDFLAGS} $(OBJS) -o ${TARGET_LIB}

# pull in dependency info for *existing* .o files
-include $(OBJS:.o=.d)

# Compiles and Generates Dependency Info
%.o: %.c
    gcc $(CFLAGS) -c $*.c -o $*.o ${LDLIBS}
    gcc -MM $(CFLAGS) $*.c > $*.d

测试代码编译的示例输出

gcc -I/path/to/libservices -L/path/to/libservices services_test.c -o test -lservices

libservices.so: undefined reference to `EVP_CipherInit'
libservices.so: undefined reference to `g_free'
libservices.so: undefined reference to `xmlFreeDoc'
libservices.so: undefined reference to `g_str_equal'
libservices.so: undefined reference to `g_hash_table_lookup'
libservices.so: undefined reference to `__gmpz_get_str'
libservices.so: undefined reference to `EVP_CIPHER_CTX_block_size'
libservices.so: undefined reference to `xmlCheckVersion'
libservices.so: undefined reference to `EVP_EncryptFinal'
libservices.so: undefined reference to `g_hash_table_new'
libservices.so: undefined reference to `EVP_CIPHER_CTX_init'
libservices.so: undefined reference to `xmlNodeGetContent'
libservices.so: undefined reference to `xmlCleanupParser'
libservices.so: undefined reference to `__gmpz_pow_ui'
libservices.so: undefined reference to `EVP_EncryptUpdate'
libservices.so: undefined reference to `__gmpz_clear'
libservices.so: undefined reference to `xmlParseDoc'
libservices.so: undefined reference to `g_hash_table_insert'
libservices.so: undefined reference to `EVP_DecryptInit_ex'
libservices.so: undefined reference to `EVP_EncryptInit_ex'
libservices.so: undefined reference to `g_hash_table_destroy'
libservices.so: undefined reference to `g_list_free'
libservices.so: undefined reference to `EVP_DecryptFinal'
libservices.so: undefined reference to `g_str_hash'
libservices.so: undefined reference to `__gmpz_init_set_str'
libservices.so: undefined reference to `EVP_DecryptUpdate'
libservices.so: undefined reference to `EVP_CipherFinal'
libservices.so: undefined reference to `__gmpz_sizeinbase'
libservices.so: undefined reference to `xmlDocGetRootElement'
libservices.so: undefined reference to `g_hash_table_foreach'
libservices.so: undefined reference to `EVP_CIPHER_CTX_cleanup'
libservices.so: undefined reference to `EVP_CipherUpdate'
collect2: error: ld returned 1 exit status

如果我将它与其他库链接:

gcc -I/path/to/libservices -L/path/to/libservices servicios_test.c -o test -lservices -lssl -lcrypto -lm -lgmp -lz `pkg-config --cflags --libs glib-2.0` `xml2-config --cflags --libs`

然后使用以下命令添加路径:

export LD_LIBRARY_PATH=/path/to/libservices:$LD_LIBRARY_PATH

测试代码工作正常。

4

2 回答 2

0

我认为你或多或少已经完成了。

共享库的全部意义在于包含任何其他库依赖项中的任何内容。这是运行时链接器的工作,而不是编译时链接器。

由于应用程序显然已经依赖于其他库,因此您的库以及您的库所依赖但应用程序尚未依赖的任何其他库只是要添加到列表中的更多库。

于 2015-05-15T18:50:14.107 回答
0

理论上这是可能的,但我想不出走这条路的合理理由。也很难做到。就代码而言,这是您应该做的:

LDLIBS = $(shell xml2-config --cflags --libs) $(shell pkg-config --cflags --libs glib-2.0) -Bstatic -lssl -lcrypto -lm -lgmp -lz

通过添加-Bstatic,您可以指示链接器使用后续库的静态版本。但是有一个问题:库的静态版本必须存在于您编译.so的系统上。其中一些可能默认存在,但您必须做额外的工作:

  • libmlibgmplibz - 操作系统中很可能有包含这些静态版本的软件包,您必须安装它们(通常软件包名称是lib${LIBNAME}-staticlib${LIBNAME}-devel
  • 对于libcryptolibssl,您必须将 OpenSSL 重新编译为静态库。
于 2015-05-15T19:09:13.453 回答