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