blob: 49d40509c97b52bacb39a78f1807acc551cb1ac9 [file] [log] [blame]
/*
Copyright (c) 2014-2016 Intel Corporation. All Rights Reserved.
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
#include "offload_table.h"
#ifdef MYO_SUPPORT
#include "offload_myo_host.h"
#endif // MYO_SUPPORT
#else
#include "compiler_if_target.h"
#include "offload_target.h"
#ifdef MYO_SUPPORT
#include "offload_myo_target.h"
#endif // MYO_SUPPORT
#endif // HOST_LIBRARY
// 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);
#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
// first element is empty
ALLOCATE(OFFLOAD_MYO_SHARED_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(SharedTableEntry)))
#endif // TARGET_WINNT
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
};
// offload myo shared var init section prolog
// first element is empty
ALLOCATE(OFFLOAD_MYO_SHARED_INIT_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(InitTableEntry)))
#endif // TARGET_WINNT
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;
}
#endif
static void* __offload_vtable_initializer_thunk_ptr = 0;
// offload myo fptr section prolog
// first element is pre-initialized to the MIC vtable initializer
ALLOCATE(OFFLOAD_MYO_FPTR_TABLE_SECTION_START)
#ifdef TARGET_WINNT
__declspec(align(sizeof(FptrTableEntry)))
#endif // TARGET_WINNT
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
};
#endif // MYO_SUPPORT
// init/fini code which adds/removes local lookup data to/from the global list
static void offload_fini();
static void offload_fini_so();
#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()
{
bool success;
// Set offload version
__offload_set_version(OFFLOAD_VERSION_17);
// register offload tables
__offload_register_tables(&__offload_entry_node,
&__offload_func_node,
&__offload_var_node);
#if HOST_LIBRARY
success = __offload_register_image(&__offload_target_image);
if (!success)
{
return;
}
#endif // HOST_LIBRARY
#ifdef MYO_SUPPORT
#if HOST_LIBRARY
// 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
);
#endif // HOST_LIBRARY
#endif // MYO_SUPPORT
}
#ifndef TARGET_WINNT
static void offload_fini_so() __attribute__((destructor(101)));
#endif // TARGET_WINNT
static void offload_fini()
{
#if HOST_LIBRARY
__offload_unregister_image(&__offload_target_image);
#endif // HOST_LIBRARY
}
static void offload_fini_so()
{
// Offload and MYO tables need to be removed from list
// to prevent invalid accesses after dlclose
// Remove offload tables
__offload_unregister_tables(&__offload_entry_node,
&__offload_func_node,
&__offload_var_node);
#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
}