1848b8605Smrg/* 2848b8605Smrg * Mesa 3-D graphics library 3848b8605Smrg * 4848b8605Smrg * Copyright (C) 2010 LunarG Inc. 5848b8605Smrg * 6848b8605Smrg * Permission is hereby granted, free of charge, to any person obtaining a 7848b8605Smrg * copy of this software and associated documentation files (the "Software"), 8848b8605Smrg * to deal in the Software without restriction, including without limitation 9848b8605Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 10848b8605Smrg * and/or sell copies of the Software, and to permit persons to whom the 11848b8605Smrg * Software is furnished to do so, subject to the following conditions: 12848b8605Smrg * 13848b8605Smrg * The above copyright notice and this permission notice shall be included 14848b8605Smrg * in all copies or substantial portions of the Software. 15848b8605Smrg * 16848b8605Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 17848b8605Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 18848b8605Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 19848b8605Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 20848b8605Smrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 21848b8605Smrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 22848b8605Smrg * DEALINGS IN THE SOFTWARE. 23848b8605Smrg * 24848b8605Smrg * Authors: 25848b8605Smrg * Chia-I Wu <olv@lunarg.com> 26848b8605Smrg */ 27848b8605Smrg 28b8e80941Smrg#ifdef HAVE_FUNC_ATTRIBUTE_VISIBILITY 29b8e80941Smrg#define HIDDEN __attribute__((visibility("hidden"))) 30b8e80941Smrg#else 31b8e80941Smrg#define HIDDEN 32b8e80941Smrg#endif 33848b8605Smrg 34848b8605Smrg__asm__(".text\n" 35b8e80941Smrg ".balign 64\n" 36848b8605Smrg "x86_64_entry_start:"); 37848b8605Smrg 38848b8605Smrg#define STUB_ASM_ENTRY(func) \ 39848b8605Smrg ".globl " func "\n" \ 40848b8605Smrg ".type " func ", @function\n" \ 41b8e80941Smrg ".balign 64\n" \ 42848b8605Smrg func ":" 43848b8605Smrg 44b8e80941Smrg#ifndef __ILP32__ 45b8e80941Smrg 46b8e80941Smrg#if defined(__NetBSD__) 47b8e80941Smrg#define STUB_ASM_CODE(slot) \ 48b8e80941Smrg "movq " ENTRY_CURRENT_TABLE "@GOTTPOFF(%rip), %rax\n\t" \ 49b8e80941Smrg "movq %fs:(%rax), %r11\n\t" \ 50b8e80941Smrg "testq %r11, %r11\n\t" \ 51b8e80941Smrg "je 1f\n\t" \ 52b8e80941Smrg "jmp *(8 * " slot ")(%r11)\n\t" \ 53b8e80941Smrg "1:\n\t" \ 54b8e80941Smrg "callq " ENTRY_CURRENT_TABLE_GET "@PLT\n\t" \ 55b8e80941Smrg "jmp *(8 * " slot ")(%rax)" 56b8e80941Smrg#else 57848b8605Smrg#define STUB_ASM_CODE(slot) \ 58848b8605Smrg "movq " ENTRY_CURRENT_TABLE "@GOTTPOFF(%rip), %rax\n\t" \ 59848b8605Smrg "movq %fs:(%rax), %r11\n\t" \ 60848b8605Smrg "jmp *(8 * " slot ")(%r11)" 61b8e80941Smrg#endif 62b8e80941Smrg 63b8e80941Smrg#else 64b8e80941Smrg 65b8e80941Smrg#define STUB_ASM_CODE(slot) \ 66b8e80941Smrg "movq " ENTRY_CURRENT_TABLE "@GOTTPOFF(%rip), %rax\n\t" \ 67b8e80941Smrg "movl %fs:(%rax), %r11d\n\t" \ 68b8e80941Smrg "movl 4*" slot "(%r11d), %r11d\n\t" \ 69b8e80941Smrg "jmp *%r11" 70b8e80941Smrg 71b8e80941Smrg#endif 72848b8605Smrg 73848b8605Smrg#define MAPI_TMP_STUB_ASM_GCC 74848b8605Smrg#include "mapi_tmp.h" 75848b8605Smrg 76848b8605Smrg#ifndef MAPI_MODE_BRIDGE 77848b8605Smrg 78848b8605Smrg#include <string.h> 79848b8605Smrg#include "u_execmem.h" 80848b8605Smrg 81848b8605Smrgvoid 82848b8605Smrgentry_patch_public(void) 83848b8605Smrg{ 84848b8605Smrg} 85848b8605Smrg 86b8e80941Smrgextern char 87b8e80941Smrgx86_64_entry_start[] HIDDEN; 88848b8605Smrg 89848b8605Smrgmapi_func 90848b8605Smrgentry_get_public(int slot) 91848b8605Smrg{ 92b8e80941Smrg return (mapi_func) (x86_64_entry_start + slot * 64); 93848b8605Smrg} 94848b8605Smrg 95848b8605Smrgvoid 96848b8605Smrgentry_patch(mapi_func entry, int slot) 97848b8605Smrg{ 98848b8605Smrg char *code = (char *) entry; 99b8e80941Smrg int offset = 12; 100b8e80941Smrg#ifdef __ILP32__ 101b8e80941Smrg offset = 13; 102b8e80941Smrg#endif 103b8e80941Smrg *((unsigned int *) (code + offset)) = slot * sizeof(mapi_func); 104848b8605Smrg} 105848b8605Smrg 106848b8605Smrgmapi_func 107848b8605Smrgentry_generate(int slot) 108848b8605Smrg{ 109b8e80941Smrg const char code_templ[] = { 110b8e80941Smrg#ifndef __ILP32__ 111848b8605Smrg /* movq %fs:0, %r11 */ 112848b8605Smrg 0x64, 0x4c, 0x8b, 0x1c, 0x25, 0x00, 0x00, 0x00, 0x00, 113848b8605Smrg /* jmp *0x1234(%r11) */ 114848b8605Smrg 0x41, 0xff, 0xa3, 0x34, 0x12, 0x00, 0x00, 115b8e80941Smrg#else 116b8e80941Smrg /* movl %fs:0, %r11d */ 117b8e80941Smrg 0x64, 0x44, 0x8b, 0x1c, 0x25, 0x00, 0x00, 0x00, 0x00, 118b8e80941Smrg /* movl 0x1234(%r11d), %r11d */ 119b8e80941Smrg 0x67, 0x45, 0x8b, 0x9b, 0x34, 0x12, 0x00, 0x00, 120b8e80941Smrg /* jmp *%r11 */ 121b8e80941Smrg 0x41, 0xff, 0xe3, 122b8e80941Smrg#endif 123848b8605Smrg }; 124b8e80941Smrg unsigned long long addr; 125b8e80941Smrg char *code; 126848b8605Smrg mapi_func entry; 127848b8605Smrg 128b8e80941Smrg __asm__("movq " ENTRY_CURRENT_TABLE "@GOTTPOFF(%%rip), %0" 129b8e80941Smrg : "=r" (addr)); 130848b8605Smrg if ((addr >> 32) != 0xffffffff) 131848b8605Smrg return NULL; 132848b8605Smrg addr &= 0xffffffff; 133848b8605Smrg 134848b8605Smrg code = u_execmem_alloc(sizeof(code_templ)); 135848b8605Smrg if (!code) 136848b8605Smrg return NULL; 137848b8605Smrg 138848b8605Smrg memcpy(code, code_templ, sizeof(code_templ)); 139848b8605Smrg 140848b8605Smrg *((unsigned int *) (code + 5)) = addr; 141848b8605Smrg entry = (mapi_func) code; 142848b8605Smrg entry_patch(entry, slot); 143848b8605Smrg 144848b8605Smrg return entry; 145848b8605Smrg} 146848b8605Smrg 147848b8605Smrg#endif /* MAPI_MODE_BRIDGE */ 148