Retro68/gcc/liboffloadmic/runtime/ofldbegin.cpp

333 lines
10 KiB
C++
Raw Normal View History

2015-08-28 15:33:40 +00:00
/*
2017-04-10 11:32:00 +00:00
Copyright (c) 2014-2015 Intel Corporation. All Rights Reserved.
2015-08-28 15:33:40 +00:00
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
* Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
* Neither the name of Intel Corporation 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 THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT
HOLDER OR 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.
*/
#if HOST_LIBRARY
2017-04-10 11:32:00 +00:00
#include "offload_table.h"
2015-08-28 15:33:40 +00:00
#include "offload_myo_host.h"
#else
#include "compiler_if_target.h"
#include "offload_target.h"
#include "offload_myo_target.h"
#endif
2017-04-10 11:32:00 +00:00
// Initializes library and registers specified offload image.
// Don't use this declarations from offload_host.h as offload_table.h
// is used instead of it. Using offload_host.h contradicts with
// STL library compiled with VS2010.
extern "C" bool __offload_register_image(const void* image);
extern "C" void __offload_unregister_image(const void* image);
extern "C" bool __offload_target_image_is_executable(const void *image);
2015-08-28 15:33:40 +00:00
#ifdef TARGET_WINNT
#define ALLOCATE(name) __declspec(allocate(name))
#define DLL_LOCAL
#else // TARGET_WINNT
#define ALLOCATE(name) __attribute__((section(name)))
#define DLL_LOCAL __attribute__((visibility("hidden")))
#endif // TARGET_WINNT
#if HOST_LIBRARY
// the host program/shared library should always have __offload_target_image
// symbol defined. This symbol specifies the beginning of the target program
// image.
extern "C" DLL_LOCAL const void* __offload_target_image;
#else // HOST_LIBRARY
// Define a weak main which would be used on target side in case usere's
// source file containing main does not have offload code.
#pragma weak main
int main(void)
{
OFFLOAD_TARGET_MAIN();
return 0;
}
#pragma weak MAIN__
extern "C" int MAIN__(void)
{
OFFLOAD_TARGET_MAIN();
return 0;
}
#endif // HOST_LIBRARY
// offload section prolog
ALLOCATE(OFFLOAD_ENTRY_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(FuncTable::Entry)))
#endif // TARGET_WINNT
static FuncTable::Entry __offload_entry_table_start = { 0 };
// list element for the current module
static FuncList::Node __offload_entry_node = {
{ &__offload_entry_table_start + 1, -1 },
0, 0
};
// offload fp section prolog
ALLOCATE(OFFLOAD_FUNC_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(FuncTable::Entry)))
#endif // TARGET_WINNT
static FuncTable::Entry __offload_func_table_start = { 0 };
// list element for the current module
static FuncList::Node __offload_func_node = {
{ &__offload_func_table_start + 1, -1 },
0, 0
};
// offload fp section prolog
ALLOCATE(OFFLOAD_VAR_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(VarTable::Entry)))
#endif // TARGET_WINNT
static VarTable::Entry __offload_var_table_start = { 0 };
// list element for the current module
static VarList::Node __offload_var_node = {
{ &__offload_var_table_start + 1 },
0, 0
};
#ifdef MYO_SUPPORT
// offload myo shared var section prolog
2017-04-10 11:32:00 +00:00
// first element is empty
2015-08-28 15:33:40 +00:00
ALLOCATE(OFFLOAD_MYO_SHARED_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(SharedTableEntry)))
#endif // TARGET_WINNT
2017-04-10 11:32:00 +00:00
static MYOVarTable::Entry __offload_myo_shared_var_start = { 0 };
// list element for the current module
// table entry pointer skips the empty first entry
static MYOVarTableList::Node __offload_myo_shared_var_node = {
{ &__offload_myo_shared_var_start + 1 },
0, 0
};
// offload myo shared vtable section prolog
// first element is empty
ALLOCATE(OFFLOAD_MYO_SHARED_VTABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(SharedTableEntry)))
#endif // TARGET_WINNT
static MYOVarTable::Entry __offload_myo_shared_vtable_start = { 0 };
// list element for the current module
// table entry pointer skips the empty first entry
static MYOVarTableList::Node __offload_myo_shared_vtable_node = {
{ &__offload_myo_shared_vtable_start + 1 },
0, 0
};
2015-08-28 15:33:40 +00:00
// offload myo shared var init section prolog
2017-04-10 11:32:00 +00:00
// first element is empty
2015-08-28 15:33:40 +00:00
ALLOCATE(OFFLOAD_MYO_SHARED_INIT_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(InitTableEntry)))
#endif // TARGET_WINNT
2017-04-10 11:32:00 +00:00
static MYOInitTable::Entry __offload_myo_init_table_start = { 0 };
// list element for the current module
// table entry pointer skips the empty first entry
static MYOInitTableList::Node __offload_myo_init_table_node = {
{ &__offload_myo_init_table_start + 1 },
0, 0
};
// The functions and variables needed for a built-in
// remote function entry for vtable initialization on MIC
#if !HOST_LIBRARY
MyoError __offload_init_vtables(void)
{
SharedTableEntry *t_start;
//OFFLOAD_DEBUG_TRACE(3, "%s\n", __func__);
t_start = &__offload_myo_shared_vtable_start + 1;
//OFFLOAD_DEBUG_TRACE(3, "%s(%p)\n", __func__, t_start);
while (t_start->varName != 0) {
//OFFLOAD_DEBUG_TRACE(4,
// "myo shared vtable \"%s\" &myo_ptr = %p myo_ptr = %p\n",
// t_start->varName,
// (void *)(t_start->sharedAddr),
// ((void **)(t_start->sharedAddr))[0]);
t_start++;
}
__offload_myo_shared_init_table_process(
&__offload_myo_init_table_start + 1);
return MYO_SUCCESS;
}
#endif // !HOST_LIBRARY
static void vtable_initializer()
{
}
#if !HOST_LIBRARY
static MyoError vtable_initializer_wrapper()
{
__offload_myoAcquire();
__offload_init_vtables();
__offload_myoRelease();
return MYO_SUCCESS;
}
2015-08-28 15:33:40 +00:00
#endif
2017-04-10 11:32:00 +00:00
static void* __offload_vtable_initializer_thunk_ptr = 0;
2015-08-28 15:33:40 +00:00
// offload myo fptr section prolog
2017-04-10 11:32:00 +00:00
// first element is pre-initialized to the MIC vtable initializer
2015-08-28 15:33:40 +00:00
ALLOCATE(OFFLOAD_MYO_FPTR_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(FptrTableEntry)))
#endif // TARGET_WINNT
2017-04-10 11:32:00 +00:00
static MYOFuncTable::Entry __offload_myo_fptr_table_start = {
#if HOST_LIBRARY
"--vtable_initializer--",
(void*)&vtable_initializer,
(void*)&__offload_vtable_initializer_thunk_ptr,
#ifdef TARGET_WINNT
// Dummy to pad up to 32 bytes
0
#endif // TARGET_WINNT
#else // HOST_LIBRARY
"--vtable_initializer--",
(void*)&vtable_initializer,
(void*)&vtable_initializer_wrapper,
&__offload_vtable_initializer_thunk_ptr,
#endif // HOST_LIBRARY
};
// list element for the current module
static MYOFuncTableList::Node __offload_myo_fptr_table_node = {
{ &__offload_myo_fptr_table_start },
0, 0
};
2015-08-28 15:33:40 +00:00
#endif // MYO_SUPPORT
// init/fini code which adds/removes local lookup data to/from the global list
static void offload_fini();
2017-04-10 11:32:00 +00:00
static void offload_fini_so();
2015-08-28 15:33:40 +00:00
#ifndef TARGET_WINNT
static void offload_init() __attribute__((constructor(101)));
#else // TARGET_WINNT
static void offload_init();
// Place offload initialization before user constructors
ALLOCATE(OFFLOAD_CRTINIT_SECTION_START)
static void (*addressof_offload_init)() = offload_init;
#endif // TARGET_WINNT
static void offload_init()
{
2017-04-10 11:32:00 +00:00
bool success;
2015-08-28 15:33:40 +00:00
// register offload tables
__offload_register_tables(&__offload_entry_node,
&__offload_func_node,
&__offload_var_node);
#if HOST_LIBRARY
2017-04-10 11:32:00 +00:00
success = __offload_register_image(&__offload_target_image);
if (!success)
{
return;
}
2015-08-28 15:33:40 +00:00
#endif // HOST_LIBRARY
#ifdef MYO_SUPPORT
#if HOST_LIBRARY
2017-04-10 11:32:00 +00:00
// If this was the main program register main atexit routine
if (__offload_myoProcessTables(
&__offload_target_image,
&__offload_myo_init_table_node,
&__offload_myo_shared_var_node,
&__offload_myo_shared_vtable_node,
&__offload_myo_fptr_table_node))
{
atexit(offload_fini);
#ifdef TARGET_WINNT
} else {
atexit(offload_fini_so);
#endif
}
#else // HOST_LIBRARY
__offload_myoProcessTables(
&__offload_myo_init_table_start + 1,
&__offload_myo_shared_var_start + 1,
&__offload_myo_shared_vtable_start + 1,
&__offload_myo_fptr_table_start
2015-08-28 15:33:40 +00:00
);
2017-04-10 11:32:00 +00:00
#endif // HOST_LIBRARY
2015-08-28 15:33:40 +00:00
#endif // MYO_SUPPORT
}
2017-04-10 11:32:00 +00:00
#ifndef TARGET_WINNT
static void offload_fini_so() __attribute__((destructor(101)));
#else // TARGET_WINNT
static void offload_init_so();
#endif // TARGET_WINNT
2015-08-28 15:33:40 +00:00
static void offload_fini()
{
#if HOST_LIBRARY
__offload_unregister_image(&__offload_target_image);
#endif // HOST_LIBRARY
2017-04-10 11:32:00 +00:00
}
2015-08-28 15:33:40 +00:00
2017-04-10 11:32:00 +00:00
static void offload_fini_so()
{
// Offload and MYO tables need to be removed from list
// to prevent invalid accesses after dlclose
// Remove offload tables
2015-08-28 15:33:40 +00:00
__offload_unregister_tables(&__offload_entry_node,
&__offload_func_node,
&__offload_var_node);
2017-04-10 11:32:00 +00:00
#if HOST_LIBRARY
if(!__offload_target_image_is_executable(&__offload_target_image)) {
__offload_unregister_image(&__offload_target_image);
}
#endif
#ifdef MYO_SUPPORT
#if HOST_LIBRARY
// Remove MYO tables
__offload_myoRemoveTables(
&__offload_myo_init_table_node,
&__offload_myo_shared_var_node,
&__offload_myo_shared_vtable_node,
&__offload_myo_fptr_table_node);
#endif // HOST_LIBRARY
#endif // MYO_SUPPORT
2015-08-28 15:33:40 +00:00
}