libc/newlib/libc/machine/cris/libcdtor.c

92 lines
3.0 KiB
C

/* Call ctors and dtors from elinux a.out shared libraries.
Copyright (C) 1999, 2000, 2003, 2004, 2005 Axis Communications.
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Neither the name of Axis Communications nor the names of its
contributors may be used to endorse or promote products derived
from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY AXIS COMMUNICATIONS AND ITS CONTRIBUTORS
``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL AXIS
COMMUNICATIONS OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE. */
typedef void (*vfnp) (void);
/* The guts of the _Libctors and _Libdtors is "optimized" away into
empty functions when the definition is visible as well. Simplest
solution is to emit the definitions as asm. We have no .previous
directive in a.out, so we rely on the fact that everything in this
file goes into the .text section. */
__asm__
(
".text\n\t.global .$global.lib.ctors\n.$global.lib.ctors:\n\t.dword 0"
);
__asm__
(
".text\n\t.global .$global.lib.dtors\n.$global.lib.dtors:\n\t.dword 0"
);
extern vfnp * const _Ctors asm(".$global.lib.ctors");
extern vfnp * const _Dtors asm(".$global.lib.dtors");
/* We better provide weak empty ctor and dtor lists, since they are
not created if the main program does not have ctor/dtors. Because
it's otherwise not used, GCC trunk "Mon Jul 25 22:33:14 UTC 2005"
thinks it can remove defaultors, so we need to artificially mark it
as used. FIXME: Perhaps a GCC bug. */
static vfnp const defaultors[] __attribute__ ((__used__)) = {0, 0};
extern vfnp * __CTOR_LIST__ __attribute__ ((weak, alias ("defaultors")));
extern vfnp * __DTOR_LIST__ __attribute__ ((weak, alias ("defaultors")));
void
_Libctors (void)
{
const vfnp *firstor = _Ctors;
const vfnp *ctors;
/* Have to find the last ctor; they will run in opposite order as in
the table. */
if (firstor != 0 && *firstor != 0)
{
for (ctors = firstor; *ctors != 0; ctors++)
;
while (--ctors != firstor)
{
(**ctors)();
}
(**ctors)();
}
}
void
_Libdtors(void)
{
const vfnp *dtors = _Dtors;
if (dtors)
while (*dtors != 0)
{
(**dtors++) ();
}
}