necessary on freebsd.
R=r, dho
CC=golang-dev
https://golang.org/cl/157069
elfstr[ElfStrDynstr] = addstring(shstrtab, ".dynstr");
elfstr[ElfStrRela] = addstring(shstrtab, ".rela");
- /* interpreter string */
- s = lookup(".interp", 0);
- s->reachable = 1;
- s->type = SDATA; // TODO: rodata
- switch(HEADTYPE) {
- case 7:
- addstring(lookup(".interp", 0), linuxdynld);
- break;
- case 9:
- addstring(lookup(".interp", 0), freebsddynld);
- break;
- }
-
/*
* hash table.
* only entries that other objects need to find when
sh->type = SHT_PROGBITS;
sh->flags = SHF_ALLOC;
sh->addralign = 1;
- shsym(sh, lookup(".interp", 0));
+ switch(HEADTYPE) {
+ case 7:
+ elfinterp(sh, startva, linuxdynld);
+ break;
+ case 9:
+ elfinterp(sh, startva, freebsddynld);
+ break;
+ }
ph = newElfPhdr();
ph->type = PT_INTERP;
a += elfwritehdr();
a += elfwritephdrs();
a += elfwriteshdrs();
- if (a > ELFRESERVE)
+ cflush();
+ if(a+elfwriteinterp() > ELFRESERVE)
diag("ELFRESERVE too small: %d > %d", a, ELFRESERVE);
break;
}
s = lookup(".interp", 0);
s->reachable = 1;
s->type = SDATA; // TODO: rodata
- switch(HEADTYPE) {
- case 7:
- addstring(lookup(".interp", 0), linuxdynld);
- break;
- case 9:
- addstring(lookup(".interp", 0), freebsddynld);
- break;
- }
/*
* hash table - empty for now.
sh->type = SHT_PROGBITS;
sh->flags = SHF_ALLOC;
sh->addralign = 1;
- shsym(sh, lookup(".interp", 0));
+ switch(HEADTYPE) {
+ case 7:
+ elfinterp(sh, startva, linuxdynld);
+ break;
+ case 9:
+ elfinterp(sh, startva, freebsddynld);
+ break;
+ }
ph = newElfPhdr();
ph->type = PT_INTERP;
a += elfwritehdr();
a += elfwritephdrs();
a += elfwriteshdrs();
- if (a > ELFRESERVE) {
- diag("ELFRESERVE too small: %d > %d", a, ELFRESERVE);
- }
cflush();
+ if(a+elfwriteinterp() > ELFRESERVE)
+ diag("ELFRESERVE too small: %d > %d", a, ELFRESERVE);
break;
}
cflush();
static ElfEhdr hdr;
static ElfPhdr *phdr[NSECT];
static ElfShdr *shdr[NSECT];
+static char *interp;
/*
Initialize the global variable that describes the ELF header. It will be updated as
adduint32(s, tag);
addsize(s, t);
}
+
+int
+elfwriteinterp(void)
+{
+ int n;
+
+ if(interp == nil)
+ return 0;
+
+ n = strlen(interp)+1;
+ seek(cout, ELFRESERVE-n, 0);
+ write(cout, interp, n);
+ return n;
+}
+
+void
+elfinterp(ElfShdr *sh, uint64 startva, char *p)
+{
+ int n;
+
+ interp = p;
+ n = strlen(interp)+1;
+ sh->addr = startva + ELFRESERVE - n;
+ sh->off = ELFRESERVE - n;
+ sh->size = n;
+}
uint64 endelf(void);
extern int numelfphdr;
extern int numelfshdr;
+int elfwriteinterp(void);
+void elfinterp(ElfShdr*, uint64, char*);
/*
* Total amount of space to reserve at the start of the file
- * for Header, PHeaders, and SHeaders.
+ * for Header, PHeaders, SHeaders, and interp.
* May waste some.
+ * On FreeBSD, cannot be larger than a page.
*/
#define ELFRESERVE 2048