Merge "Put __*_ARRAY__ symbols before prioritized init/fini funcs" am: e9eae51539
Original change: https://android-review.googlesource.com/c/platform/bionic/+/1472582 Change-Id: I33424f313d7b8906f177d4046d39c91fe32c8d93
This commit is contained in:
commit
9d5e9ea82a
5 changed files with 55 additions and 10 deletions
|
@ -31,9 +31,9 @@
|
|||
#include <stdint.h>
|
||||
|
||||
#define SECTION(name) __attribute__((__section__(name)))
|
||||
SECTION(".preinit_array") void (*__PREINIT_ARRAY__)(void) = (void (*)(void)) -1;
|
||||
SECTION(".init_array") void (*__INIT_ARRAY__)(void) = (void (*)(void)) -1;
|
||||
SECTION(".fini_array") void (*__FINI_ARRAY__)(void) = (void (*)(void)) -1;
|
||||
SECTION(".preinit_array") init_func_t* __PREINIT_ARRAY__ = (init_func_t*)-1;
|
||||
SECTION(".init_array.0") init_func_t* __INIT_ARRAY__ = (init_func_t*)-1;
|
||||
SECTION(".fini_array.0") fini_func_t* __FINI_ARRAY__ = (fini_func_t*)-1;
|
||||
#undef SECTION
|
||||
|
||||
__used static void _start_main(void* raw_args) {
|
||||
|
|
|
@ -30,10 +30,13 @@
|
|||
|
||||
#include <sys/cdefs.h>
|
||||
|
||||
typedef void init_func_t(int, char*[], char*[]);
|
||||
typedef void fini_func_t(void);
|
||||
|
||||
typedef struct {
|
||||
void (**preinit_array)(void);
|
||||
void (**init_array)(void);
|
||||
void (**fini_array)(void);
|
||||
init_func_t** preinit_array;
|
||||
init_func_t** init_array;
|
||||
fini_func_t** fini_array;
|
||||
} structors_array_t;
|
||||
|
||||
__BEGIN_DECLS
|
||||
|
|
|
@ -60,10 +60,10 @@ __LIBC_HIDDEN__ void* __libc_sysinfo;
|
|||
|
||||
extern "C" int __cxa_atexit(void (*)(void *), void *, void *);
|
||||
|
||||
static void call_array(void(**list)()) {
|
||||
static void call_array(init_func_t** list, int argc, char* argv[], char* envp[]) {
|
||||
// First element is -1, list is null-terminated
|
||||
while (*++list) {
|
||||
(*list)();
|
||||
(*list)(argc, argv, envp);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -183,8 +183,8 @@ __noreturn static void __real_libc_init(void *raw_args,
|
|||
// Several Linux ABIs don't pass the onexit pointer, and the ones that
|
||||
// do never use it. Therefore, we ignore it.
|
||||
|
||||
call_array(structors->preinit_array);
|
||||
call_array(structors->init_array);
|
||||
call_array(structors->preinit_array, args.argc, args.argv, args.envp);
|
||||
call_array(structors->init_array, args.argc, args.argv, args.envp);
|
||||
|
||||
// The executable may have its own destructors listed in its .fini_array
|
||||
// so we need to ensure that these are called when the program exits
|
||||
|
|
|
@ -366,6 +366,7 @@ cc_test_library {
|
|||
"netinet_udp_test.cpp",
|
||||
"nl_types_test.cpp",
|
||||
"poll_test.cpp",
|
||||
"prio_ctor_test.cpp",
|
||||
"pthread_test.cpp",
|
||||
"pty_test.cpp",
|
||||
"regex_test.cpp",
|
||||
|
|
41
tests/prio_ctor_test.cpp
Normal file
41
tests/prio_ctor_test.cpp
Normal file
|
@ -0,0 +1,41 @@
|
|||
/*
|
||||
* Copyright (C) 2020 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 <gtest/gtest.h>
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
static const char* record[4] = {};
|
||||
static int idx = 0;
|
||||
|
||||
__attribute__((constructor(1000))) static void prio1000() {
|
||||
record[idx++] = "prio1000";
|
||||
}
|
||||
|
||||
__attribute__((constructor(1))) static void prio1() {
|
||||
record[idx++] = "prio1";
|
||||
}
|
||||
|
||||
__attribute__((constructor)) static void noprio() {
|
||||
record[idx++] = "noprio";
|
||||
}
|
||||
|
||||
TEST(prio_ctor, order) {
|
||||
EXPECT_EQ(idx, 3);
|
||||
EXPECT_STREQ(record[0], "prio1");
|
||||
EXPECT_STREQ(record[1], "prio1000");
|
||||
EXPECT_STREQ(record[2], "noprio");
|
||||
}
|
Loading…
Reference in a new issue