mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 10:10:22 +00:00
d4bc590c27
The RISC-V frame pointer convention is different from that of x86 and ARM. FP is the caller's SP, and the saved FP is at FP[-2] instead of FP[0]. Making Dart frames match the convention of C frames allows stack walkers to continue their traversals through transitions between Dart and C (e.g., for stack dumps or profiling). TEST=ci Change-Id: I463348beba70c1a75bfb0d902b3391be524de0fe Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/235960 Reviewed-by: Daco Harkes <dacoharkes@google.com> Commit-Queue: Ryan Macnak <rmacnak@google.com>
75 lines
2.9 KiB
C++
75 lines
2.9 KiB
C++
// Copyright (c) 2021, the Dart project authors. Please see the AUTHORS file
|
|
// for details. All rights reserved. Use of this source code is governed by a
|
|
// BSD-style license that can be found in the LICENSE file.
|
|
|
|
#ifndef RUNTIME_VM_STACK_FRAME_RISCV_H_
|
|
#define RUNTIME_VM_STACK_FRAME_RISCV_H_
|
|
|
|
#if !defined(RUNTIME_VM_STACK_FRAME_H_)
|
|
#error Do not include stack_frame_riscv.h directly; use stack_frame.h instead.
|
|
#endif
|
|
|
|
namespace dart {
|
|
|
|
/* RISC-V Dart Frame Layout
|
|
| | <- TOS
|
|
Callee frame | ... |
|
|
| saved PP |
|
|
| code object |
|
|
| saved FP | (FP of current frame)
|
|
| saved PC | (PC of current frame)
|
|
+--------------------+
|
|
Current frame | ... T| <- SP of current frame
|
|
| first local T|
|
|
| caller's PP T|
|
|
| code object T| (current frame's code object)
|
|
| caller's FP |
|
|
| caller's RA | (PC of caller frame)
|
|
+--------------------+
|
|
Caller frame | last parameter | <- SP of caller frame, FP of current frame
|
|
| ... |
|
|
|
|
T against a slot indicates it needs to be traversed during GC.
|
|
*/
|
|
|
|
static const int kDartFrameFixedSize = 4; // PP, FP, RA, PC marker.
|
|
static const int kSavedPcSlotFromSp = -1;
|
|
|
|
static const int kFirstObjectSlotFromFp = -3; // Used by GC to traverse stack.
|
|
static const int kLastFixedObjectSlotFromFp = -4;
|
|
|
|
static const int kFirstLocalSlotFromFp = -5;
|
|
static const int kSavedCallerPpSlotFromFp = -4;
|
|
static const int kPcMarkerSlotFromFp = -3;
|
|
static const int kSavedCallerFpSlotFromFp = -2;
|
|
static const int kSavedCallerPcSlotFromFp = -1;
|
|
|
|
static const int kParamEndSlotFromFp = -1; // One slot past last parameter.
|
|
static const int kCallerSpSlotFromFp = 0;
|
|
static const int kLastParamSlotFromEntrySp = 0;
|
|
|
|
// Entry and exit frame layout.
|
|
#if defined(TARGET_ARCH_RISCV64)
|
|
static const int kExitLinkSlotFromEntryFp = -30;
|
|
#elif defined(TARGET_ARCH_RISCV32)
|
|
static const int kExitLinkSlotFromEntryFp = -42;
|
|
#endif
|
|
COMPILE_ASSERT(kAbiPreservedCpuRegCount == 11);
|
|
COMPILE_ASSERT(kAbiPreservedFpuRegCount == 12);
|
|
|
|
// For FFI native -> Dart callbacks, this is the number of stack slots between
|
|
// arguments passed on stack and arguments saved in callback prologue.
|
|
//
|
|
// 2 = return adddress (1) + saved frame pointer (1).
|
|
//
|
|
// If NativeCallbackTrampolines::Enabled(), then
|
|
// kNativeCallbackTrampolineStackDelta must be added as well.
|
|
constexpr intptr_t kCallbackSlotsBeforeSavedArguments = 2;
|
|
|
|
// For FFI calls passing in TypedData, we save it on the stack before entering
|
|
// a Dart frame. This denotes how to get to the backed up typed data.
|
|
static const int kFfiCallerTypedDataSlotFromFp = kCallerSpSlotFromFp;
|
|
|
|
} // namespace dart
|
|
|
|
#endif // RUNTIME_VM_STACK_FRAME_RISCV_H_
|