Связывая статическую (.a) библиотеку с общей (.so) библиотекой, получая ошибку "перемещение R_X86_64_32S по локальному символу; перекомпиляция с -fPIC"

Компиляция с g++ 64 bit в redhat с использованием eclipse.

Ошибка компиляции: ../lib-EL5-64bit/libskd3_clnt_30134500.a(skd_clnt.o): перемещение R_X86_64_32S к `локальному символу 'не может использоваться при создании общего объекта; перекомпилировать с -fPIC

Вещи, которые я пробовал, я добавил -fPIC как к флагам компилятора C++, так и к флагам компоновщика. Он уже существовал в флагах компилятора, я просто переместил его непосредственно после g ++ и в конце списка.

Я попросил создателя библиотеки.a перекомпилировать с -fPIC, и она по-прежнему выдает ту же ошибку

Есть ли способ убедиться, что их библиотека была скомпилирована с -fPIC? Сообщение об ошибке говорит, что мне нужно скомпилировать мой.so с -fPIC, или файл.a должен быть скомпилирован с -fPIC? Что еще я могу проверить?

Вот файл make (автоматически сгенерированный eclipse)

Сделать файл:

################################################################################
# Automatically-generated file. Do not edit!
################################################################################

-include ../makefile.init

RM := rm -rf

# All of the sources participating in the build are defined here
-include sources.mk
-include subdir.mk
-include generic_d2s/subdir.mk
-include objects.mk

ifneq ($(MAKECMDGOALS),clean)
ifneq ($(strip $(C++_DEPS)),)
-include $(C++_DEPS)
endif
ifneq ($(strip $(C_DEPS)),)
-include $(C_DEPS)
endif
ifneq ($(strip $(CC_DEPS)),)
-include $(CC_DEPS)
endif
ifneq ($(strip $(CPP_DEPS)),)
-include $(CPP_DEPS)
endif
ifneq ($(strip $(CXX_DEPS)),)
-include $(CXX_DEPS)
endif
ifneq ($(strip $(C_UPPER_DEPS)),)
-include $(C_UPPER_DEPS)
endif
endif

-include ../makefile.defs

# Add inputs and outputs from these tool invocations to the build variables 

# All Target
all: libFUSE.so

# Tool invocations
libFUSE.so: $(OBJS) $(USER_OBJS)
    @echo 'Building target: $@'
    @echo 'Invoking: GCC C++ Linker'
    g++ -fPIC -L../../CRYPTOPP/lib-EL5-64bit -L../lib-EL5-64bit -Wl,-rpath /lib -shared -o"libFUSE.so" $(OBJS) $(USER_OBJS) $(LIBS)
    @echo 'Finished building target: $@'
    @echo ' '

# Other Targets
clean:
    -$(RM) $(OBJS)$(C++_DEPS)$(C_DEPS)$(CC_DEPS)$(LIBRARIES)$(CPP_DEPS)$(CXX_DEPS)$(C_UPPER_DEPS) libFUSE.so
    -@echo ' '

.PHONY: all clean dependents
.SECONDARY:

-include ../makefile.targets

subdir.mk

################################################################################
# Automatically-generated file. Do not edit!
################################################################################

# Add inputs and outputs from these tool invocations to the build variables 
CPP_SRCS += \
../CRI.cpp 


C_SRCS += \
../client_rsa.c 

OBJS += \
./CRI.o 


C_DEPS += \
./client_rsa.d 

CPP_DEPS += \
./CRI.d 


# Each subdirectory must supply rules for building sources it contributes
%.o: ../%.cpp
    @echo 'Building file: $<'
    @echo 'Invoking: GCC C++ Compiler'
    g++ -D_TM_CDIR='"/proj/kronos/test/hho/testprograms/trunk/testmethods/FUSE"' -I../../CRYPTOPP/include/cryptopp -I"/opt/hp93000/soc/pws/lib" -I"/opt/hp93000/soc/com/include" -I/opt/hp93000/soc/prod_com/include -I/opt/hp93000/soc/prod_com/include/MAPI -I/opt/hp93000/soc/testmethod/include -O0 -g3 -Wall -c -fmessage-length=0 -shared -fPIC  -Wreturn-type -Wmissing-braces -Wparentheses -Wswitch -Wunused-function -Wunused-label -Wunused-parameter -Wunused-variable -Wunused-value -Wunknown-pragmas -Wsign-compare -Wconversion -fPIC -MMD -MP -MF"$(@:%.o=%.d)" -MT"$(@:%.o=%.d)" -o"$@" "$<"
    @echo 'Finished building: $<'
    @echo ' '

%.o: ../%.c
    @echo 'Building file: $<'
    @echo 'Invoking: GCC C Compiler'
    gcc -D_TM_CDIR='"/proj/kronos/test/hho/testprograms/trunk/testmethods/FUSE"' -I/opt/hp93000/soc/prod_com/include -I/opt/hp93000/soc/prod_com/include/MAPI -I/opt/hp93000/soc/testmethod/include -I/opt/hp93000/soc/com/include -I/opt/hp93000/soc/pws/lib -O0 -g3 -Wall -c -fmessage-length=0 -fPIC  -Wreturn-type -Wmissing-braces -Wparentheses -Wswitch -Wunused-function -Wunused-label -Wunused-parameter -Wunused-variable -Wunused-value -Wunknown-pragmas -Wsign-compare -Wconversion -MMD -MP -MF"$(@:%.o=%.d)" -MT"$(@:%.o=%.d)" -o"$@" "$<"
    @echo 'Finished building: $<'
    @echo ' '

objects.mk

################################################################################
# Automatically-generated file. Do not edit!
################################################################################

USER_OBJS :=

LIBS := -lBB_IO_lib.0.1 -lskd3_clnt_30134500 -lcryptopp

sources.mk

################################################################################
# Automatically-generated file. Do not edit!
################################################################################

O_SRCS := 
CPP_SRCS := 
C_UPPER_SRCS := 
C_SRCS := 
S_UPPER_SRCS := 
OBJ_SRCS := 
ASM_SRCS := 
CXX_SRCS := 
C++_SRCS := 
CC_SRCS := 
OBJS := 
C++_DEPS := 
C_DEPS := 
CC_DEPS := 
LIBRARIES := 
CPP_DEPS := 
CXX_DEPS := 
C_UPPER_DEPS := 

# Every subdirectory with source files must be described here
SUBDIRS := \
. \
generic_d2s \

2 ответа

Когда вы компилируете статически, инструкции на ассемблере будут предполагать определенные вещи относительно локальности кода, например, любой переход находится в пределах 4 ГБ от инструкции перехода / вызова. (Предполагая 64b x86). Общие библиотеки могут быть загружены дальше, чем 4 ГБ, так что переход / вызов не подойдут.

Когда вы компилируете с -fPIC (позиционно-независимый код) - компилятор гарантирует, что код и ссылки на локальные данные не зависят от позиции, и организует, чтобы вызовы / переходы вне себя происходили с наихудшим случаем / режимом максимальной адресации (например, полная 64b),

Когда вы получаете эту ошибку, как у вас, статическая библиотека не может перейти к общей библиотеке - ярлыки выше были приняты. Единственное решение - это перекомпиляция с -fPIC. (Технически, 32-битный переход может быть только 5 байтов, но вам нужно 9 байтов [1 код операции + 8 для адреса]), поэтому никакие умения ассемблера не могут обойти это - не без какой-либо другой подлой операции, такой как использование инструкций INT обеспечить короткий прыжок рукой).

В настоящее время большинство всего может / должно быть скомпилировано с -fPIC независимо от общего доступа - если производительность не важна.

Я нашел совершенно другую причину этой ошибки. Стоит отметить, что помимо сообщения в заголовке этой темы у меня была еще одна ошибка, а именно:

      :-1: error: [path]/lib.a(xxxxxx.cpp.o): warning: relocation against `_ZN7N........kEPcmmPv' in read-only section `.text'

в моемCMakeLists.txt, у меня был такой код:

      target_sources(${PROJECT_NAME}
    PUBLIC  ${HEADERS}
    PRIVATE ${SOURCES}
    )

target_link_libraries(${PROJECT_NAME} PRIVATE
   AnotherLib
)

А.cpp(упомянутый в ИСТОЧНИКАХ) был виновником. УдалениеPRIVATEисправил проблему.

Другие вопросы по тегам