summaryrefslogtreecommitdiff
path: root/src/elfops/last-section-in-segment.c
diff options
context:
space:
mode:
authornorly <ny-git@enpas.org>2013-02-23 15:41:35 +0000
committernorly <ny-git@enpas.org>2013-02-23 15:42:42 +0000
commit5b7078265bb864ddec23162b2dfc5c02dacb0e1d (patch)
treef8973076c5f93c0e8e31d432b32a808c72bdb8db /src/elfops/last-section-in-segment.c
parent6f12fafdc4102a9b8c3ffee060cab48f50f4c37d (diff)
Move ELF-related files together
Diffstat (limited to 'src/elfops/last-section-in-segment.c')
-rw-r--r--src/elfops/last-section-in-segment.c54
1 files changed, 54 insertions, 0 deletions
diff --git a/src/elfops/last-section-in-segment.c b/src/elfops/last-section-in-segment.c
new file mode 100644
index 0000000..53323f6
--- /dev/null
+++ b/src/elfops/last-section-in-segment.c
@@ -0,0 +1,54 @@
+#include <stdio.h>
+#include <stdlib.h>
+
+#include <libelf.h>
+#include <gelf.h>
+
+#include <libelfu/libelfu.h>
+
+
+/*
+ * Returns the first section that is contained in the segment and
+ * ends as close to its memory image of as possible (the "last"
+ * section in the segment).
+ *
+ * If no section fits, NULL is returned.
+ */
+Elf_Scn* elfu_lastSectionInSegment(Elf *e, GElf_Phdr *phdr)
+{
+ Elf_Scn *last = NULL;
+ Elf_Scn *scn;
+
+
+ scn = elf_getscn(e, 1);
+ while (scn) {
+ GElf_Shdr shdr;
+
+ if (gelf_getshdr(scn, &shdr) != &shdr) {
+ fprintf(stderr, "gelf_getshdr() failed: %s\n", elf_errmsg(-1));
+ continue;
+ }
+
+ if (elfu_segmentContainsSection(phdr, &shdr)) {
+ if (!last) {
+ last = scn;
+ } else {
+ GElf_Shdr shdrOld;
+
+ if (gelf_getshdr(last, &shdrOld) != &shdrOld) {
+ continue;
+ }
+
+ if (shdr.sh_offset + shdr.sh_size
+ > shdrOld.sh_offset + shdrOld.sh_size) {
+ // TODO: Check (leftover space in memory image) < (p_align)
+ last = scn;
+ }
+ }
+ }
+
+ scn = elf_nextscn(e, scn);
+ }
+
+ return last;
+}