Initial import from FreeBSD RELENG_4:
[dragonfly.git] / lib / csu / i386-elf / crt1.c
1 /*-
2  * Copyright 1996-1998 John D. Polstra.
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
15  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
16  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
17  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
18  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
19  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
20  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
21  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
23  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  *
25  * $FreeBSD: src/lib/csu/i386-elf/crt1.c,v 1.4.2.2 2002/11/18 04:57:13 bde Exp $
26  */
27
28 #ifndef __GNUC__
29 #error "GCC is needed to compile this file"
30 #endif
31
32 #include <stddef.h>
33 #include <stdlib.h>
34 #include "crtbrand.c"
35
36 typedef void (*fptr)(void);
37
38 extern void _fini(void);
39 extern void _init(void);
40 extern int main(int, char **, char **);
41
42 #ifdef GCRT
43 extern void _mcleanup(void);
44 extern void monstartup(void *, void *);
45 extern int eprol;
46 extern int etext;
47 #endif
48
49 extern int _DYNAMIC;
50 #pragma weak _DYNAMIC
51
52 #ifdef __i386__
53 #define get_rtld_cleanup()                              \
54     ({ fptr __value;                                    \
55        __asm__("movl %%edx,%0" : "=rm"(__value));       \
56        __value; })
57 #else
58 #error "This file only supports the i386 architecture"
59 #endif
60
61 char **environ;
62 char *__progname = "";
63
64 void
65 _start(char *arguments, ...)
66 {
67     fptr rtld_cleanup;
68     int argc;
69     char **argv;
70     char **env;
71
72     rtld_cleanup = get_rtld_cleanup();
73     argv = &arguments;
74     argc = * (int *) (argv - 1);
75     env = argv + argc + 1;
76     environ = env;
77     if(argc > 0 && argv[0] != NULL) {
78         char *s;
79         __progname = argv[0];
80         for (s = __progname; *s != '\0'; s++)
81             if (*s == '/')
82                 __progname = s + 1;
83     }
84
85     if(&_DYNAMIC != NULL)
86         atexit(rtld_cleanup);
87
88 #ifdef GCRT
89     atexit(_mcleanup);
90 #endif
91     atexit(_fini);
92 #ifdef GCRT
93     monstartup(&eprol, &etext);
94 #endif
95     _init();
96 #ifndef __GNUC__
97     exit( main(argc, argv, env) );
98 #else
99         /*
100          * Some versions of gcc-2 expect the stack frame to be aligned as
101          * follows after it is set up in main():
102          *
103          *  +--------------+ <--- aligned by PREFERRED_STACK_BOUNDARY
104          *  +%ebp (if any) +
105          *  +--------------+
106          *  |return address|
107          *  +--------------+
108          *  |  arguments   |
109          *  |      :       |
110          *  |      :       |
111          *  +--------------+
112          *
113          * We implement the above to fix just the usual case in FreeBSD-4.
114          * Alignment for main() is too compiler-dependent to handle correctly
115          * in all cases here (or in the kernel).  E.g., a different alignment
116          * is required for at least gcc-2.95.4 even for the small variation
117          * of compiling main() with -fomit-frame-pointer.
118          */
119         __asm__("
120         andl    $~0xf, %%esp            # align stack to 16-byte boundary
121         subl    $12+12, %%esp           # space for args and padding
122         movl    %0, 0(%%esp)
123         movl    %1, 4(%%esp)
124         movl    %2, 8(%%esp)
125         call    main
126         movl    %%eax, 0(%%esp)
127         call    exit
128         " : : "r" (argc), "r" (argv), "r" (env) : "ax", "cx", "dx", "memory");
129 #endif
130 }
131
132 #ifdef GCRT
133 __asm__(".text");
134 __asm__("eprol:");
135 __asm__(".previous");
136 #endif