Merge branch 'vendor/TNFTP'
[dragonfly.git] / lib / csu / i386 / crt1_c.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, svn 200038 2009/12/02 kib $
26  */
27
28 #ifndef __GNUC__
29 #error "GCC is needed to compile this file"
30 #endif
31
32 #include <machine/tls.h>
33 #include <stddef.h>
34 #include <stdlib.h>
35
36 #include "libc_private.h"
37 #include "crtbrand.c"
38
39 extern int _DYNAMIC;
40 #pragma weak _DYNAMIC
41
42 typedef void (*fptr)(void);
43
44 extern void _fini(void);
45 extern void _init(void);
46 extern int main(int, char **, char **);
47
48 #ifdef GCRT
49 extern void _mcleanup(void);
50 extern void monstartup(void *, void *);
51 extern int eprol;
52 extern int etext;
53 #endif
54
55 char **environ;
56 const char *__progname = "";
57
58 void _start1(fptr, int, char *[]) __dead2;
59
60 /* The entry function, C part. */
61 void
62 _start1(fptr cleanup, int argc, char *argv[])
63 {
64     char **env;
65     const char *s;
66
67     env = argv + argc + 1;
68     environ = env;
69     if (argc > 0 && argv[0] != NULL) {
70         __progname = argv[0];
71         for (s = __progname; *s != '\0'; s++)
72             if (*s == '/')
73                 __progname = s + 1;
74     }
75
76     /*
77      * Setup the initial TLS space.  The RTLD does not set up our TLS
78      * (it can't, it doesn't know how our errno is declared).  It simply
79      * does all the groundwork required so that we can call
80      * _rtld_allocate_tls().
81      */
82     _init_tls();
83     _rtld_call_init();
84
85     if(&_DYNAMIC != NULL)
86         atexit(cleanup);
87
88 #ifdef GCRT
89     atexit(_mcleanup);
90 #endif
91     atexit(_fini);
92 #ifdef GCRT
93     monstartup(&eprol, &etext);
94 __asm__("eprol:");
95 #endif
96     _init();
97     exit( main(argc, argv, env) );
98 }
99
100 __asm(".hidden  _start1");