Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(144)

Unified Diff: runtime/vm/assembler_x64.cc

Issue 22825023: Uses an object pool on x64 (Closed) Base URL: http://dart.googlecode.com/svn/branches/bleeding_edge/dart/
Patch Set: Created 7 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « runtime/vm/assembler_x64.h ('k') | runtime/vm/code_patcher_x64.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: runtime/vm/assembler_x64.cc
===================================================================
--- runtime/vm/assembler_x64.cc (revision 26436)
+++ runtime/vm/assembler_x64.cc (working copy)
@@ -95,9 +95,14 @@
}
+void Assembler::LoadExternalLabel(const ExternalLabel* label) {
+ const int32_t offset = Array::element_offset(AddExternalLabel(label));
+ LoadWordFromPoolOffset(TMP, offset - kHeapObjectTag);
+}
+
+
void Assembler::call(const ExternalLabel* label) {
AssemblerBuffer::EnsureCapacity ensured(&buffer_);
- intptr_t call_start = buffer_.GetPosition();
// Encode movq(TMP, Immediate(label->address())), but always as imm64.
EmitRegisterREX(TMP, REX_W);
@@ -109,11 +114,21 @@
EmitOperandREX(2, operand, REX_NONE);
EmitUint8(0xFF);
EmitOperand(2, operand);
+}
- ASSERT((buffer_.GetPosition() - call_start) == kCallExternalLabelSize);
+
+void Assembler::CallPatchable(const ExternalLabel* label) {
+ LoadExternalLabel(label);
+ {
+ AssemblerBuffer::EnsureCapacity ensured(&buffer_);
+ // Encode call(TMP).
+ Operand operand(TMP);
+ EmitOperandREX(2, operand, REX_NONE);
+ EmitUint8(0xFF);
+ EmitOperand(2, operand);
+ }
}
-
void Assembler::pushq(Register reg) {
AssemblerBuffer::EnsureCapacity ensured(&buffer_);
EmitRegisterREX(reg, REX_NONE);
@@ -1995,7 +2010,6 @@
void Assembler::jmp(const ExternalLabel* label) {
AssemblerBuffer::EnsureCapacity ensured(&buffer_);
- intptr_t call_start = buffer_.GetPosition();
// Encode movq(TMP, Immediate(label->address())), but always as imm64.
EmitRegisterREX(TMP, REX_W);
@@ -2007,8 +2021,6 @@
EmitOperandREX(4, operand, REX_NONE);
EmitUint8(0xFF);
EmitOperand(4, operand);
-
- ASSERT((buffer_.GetPosition() - call_start) == kCallExternalLabelSize);
}
@@ -2091,16 +2103,59 @@
}
+int32_t Assembler::AddObject(const Object& obj) {
+ ASSERT(obj.IsNotTemporaryScopedHandle());
+ ASSERT(obj.IsOld());
+ if (object_pool_.IsNull()) {
+ // The object pool cannot be used in the vm isolate.
+ ASSERT(Isolate::Current() != Dart::vm_isolate());
+ object_pool_ = GrowableObjectArray::New(Heap::kOld);
+ }
+ for (int i = 0; i < object_pool_.Length(); i++) {
+ if (object_pool_.At(i) == obj.raw()) {
+ return i;
+ }
+ }
+ object_pool_.Add(obj, Heap::kOld);
+ return object_pool_.Length() - 1;
+}
+
+
+int32_t Assembler::AddExternalLabel(const ExternalLabel* label) {
+ if (object_pool_.IsNull()) {
+ // The object pool cannot be used in the vm isolate.
+ ASSERT(Isolate::Current() != Dart::vm_isolate());
+ object_pool_ = GrowableObjectArray::New(Heap::kOld);
+ }
+ const word address = label->address();
+ ASSERT(Utils::IsAligned(address, 4));
+ // The address is stored in the object array as a RawSmi.
+ const Smi& smi = Smi::Handle(Smi::New(address >> kSmiTagShift));
+ // Do not reuse an existing entry, since each reference may be patched
+ // independently.
+ object_pool_.Add(smi, Heap::kOld);
+ return object_pool_.Length() - 1;
+}
+
+
+void Assembler::LoadWordFromPoolOffset(Register dst, int32_t offset) {
+ movq(dst, Address(PP, offset));
+ // This sequence must be of fixed size. If offset fits in a signed byte we
+ // have to pad with nops.
+ if (Utils::IsInt(8, offset)) {
+ nop(3);
+ }
+}
+
+
void Assembler::LoadObject(Register dst, const Object& object) {
if (object.IsSmi() || object.InVMHeap()) {
movq(dst, Immediate(reinterpret_cast<int64_t>(object.raw())));
} else {
ASSERT(object.IsNotTemporaryScopedHandle());
ASSERT(object.IsOld());
- AssemblerBuffer::EnsureCapacity ensured(&buffer_);
- EmitRegisterREX(dst, REX_W);
- EmitUint8(0xB8 | (dst & 7));
- buffer_.EmitObject(object);
+ const int32_t offset = Array::element_offset(AddObject(object));
+ LoadWordFromPoolOffset(dst, offset - kHeapObjectTag);
}
}
@@ -2292,9 +2347,15 @@
}
-void Assembler::LeaveFrame() {
- movq(RSP, RBP);
- popq(RBP);
+void Assembler::LeaveFrame(bool restore_pp) {
+ if (restore_pp) {
+ leaq(RSP, Address(RBP, -2 * kWordSize)); // Restore stack pointer.
Ivan Posva 2013/08/22 05:26:25 Wouldn't this be shorter? if (restore_pp) { mov
+ popq(PP); // Restore PP.
+ popq(RBP); // Ignore PC marker.
+ } else {
+ movq(RSP, RBP); // Restore stack pointer.
+ }
+ popq(RBP); // Restore frame pointer.
}
@@ -2377,6 +2438,21 @@
}
+void Assembler::LoadPoolPointer() {
+ Label next;
+ call(&next);
+ Bind(&next);
+
+ // Load new pool pointer.
+ const intptr_t object_pool_pc_dist =
+ Instructions::HeaderSize() - Instructions::object_pool_offset() +
+ CodeSize();
+ addq(Address(RSP, 0), Immediate(-object_pool_pc_dist));
+ popq(PP);
+ movq(PP, Address(PP, 0));
Ivan Posva 2013/08/22 05:26:25 popq(PP); movq(PP, Address(PP, -object_pool_pc_dis
+}
+
+
void Assembler::EnterDartFrame(intptr_t frame_size) {
EnterFrame(0);
Label dart_entry;
@@ -2385,10 +2461,21 @@
// The runtime system assumes that the code marker address is
// kEntryPointToPcMarkerOffset bytes from the entry. If there is any code
// generated before entering the frame, the address needs to be adjusted.
+ const intptr_t object_pool_pc_dist =
+ Instructions::HeaderSize() - Instructions::object_pool_offset() +
+ CodeSize();
const intptr_t offset = kEntryPointToPcMarkerOffset - CodeSize();
if (offset != 0) {
addq(Address(RSP, 0), Immediate(offset));
}
+ // Save caller's pool pointer
+ pushq(PP);
+
+ // Load callee's pool pointer.
+ movq(PP, Address(RSP, 1 * kWordSize));
+ addq(PP, Immediate(-object_pool_pc_dist - offset));
+ movq(PP, Address(PP, 0));
Ivan Posva 2013/08/22 05:26:25 Wouldn't this be shorter? movq(PP, Address(RSP, 1*
+
if (frame_size != 0) {
subq(RSP, Immediate(frame_size));
}
@@ -2407,20 +2494,37 @@
// The runtime system assumes that the code marker address is
// kEntryPointToPcMarkerOffset bytes from the entry. Since there is no
// code to set up the frame pointer, the address needs to be adjusted.
+ const intptr_t object_pool_pc_dist =
+ Instructions::HeaderSize() - Instructions::object_pool_offset() +
+ CodeSize();
const intptr_t offset = kEntryPointToPcMarkerOffset - CodeSize();
if (offset != 0) {
addq(Address(RSP, 0), Immediate(offset));
}
+
+ // Load callee's pool pointer.
+ movq(PP, Address(RSP, 0));
+ addq(PP, Immediate(-object_pool_pc_dist - offset));
+ movq(PP, Address(PP, 0));
+
popq(Address(RBP, kPcMarkerSlotFromFp * kWordSize));
+
if (extra_size != 0) {
subq(RSP, Immediate(extra_size));
}
}
-void Assembler::EnterStubFrame() {
- EnterFrame(0);
- pushq(Immediate(0)); // Push 0 in the saved PC area for stub frames.
+void Assembler::EnterStubFrame(bool save_pp) {
+ if (save_pp) {
+ EnterFrame(0);
+ pushq(Immediate(0)); // Push 0 in the saved PC area for stub frames.
+ pushq(PP); // Save caller's pool pointer
+ LoadPoolPointer();
+ } else {
+ EnterFrame(0);
+ pushq(Immediate(0)); // Push 0 in the saved PC area for stub frames.
+ }
}
« no previous file with comments | « runtime/vm/assembler_x64.h ('k') | runtime/vm/code_patcher_x64.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698