|  | /* | 
|  | 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 | 
|  | } |