summaryrefslogtreecommitdiff
path: root/liboffloadmic/runtime/ofldbegin.cpp
blob: 236500d011a7396c1fb8c98211444032db850dcd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
/*
    Copyright (c) 2014-2015 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"
#include "offload_myo_host.h"
#else
#include "compiler_if_target.h"
#include "offload_target.h"
#include "offload_myo_target.h"
#endif

// 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;

    // 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)));
#else // TARGET_WINNT
static void offload_init_so();
#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
}