Changeset 189103 in webkit
- Timestamp:
- Aug 28, 2015, 10:52:35 AM (10 years ago)
- Location:
- trunk/Source/JavaScriptCore
- Files:
-
- 1 added
- 4 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/Source/JavaScriptCore/ChangeLog
r189091 r189103 1 2015-08-28 Mark Lam <mark.lam@apple.com> 2 3 ScratchRegisterAllocator::preserveReusedRegistersByPushing() should allow room for C helper calls and keep sp properly aligned. 4 https://bugs.webkit.org/show_bug.cgi?id=148564 5 6 Reviewed by Saam Barati. 7 8 ScratchRegisterAllocator::preserveReusedRegistersByPushing() pushes registers on 9 the stack in order to preserve them. But emitPutTransitionStub() (which uses 10 preserveReusedRegistersByPushing()) may also emit a call to a C helper function 11 to flush the heap write barrier buffer. The code for emitting a C helper call 12 expects the stack pointer (sp) to already be pointing to a location on the stack 13 where there's adequate space reserved for storing the arguments to the C helper, 14 and that space is expected to be at the top of the stack. Hence, there is a 15 conflict of expectations. As a result, the arguments for the C helper will 16 overwrite and corrupt the values that are pushed on the stack by 17 preserveReusedRegistersByPushing(). 18 19 In addition, JIT compiled functions always position the sp such that it will be 20 aligned (according to platform ABI dictates) after a C call is made (i.e. after 21 the frame pointer and return address is pushed on to the stack). 22 preserveReusedRegistersByPushing()'s arbitrary pushing of a number of saved 23 register values may mess up this alignment. 24 25 The fix is to have preserveReusedRegistersByPushing(), after it has pushed the 26 saved register values, adjust the sp to reserve an additional amount of stack 27 space needed for C call helpers plus any padding needed to restore proper sp 28 alignment. The stack's ReservedZone will ensure that we have enough stack space 29 for this. ScratchRegisterAllocator::restoreReusedRegistersByPopping() also 30 needs to be updated to perform the complement of this behavior. 31 32 * jit/Repatch.cpp: 33 (JSC::emitPutReplaceStub): 34 (JSC::emitPutTransitionStub): 35 * jit/ScratchRegisterAllocator.cpp: 36 (JSC::ScratchRegisterAllocator::allocateScratchGPR): 37 (JSC::ScratchRegisterAllocator::allocateScratchFPR): 38 (JSC::ScratchRegisterAllocator::preserveReusedRegistersByPushing): 39 (JSC::ScratchRegisterAllocator::restoreReusedRegistersByPopping): 40 * jit/ScratchRegisterAllocator.h: 41 (JSC::ScratchRegisterAllocator::numberOfReusedRegisters): 42 * tests/stress/regress-148564.js: Added. 43 (test): 44 (runTest): 45 1 46 2015-08-28 Csaba Osztrogonác <ossy@webkit.org> 2 47 -
trunk/Source/JavaScriptCore/jit/Repatch.cpp
r189082 r189103 917 917 CCallHelpers stubJit(vm, exec->codeBlock()); 918 918 919 allocator.preserveReusedRegistersByPushing(stubJit);919 size_t numberOfPaddingBytes = allocator.preserveReusedRegistersByPushing(stubJit); 920 920 921 921 MacroAssembler::Jump badStructure = branchStructure(stubJit, … … 946 946 947 947 if (allocator.didReuseRegisters()) { 948 allocator.restoreReusedRegistersByPopping(stubJit );948 allocator.restoreReusedRegistersByPopping(stubJit, numberOfPaddingBytes); 949 949 success = stubJit.jump(); 950 950 951 951 badStructure.link(&stubJit); 952 allocator.restoreReusedRegistersByPopping(stubJit );952 allocator.restoreReusedRegistersByPopping(stubJit, numberOfPaddingBytes); 953 953 failure = stubJit.jump(); 954 954 } else { … … 1054 1054 scratchGPR3 = InvalidGPRReg; 1055 1055 1056 allocator.preserveReusedRegistersByPushing(stubJit);1056 size_t numberOfPaddingBytes = allocator.preserveReusedRegistersByPushing(stubJit); 1057 1057 1058 1058 MacroAssembler::JumpList failureCases; … … 1170 1170 1171 1171 if (allocator.didReuseRegisters()) { 1172 allocator.restoreReusedRegistersByPopping(stubJit );1172 allocator.restoreReusedRegistersByPopping(stubJit, numberOfPaddingBytes); 1173 1173 success = stubJit.jump(); 1174 1174 1175 1175 failureCases.link(&stubJit); 1176 allocator.restoreReusedRegistersByPopping(stubJit );1176 allocator.restoreReusedRegistersByPopping(stubJit, numberOfPaddingBytes); 1177 1177 failure = stubJit.jump(); 1178 1178 } else … … 1185 1185 slowPath.link(&stubJit); 1186 1186 1187 allocator.restoreReusedRegistersByPopping(stubJit );1187 allocator.restoreReusedRegistersByPopping(stubJit, numberOfPaddingBytes); 1188 1188 if (!scratchBuffer) 1189 1189 scratchBuffer = vm->scratchBufferForSize(allocator.desiredScratchBufferSizeForCall()); -
trunk/Source/JavaScriptCore/jit/ScratchRegisterAllocator.cpp
r167187 r189103 30 30 31 31 #include "JSCInlines.h" 32 #include "MaxFrameExtentForSlowPathCall.h" 32 33 #include "VM.h" 33 34 … … 92 93 FPRReg ScratchRegisterAllocator::allocateScratchFPR() { return allocateScratch<FPRInfo>(); } 93 94 94 voidScratchRegisterAllocator::preserveReusedRegistersByPushing(MacroAssembler& jit)95 size_t ScratchRegisterAllocator::preserveReusedRegistersByPushing(MacroAssembler& jit) 95 96 { 96 97 if (!didReuseRegisters()) 97 return; 98 98 return 0; 99 100 size_t numberOfBytesPushed = 0; 101 99 102 for (unsigned i = 0; i < FPRInfo::numberOfRegisters; ++i) { 100 103 FPRReg reg = FPRInfo::toRegister(i); 101 if (m_scratchRegisters.getFPRByIndex(i) && m_usedRegisters.get(reg)) 104 if (m_scratchRegisters.getFPRByIndex(i) && m_usedRegisters.get(reg)) { 102 105 jit.pushToSave(reg); 106 numberOfBytesPushed += sizeof(double); 107 } 103 108 } 104 109 for (unsigned i = 0; i < GPRInfo::numberOfRegisters; ++i) { 105 110 GPRReg reg = GPRInfo::toRegister(i); 106 if (m_scratchRegisters.getGPRByIndex(i) && m_usedRegisters.get(reg)) 111 if (m_scratchRegisters.getGPRByIndex(i) && m_usedRegisters.get(reg)) { 107 112 jit.pushToSave(reg); 108 } 109 } 110 111 void ScratchRegisterAllocator::restoreReusedRegistersByPopping(MacroAssembler& jit) 113 numberOfBytesPushed += sizeof(uintptr_t); 114 } 115 } 116 117 size_t totalStackAdjustmentBytes = numberOfBytesPushed + maxFrameExtentForSlowPathCall; 118 totalStackAdjustmentBytes = WTF::roundUpToMultipleOf(stackAlignmentBytes(), totalStackAdjustmentBytes); 119 120 size_t numberOfPaddingBytes = totalStackAdjustmentBytes - numberOfBytesPushed; 121 jit.subPtr(MacroAssembler::TrustedImm32(numberOfPaddingBytes), MacroAssembler::stackPointerRegister); 122 123 return numberOfPaddingBytes; 124 } 125 126 void ScratchRegisterAllocator::restoreReusedRegistersByPopping(MacroAssembler& jit, size_t numberOfPaddingBytes) 112 127 { 113 128 if (!didReuseRegisters()) 114 129 return; 115 130 131 jit.addPtr(MacroAssembler::TrustedImm32(numberOfPaddingBytes), MacroAssembler::stackPointerRegister); 132 116 133 for (unsigned i = GPRInfo::numberOfRegisters; i--;) { 117 134 GPRReg reg = GPRInfo::toRegister(i); -
trunk/Source/JavaScriptCore/jit/ScratchRegisterAllocator.h
r165414 r189103 63 63 } 64 64 65 void preserveReusedRegistersByPushing(MacroAssembler& jit); 66 void restoreReusedRegistersByPopping(MacroAssembler& jit); 65 // preserveReusedRegistersByPushing() returns the number of padding bytes used to keep the stack 66 // pointer properly aligned and to reserve room for calling a C helper. This number of padding 67 // bytes must be provided to restoreReusedRegistersByPopping() in order to reverse the work done 68 // by preserveReusedRegistersByPushing(). 69 size_t preserveReusedRegistersByPushing(MacroAssembler& jit); 70 void restoreReusedRegistersByPopping(MacroAssembler& jit, size_t numberOfPaddingBytes); 67 71 68 72 RegisterSet usedRegistersForCall() const;
Note:
See TracChangeset
for help on using the changeset viewer.