1 /* 2 * Copyright (C) 2009 Apple Inc. All rights reserved. 3 * 4 * Redistribution and use in source and binary forms, with or without 5 * modification, are permitted provided that the following conditions 6 * are met: 7 * 1. Redistributions of source code must retain the above copyright 8 * notice, this list of conditions and the following disclaimer. 9 * 2. Redistributions in binary form must reproduce the above copyright 10 * notice, this list of conditions and the following disclaimer in the 11 * documentation and/or other materials provided with the distribution. 12 * 13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY 14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR 17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY 21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 24 */ 25 26 #ifndef MacroAssemblerCodeRef_h 27 #define MacroAssemblerCodeRef_h 28 29 #include <wtf/Platform.h> 30 31 #include "ExecutableAllocator.h" 32 #include "PassRefPtr.h" 33 #include "RefPtr.h" 34 #include "UnusedParam.h" 35 36 #if ENABLE(ASSEMBLER) 37 38 // ASSERT_VALID_CODE_POINTER checks that ptr is a non-null pointer, and that it is a valid 39 // instruction address on the platform (for example, check any alignment requirements). 40 #if PLATFORM_ARM_ARCH(7) 41 // ARM/thumb instructions must be 16-bit aligned, but all code pointers to be loaded 42 // into the processor are decorated with the bottom bit set, indicating that this is 43 // thumb code (as oposed to 32-bit traditional ARM). The first test checks for both 44 // decorated and undectorated null, and the second test ensures that the pointer is 45 // decorated. 46 #define ASSERT_VALID_CODE_POINTER(ptr) \ 47 ASSERT(reinterpret_cast<intptr_t>(ptr) & ~1); \ 48 ASSERT(reinterpret_cast<intptr_t>(ptr) & 1) 49 #define ASSERT_VALID_CODE_OFFSET(offset) \ 50 ASSERT(!(offset & 1)) // Must be multiple of 2. 51 #else 52 #define ASSERT_VALID_CODE_POINTER(ptr) \ 53 ASSERT(ptr) 54 #define ASSERT_VALID_CODE_OFFSET(offset) // Anything goes! 55 #endif 56 57 namespace JSC { 58 59 // FunctionPtr: 60 // 61 // FunctionPtr should be used to wrap pointers to C/C++ functions in JSC 62 // (particularly, the stub functions). 63 class FunctionPtr { 64 public: FunctionPtr()65 FunctionPtr() 66 : m_value(0) 67 { 68 } 69 70 template<typename FunctionType> FunctionPtr(FunctionType * value)71 explicit FunctionPtr(FunctionType* value) 72 : m_value(reinterpret_cast<void*>(value)) 73 { 74 ASSERT_VALID_CODE_POINTER(m_value); 75 } 76 value()77 void* value() const { return m_value; } executableAddress()78 void* executableAddress() const { return m_value; } 79 80 81 private: 82 void* m_value; 83 }; 84 85 // ReturnAddressPtr: 86 // 87 // ReturnAddressPtr should be used to wrap return addresses generated by processor 88 // 'call' instructions exectued in JIT code. We use return addresses to look up 89 // exception and optimization information, and to repatch the call instruction 90 // that is the source of the return address. 91 class ReturnAddressPtr { 92 public: ReturnAddressPtr()93 ReturnAddressPtr() 94 : m_value(0) 95 { 96 } 97 ReturnAddressPtr(void * value)98 explicit ReturnAddressPtr(void* value) 99 : m_value(value) 100 { 101 ASSERT_VALID_CODE_POINTER(m_value); 102 } 103 ReturnAddressPtr(FunctionPtr function)104 explicit ReturnAddressPtr(FunctionPtr function) 105 : m_value(function.value()) 106 { 107 ASSERT_VALID_CODE_POINTER(m_value); 108 } 109 value()110 void* value() const { return m_value; } 111 112 private: 113 void* m_value; 114 }; 115 116 // MacroAssemblerCodePtr: 117 // 118 // MacroAssemblerCodePtr should be used to wrap pointers to JIT generated code. 119 class MacroAssemblerCodePtr { 120 public: MacroAssemblerCodePtr()121 MacroAssemblerCodePtr() 122 : m_value(0) 123 { 124 } 125 MacroAssemblerCodePtr(void * value)126 explicit MacroAssemblerCodePtr(void* value) 127 #if PLATFORM_ARM_ARCH(7) 128 // Decorate the pointer as a thumb code pointer. 129 : m_value(reinterpret_cast<char*>(value) + 1) 130 #else 131 : m_value(value) 132 #endif 133 { 134 ASSERT_VALID_CODE_POINTER(m_value); 135 } 136 MacroAssemblerCodePtr(ReturnAddressPtr ra)137 explicit MacroAssemblerCodePtr(ReturnAddressPtr ra) 138 : m_value(ra.value()) 139 { 140 ASSERT_VALID_CODE_POINTER(m_value); 141 } 142 executableAddress()143 void* executableAddress() const { return m_value; } 144 #if PLATFORM_ARM_ARCH(7) 145 // To use this pointer as a data address remove the decoration. dataLocation()146 void* dataLocation() const { ASSERT_VALID_CODE_POINTER(m_value); return reinterpret_cast<char*>(m_value) - 1; } 147 #else dataLocation()148 void* dataLocation() const { ASSERT_VALID_CODE_POINTER(m_value); return m_value; } 149 #endif 150 151 bool operator!() 152 { 153 return !m_value; 154 } 155 156 private: 157 void* m_value; 158 }; 159 160 // MacroAssemblerCodeRef: 161 // 162 // A reference to a section of JIT generated code. A CodeRef consists of a 163 // pointer to the code, and a ref pointer to the pool from within which it 164 // was allocated. 165 class MacroAssemblerCodeRef { 166 public: MacroAssemblerCodeRef()167 MacroAssemblerCodeRef() 168 : m_size(0) 169 { 170 } 171 MacroAssemblerCodeRef(void * code,PassRefPtr<ExecutablePool> executablePool,size_t size)172 MacroAssemblerCodeRef(void* code, PassRefPtr<ExecutablePool> executablePool, size_t size) 173 : m_code(code) 174 , m_executablePool(executablePool) 175 , m_size(size) 176 { 177 } 178 179 MacroAssemblerCodePtr m_code; 180 RefPtr<ExecutablePool> m_executablePool; 181 size_t m_size; 182 }; 183 184 } // namespace JSC 185 186 #endif // ENABLE(ASSEMBLER) 187 188 #endif // MacroAssemblerCodeRef_h 189