Compensación de datos de ID de compilación en el archivo ELF
Necesito modificar la build-id
sección de notas de ELF. Descubrí que es posible aquí . También descubrí que puedo hacerlo modificando este código . Lo que no puedo entender es la ubicación de los datos. Esto es de lo que estoy hablando.
$ eu-readelf -S myelffile
Section Headers:
[Nr] Name Type Addr Off Size ES Flags Lk Inf Al
...
[ 2] .note.ABI-tag NOTE 000000000000028c 0000028c 00000020 0 A 0 0 4
[ 3] .note.gnu.build-id NOTE 00000000000002ac 000002ac 00000024 0 A 0 0 4
...
$ eu-readelf -n myelffile
Note section [ 2] '.note.ABI-tag' of 32 bytes at offset 0x28c:
Owner Data size Type
GNU 16 GNU_ABI_TAG
OS: Linux, ABI: 3.14.0
Note section [ 3] '.note.gnu.build-id' of 36 bytes at offset 0x2ac:
Owner Data size Type
GNU 20 GNU_BUILD_ID
Build ID: d75a086c288c582036b0562908304bc3a8033235
.note.gnu.build-id
la sección es de 36 bytes. El ID de compilación es de 20 bytes. ¿Cuáles son los otros 16 bytes?
Jugué un poco con el código y leí 36 bytes de myelffile
at offset 0x2ac
. Tengo lo siguiente 040000001400000003000000474e5500d75a086c288c582036b0562908304bc3a8033235
.
Luego decidí usar la Elf64_Shdrdefinición , así que leí los datos en la dirección 0x2ac + sizeof(Elf64_Shdr.sh_name) + sizeof(Elf64_Shdr.sh_type) + sizeof(Elf64_Shdr.sh_flags)
y obtuve mi ID de compilación, d75a086c288c582036b0562908304bc3a8033235
. Tiene sentido por qué lo obtuve sizeof(Elf64_Shdr.sh_name) + sizeof(Elf64_Shdr.sh_type) + sizeof(Elf64_Shdr.sh_flags) = 16 bytes
, pero de acuerdo con la Elf64_Shdrdefinición , debería señalar Elf64_Addr sh_addr
, es decir, la dirección virtual de la sección.
Entonces, lo que no me queda claro es ¿cuáles son los otros 16 bytes de la sección? ¿Qué representan? No puedo conciliar la Elf64_Shdrdefinición y los resultados que obtengo de mis experimentos.
Respuestas
La sección .note.gnu.build-id tiene 36 bytes. El ID de compilación es de 20 bytes. ¿Cuáles son los otros 16 bytes?
Cada .note.*
sección comienza con Elf64_Nhdr
(12 bytes), seguido del nombre de la nota (4 bytes alineados) de tamaño variable ( GNU\0
aquí), seguido de los datos reales de la nota (4 bytes alineados). Documentación .
Mirando en /bin/date
mi sistema:
eu-readelf -Wn /bin/date
Note section [ 2] '.note.ABI-tag' of 32 bytes at offset 0x2c4:
Owner Data size Type
GNU 16 GNU_ABI_TAG
OS: Linux, ABI: 3.2.0
Note section [ 3] '.note.gnu.build-id' of 36 bytes at offset 0x2e4:
Owner Data size Type
GNU 20 GNU_BUILD_ID
Build ID: 979ae4616ae71af565b123da2f994f4261748cc9
¿Cuáles son los bytes en el desplazamiento 0x2e4
?
dd bs=1 skip=$((0x2e4)) count=36 < /bin/date | xxd
00000000: 0400 0000 1400 0000 0300 0000 474e 5500 ............GNU.
00000010: 979a e461 6ae7 1af5 65b1 23da 2f99 4f42 ...aj...e.#./.OB
00000020: 6174 8cc9 at..
Así que tenemos: .n_namesz == 4
, .n_descsz == 20
, .n_type == 3 == NT_GNU_BUILD_ID
, seguido de un GNU\0
nombre de nota de 4 bytes, seguido de 20 bytes de bytes de ID de compilación reales 0x97
, 0x9a
, etc.