1 // Copyright (c) 2012, Google Inc. 2 // 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 are 6 // met: 7 // 8 // * Redistributions of source code must retain the above copyright 9 // notice, this list of conditions and the following disclaimer. 10 // * Redistributions in binary form must reproduce the above 11 // copyright notice, this list of conditions and the following disclaimer 12 // in the documentation and/or other materials provided with the 13 // distribution. 14 // * Neither the name of Google Inc. nor the names of its 15 // contributors may be used to endorse or promote products derived from 16 // this software without specific prior written permission. 17 // 18 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 19 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 20 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 21 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 22 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 23 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 24 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 25 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 26 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 27 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 28 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 29 30 #if defined(__x86_64__) 31 #include <asm/sigcontext.h> 32 #endif 33 34 #include <sys/ucontext.h> 35 36 #include "breakpad_googletest_includes.h" 37 #include "common/android/ucontext_constants.h" 38 39 template <int left, int right> 40 struct CompileAssertEquals { 41 // a compilation error here indicates left and right are not equal. 42 char left_too_large[right - left]; 43 // a compilation error here indicates left and right are not equal. 44 char right_too_large[left - right]; 45 }; 46 47 #define COMPILE_ASSERT_EQ(left, right, tag) \ 48 CompileAssertEquals<left, right> tag; 49 50 TEST(AndroidUContext, GRegsOffset) { 51 #if defined(__arm__) 52 // There is no gregs[] array on ARM, so compare to the offset of 53 // first register fields, since they're stored in order. 54 ASSERT_EQ(static_cast<size_t>(MCONTEXT_GREGS_OFFSET), 55 offsetof(ucontext_t,uc_mcontext.arm_r0)); 56 #elif defined(__aarch64__) 57 // There is no gregs[] array on ARM, so compare to the offset of 58 // first register fields, since they're stored in order. 59 ASSERT_EQ(static_cast<size_t>(MCONTEXT_GREGS_OFFSET), 60 offsetof(ucontext_t,uc_mcontext.regs[0])); 61 ASSERT_EQ(static_cast<size_t>(MCONTEXT_SP_OFFSET), 62 offsetof(ucontext_t,uc_mcontext.sp)); 63 ASSERT_EQ(static_cast<size_t>(MCONTEXT_PC_OFFSET), 64 offsetof(ucontext_t,uc_mcontext.pc)); 65 ASSERT_EQ(static_cast<size_t>(MCONTEXT_PSTATE_OFFSET), 66 offsetof(ucontext_t,uc_mcontext.pstate)); 67 ASSERT_EQ(static_cast<size_t>(MCONTEXT_EXTENSION_OFFSET), 68 offsetof(ucontext_t,uc_mcontext.__reserved)); 69 #elif defined(__i386__) 70 ASSERT_EQ(static_cast<size_t>(MCONTEXT_GREGS_OFFSET), 71 offsetof(ucontext_t,uc_mcontext.gregs)); 72 #define CHECK_REG(x) \ 73 ASSERT_EQ(static_cast<size_t>(MCONTEXT_##x##_OFFSET), \ 74 offsetof(ucontext_t,uc_mcontext.gregs[REG_##x])) 75 CHECK_REG(GS); 76 CHECK_REG(FS); 77 CHECK_REG(ES); 78 CHECK_REG(DS); 79 CHECK_REG(EDI); 80 CHECK_REG(ESI); 81 CHECK_REG(EBP); 82 CHECK_REG(ESP); 83 CHECK_REG(EBX); 84 CHECK_REG(EDX); 85 CHECK_REG(ECX); 86 CHECK_REG(EAX); 87 CHECK_REG(TRAPNO); 88 CHECK_REG(ERR); 89 CHECK_REG(EIP); 90 CHECK_REG(CS); 91 CHECK_REG(EFL); 92 CHECK_REG(UESP); 93 CHECK_REG(SS); 94 95 ASSERT_EQ(static_cast<size_t>(UCONTEXT_FPREGS_OFFSET), 96 offsetof(ucontext_t,uc_mcontext.fpregs)); 97 98 ASSERT_EQ(static_cast<size_t>(UCONTEXT_FPREGS_MEM_OFFSET), 99 offsetof(ucontext_t,__fpregs_mem)); 100 #elif defined(__mips__) 101 ASSERT_EQ(static_cast<size_t>(MCONTEXT_GREGS_OFFSET), 102 offsetof(ucontext_t,uc_mcontext.gregs)); 103 104 // PC for mips is not part of gregs. 105 ASSERT_EQ(static_cast<size_t>(MCONTEXT_PC_OFFSET), 106 offsetof(ucontext_t,uc_mcontext.pc)); 107 108 ASSERT_EQ(static_cast<size_t>(MCONTEXT_FPREGS_OFFSET), 109 offsetof(ucontext_t,uc_mcontext.fpregs)); 110 111 ASSERT_EQ(static_cast<size_t>(MCONTEXT_FPC_CSR), 112 offsetof(ucontext_t,uc_mcontext.fpc_csr)); 113 #elif defined(__x86_64__) 114 115 COMPILE_ASSERT_EQ(static_cast<size_t>(MCONTEXT_GREGS_OFFSET), 116 offsetof(ucontext_t,uc_mcontext.gregs), 117 mcontext_gregs_offset); 118 #define CHECK_REG(x) \ 119 COMPILE_ASSERT_EQ(static_cast<size_t>(MCONTEXT_GREGS_##x), \ 120 offsetof(ucontext_t,uc_mcontext.gregs[REG_##x]), reg_##x) 121 CHECK_REG(R8); 122 CHECK_REG(R9); 123 CHECK_REG(R10); 124 CHECK_REG(R11); 125 CHECK_REG(R12); 126 CHECK_REG(R13); 127 CHECK_REG(R14); 128 CHECK_REG(R15); 129 CHECK_REG(RDI); 130 CHECK_REG(RSI); 131 CHECK_REG(RBP); 132 CHECK_REG(RBX); 133 CHECK_REG(RDX); 134 CHECK_REG(RAX); 135 CHECK_REG(RCX); 136 CHECK_REG(RSP); 137 CHECK_REG(RIP); 138 139 // sigcontext is an analog to mcontext_t. The layout should be the same. 140 COMPILE_ASSERT_EQ(offsetof(mcontext_t,fpregs), 141 offsetof(sigcontext,fpstate), sigcontext_fpstate); 142 // Check that _fpstate from asm/sigcontext.h is essentially the same 143 // as _libc_fpstate. 144 COMPILE_ASSERT_EQ(sizeof(_libc_fpstate), sizeof(_fpstate), 145 sigcontext_fpstate_size); 146 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,cwd),offsetof(_fpstate,cwd), 147 sigcontext_fpstate_cwd); 148 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,swd),offsetof(_fpstate,swd), 149 sigcontext_fpstate_swd); 150 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,ftw),offsetof(_fpstate,twd), 151 sigcontext_fpstate_twd); 152 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,fop),offsetof(_fpstate,fop), 153 sigcontext_fpstate_fop); 154 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,rip),offsetof(_fpstate,rip), 155 sigcontext_fpstate_rip); 156 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,rdp),offsetof(_fpstate,rdp), 157 sigcontext_fpstate_rdp); 158 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,mxcsr),offsetof(_fpstate,mxcsr), 159 sigcontext_fpstate_mxcsr); 160 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,mxcr_mask), 161 offsetof(_fpstate,mxcsr_mask), 162 sigcontext_fpstate_mxcsr_mask); 163 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,_st), offsetof(_fpstate,st_space), 164 sigcontext_fpstate_stspace); 165 COMPILE_ASSERT_EQ(offsetof(_libc_fpstate,_xmm), offsetof(_fpstate,xmm_space), 166 sigcontext_fpstate_xmm_space); 167 168 COMPILE_ASSERT_EQ(MCONTEXT_FPREGS_PTR, 169 offsetof(ucontext_t,uc_mcontext.fpregs), 170 mcontext_fpregs_ptr); 171 COMPILE_ASSERT_EQ(MCONTEXT_FPREGS_MEM, offsetof(ucontext_t,__fpregs_mem), 172 mcontext_fpregs_mem); 173 COMPILE_ASSERT_EQ(FPREGS_OFFSET_MXCSR, offsetof(_libc_fpstate,mxcsr), 174 fpregs_offset_mxcsr); 175 COMPILE_ASSERT_EQ(UCONTEXT_SIGMASK_OFFSET, offsetof(ucontext_t, uc_sigmask), 176 ucontext_sigmask); 177 #else 178 ASSERT_EQ(static_cast<size_t>(MCONTEXT_GREGS_OFFSET), 179 offsetof(ucontext_t,uc_mcontext.gregs)); 180 #endif 181 } 182 183 TEST(AndroidUContext, SigmakOffset) { 184 ASSERT_EQ(static_cast<size_t>(UCONTEXT_SIGMASK_OFFSET), 185 offsetof(ucontext_t,uc_sigmask)); 186 } 187