Skip to content

Commit

Permalink
Stop calling _init/_fini methods from crt1 for dynamic binaries. Do
Browse files Browse the repository at this point in the history
call preinit, init and fini arrays methods from crt1 for static binaries.

Mark new crt1 with FreeBSD-specific ELF note.

Move some common crt1 code into new MI file ignore_init.c, to reduce
duplication.  Also, conservatively adjust nearby sources for style.

Reviewed by:	kan
Tested by:	andrew (arm), flo (sparc64)
MFC after:	3 weeks
  • Loading branch information
kostikbel committed Mar 11, 2012
1 parent 95d1e3d commit 3a09450
Show file tree
Hide file tree
Showing 16 changed files with 187 additions and 180 deletions.
2 changes: 2 additions & 0 deletions lib/csu/Makefile.inc
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,6 @@

SSP_CFLAGS=

SED_FIX_NOTE = -i "" -e '/\.note\.tag/s/progbits/note/'

.include "../Makefile.inc"
6 changes: 3 additions & 3 deletions lib/csu/amd64/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -19,21 +19,21 @@ CLEANFILES+= crt1.s gcrt1.s Scrt1.s

crt1.s: crt1.c
${CC} ${CFLAGS} -S -o ${.TARGET} ${.CURDIR}/crt1.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

crt1.o: crt1.s
${CC} ${CFLAGS} -c -o ${.TARGET} crt1.s

gcrt1.s: crt1.c
${CC} ${CFLAGS} -DGCRT -S -o ${.TARGET} ${.CURDIR}/crt1.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

gcrt1.o: gcrt1.s
${CC} ${CFLAGS} -c -o ${.TARGET} gcrt1.s

Scrt1.s: crt1.c
${CC} ${CFLAGS} -fPIC -DPIC -S -o ${.TARGET} ${.CURDIR}/crt1.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

Scrt1.o: Scrt1.s
${CC} ${CFLAGS} -c -o ${.TARGET} Scrt1.s
Expand Down
28 changes: 6 additions & 22 deletions lib/csu/amd64/crt1.c
Original file line number Diff line number Diff line change
Expand Up @@ -37,26 +37,17 @@ __FBSDID("$FreeBSD$");

#include "libc_private.h"
#include "crtbrand.c"

extern int _DYNAMIC;
#pragma weak _DYNAMIC
#include "ignore_init.c"

typedef void (*fptr)(void);

extern void _fini(void);
extern void _init(void);
extern int main(int, char **, char **);

#ifdef GCRT
extern void _mcleanup(void);
extern void monstartup(void *, void *);
extern int eprol;
extern int etext;
#endif

char **environ;
const char *__progname = "";

void _start(char **, void (*)(void));

/* The entry function. */
Expand All @@ -66,18 +57,13 @@ _start(char **ap, void (*cleanup)(void))
int argc;
char **argv;
char **env;
const char *s;

argc = *(long *)(void *)ap;
argv = ap + 1;
env = ap + 2 + argc;
environ = env;
if (argc > 0 && argv[0] != NULL) {
__progname = argv[0];
for (s = __progname; *s != '\0'; s++)
if (*s == '/')
__progname = s + 1;
}
if (argc > 0 && argv[0] != NULL)
handle_progname(argv[0]);

if (&_DYNAMIC != NULL)
atexit(cleanup);
Expand All @@ -86,12 +72,10 @@ _start(char **ap, void (*cleanup)(void))

#ifdef GCRT
atexit(_mcleanup);
#endif
atexit(_fini);
#ifdef GCRT
monstartup(&eprol, &etext);
__asm__("eprol:");
#endif
_init();
exit( main(argc, argv, env) );

handle_static_init(argc, argv, env);
exit(main(argc, argv, env));
}
6 changes: 3 additions & 3 deletions lib/csu/arm/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -18,21 +18,21 @@ CLEANFILES+= crt1.s gcrt1.s Scrt1.s

crt1.s: crt1.c
${CC} ${CFLAGS} -S -o ${.TARGET} ${.CURDIR}/crt1.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

crt1.o: crt1.s
${CC} ${CFLAGS} -c -o ${.TARGET} crt1.s

gcrt1.s: crt1.c
${CC} ${CFLAGS} -DGCRT -S -o ${.TARGET} ${.CURDIR}/crt1.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

gcrt1.o: gcrt1.s
${CC} ${CFLAGS} -c -o ${.TARGET} gcrt1.s

Scrt1.s: crt1.c
${CC} ${CFLAGS} -fPIC -DPIC -S -o ${.TARGET} ${.CURDIR}/crt1.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

Scrt1.o: Scrt1.s
${CC} ${CFLAGS} -c -o ${.TARGET} Scrt1.s
Expand Down
25 changes: 5 additions & 20 deletions lib/csu/arm/crt1.c
Original file line number Diff line number Diff line change
Expand Up @@ -52,16 +52,11 @@ __FBSDID("$FreeBSD$");

#include "libc_private.h"
#include "crtbrand.c"
#include "ignore_init.c"

struct Struct_Obj_Entry;
struct ps_strings;

extern int _DYNAMIC;
#pragma weak _DYNAMIC

extern void _fini(void);
extern void _init(void);
extern int main(int, char **, char **);
extern void _start(int, char **, char **, const struct Struct_Obj_Entry *,
void (*)(void), struct ps_strings *);

Expand All @@ -72,8 +67,6 @@ extern int eprol;
extern int etext;
#endif

char **environ;
const char *__progname = "";
struct ps_strings *__ps_strings;

void __start(int, char **, char **, struct ps_strings *,
Expand Down Expand Up @@ -104,16 +97,11 @@ void
__start(int argc, char **argv, char **env, struct ps_strings *ps_strings,
const struct Struct_Obj_Entry *obj __unused, void (*cleanup)(void))
{
const char *s;

environ = env;

if (argc > 0 && argv[0] != NULL) {
__progname = argv[0];
for (s = __progname; *s != '\0'; s++)
if (*s == '/')
__progname = s + 1;
}
if (argc > 0 && argv[0] != NULL)
handle_progname(argv[0]);

if (ps_strings != (struct ps_strings *)0)
__ps_strings = ps_strings;
Expand All @@ -124,13 +112,10 @@ __start(int argc, char **argv, char **env, struct ps_strings *ps_strings,
_init_tls();
#ifdef GCRT
atexit(_mcleanup);
#endif
atexit(_fini);
#ifdef GCRT
monstartup(&eprol, &etext);
#endif
_init();
exit( main(argc, argv, env) );
handle_static_init(argc, argv, env);
exit(main(argc, argv, env));
}

#ifdef GCRT
Expand Down
27 changes: 12 additions & 15 deletions lib/csu/common/crtbrand.c
Original file line number Diff line number Diff line change
Expand Up @@ -27,10 +27,7 @@
__FBSDID("$FreeBSD$");

#include <sys/param.h>

#define ABI_VENDOR "FreeBSD"
#define ABI_SECTION ".note.ABI-tag"
#define ABI_NOTETYPE 1
#include "notes.h"

/*
* Special ".note" entry specifying the ABI version. See
Expand All @@ -55,15 +52,15 @@ __FBSDID("$FreeBSD$");
* These steps are done in the invididual Makefiles for each applicable arch.
*/
static const struct {
int32_t namesz;
int32_t descsz;
int32_t type;
char name[sizeof ABI_VENDOR];
int32_t desc;
} abitag __attribute__ ((section (ABI_SECTION), aligned(4))) __used = {
sizeof ABI_VENDOR,
sizeof(int32_t),
ABI_NOTETYPE,
ABI_VENDOR,
__FreeBSD_version
int32_t namesz;
int32_t descsz;
int32_t type;
char name[sizeof(NOTE_FREEBSD_VENDOR)];
int32_t desc;
} abitag __attribute__ ((section (NOTE_SECTION), aligned(4))) __used = {
.namesz = sizeof(NOTE_FREEBSD_VENDOR),
.descsz = sizeof(int32_t),
.type = ABI_NOTETYPE,
.name = NOTE_FREEBSD_VENDOR,
.desc = __FreeBSD_version
};
114 changes: 114 additions & 0 deletions lib/csu/common/ignore_init.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
/*-
* Copyright 2012 Konstantin Belousov <[email protected]>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/

#include <sys/cdefs.h>
__FBSDID("$FreeBSD$");

#include "notes.h"

extern int main(int, char **, char **);

extern void (*__preinit_array_start[])(int, char **, char **) __hidden;
extern void (*__preinit_array_end[])(int, char **, char **) __hidden;
extern void (*__init_array_start[])(int, char **, char **) __hidden;
extern void (*__init_array_end[])(int, char **, char **) __hidden;
extern void (*__fini_array_start[])(void) __hidden;
extern void (*__fini_array_end[])(void) __hidden;
extern void _fini(void) __hidden;
extern void _init(void) __hidden;

extern int _DYNAMIC;
#pragma weak _DYNAMIC

char **environ;
const char *__progname = "";

static void
finalizer(void)
{
void (*fn)(void);
size_t array_size, n;

array_size = __fini_array_end - __fini_array_start;
for (n = array_size; n > 0; n--) {
fn = __fini_array_start[n - 1];
if ((uintptr_t)fn != 0 && (uintptr_t)fn != 1)
(fn)();
}
_fini();
}

static inline void
handle_static_init(int argc, char **argv, char **env)
{
void (*fn)(int, char **, char **);
size_t array_size, n;

if (&_DYNAMIC != NULL)
return;

atexit(finalizer);

array_size = __preinit_array_end - __preinit_array_start;
for (n = 0; n < array_size; n++) {
fn = __preinit_array_start[n];
if ((uintptr_t)fn != 0 && (uintptr_t)fn != 1)
fn(argc, argv, env);
}
_init();
array_size = __init_array_end - __init_array_start;
for (n = 0; n < array_size; n++) {
fn = __init_array_start[n];
if ((uintptr_t)fn != 0 && (uintptr_t)fn != 1)
fn(argc, argv, env);
}
}

static inline void
handle_progname(const char *v)
{
const char *s;

__progname = v;
for (s = __progname; *s != '\0'; s++) {
if (*s == '/')
__progname = s + 1;
}
}

static const struct {
int32_t namesz;
int32_t descsz;
int32_t type;
char name[sizeof(NOTE_FREEBSD_VENDOR)];
uint32_t desc;
} crt_noinit_tag __attribute__ ((section (NOTE_SECTION),
aligned(4))) __used = {
.namesz = sizeof(NOTE_FREEBSD_VENDOR),
.descsz = sizeof(uint32_t),
.type = CRT_NOINIT_NOTETYPE,
.name = NOTE_FREEBSD_VENDOR,
.desc = 0
};
6 changes: 3 additions & 3 deletions lib/csu/i386-elf/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ CLEANFILES+= crt1_c.s gcrt1_c.s Scrt1_c.s

gcrt1_c.s: crt1_c.c
${CC} ${CFLAGS} -DGCRT -S -o ${.TARGET} ${.CURDIR}/crt1_c.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

gcrt1_c.o: gcrt1_c.s
${CC} ${CFLAGS} -c -o ${.TARGET} gcrt1_c.s
Expand All @@ -28,7 +28,7 @@ gcrt1.o: gcrt1_c.o crt1_s.o

crt1_c.s: crt1_c.c
${CC} ${CFLAGS} -S -o ${.TARGET} ${.CURDIR}/crt1_c.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

crt1_c.o: crt1_c.s
${CC} ${CFLAGS} -c -o ${.TARGET} crt1_c.s
Expand All @@ -39,7 +39,7 @@ crt1.o: crt1_c.o crt1_s.o

Scrt1_c.s: crt1_c.c
${CC} ${CFLAGS} -fPIC -DPIC -S -o ${.TARGET} ${.CURDIR}/crt1_c.c
sed -i "" -e '/\.note\.ABI-tag/s/progbits/note/' ${.TARGET}
sed ${SED_FIX_NOTE} ${.TARGET}

Scrt1_c.o: Scrt1_c.s
${CC} ${CFLAGS} -c -o ${.TARGET} Scrt1_c.s
Expand Down
Loading

0 comments on commit 3a09450

Please sign in to comment.