/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <stdint.h>
#include <stdlib.h>
#include <crt_priv.h>
static void callVectors(const void *from_addr, const void *to_addr)
{
typedef void (* const callVect)(void);
callVect *start = (callVect *)from_addr;
callVect *end = (callVect *)to_addr;
const int32_t step = from_addr < to_addr ? 1 : -1;
const int32_t count = step > 0 ? end - start : start - end;
// basic sanity check
if (&start[step * count] != end)
return;
for (; start != end; start += step) {
callVect vec = *start;
if (vec != NULL)
vec();
}
}
void __crt_init(void)
{
extern uint32_t __init_array_start[];
extern uint32_t __init_array_end[];
callVectors(__init_array_start, __init_array_end);
}
void __crt_exit(void)
{
extern uint32_t __fini_array_start[];
extern uint32_t __fini_array_end[];
extern uint32_t __bss_end[];
extern uint32_t __got_start[];
// call global destructors
callVectors(__fini_array_start, __fini_array_end);
if (&__fini_array_end[1] <= __got_start) {
// call registered static destructors
callVectors(__bss_end + __fini_array_end[0] * (sizeof(uint32_t)), __bss_end);
}
}