source: mainline/uspace/lib/c/generic/libc.c@ 6f025a8

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 6f025a8 was 199b6d8, checked in by Dzejrou <dzejrou@…>, 7 years ago

c: fixed constructor calls for c++ objects in program loader (off-by-one error)

  • Property mode set to 100644
File size: 4.2 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
29/** @addtogroup lc Libc
[433131d]30 * @brief HelenOS C library
[a46da63]31 * @{
32 * @}
33 */
[433131d]34
[a46da63]35/** @addtogroup libc generic
36 * @ingroup lc
[b2951e2]37 * @{
38 */
[433131d]39
[b2951e2]40/** @file
[433131d]41 */
[3eddaff]42
[153c7a29]43#include <errno.h>
[3eddaff]44#include <libc.h>
[76d0981d]45#include <stdbool.h>
[47b7006]46#include <stdlib.h>
[fa23560]47#include <tls.h>
[bc1f1c2]48#include <fibril.h>
[47b7006]49#include <task.h>
[c98e6ee]50#include <loader/pcb.h>
[5126f80]51#include <vfs/vfs.h>
52#include <vfs/inbox.h>
[e26a4633]53#include "private/libc.h"
[fc5f7a8]54#include "private/async.h"
[47b7006]55#include "private/malloc.h"
56#include "private/io.h"
[d73d992]57#include "private/fibril.h"
[d54b303]58
[7fb3f1c]59#ifdef CONFIG_RTLD
[8a1fb09]60#include <rtld/rtld.h>
[7fb3f1c]61#endif
[1ea99cc]62
[47b7006]63static bool env_setup = false;
[350514c]64
[c4049e6]65/**
66 * Used for C++ constructors/destructors
67 * and the GCC constructor/destructor extension.
68 */
69typedef void (*init_array_entry_t)();
70extern init_array_entry_t __init_array_start[];
71extern init_array_entry_t __init_array_end[];
72typedef void (*fini_array_entry_t)();
73extern fini_array_entry_t __fini_array_start[];
74extern fini_array_entry_t __fini_array_end[];
75
[7148abf]76void __libc_main(void *pcb_ptr)
[a46da63]77{
[153c7a29]78 /* Initialize user task run-time environment */
79 __malloc_init();
[a35b458]80
[91e4567]81 /* Save the PCB pointer */
82 __pcb = (pcb_t *) pcb_ptr;
[a35b458]83
[91e4567]84#ifdef CONFIG_RTLD
85 if (__pcb != NULL && __pcb->rtld_runtime != NULL) {
86 runtime_env = (rtld_t *) __pcb->rtld_runtime;
[153c7a29]87 } else {
88 if (rtld_init_static() != EOK)
89 abort();
[91e4567]90 }
91#endif
[a35b458]92
[433131d]93 fibril_t *fibril = fibril_setup();
[47b7006]94 if (fibril == NULL)
95 abort();
[a35b458]96
[433131d]97 __tcb_set(fibril->tcb);
[a35b458]98
[49a796f1]99 __async_server_init();
100 __async_client_init();
101 __async_ports_init();
[a35b458]102
[47b7006]103 /* The basic run-time environment is setup */
104 env_setup = true;
[a35b458]105
[433131d]106 int argc;
107 char **argv;
[a35b458]108
[47b7006]109 /*
110 * Get command line arguments and initialize
111 * standard input and output
112 */
[c98e6ee]113 if (__pcb == NULL) {
114 argc = 0;
115 argv = NULL;
[bb9ec2d]116 __stdio_init();
[c98e6ee]117 } else {
118 argc = __pcb->argc;
119 argv = __pcb->argv;
[bb9ec2d]120 __inbox_init(__pcb->inbox, __pcb->inbox_entries);
121 __stdio_init();
[5126f80]122 vfs_root_set(inbox_get("root"));
[d96d9bc]123 (void) vfs_cwd_set(__pcb->cwd);
[c98e6ee]124 }
[a35b458]125
[c4049e6]126 /*
127 * C++ Static constructor calls.
128 */
129 ptrdiff_t init_array_entries = (__init_array_end - __init_array_start);
130
[199b6d8]131 for (int i = init_array_entries - 1; i >= 0; --i)
[c4049e6]132 __init_array_start[i]();
133
[47b7006]134 /*
135 * Run main() and set task return value
136 * according the result
137 */
138 int retval = main(argc, argv);
139 exit(retval);
140}
[7114d83]141
[099c834]142void __libc_exit(int status)
[47b7006]143{
[c4049e6]144 /*
145 * GCC extension __attribute__((destructor)),
146 * C++ destructors are added to __cxa_finalize call
147 * when the respective constructor is called.
148 */
149 ptrdiff_t fini_array_entries = (__fini_array_end - __fini_array_start);
150
151 for (int i = 0; i < fini_array_entries; ++i)
152 __fini_array_start[i]();
153
[47b7006]154 if (env_setup) {
155 __stdio_done();
156 task_retval(status);
[4d11204]157 fibril_teardown(__tcb_get()->fibril_data, false);
[47b7006]158 }
[a35b458]159
[47b7006]160 __SYSCALL1(SYS_TASK_EXIT, false);
[a35b458]161
[47b7006]162 /* Unreachable */
[76d0981d]163 while (true)
[1433ecda]164 ;
[3eddaff]165}
166
[099c834]167void __libc_abort(void)
[a46da63]168{
[47b7006]169 __SYSCALL1(SYS_TASK_EXIT, true);
[a35b458]170
[47b7006]171 /* Unreachable */
[76d0981d]172 while (true)
[1433ecda]173 ;
[3eddaff]174}
[b2951e2]175
[a46da63]176/** @}
[b2951e2]177 */
Note: See TracBrowser for help on using the repository browser.