| OLD | NEW |
| 1 // Copyright 2006-2008 the V8 project authors. All rights reserved. | 1 // Copyright 2006-2008 the V8 project authors. All rights reserved. |
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
| 4 // met: | 4 // met: |
| 5 // | 5 // |
| 6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
| 7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
| 8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
| 9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
| 10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
| (...skipping 18 matching lines...) Expand all Loading... |
| 29 | 29 |
| 30 #include "frames-inl.h" | 30 #include "frames-inl.h" |
| 31 #include "assembler-arm-inl.h" | 31 #include "assembler-arm-inl.h" |
| 32 | 32 |
| 33 | 33 |
| 34 namespace v8 { namespace internal { | 34 namespace v8 { namespace internal { |
| 35 | 35 |
| 36 | 36 |
| 37 StackFrame::Type StackFrame::ComputeType(State* state) { | 37 StackFrame::Type StackFrame::ComputeType(State* state) { |
| 38 ASSERT(state->fp != NULL); | 38 ASSERT(state->fp != NULL); |
| 39 if (state->pp == NULL) { | 39 if (StandardFrame::IsArgumentsAdaptorFrame(state->fp)) { |
| 40 if (Memory::Address_at(state->fp + | |
| 41 EntryFrameConstants::kConstructMarkOffset) != 0) { | |
| 42 return ENTRY_CONSTRUCT; | |
| 43 } else { | |
| 44 return ENTRY; | |
| 45 } | |
| 46 } else if (StandardFrame::IsArgumentsAdaptorFrame(state->fp)) { | |
| 47 return ARGUMENTS_ADAPTOR; | 40 return ARGUMENTS_ADAPTOR; |
| 48 } else if ( | |
| 49 Memory::Object_at(state->fp + | |
| 50 StandardFrameConstants::kFunctionOffset)->IsSmi()) { | |
| 51 return INTERNAL; | |
| 52 } else { | |
| 53 return JAVA_SCRIPT; | |
| 54 } | 41 } |
| 42 // The marker and function offsets overlap. If the marker isn't a |
| 43 // smi then the frame is a JavaScript frame -- and the marker is |
| 44 // really the function. |
| 45 const int offset = StandardFrameConstants::kMarkerOffset; |
| 46 Object* marker = Memory::Object_at(state->fp + offset); |
| 47 if (!marker->IsSmi()) return JAVA_SCRIPT; |
| 48 return static_cast<StackFrame::Type>(Smi::cast(marker)->value()); |
| 55 } | 49 } |
| 56 | 50 |
| 57 | 51 |
| 58 StackFrame::Type ExitFrame::GetStateForFramePointer(Address fp, State* state) { | 52 StackFrame::Type ExitFrame::GetStateForFramePointer(Address fp, State* state) { |
| 59 if (fp == 0) return NONE; | 53 if (fp == 0) return NONE; |
| 60 // Compute frame type and stack pointer. | 54 // Compute frame type and stack pointer. |
| 61 Address sp = fp + ExitFrameConstants::kSPDisplacement; | 55 Address sp = fp + ExitFrameConstants::kSPDisplacement; |
| 62 Type type; | 56 Type type; |
| 63 if (Memory::Address_at(fp + ExitFrameConstants::kDebugMarkOffset) != 0) { | 57 if (Memory::Address_at(fp + ExitFrameConstants::kDebugMarkOffset) != 0) { |
| 64 type = EXIT_DEBUG; | 58 type = EXIT_DEBUG; |
| 65 sp -= kNumJSCallerSaved * kPointerSize; | 59 sp -= kNumJSCallerSaved * kPointerSize; |
| 66 } else { | 60 } else { |
| 67 type = EXIT; | 61 type = EXIT; |
| 68 } | 62 } |
| 69 // Fill in the state. | 63 // Fill in the state. |
| 70 state->sp = sp; | 64 state->sp = sp; |
| 71 state->fp = fp; | 65 state->fp = fp; |
| 72 state->pp = fp + ExitFrameConstants::kPPDisplacement; | |
| 73 state->pc_address = reinterpret_cast<Address*>(sp - 1 * kPointerSize); | 66 state->pc_address = reinterpret_cast<Address*>(sp - 1 * kPointerSize); |
| 74 return type; | 67 return type; |
| 75 } | 68 } |
| 76 | 69 |
| 77 | 70 |
| 78 void ExitFrame::Iterate(ObjectVisitor* v) const { | 71 void ExitFrame::Iterate(ObjectVisitor* v) const { |
| 79 // Do nothing | 72 // Do nothing |
| 80 } | 73 } |
| 81 | 74 |
| 82 | 75 |
| 83 int JavaScriptFrame::GetProvidedParametersCount() const { | 76 int JavaScriptFrame::GetProvidedParametersCount() const { |
| 84 const int offset = JavaScriptFrameConstants::kArgsLengthOffset; | 77 return ComputeParametersCount(); |
| 85 int result = Memory::int_at(fp() + offset); | |
| 86 // We never remove extra parameters provided on the stack; we only | |
| 87 // fill in undefined values for parameters not provided. | |
| 88 ASSERT(0 <= result && result <= ComputeParametersCount()); | |
| 89 return result; | |
| 90 } | 78 } |
| 91 | 79 |
| 92 | 80 |
| 93 Address JavaScriptFrame::GetCallerStackPointer() const { | 81 Address JavaScriptFrame::GetCallerStackPointer() const { |
| 94 return state_.pp; | 82 int arguments; |
| 83 if (Heap::gc_state() != Heap::NOT_IN_GC) { |
| 84 // The arguments for cooked frames are traversed as if they were |
| 85 // expression stack elements of the calling frame. The reason for |
| 86 // this rather strange decision is that we cannot access the |
| 87 // function during mark-compact GCs when the stack is cooked. |
| 88 // In fact accessing heap objects (like function->shared() below) |
| 89 // at all during GC is problematic. |
| 90 arguments = 0; |
| 91 } else { |
| 92 // Compute the number of arguments by getting the number of formal |
| 93 // parameters of the function. We must remember to take the |
| 94 // receiver into account (+1). |
| 95 JSFunction* function = JSFunction::cast(this->function()); |
| 96 arguments = function->shared()->formal_parameter_count() + 1; |
| 97 } |
| 98 const int offset = StandardFrameConstants::kCallerSPOffset; |
| 99 return fp() + offset + (arguments * kPointerSize); |
| 95 } | 100 } |
| 96 | 101 |
| 97 | 102 |
| 98 Address ArgumentsAdaptorFrame::GetCallerStackPointer() const { | 103 Address ArgumentsAdaptorFrame::GetCallerStackPointer() const { |
| 99 // Argument adaptor frames aren't used on ARM (yet). | 104 const int arguments = Smi::cast(GetExpression(0))->value(); |
| 100 UNIMPLEMENTED(); | 105 const int offset = StandardFrameConstants::kCallerSPOffset; |
| 101 return 0; | 106 return fp() + offset + (arguments + 1) * kPointerSize; |
| 102 } | 107 } |
| 103 | 108 |
| 104 | 109 |
| 105 Address InternalFrame::GetCallerStackPointer() const { | 110 Address InternalFrame::GetCallerStackPointer() const { |
| 106 return state_.pp; | 111 // Internal frames have no arguments. The stack pointer of the |
| 112 // caller is at a fixed offset from the frame pointer. |
| 113 return fp() + StandardFrameConstants::kCallerSPOffset; |
| 107 } | 114 } |
| 108 | 115 |
| 109 | 116 |
| 110 Code* JavaScriptFrame::FindCode() const { | 117 Code* JavaScriptFrame::FindCode() const { |
| 111 const int offset = StandardFrameConstants::kCodeOffset; | 118 JSFunction* function = JSFunction::cast(this->function()); |
| 112 Object* code = Memory::Object_at(fp() + offset); | 119 return function->shared()->code(); |
| 113 if (code == NULL) { | |
| 114 // The code object isn't set; find it and set it. | |
| 115 code = Heap::FindCodeObject(pc()); | |
| 116 ASSERT(!code->IsFailure()); | |
| 117 Memory::Object_at(fp() + offset) = code; | |
| 118 } | |
| 119 ASSERT(code != NULL); | |
| 120 return Code::cast(code); | |
| 121 } | 120 } |
| 122 | 121 |
| 123 | 122 |
| 124 } } // namespace v8::internal | 123 } } // namespace v8::internal |
| OLD | NEW |