Как сгенерировать символ отладки gcc вне цели сборки?

Я знаю, что могу генерировать символ отладки с помощью опции -g. Однако символ помещается в целевой файл. Может ли gcc генерировать символ отладки вне исполняемого файла/библиотеки результата? Как файл .pdb для Windows VС++ компилятор сделал.

Ответ 1

Вам нужно использовать objcopy для отделить отладочную информацию:

objcopy --only-keep-debug "${tostripfile}" "${debugdir}/${debugfile}"
strip --strip-debug --strip-unneeded "${tostripfile}"
objcopy --add-gnu-debuglink="${debugdir}/${debugfile}" "${tostripfile}"

Я использую bash script ниже, чтобы отделить отладочную информацию в файлах с расширением .debug в каталоге .debug. Таким образом, я могу tar библиотеки и исполняемые файлы в одном файле tar и каталоги .debug в другом. Если я хочу добавить информацию об отладке позже, я просто извлекаю файл debug tar и voila, у меня есть символическая информация об отладке.

Это bash script:

#!/bin/bash

scriptdir=`dirname ${0}`
scriptdir=`(cd ${scriptdir}; pwd)`
scriptname=`basename ${0}`

set -e

function errorexit()
{
  errorcode=${1}
  shift
  echo [email protected]
  exit ${errorcode}
}

function usage()
{
  echo "USAGE ${scriptname} <tostrip>"
}

tostripdir=`dirname "$1"`
tostripfile=`basename "$1"`


if [ -z ${tostripfile} ] ; then
  usage
  errorexit 0 "tostrip must be specified"
fi

cd "${tostripdir}"

debugdir=.debug
debugfile="${tostripfile}.debug"

if [ ! -d "${debugdir}" ] ; then
  echo "creating dir ${tostripdir}/${debugdir}"
  mkdir -p "${debugdir}"
fi
echo "stripping ${tostripfile}, putting debug info into ${debugfile}"
objcopy --only-keep-debug "${tostripfile}" "${debugdir}/${debugfile}"
strip --strip-debug --strip-unneeded "${tostripfile}"
objcopy --add-gnu-debuglink="${debugdir}/${debugfile}" "${tostripfile}"
chmod -x "${debugdir}/${debugfile}"

Ответ 2

Компилировать с информацией об отладке:

gcc -g -o main main.c

Отделить информацию об отладке:

objcopy --only-keep-debug main main.debug

или

cp main main.debug
strip --only-keep-debug main.debug

Отладить информацию об отладке от исходного файла:

objcopy --strip-debug main

или

strip --strip-debug --strip-unneeded main

отладка в режиме debuglink:

objcopy --add-gnu-debuglink main.debug main
gdb main

Вы также можете использовать файл exec и файл символа отдельно:

gdb -s main.debug -e main

или

gdb
(gdb) exec-file main
(gdb) symbol-file main.debug

Подробнее:

(gdb) help exec-file
(gdb) help symbol-file

Ref:
https://sourceware.org/gdb/onlinedocs/gdb/Files.html#Files https://sourceware.org/gdb/onlinedocs/gdb/Separate-Debug-Files.html

Ответ 3

Отметьте опцию "-only-keep-debug" команды strip.

Из ссылки:

Предполагается, что этот параметр будет использоваться совместно с --add-gnu-debuglink для создания исполняемого файла из двух частей. Один лишенный двоичный код, который будет занимать меньше места в ОЗУ и в дистрибутиве, а второй - файл информации отладки, который необходим только тогда, когда требуются возможности отладки.

Ответ 4

ПРИМЕЧАНИЕ. Программы, скомпилированные с уровнями высокой оптимизации (-O3, -O4), не могут генерировать много символов отладки для оптимизированных переменных, встроенных функций и развернутых циклов, независимо от введенных символов (-g) или извлечения (objcopy ) в файл .debug.

Альтернативные подходы
a) Вставьте данные версии (VCS, git, svn) в программу, для оптимизированных для компилятора исполняемых файлов (-O3, -O4).
b) Создайте вторую неоптимизированную версию исполняемого файла.

Первый вариант предоставляет возможность перестроить производственный код с полной отладкой и символами на более позднюю дату. Возможность повторной сборки исходного производственного кода без оптимизации - огромная помощь для отладки. (ПРИМЕЧАНИЕ. Предполагается, что тестирование проводилось с оптимизированной версией программы).

Ваша система сборки может создать файл .c, загруженный с датой компиляции, фиксацией и другими данными VCS. Вот пример "make + git":

program: program.o version.o 

program.o: program.cpp program.h 

build_version.o: build_version.c    

build_version.c: 
    @echo "const char *build1=\"VCS: Commit: $(shell git log -1 --pretty=%H)\";" > "[email protected]"
    @echo "const char *build2=\"VCS: Date: $(shell git log -1 --pretty=%cd)\";" >> "[email protected]"
    @echo "const char *build3=\"VCS: Author: $(shell git log -1 --pretty="%an %ae")\";" >> "[email protected]"
    @echo "const char *build4=\"VCS: Branch: $(shell git symbolic-ref HEAD)\";" >> "[email protected]"
    # TODO: Add compiler options and other build details

.TEMPORARY: build_version.c

После компиляции программы вы можете найти оригинал "commit" для своего кода, используя команду: strings -a my_program | grep VCS

VCS: PROGRAM_NAME=my_program
VCS: Commit=190aa9cace3b12e2b58b692f068d4f5cf22b0145
VCS: BRANCH=refs/heads/PRJ123_feature_desc
VCS: AUTHOR=Joe Developer  [email protected]
VCS: COMMIT_DATE=2013-12-19

Осталось только проверить исходный код, перекомпилировать без оптимизации и начать отладку.