source: mainline/uspace/lib/c/generic/libc.c@ e43acd3

Last change on this file since e43acd3 was 32254d6, checked in by GitHub <noreply@…>, 5 months ago

init RTLD runtime at load time even for statically linked binaries (#242)

init RTLD runtime at load time even for statically linked binaries

  • Property mode set to 100644
File size: 4.5 KB
RevLine 
[3eddaff]1/*
[df4ed85]2 * Copyright (c) 2005 Martin Decky
[3eddaff]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 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
[b2951e2]27 */
[a46da63]28
[4122410]29/** @addtogroup libc
[b2951e2]30 * @{
31 */
[433131d]32
[b2951e2]33/** @file
[433131d]34 */
[3eddaff]35
[153c7a29]36#include <errno.h>
[3eddaff]37#include <libc.h>
[76d0981d]38#include <stdbool.h>
[47b7006]39#include <stdlib.h>
[fa23560]40#include <tls.h>
[bc1f1c2]41#include <fibril.h>
[45c8eea]42#include <fibril_synch.h>
[47b7006]43#include <task.h>
[c98e6ee]44#include <loader/pcb.h>
[5126f80]45#include <vfs/vfs.h>
46#include <vfs/inbox.h>
[45c8eea]47#include <io/kio.h>
[e26a4633]48#include "private/libc.h"
[fc5f7a8]49#include "private/async.h"
[47b7006]50#include "private/malloc.h"
51#include "private/io.h"
[d73d992]52#include "private/fibril.h"
[d54b303]53
[7fb3f1c]54#ifdef CONFIG_RTLD
[8a1fb09]55#include <rtld/rtld.h>
[7fb3f1c]56#endif
[1ea99cc]57
[2eadda9]58progsymbols_t __progsymbols;
59
[40abf56]60static bool env_setup;
61static fibril_t main_fibril;
[350514c]62
[7148abf]63void __libc_main(void *pcb_ptr)
[a46da63]64{
[45c8eea]65 __kio_init();
66
[40abf56]67 assert(!__tcb_is_set());
[a35b458]68
[91e4567]69 __pcb = (pcb_t *) pcb_ptr;
[a35b458]70
[40abf56]71 if (__pcb) {
72 main_fibril.tcb = __pcb->tcb;
73 } else {
74 /*
75 * Loaded by kernel, not the loader.
76 * Kernel only supports loading fully static binaries,
77 * so we can do basic initialization without worrying about
78 * dynamic libraries.
79 */
80
81 main_fibril.tcb = tls_make_initial(__progsymbols.elfstart);
82 }
83
84 assert(main_fibril.tcb);
85
[514d561]86 __fibrils_init();
[45c8eea]87 __fibril_synch_init();
[514d561]88
[40abf56]89 /* Initialize the fibril. */
90 main_fibril.tcb->fibril_data = &main_fibril;
91 __tcb_set(main_fibril.tcb);
92 fibril_setup(&main_fibril);
93
94 /* Initialize user task run-time environment */
95 __malloc_init();
96
[91e4567]97#ifdef CONFIG_RTLD
[32254d6]98 if (__pcb == NULL) {
99 /*
100 * A binary loaded by kernel, not the loader.
101 * Noop - code loaded by kernel doesn't need RTLD.
102 */
[153c7a29]103 } else {
[32254d6]104 assert(__pcb->rtld_runtime != NULL);
105 runtime_env = (rtld_t *) __pcb->rtld_runtime;
[91e4567]106 }
107#endif
[a35b458]108
[49a796f1]109 __async_server_init();
110 __async_client_init();
111 __async_ports_init();
[a35b458]112
[47b7006]113 /* The basic run-time environment is setup */
114 env_setup = true;
[a35b458]115
[433131d]116 int argc;
117 char **argv;
[a35b458]118
[47b7006]119 /*
120 * Get command line arguments and initialize
121 * standard input and output
122 */
[c98e6ee]123 if (__pcb == NULL) {
124 argc = 0;
125 argv = NULL;
[bb9ec2d]126 __stdio_init();
[c98e6ee]127 } else {
128 argc = __pcb->argc;
129 argv = __pcb->argv;
[bb9ec2d]130 __inbox_init(__pcb->inbox, __pcb->inbox_entries);
131 __stdio_init();
[5126f80]132 vfs_root_set(inbox_get("root"));
[d96d9bc]133 (void) vfs_cwd_set(__pcb->cwd);
[c98e6ee]134 }
[a35b458]135
[c4049e6]136 /*
137 * C++ Static constructor calls.
138 */
139
[2eadda9]140 if (__progsymbols.preinit_array) {
141 for (int i = __progsymbols.preinit_array_len - 1; i >= 0; --i)
142 __progsymbols.preinit_array[i]();
143 }
144
145 if (__progsymbols.init_array) {
146 for (int i = __progsymbols.init_array_len - 1; i >= 0; --i)
147 __progsymbols.init_array[i]();
148 }
[c4049e6]149
[47b7006]150 /*
151 * Run main() and set task return value
152 * according the result
153 */
[2eadda9]154 int retval = __progsymbols.main(argc, argv);
[47b7006]155 exit(retval);
156}
[7114d83]157
[25f6bddb]158void __libc_fini(void)
159{
160 __async_client_fini();
161 __async_server_fini();
162 __async_ports_fini();
163
164 __fibril_synch_fini();
165 __fibrils_fini();
166
167 __malloc_fini();
168
169 __kio_fini();
170}
171
[099c834]172void __libc_exit(int status)
[47b7006]173{
[c4049e6]174 /*
175 * GCC extension __attribute__((destructor)),
176 * C++ destructors are added to __cxa_finalize call
177 * when the respective constructor is called.
178 */
179
[2eadda9]180 for (int i = 0; i < __progsymbols.fini_array_len; ++i)
181 __progsymbols.fini_array[i]();
[c4049e6]182
[47b7006]183 if (env_setup) {
184 __stdio_done();
185 task_retval(status);
186 }
[a35b458]187
[47b7006]188 __SYSCALL1(SYS_TASK_EXIT, false);
[2eadda9]189 __builtin_unreachable();
[3eddaff]190}
191
[099c834]192void __libc_abort(void)
[a46da63]193{
[47b7006]194 __SYSCALL1(SYS_TASK_EXIT, true);
[2eadda9]195 __builtin_unreachable();
[3eddaff]196}
[b2951e2]197
[a46da63]198/** @}
[b2951e2]199 */
Note: See TracBrowser for help on using the repository browser.