xref: /aosp_15_r20/external/mesa3d/src/mapi/entry_x86_tls.h (revision 6104692788411f58d303aa86923a9ff6ecaded22)
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