2019-06-28 17:15:23 +00:00
|
|
|
// Copyright 2019 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
// The C++ style guide recommends using <re2> instead of <regex>. However, the
|
|
|
|
// former isn't available in V8.
|
|
|
|
#include <regex> // NOLINT(build/c++11)
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "src/codegen/arm64/register-arm64.h"
|
|
|
|
#include "test/cctest/cctest.h"
|
|
|
|
#include "test/cctest/disasm-regex-helper.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
// Poison register.
|
|
|
|
const int kPRegCode = kSpeculationPoisonRegister.code();
|
|
|
|
const std::string kPReg = // NOLINT(runtime/string)
|
|
|
|
"x" + std::to_string(kPRegCode);
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TEST(DisasmPoisonMonomorphicLoad) {
|
|
|
|
#ifdef ENABLE_DISASSEMBLER
|
|
|
|
if (i::FLAG_always_opt || !i::FLAG_opt) return;
|
2019-09-26 14:57:43 +00:00
|
|
|
// TODO(9684): Re-enable for TurboProp if necessary.
|
|
|
|
if (i::FLAG_turboprop) return;
|
2019-06-28 17:15:23 +00:00
|
|
|
|
|
|
|
i::FLAG_allow_natives_syntax = true;
|
|
|
|
i::FLAG_untrusted_code_mitigations = true;
|
2019-11-19 10:13:20 +00:00
|
|
|
i::FLAG_debug_code = false;
|
2019-06-28 17:15:23 +00:00
|
|
|
|
|
|
|
CcTest::InitializeVM();
|
|
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
|
|
|
|
CompileRun(
|
|
|
|
"function mono(o) { return o.x; };"
|
|
|
|
"%PrepareFunctionForOptimization(mono);"
|
|
|
|
"mono({ x : 1 });"
|
|
|
|
"mono({ x : 1 });"
|
|
|
|
"%OptimizeFunctionOnNextCall(mono);"
|
|
|
|
"mono({ x : 1 });");
|
|
|
|
|
|
|
|
// Matches that the property access sequence is instrumented with
|
|
|
|
// poisoning.
|
|
|
|
#if defined(V8_COMPRESS_POINTERS)
|
|
|
|
std::vector<std::string> patterns_array = {
|
|
|
|
"ldur <<Map:w[0-9]+>>, \\[<<Obj:x[0-9]+>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap:w[0-9]+>>, pc", // load expected map
|
|
|
|
"cmp <<Map>>, <<ExpMap>>", // compare maps
|
|
|
|
"b.ne", // deopt if different
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
2019-10-09 12:58:35 +00:00
|
|
|
"ldur w<<Field:[0-9]+>>, \\[<<Obj>>, #[0-9]+\\]", // load the field
|
|
|
|
"and x<<Field>>, x<<Field>>, " + kPReg, // apply the poison
|
2019-06-28 17:15:23 +00:00
|
|
|
};
|
|
|
|
#else
|
|
|
|
std::vector<std::string> patterns_array = {
|
|
|
|
"ldur <<Map:x[0-9]+>>, \\[<<Obj:x[0-9]+>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap:x[0-9]+>>, pc", // load expected map
|
|
|
|
"cmp <<Map>>, <<ExpMap>>", // compare maps
|
|
|
|
"b.ne", // deopt if different
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
|
|
|
"ldur <<Field:x[0-9]+>>, \\[<<Obj>>, #[0-9]+\\]", // load the field
|
|
|
|
"and <<Field>>, <<Field>>, " + kPReg, // apply the poison
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
CHECK(CheckDisassemblyRegexPatterns("mono", patterns_array));
|
|
|
|
#endif // ENABLE_DISASSEMBLER
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(DisasmPoisonPolymorphicLoad) {
|
|
|
|
#ifdef ENABLE_DISASSEMBLER
|
|
|
|
if (i::FLAG_always_opt || !i::FLAG_opt) return;
|
2019-09-26 14:57:43 +00:00
|
|
|
// TODO(9684): Re-enable for TurboProp if necessary.
|
|
|
|
if (i::FLAG_turboprop) return;
|
2019-06-28 17:15:23 +00:00
|
|
|
|
|
|
|
i::FLAG_allow_natives_syntax = true;
|
|
|
|
i::FLAG_untrusted_code_mitigations = true;
|
2019-11-19 10:13:20 +00:00
|
|
|
i::FLAG_debug_code = false;
|
2019-06-28 17:15:23 +00:00
|
|
|
|
|
|
|
CcTest::InitializeVM();
|
|
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
|
|
|
|
CompileRun(
|
|
|
|
"function poly(o) { return o.x + 1; };"
|
|
|
|
"let o1 = { x : 1 };"
|
|
|
|
"let o2 = { y : 1 };"
|
|
|
|
"o2.x = 2;"
|
|
|
|
"%PrepareFunctionForOptimization(poly);"
|
|
|
|
"poly(o2);"
|
|
|
|
"poly(o1);"
|
|
|
|
"poly(o2);"
|
|
|
|
"%OptimizeFunctionOnNextCall(poly);"
|
|
|
|
"poly(o1);");
|
|
|
|
|
|
|
|
// Matches that the property access sequence is instrumented with
|
|
|
|
// poisoning.
|
|
|
|
#if defined(V8_COMPRESS_POINTERS)
|
|
|
|
std::vector<std::string> patterns_array = {
|
|
|
|
"ldur <<Map0:w[0-9]+>>, \\[<<Obj:x[0-9]+>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap:w[0-9]+>>, pc", // load map const #1
|
|
|
|
"cmp <<Map0>>, <<ExpMap>>", // compare maps
|
|
|
|
"b.eq", // ? go to the load
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", eq", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
|
|
|
"ldur <<Map1:w[0-9]+>>, \\[<<Obj>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap1:w[0-9]+>>, pc", // load map const #2
|
|
|
|
"cmp <<Map1>>, <<ExpMap1>>", // compare maps
|
|
|
|
"b.ne", // deopt if different
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
|
|
|
"ldur w<<Field:[0-9]+>>, \\[<<Obj>>, #[0-9]+\\]", // load the field
|
|
|
|
"and x<<Field>>, x<<Field>>, " + kPReg, // apply the poison
|
2019-07-18 14:24:03 +00:00
|
|
|
"asr w[0-9]+, w<<Field>>, #1", // untag
|
|
|
|
"b", // goto merge point
|
2019-06-28 17:15:23 +00:00
|
|
|
// Lcase1:
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
2019-10-09 12:58:35 +00:00
|
|
|
"ldur w<<BSt:[0-9]+>>, \\[<<Obj>>, #[0-9]+\\]", // load backing store
|
2019-08-13 14:08:45 +00:00
|
|
|
// branchful decompress
|
2019-07-18 14:24:03 +00:00
|
|
|
"add x<<BSt>>, x26, x<<BSt>>", // Add root to ref
|
2019-06-28 17:15:23 +00:00
|
|
|
"and x<<BSt>>, x<<BSt>>, " + kPReg, // apply the poison
|
|
|
|
"ldur w<<Prop:[0-9]+>>, \\[x<<BSt>>, #[0-9]+\\]", // load the property
|
|
|
|
"and x<<Prop>>, x<<Prop>>, " + kPReg, // apply the poison
|
|
|
|
// Ldone:
|
|
|
|
};
|
|
|
|
#else
|
|
|
|
std::vector<std::string> patterns_array = {
|
|
|
|
"ldur <<Map0:x[0-9]+>>, \\[<<Obj:x[0-9]+>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap0:x[0-9]+>>, pc", // load map const #1
|
|
|
|
"cmp <<Map0>>, <<ExpMap0>>", // compare maps
|
|
|
|
"b.eq", // ? go to the load
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", eq", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
|
|
|
"ldur <<Map1:x[0-9]+>>, \\[<<Obj>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap1:x[0-9]+>>, pc", // load map const #2
|
|
|
|
"cmp <<Map1>>, <<ExpMap1>>", // compare maps
|
|
|
|
"b.ne", // deopt if different
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
2019-09-24 09:19:57 +00:00
|
|
|
"ldur x<<Field:[0-9]+>>, \\[<<Obj>>, #[0-9]+\\]", // load the field
|
|
|
|
"and x<<Field>>, x<<Field>>, " + kPReg, // apply the poison
|
|
|
|
#ifdef V8_31BIT_SMIS_ON_64BIT_ARCH
|
|
|
|
"asr w<<Field>>, w<<Field>>, #1", // untag
|
|
|
|
#else
|
|
|
|
"asr x[0-9]+, x<<Field>>, #32", // untag
|
|
|
|
#endif
|
2019-06-28 17:15:23 +00:00
|
|
|
"b", // goto merge point
|
|
|
|
// Lcase1:
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
|
|
|
"ldur <<BSt:x[0-9]+>>, \\[<<Obj>>, #[0-9]+\\]", // load backing store
|
|
|
|
"and <<BSt>>, <<BSt>>, " + kPReg, // apply the poison
|
|
|
|
"ldur <<Prop:x[0-9]+>>, \\[<<BSt>>, #[0-9]+\\]", // load the property
|
|
|
|
"and <<Prop>>, <<Prop>>, " + kPReg, // apply the poison
|
|
|
|
// Ldone:
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
CHECK(CheckDisassemblyRegexPatterns("poly", patterns_array));
|
|
|
|
#endif // ENABLE_DISASSEMBLER
|
|
|
|
}
|
|
|
|
|
2019-09-13 12:57:53 +00:00
|
|
|
TEST(DisasmPoisonMonomorphicLoadFloat64) {
|
|
|
|
#ifdef ENABLE_DISASSEMBLER
|
|
|
|
if (i::FLAG_always_opt || !i::FLAG_opt) return;
|
2019-10-07 16:17:21 +00:00
|
|
|
// TODO(9684): Re-enable for TurboProp if necessary.
|
|
|
|
if (i::FLAG_turboprop) return;
|
2019-09-13 12:57:53 +00:00
|
|
|
|
|
|
|
i::FLAG_allow_natives_syntax = true;
|
|
|
|
i::FLAG_untrusted_code_mitigations = true;
|
|
|
|
|
|
|
|
CcTest::InitializeVM();
|
|
|
|
v8::HandleScope scope(CcTest::isolate());
|
|
|
|
|
|
|
|
CompileRun(
|
|
|
|
"function mono(o) { return o.x; }"
|
|
|
|
"%PrepareFunctionForOptimization(mono);"
|
|
|
|
"mono({ x : 1.1 });"
|
|
|
|
"mono({ x : 1.1 });"
|
|
|
|
"%OptimizeFunctionOnNextCall(mono);"
|
|
|
|
"mono({ x : 1.1 });");
|
|
|
|
|
|
|
|
// Matches that the property access sequence is instrumented with
|
|
|
|
// poisoning.
|
|
|
|
#if defined(V8_COMPRESS_POINTERS)
|
|
|
|
std::vector<std::string> patterns_array = {
|
|
|
|
"ldur <<Map:w[0-9]+>>, \\[<<Obj:x[0-9]+>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap:w[0-9]+>>, pc", // load expected map
|
|
|
|
"cmp <<Map>>, <<ExpMap>>", // compare maps
|
|
|
|
"b.ne", // deopt if differ
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
2019-10-09 12:58:35 +00:00
|
|
|
"ldur w<<F1:[0-9]+>>, \\[<<Obj>>, #11\\]", // load heap number
|
|
|
|
"add x<<F1>>, x26, x<<F1>>", // Decompress ref
|
|
|
|
"and x<<F1>>, x<<F1>>, " + kPReg, // apply the poison
|
|
|
|
"add <<Addr:x[0-9]+>>, x<<F1>>, #0x[0-9a-f]+", // addr. calculation
|
2019-09-13 12:57:53 +00:00
|
|
|
"and <<Addr>>, <<Addr>>, " + kPReg, // apply the poison
|
|
|
|
"ldr d[0-9]+, \\[<<Addr>>\\]", // load Float64
|
|
|
|
};
|
|
|
|
#else
|
|
|
|
std::vector<std::string> patterns_array = {
|
|
|
|
"ldur <<Map:x[0-9]+>>, \\[<<Obj:x[0-9]+>>, #-1\\]", // load map
|
|
|
|
"ldr <<ExpMap:x[0-9]+>>, pc", // load expected map
|
|
|
|
"cmp <<Map>>, <<ExpMap>>", // compare maps
|
|
|
|
"b.ne", // deopt if differ
|
|
|
|
"csel " + kPReg + ", xzr, " + kPReg + ", ne", // update the poison
|
|
|
|
"csdb", // spec. barrier
|
2019-10-04 11:11:36 +00:00
|
|
|
#if V8_DOUBLE_FIELDS_UNBOXING
|
2019-09-13 12:57:53 +00:00
|
|
|
"add <<Addr:x[0-9]+>>, <<Obj>>, #0x[0-9a-f]+", // addr. calculation
|
2019-10-04 11:11:36 +00:00
|
|
|
#else
|
|
|
|
"ldur <<F1:x[0-9]+>>, \\[<<Obj>>, #23\\]", // load heap number
|
|
|
|
"and <<F1>>, <<F1>>, " + kPReg, // apply the poison
|
|
|
|
"add <<Addr:x[0-9]+>>, <<F1>>, #0x7", // addr. calculation
|
|
|
|
#endif
|
2019-09-13 12:57:53 +00:00
|
|
|
"and <<Addr>>, <<Addr>>, " + kPReg, // apply the poison
|
|
|
|
"ldr d[0-9]+, \\[<<Addr>>\\]", // load Float64
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
CHECK(CheckDisassemblyRegexPatterns("mono", patterns_array));
|
|
|
|
#endif // ENABLE_DISASSEMBLER
|
|
|
|
}
|
|
|
|
|
2019-06-28 17:15:23 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|