mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 10:49:00 +00:00
63e6041ca9
TEST=ci Change-Id: Ic6bc784605e10760bb28ea6df34242336a33b4d0 Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/286947 Reviewed-by: Alexander Aprelev <aam@google.com> Commit-Queue: Ryan Macnak <rmacnak@google.com>
110 lines
3.6 KiB
C++
110 lines
3.6 KiB
C++
// Copyright (c) 2012, 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.
|
|
|
|
#include "vm/globals.h" // Needed here to get TARGET_ARCH_X64.
|
|
#if defined(TARGET_ARCH_X64)
|
|
|
|
#include "platform/unaligned.h"
|
|
|
|
#include "vm/code_patcher.h"
|
|
#include "vm/instructions.h"
|
|
#include "vm/instructions_x64.h"
|
|
|
|
#include "vm/constants.h"
|
|
#include "vm/cpu.h"
|
|
#include "vm/object.h"
|
|
#include "vm/object_store.h"
|
|
|
|
namespace dart {
|
|
|
|
// [start] is the address of a displacement inside a load instruction
|
|
intptr_t IndexFromPPLoadDisp8(uword start) {
|
|
int8_t offset = *reinterpret_cast<int8_t*>(start);
|
|
return ObjectPool::IndexFromOffset(offset);
|
|
}
|
|
|
|
intptr_t IndexFromPPLoadDisp32(uword start) {
|
|
int32_t offset = LoadUnaligned(reinterpret_cast<int32_t*>(start));
|
|
return ObjectPool::IndexFromOffset(offset);
|
|
}
|
|
|
|
bool DecodeLoadObjectFromPoolOrThread(uword pc, const Code& code, Object* obj) {
|
|
ASSERT(code.ContainsInstructionAt(pc));
|
|
|
|
uint8_t* bytes = reinterpret_cast<uint8_t*>(pc);
|
|
|
|
COMPILE_ASSERT(THR == R14);
|
|
if ((bytes[0] == 0x49) || (bytes[0] == 0x4d)) {
|
|
if ((bytes[1] == 0x8b) || (bytes[1] == 0x3b)) { // movq, cmpq
|
|
if ((bytes[2] & 0xc7) == (0x80 | (THR & 7))) { // [r14+disp32]
|
|
int32_t offset = LoadUnaligned(reinterpret_cast<int32_t*>(pc + 3));
|
|
return Thread::ObjectAtOffset(offset, obj);
|
|
}
|
|
if ((bytes[2] & 0xc7) == (0x40 | (THR & 7))) { // [r14+disp8]
|
|
uint8_t offset = *reinterpret_cast<uint8_t*>(pc + 3);
|
|
return Thread::ObjectAtOffset(offset, obj);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (((bytes[0] == 0x41) && (bytes[1] == 0xff) && (bytes[2] == 0x76))) {
|
|
// push [r14+disp8]
|
|
uint8_t offset = *reinterpret_cast<uint8_t*>(pc + 3);
|
|
return Thread::ObjectAtOffset(offset, obj);
|
|
}
|
|
|
|
COMPILE_ASSERT(PP == R15);
|
|
if ((bytes[0] == 0x49) || (bytes[0] == 0x4d)) {
|
|
if ((bytes[1] == 0x8b) || (bytes[1] == 0x3b)) { // movq, cmpq
|
|
if ((bytes[2] & 0xc7) == (0x80 | (PP & 7))) { // [r15+disp32]
|
|
intptr_t index = IndexFromPPLoadDisp32(pc + 3);
|
|
return ObjectAtPoolIndex(code, index, obj);
|
|
}
|
|
if ((bytes[2] & 0xc7) == (0x40 | (PP & 7))) { // [r15+disp8]
|
|
intptr_t index = IndexFromPPLoadDisp8(pc + 3);
|
|
return ObjectAtPoolIndex(code, index, obj);
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
intptr_t TypeTestingStubCallPattern::GetSubtypeTestCachePoolIndex() {
|
|
static int16_t indirect_call_pattern[] = {
|
|
0xff, -1 /* 0x53 or 0x56 */, 0x07, // callq [RBX/RSI + 0x7]
|
|
};
|
|
static int16_t direct_call_pattern[] = {
|
|
0xe8, -1, -1, -1, -1, // callq [PC + <offset>]
|
|
};
|
|
static int16_t pattern_disp8[] = {
|
|
0x4d, 0x8b, 0x4f, -1, // movq R9, [PP + offset]
|
|
};
|
|
static int16_t pattern_disp32[] = {
|
|
0x4d, 0x8b, 0x8f, -1, -1, -1, -1, // movq R9, [PP + offset]
|
|
};
|
|
|
|
uword pc = pc_;
|
|
if (MatchesPattern(pc, direct_call_pattern,
|
|
ARRAY_SIZE(direct_call_pattern))) {
|
|
pc -= ARRAY_SIZE(direct_call_pattern);
|
|
} else if (MatchesPattern(pc, indirect_call_pattern,
|
|
ARRAY_SIZE(indirect_call_pattern))) {
|
|
pc -= ARRAY_SIZE(indirect_call_pattern);
|
|
} else {
|
|
FATAL("Failed to decode at %" Px, pc_);
|
|
}
|
|
|
|
if (MatchesPattern(pc, pattern_disp8, ARRAY_SIZE(pattern_disp8))) {
|
|
return IndexFromPPLoadDisp8(pc - 1);
|
|
} else if (MatchesPattern(pc, pattern_disp32, ARRAY_SIZE(pattern_disp32))) {
|
|
return IndexFromPPLoadDisp32(pc - 4);
|
|
} else {
|
|
FATAL("Failed to decode at %" Px, pc);
|
|
}
|
|
}
|
|
|
|
} // namespace dart
|
|
|
|
#endif // defined TARGET_ARCH_X64
|