1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker * Mesa 3-D graphics library
3*61046927SAndroid Build Coastguard Worker *
4*61046927SAndroid Build Coastguard Worker * Copyright (C) 2010 LunarG Inc.
5*61046927SAndroid Build Coastguard Worker *
6*61046927SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
7*61046927SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
8*61046927SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
9*61046927SAndroid Build Coastguard Worker * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10*61046927SAndroid Build Coastguard Worker * and/or sell copies of the Software, and to permit persons to whom the
11*61046927SAndroid Build Coastguard Worker * Software is furnished to do so, subject to the following conditions:
12*61046927SAndroid Build Coastguard Worker *
13*61046927SAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included
14*61046927SAndroid Build Coastguard Worker * in all copies or substantial portions of the Software.
15*61046927SAndroid Build Coastguard Worker *
16*61046927SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*61046927SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*61046927SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19*61046927SAndroid Build Coastguard Worker * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*61046927SAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*61046927SAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22*61046927SAndroid Build Coastguard Worker * DEALINGS IN THE SOFTWARE.
23*61046927SAndroid Build Coastguard Worker *
24*61046927SAndroid Build Coastguard Worker * Authors:
25*61046927SAndroid Build Coastguard Worker * Chia-I Wu <[email protected]>
26*61046927SAndroid Build Coastguard Worker */
27*61046927SAndroid Build Coastguard Worker
28*61046927SAndroid Build Coastguard Worker #include <string.h>
29*61046927SAndroid Build Coastguard Worker
30*61046927SAndroid Build Coastguard Worker #ifdef __CET__
31*61046927SAndroid Build Coastguard Worker #define ENDBR "endbr32\n\t"
32*61046927SAndroid Build Coastguard Worker #else
33*61046927SAndroid Build Coastguard Worker #define ENDBR
34*61046927SAndroid Build Coastguard Worker #endif
35*61046927SAndroid Build Coastguard Worker
36*61046927SAndroid Build Coastguard Worker #ifdef HAVE_FUNC_ATTRIBUTE_VISIBILITY
37*61046927SAndroid Build Coastguard Worker #define HIDDEN __attribute__((visibility("hidden")))
38*61046927SAndroid Build Coastguard Worker #else
39*61046927SAndroid Build Coastguard Worker #define HIDDEN
40*61046927SAndroid Build Coastguard Worker #endif
41*61046927SAndroid Build Coastguard Worker
42*61046927SAndroid Build Coastguard Worker #define X86_ENTRY_SIZE 32
43*61046927SAndroid Build Coastguard Worker
44*61046927SAndroid Build Coastguard Worker __asm__(".text");
45*61046927SAndroid Build Coastguard Worker
46*61046927SAndroid Build Coastguard Worker __asm__("x86_current_tls:\n\t"
47*61046927SAndroid Build Coastguard Worker "call 1f\n"
48*61046927SAndroid Build Coastguard Worker "1:\n\t"
49*61046927SAndroid Build Coastguard Worker "popl %eax\n\t"
50*61046927SAndroid Build Coastguard Worker "addl $_GLOBAL_OFFSET_TABLE_+[.-1b], %eax\n\t"
51*61046927SAndroid Build Coastguard Worker "movl _glapi_tls_Dispatch@GOTNTPOFF(%eax), %eax\n\t"
52*61046927SAndroid Build Coastguard Worker "ret");
53*61046927SAndroid Build Coastguard Worker
54*61046927SAndroid Build Coastguard Worker #ifndef GLX_X86_READONLY_TEXT
55*61046927SAndroid Build Coastguard Worker __asm__(".section wtext, \"awx\", @progbits");
56*61046927SAndroid Build Coastguard Worker #endif /* GLX_X86_READONLY_TEXT */
57*61046927SAndroid Build Coastguard Worker
58*61046927SAndroid Build Coastguard Worker __asm__(".balign 16\n"
59*61046927SAndroid Build Coastguard Worker "x86_entry_start:");
60*61046927SAndroid Build Coastguard Worker
61*61046927SAndroid Build Coastguard Worker #define STUB_ASM_ENTRY(func) \
62*61046927SAndroid Build Coastguard Worker ".globl " func "\n" \
63*61046927SAndroid Build Coastguard Worker ".type " func ", @function\n" \
64*61046927SAndroid Build Coastguard Worker ".balign 16\n" \
65*61046927SAndroid Build Coastguard Worker func ":"
66*61046927SAndroid Build Coastguard Worker
67*61046927SAndroid Build Coastguard Worker #define STUB_ASM_CODE(slot) \
68*61046927SAndroid Build Coastguard Worker ENDBR \
69*61046927SAndroid Build Coastguard Worker "call 1f\n" \
70*61046927SAndroid Build Coastguard Worker "1:\n\t" \
71*61046927SAndroid Build Coastguard Worker "popl %eax\n\t" \
72*61046927SAndroid Build Coastguard Worker "addl $_GLOBAL_OFFSET_TABLE_+[.-1b], %eax\n\t" \
73*61046927SAndroid Build Coastguard Worker "movl _glapi_tls_Dispatch@GOTNTPOFF(%eax), %eax\n\t" \
74*61046927SAndroid Build Coastguard Worker "movl %gs:(%eax), %eax\n\t" \
75*61046927SAndroid Build Coastguard Worker "jmp *(4 * " slot ")(%eax)"
76*61046927SAndroid Build Coastguard Worker
77*61046927SAndroid Build Coastguard Worker #define MAPI_TMP_STUB_ASM_GCC
78*61046927SAndroid Build Coastguard Worker #include "mapi_tmp.h"
79*61046927SAndroid Build Coastguard Worker
80*61046927SAndroid Build Coastguard Worker #ifndef GLX_X86_READONLY_TEXT
81*61046927SAndroid Build Coastguard Worker __asm__(".balign 16\n"
82*61046927SAndroid Build Coastguard Worker "x86_entry_end:");
83*61046927SAndroid Build Coastguard Worker __asm__(".text");
84*61046927SAndroid Build Coastguard Worker #endif /* GLX_X86_READONLY_TEXT */
85*61046927SAndroid Build Coastguard Worker
86*61046927SAndroid Build Coastguard Worker #ifndef MAPI_MODE_BRIDGE
87*61046927SAndroid Build Coastguard Worker
88*61046927SAndroid Build Coastguard Worker extern unsigned long
89*61046927SAndroid Build Coastguard Worker x86_current_tls();
90*61046927SAndroid Build Coastguard Worker
91*61046927SAndroid Build Coastguard Worker extern char x86_entry_start[] HIDDEN;
92*61046927SAndroid Build Coastguard Worker extern char x86_entry_end[] HIDDEN;
93*61046927SAndroid Build Coastguard Worker
94*61046927SAndroid Build Coastguard Worker static inline mapi_func
95*61046927SAndroid Build Coastguard Worker entry_generate_or_patch(int, char *, size_t);
96*61046927SAndroid Build Coastguard Worker
97*61046927SAndroid Build Coastguard Worker void
entry_patch_public(void)98*61046927SAndroid Build Coastguard Worker entry_patch_public(void)
99*61046927SAndroid Build Coastguard Worker {
100*61046927SAndroid Build Coastguard Worker #ifndef GLX_X86_READONLY_TEXT
101*61046927SAndroid Build Coastguard Worker char *entry;
102*61046927SAndroid Build Coastguard Worker int slot = 0;
103*61046927SAndroid Build Coastguard Worker for (entry = x86_entry_start; entry < x86_entry_end;
104*61046927SAndroid Build Coastguard Worker entry += X86_ENTRY_SIZE, ++slot)
105*61046927SAndroid Build Coastguard Worker entry_generate_or_patch(slot, entry, X86_ENTRY_SIZE);
106*61046927SAndroid Build Coastguard Worker #endif
107*61046927SAndroid Build Coastguard Worker }
108*61046927SAndroid Build Coastguard Worker
109*61046927SAndroid Build Coastguard Worker mapi_func
entry_get_public(int slot)110*61046927SAndroid Build Coastguard Worker entry_get_public(int slot)
111*61046927SAndroid Build Coastguard Worker {
112*61046927SAndroid Build Coastguard Worker return (mapi_func) (x86_entry_start + slot * X86_ENTRY_SIZE);
113*61046927SAndroid Build Coastguard Worker }
114*61046927SAndroid Build Coastguard Worker
115*61046927SAndroid Build Coastguard Worker static void
entry_patch(mapi_func entry,int slot)116*61046927SAndroid Build Coastguard Worker entry_patch(mapi_func entry, int slot)
117*61046927SAndroid Build Coastguard Worker {
118*61046927SAndroid Build Coastguard Worker char *code = (char *) entry;
119*61046927SAndroid Build Coastguard Worker *((unsigned long *) (code + 8)) = slot * sizeof(mapi_func);
120*61046927SAndroid Build Coastguard Worker }
121*61046927SAndroid Build Coastguard Worker
122*61046927SAndroid Build Coastguard Worker static inline mapi_func
entry_generate_or_patch(int slot,char * code,size_t size)123*61046927SAndroid Build Coastguard Worker entry_generate_or_patch(int slot, char *code, size_t size)
124*61046927SAndroid Build Coastguard Worker {
125*61046927SAndroid Build Coastguard Worker const char code_templ[16] = {
126*61046927SAndroid Build Coastguard Worker 0x65, 0xa1, 0x00, 0x00, 0x00, 0x00, /* movl %gs:0x0, %eax */
127*61046927SAndroid Build Coastguard Worker 0xff, 0xa0, 0x34, 0x12, 0x00, 0x00, /* jmp *0x1234(%eax) */
128*61046927SAndroid Build Coastguard Worker 0x90, 0x90, 0x90, 0x90 /* nop's */
129*61046927SAndroid Build Coastguard Worker };
130*61046927SAndroid Build Coastguard Worker mapi_func entry;
131*61046927SAndroid Build Coastguard Worker
132*61046927SAndroid Build Coastguard Worker if (size < sizeof(code_templ))
133*61046927SAndroid Build Coastguard Worker return NULL;
134*61046927SAndroid Build Coastguard Worker
135*61046927SAndroid Build Coastguard Worker memcpy(code, code_templ, sizeof(code_templ));
136*61046927SAndroid Build Coastguard Worker
137*61046927SAndroid Build Coastguard Worker *((unsigned long *) (code + 2)) = x86_current_tls();
138*61046927SAndroid Build Coastguard Worker entry = (mapi_func) code;
139*61046927SAndroid Build Coastguard Worker entry_patch(entry, slot);
140*61046927SAndroid Build Coastguard Worker
141*61046927SAndroid Build Coastguard Worker return entry;
142*61046927SAndroid Build Coastguard Worker }
143*61046927SAndroid Build Coastguard Worker
144*61046927SAndroid Build Coastguard Worker #endif /* MAPI_MODE_BRIDGE */
145