Clean multiline string literal for gcc3
[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  * $DragonFly: src/lib/csu/i386-elf/Attic/crt1.c,v 1.3 2004/01/23 11:01:58 joerg Exp $
27  */
28
29 #ifndef __GNUC__
30 #error "GCC is needed to compile this file"
31 #endif
32
33 #include <stddef.h>
34 #include <stdlib.h>
35 #include "crtbrand.c"
36
37 typedef void (*fptr)(void);
38
39 extern void _fini(void);
40 extern void _init(void);
41 extern int main(int, char **, char **);
42
43 #ifdef GCRT
44 extern void _mcleanup(void);
45 extern void monstartup(void *, void *);
46 extern int eprol;
47 extern int etext;
48 #endif
49
50 extern int _DYNAMIC;
51 #pragma weak _DYNAMIC
52
53 #ifdef __i386__
54 #define get_rtld_cleanup()                              \
55     ({ fptr __value;                                    \
56        __asm__("movl %%edx,%0" : "=rm"(__value));       \
57        __value; })
58 #else
59 #error "This file only supports the i386 architecture"
60 #endif
61
62 char **environ;
63 char *__progname = "";
64
65 void
66 _start(char *arguments, ...)
67 {
68     fptr rtld_cleanup;
69     int argc;
70     char **argv;
71     char **env;
72
73     rtld_cleanup = get_rtld_cleanup();
74     argv = &arguments;
75     argc = * (int *) (argv - 1);
76     env = argv + argc + 1;
77     environ = env;
78     if(argc > 0 && argv[0] != NULL) {
79         char *s;
80         __progname = argv[0];
81         for (s = __progname; *s != '\0'; s++)
82             if (*s == '/')
83                 __progname = s + 1;
84     }
85
86     if(&_DYNAMIC != NULL)
87         atexit(rtld_cleanup);
88
89 #ifdef GCRT
90     atexit(_mcleanup);
91 #endif
92     atexit(_fini);
93 #ifdef GCRT
94     monstartup(&eprol, &etext);
95 #endif
96     _init();
97 #ifndef __GNUC__
98     exit( main(argc, argv, env) );
99 #else
100         /*
101          * Some versions of gcc-2 expect the stack frame to be aligned as
102          * follows after it is set up in main():
103          *
104          *  +--------------+ <--- aligned by PREFERRED_STACK_BOUNDARY
105          *  +%ebp (if any) +
106          *  +--------------+
107          *  |return address|
108          *  +--------------+
109          *  |  arguments   |
110          *  |      :       |
111          *  |      :       |
112          *  +--------------+
113          *
114          * We implement the above to fix just the usual case in FreeBSD-4.
115          * Alignment for main() is too compiler-dependent to handle correctly
116          * in all cases here (or in the kernel).  E.g., a different alignment
117          * is required for at least gcc-2.95.4 even for the small variation
118          * of compiling main() with -fomit-frame-pointer.
119          */
120         __asm__("\n"
121         "andl   $~0xf, %%esp            # align stack to 16-byte boundary\n"
122         "subl   $12+12, %%esp           # space for args and padding\n"
123         "movl   %0, 0(%%esp)\n"
124         "movl   %1, 4(%%esp)\n"
125         "movl   %2, 8(%%esp)\n"
126         "call   main\n"
127         "movl   %%eax, 0(%%esp)\n"
128         "call   exit\n"
129         : : "r" (argc), "r" (argv), "r" (env) : "ax", "cx", "dx", "memory");
130 #endif
131 }
132
133 #ifdef GCRT
134 __asm__(".text");
135 __asm__("eprol:");
136 __asm__(".previous");
137 #endif