summaryrefslogtreecommitdiff
path: root/js/src/methodjit/InlineFrameAssembler.h
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/methodjit/InlineFrameAssembler.h')
-rw-r--r--js/src/methodjit/InlineFrameAssembler.h157
1 files changed, 157 insertions, 0 deletions
diff --git a/js/src/methodjit/InlineFrameAssembler.h b/js/src/methodjit/InlineFrameAssembler.h
new file mode 100644
index 0000000..0e60478
--- /dev/null
+++ b/js/src/methodjit/InlineFrameAssembler.h
@@ -0,0 +1,157 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
+ * vim: set ts=4 sw=4 et tw=99:
+ *
+ * ***** BEGIN LICENSE BLOCK *****
+ * Version: MPL 1.1/GPL 2.0/LGPL 2.1
+ *
+ * The contents of this file are subject to the Mozilla Public License Version
+ * 1.1 (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ * http://www.mozilla.org/MPL/
+ *
+ * Software distributed under the License is distributed on an "AS IS" basis,
+ * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
+ * for the specific language governing rights and limitations under the
+ * License.
+ *
+ * The Original Code is Mozilla SpiderMonkey JavaScript 1.9 code, released
+ * May 28, 2008.
+ *
+ * The Initial Developer of the Original Code is
+ * Brendan Eich <brendan@mozilla.org>
+ *
+ * Contributor(s):
+ * David Anderson <danderson@mozilla.com>
+ *
+ * Alternatively, the contents of this file may be used under the terms of
+ * either of the GNU General Public License Version 2 or later (the "GPL"),
+ * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
+ * in which case the provisions of the GPL or the LGPL are applicable instead
+ * of those above. If you wish to allow use of your version of this file only
+ * under the terms of either the GPL or the LGPL, and not to allow others to
+ * use your version of this file under the terms of the MPL, indicate your
+ * decision by deleting the provisions above and replace them with the notice
+ * and other provisions required by the GPL or the LGPL. If you do not delete
+ * the provisions above, a recipient may use your version of this file under
+ * the terms of any one of the MPL, the GPL or the LGPL.
+ *
+ * ***** END LICENSE BLOCK ***** */
+
+#if !defined jsjaeger_inl_frame_asm_h__ && defined JS_METHODJIT && defined JS_MONOIC
+#define jsjaeger_inl_frame_asm_h__
+
+#include "assembler/assembler/MacroAssembler.h"
+#include "assembler/assembler/CodeLocation.h"
+#include "methodjit/MethodJIT.h"
+#include "CodeGenIncludes.h"
+
+namespace js {
+namespace mjit {
+
+struct AdjustedFrame {
+ AdjustedFrame(uint32 baseOffset)
+ : baseOffset(baseOffset)
+ { }
+
+ uint32 baseOffset;
+
+ JSC::MacroAssembler::Address addrOf(uint32 offset) {
+ return JSC::MacroAssembler::Address(JSFrameReg, baseOffset + offset);
+ }
+};
+
+/*
+ * This is used for emitting code to inline callee-side frame creation and
+ * should jit code equivalent to JSStackFrame::initCallFrameCallerHalf.
+ *
+ * Once finished, JSFrameReg is advanced to be the new fp.
+ */
+class InlineFrameAssembler {
+ typedef JSC::MacroAssembler::RegisterID RegisterID;
+ typedef JSC::MacroAssembler::Address Address;
+ typedef JSC::MacroAssembler::Imm32 Imm32;
+ typedef JSC::MacroAssembler::ImmPtr ImmPtr;
+ typedef JSC::MacroAssembler::DataLabelPtr DataLabelPtr;
+
+ Assembler &masm;
+ FrameSize frameSize; // size of the caller's frame
+ RegisterID funObjReg; // register containing the function object (callee)
+ jsbytecode *pc; // bytecode location at the caller call site
+ uint32 flags; // frame flags
+
+ public:
+ /*
+ * Register state, so consumers of this class can restrict which registers
+ * can and can't be clobbered.
+ */
+ Registers tempRegs;
+
+ InlineFrameAssembler(Assembler &masm, ic::CallICInfo &ic, uint32 flags)
+ : masm(masm), pc(ic.pc), flags(flags)
+ {
+ frameSize = ic.frameSize;
+ funObjReg = ic.funObjReg;
+ tempRegs.takeReg(ic.funPtrReg);
+ tempRegs.takeReg(funObjReg);
+ }
+
+ InlineFrameAssembler(Assembler &masm, Compiler::CallGenInfo &gen, uint32 flags)
+ : masm(masm), pc(gen.pc), flags(flags)
+ {
+ frameSize = gen.frameSize;
+ funObjReg = gen.funObjReg;
+ tempRegs.takeReg(funObjReg);
+ }
+
+ DataLabelPtr assemble(void *ncode)
+ {
+ JS_ASSERT((flags & ~JSFRAME_CONSTRUCTING) == 0);
+
+ /* Generate JSStackFrame::initCallFrameCallerHalf. */
+
+ DataLabelPtr ncodePatch;
+ if (frameSize.isStatic()) {
+ uint32 frameDepth = frameSize.staticLocalSlots();
+ AdjustedFrame newfp(sizeof(JSStackFrame) + frameDepth * sizeof(Value));
+
+ Address flagsAddr = newfp.addrOf(JSStackFrame::offsetOfFlags());
+ masm.store32(Imm32(JSFRAME_FUNCTION | flags), flagsAddr);
+ Address prevAddr = newfp.addrOf(JSStackFrame::offsetOfPrev());
+ masm.storePtr(JSFrameReg, prevAddr);
+ Address ncodeAddr = newfp.addrOf(JSStackFrame::offsetOfncode());
+ ncodePatch = masm.storePtrWithPatch(ImmPtr(ncode), ncodeAddr);
+
+ masm.addPtr(Imm32(sizeof(JSStackFrame) + frameDepth * sizeof(Value)), JSFrameReg);
+ } else {
+ /*
+ * If the frame size is dynamic, then the fast path generated by
+ * generateFullCallStub must be used. Thus, this code is executed
+ * after stubs::SplatApplyArgs has been called. SplatApplyArgs
+ * stores the dynamic stack pointer (i.e., regs.sp after pushing a
+ * dynamic number of arguments) to VMFrame.regs, so we just load it
+ * here to get the new frame pointer.
+ */
+ RegisterID newfp = tempRegs.takeAnyReg();
+ masm.loadPtr(FrameAddress(offsetof(VMFrame, regs.sp)), newfp);
+
+ Address flagsAddr(newfp, JSStackFrame::offsetOfFlags());
+ masm.store32(Imm32(JSFRAME_FUNCTION | flags), flagsAddr);
+ Address prevAddr(newfp, JSStackFrame::offsetOfPrev());
+ masm.storePtr(JSFrameReg, prevAddr);
+ Address ncodeAddr(newfp, JSStackFrame::offsetOfncode());
+ ncodePatch = masm.storePtrWithPatch(ImmPtr(ncode), ncodeAddr);
+
+ masm.move(newfp, JSFrameReg);
+ tempRegs.putReg(newfp);
+ }
+
+ return ncodePatch;
+ }
+};
+
+
+} /* namespace mjit */
+} /* namespace js */
+
+#endif /* jsjaeger_inl_frame_asm_h__ */
+