2018-10-15 22:31:24 +00:00
|
|
|
// Copyright 2018 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.
|
|
|
|
|
2019-04-24 13:25:27 +00:00
|
|
|
#include "src/torque/torque-compiler.h"
|
2018-10-15 22:31:24 +00:00
|
|
|
#include "src/torque/utils.h"
|
|
|
|
#include "test/unittests/test-utils.h"
|
2019-04-24 13:25:27 +00:00
|
|
|
#include "testing/gmock-support.h"
|
2018-10-15 22:31:24 +00:00
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace torque {
|
|
|
|
|
2019-05-02 07:54:08 +00:00
|
|
|
namespace {
|
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
// This is a simplified version of the basic Torque type definitions.
|
|
|
|
// Some class types are replaced by abstact types to keep it self-contained and
|
|
|
|
// small.
|
|
|
|
constexpr const char* kTestTorquePrelude = R"(
|
|
|
|
type void;
|
|
|
|
type never;
|
|
|
|
|
2019-07-26 13:30:02 +00:00
|
|
|
namespace torque_internal {
|
|
|
|
struct Reference<T: type> {
|
|
|
|
const object: HeapObject;
|
|
|
|
const offset: intptr;
|
|
|
|
}
|
2020-03-17 18:53:51 +00:00
|
|
|
type ConstReference<T : type> extends Reference<T>;
|
|
|
|
type MutableReference<T : type> extends ConstReference<T>;
|
|
|
|
|
2019-12-09 10:59:33 +00:00
|
|
|
type UninitializedHeapObject extends HeapObject;
|
2019-07-26 13:30:02 +00:00
|
|
|
}
|
|
|
|
|
2019-11-08 18:53:48 +00:00
|
|
|
type Tagged generates 'TNode<MaybeObject>' constexpr 'MaybeObject';
|
2019-11-14 17:39:19 +00:00
|
|
|
type StrongTagged extends Tagged
|
|
|
|
generates 'TNode<Object>' constexpr 'ObjectPtr';
|
2019-11-08 18:53:48 +00:00
|
|
|
type Smi extends StrongTagged generates 'TNode<Smi>' constexpr 'Smi';
|
2019-11-14 17:39:19 +00:00
|
|
|
type WeakHeapObject extends Tagged;
|
|
|
|
type Weak<T : type extends HeapObject> extends WeakHeapObject;
|
2019-11-08 18:53:48 +00:00
|
|
|
type Uninitialized extends Tagged;
|
2019-05-23 20:23:36 +00:00
|
|
|
|
|
|
|
@abstract
|
2019-11-08 18:53:48 +00:00
|
|
|
extern class HeapObject extends StrongTagged {
|
2019-05-23 20:23:36 +00:00
|
|
|
map: Map;
|
|
|
|
}
|
|
|
|
type Map extends HeapObject generates 'TNode<Map>';
|
|
|
|
type Object = Smi | HeapObject;
|
2020-03-17 18:53:51 +00:00
|
|
|
type Number = Smi|HeapNumber;
|
2019-05-23 20:23:36 +00:00
|
|
|
type JSReceiver extends HeapObject generates 'TNode<JSReceiver>';
|
|
|
|
type JSObject extends JSReceiver generates 'TNode<JSObject>';
|
|
|
|
type int32 generates 'TNode<Int32T>' constexpr 'int32_t';
|
|
|
|
type uint32 generates 'TNode<Uint32T>' constexpr 'uint32_t';
|
|
|
|
type int31 extends int32
|
|
|
|
generates 'TNode<Int32T>' constexpr 'int31_t';
|
|
|
|
type uint31 extends uint32
|
|
|
|
generates 'TNode<Uint32T>' constexpr 'uint31_t';
|
|
|
|
type int16 extends int31
|
|
|
|
generates 'TNode<Int16T>' constexpr 'int16_t';
|
|
|
|
type uint16 extends uint31
|
|
|
|
generates 'TNode<Uint16T>' constexpr 'uint16_t';
|
|
|
|
type int8 extends int16 generates 'TNode<Int8T>' constexpr 'int8_t';
|
|
|
|
type uint8 extends uint16
|
|
|
|
generates 'TNode<Uint8T>' constexpr 'uint8_t';
|
|
|
|
type int64 generates 'TNode<Int64T>' constexpr 'int64_t';
|
|
|
|
type intptr generates 'TNode<IntPtrT>' constexpr 'intptr_t';
|
|
|
|
type uintptr generates 'TNode<UintPtrT>' constexpr 'uintptr_t';
|
|
|
|
type float32 generates 'TNode<Float32T>' constexpr 'float';
|
|
|
|
type float64 generates 'TNode<Float64T>' constexpr 'double';
|
|
|
|
type bool generates 'TNode<BoolT>' constexpr 'bool';
|
|
|
|
type bint generates 'TNode<BInt>' constexpr 'BInt';
|
|
|
|
type string constexpr 'const char*';
|
|
|
|
type RawPtr generates 'TNode<RawPtrT>' constexpr 'void*';
|
2020-04-29 15:28:17 +00:00
|
|
|
type ExternalPointer
|
|
|
|
generates 'TNode<ExternalPointerT>' constexpr 'ExternalPointer_t';
|
2019-05-23 20:23:36 +00:00
|
|
|
type Code extends HeapObject generates 'TNode<Code>';
|
|
|
|
type BuiltinPtr extends Smi generates 'TNode<BuiltinPtr>';
|
|
|
|
type Context extends HeapObject generates 'TNode<Context>';
|
|
|
|
type NativeContext extends Context;
|
2020-02-24 18:32:34 +00:00
|
|
|
type SmiTagged<T : type extends uint31> extends Smi;
|
2020-03-17 18:53:51 +00:00
|
|
|
type String extends HeapObject;
|
|
|
|
type HeapNumber extends HeapObject;
|
|
|
|
type FixedArrayBase extends HeapObject;
|
2019-10-24 07:51:58 +00:00
|
|
|
|
2020-01-16 11:58:31 +00:00
|
|
|
struct float64_or_hole {
|
|
|
|
is_hole: bool;
|
|
|
|
value: float64;
|
|
|
|
}
|
|
|
|
|
2020-03-17 18:53:51 +00:00
|
|
|
extern operator '+' macro IntPtrAdd(intptr, intptr): intptr;
|
|
|
|
|
2019-12-18 15:33:16 +00:00
|
|
|
intrinsic %FromConstexpr<To: type, From: type>(b: From): To;
|
|
|
|
intrinsic %RawDownCast<To: type, From: type>(x: From): To;
|
|
|
|
intrinsic %RawConstexprCast<To: type, From: type>(f: From): To;
|
|
|
|
extern macro SmiConstant(constexpr Smi): Smi;
|
|
|
|
extern macro TaggedToSmi(Object): Smi
|
|
|
|
labels CastError;
|
|
|
|
extern macro TaggedToHeapObject(Object): HeapObject
|
|
|
|
labels CastError;
|
2020-03-17 18:53:51 +00:00
|
|
|
extern macro Float64SilenceNaN(float64): float64;
|
2019-12-18 15:33:16 +00:00
|
|
|
|
2019-12-20 11:45:14 +00:00
|
|
|
extern macro IntPtrConstant(constexpr int31): intptr;
|
|
|
|
|
2019-10-24 07:51:58 +00:00
|
|
|
macro FromConstexpr<To: type, From: type>(o: From): To;
|
2019-12-18 15:33:16 +00:00
|
|
|
FromConstexpr<Smi, constexpr Smi>(s: constexpr Smi): Smi {
|
|
|
|
return SmiConstant(s);
|
|
|
|
}
|
|
|
|
FromConstexpr<Smi, constexpr int31>(s: constexpr int31): Smi {
|
|
|
|
return %FromConstexpr<Smi>(s);
|
|
|
|
}
|
2019-12-20 11:45:14 +00:00
|
|
|
FromConstexpr<intptr, constexpr int31>(i: constexpr int31): intptr {
|
|
|
|
return IntPtrConstant(i);
|
|
|
|
}
|
2020-03-17 18:53:51 +00:00
|
|
|
FromConstexpr<intptr, constexpr intptr>(i: constexpr intptr): intptr {
|
|
|
|
return %FromConstexpr<intptr>(i);
|
|
|
|
}
|
2019-12-18 15:33:16 +00:00
|
|
|
|
|
|
|
macro Cast<A : type extends Object>(implicit context: Context)(o: Object): A
|
|
|
|
labels CastError {
|
|
|
|
return Cast<A>(TaggedToHeapObject(o) otherwise CastError)
|
|
|
|
otherwise CastError;
|
|
|
|
}
|
|
|
|
Cast<Smi>(o: Object): Smi
|
|
|
|
labels CastError {
|
|
|
|
return TaggedToSmi(o) otherwise CastError;
|
|
|
|
}
|
2019-05-23 20:23:36 +00:00
|
|
|
)";
|
|
|
|
|
|
|
|
TorqueCompilerResult TestCompileTorque(std::string source) {
|
2019-05-02 07:54:08 +00:00
|
|
|
TorqueCompilerOptions options;
|
|
|
|
options.output_directory = "";
|
|
|
|
options.collect_language_server_data = false;
|
2019-05-08 08:26:07 +00:00
|
|
|
options.force_assert_statements = false;
|
2019-06-11 08:57:18 +00:00
|
|
|
options.v8_root = ".";
|
2019-05-02 07:54:08 +00:00
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
source = kTestTorquePrelude + source;
|
2019-05-02 07:54:08 +00:00
|
|
|
return CompileTorque(source, options);
|
|
|
|
}
|
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
void ExpectSuccessfulCompilation(std::string source) {
|
|
|
|
TorqueCompilerResult result = TestCompileTorque(std::move(source));
|
|
|
|
std::vector<std::string> messages;
|
|
|
|
for (const auto& message : result.messages) {
|
|
|
|
messages.push_back(message.message);
|
|
|
|
}
|
|
|
|
EXPECT_EQ(messages, std::vector<std::string>{});
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
2019-11-15 16:27:27 +00:00
|
|
|
using MatcherVector =
|
|
|
|
std::vector<std::pair<::testing::PolymorphicMatcher<T>, LineAndColumn>>;
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
void ExpectFailingCompilation(std::string source,
|
|
|
|
MatcherVector<T> message_patterns) {
|
2019-05-23 20:23:36 +00:00
|
|
|
TorqueCompilerResult result = TestCompileTorque(std::move(source));
|
|
|
|
ASSERT_FALSE(result.messages.empty());
|
2019-11-15 16:27:27 +00:00
|
|
|
EXPECT_GE(result.messages.size(), message_patterns.size());
|
|
|
|
size_t limit = message_patterns.size();
|
|
|
|
if (result.messages.size() < limit) {
|
|
|
|
limit = result.messages.size();
|
|
|
|
}
|
|
|
|
for (size_t i = 0; i < limit; ++i) {
|
|
|
|
EXPECT_THAT(result.messages[i].message, message_patterns[i].first);
|
|
|
|
if (message_patterns[i].second != LineAndColumn::Invalid()) {
|
|
|
|
base::Optional<SourcePosition> actual = result.messages[i].position;
|
|
|
|
EXPECT_TRUE(actual.has_value());
|
|
|
|
EXPECT_EQ(actual->start, message_patterns[i].second);
|
|
|
|
}
|
|
|
|
}
|
2019-05-23 20:23:36 +00:00
|
|
|
}
|
|
|
|
|
2019-11-15 16:27:27 +00:00
|
|
|
template <class T>
|
|
|
|
void ExpectFailingCompilation(
|
|
|
|
std::string source, ::testing::PolymorphicMatcher<T> message_pattern) {
|
|
|
|
ExpectFailingCompilation(
|
|
|
|
source, MatcherVector<T>{{message_pattern, LineAndColumn::Invalid()}});
|
|
|
|
}
|
|
|
|
|
2019-12-05 13:30:12 +00:00
|
|
|
// TODO(almuthanna): the definition of this function is skipped on Fuchsia
|
|
|
|
// because it causes an 'unused function' exception upon buidling gn
|
|
|
|
// Ticket: https://crbug.com/1028617
|
|
|
|
#if !defined(V8_TARGET_OS_FUCHSIA)
|
2019-11-15 16:27:27 +00:00
|
|
|
int CountPreludeLines() {
|
|
|
|
static int result = -1;
|
|
|
|
if (result == -1) {
|
|
|
|
std::string prelude(kTestTorquePrelude);
|
|
|
|
result = static_cast<int>(std::count(prelude.begin(), prelude.end(), '\n'));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2019-12-05 13:30:12 +00:00
|
|
|
#endif
|
2019-11-15 16:27:27 +00:00
|
|
|
|
|
|
|
using SubstrWithPosition =
|
|
|
|
std::pair<::testing::PolymorphicMatcher<
|
|
|
|
::testing::internal::HasSubstrMatcher<std::string>>,
|
|
|
|
LineAndColumn>;
|
|
|
|
|
2019-12-05 13:30:12 +00:00
|
|
|
// TODO(almuthanna): the definition of this function is skipped on Fuchsia
|
|
|
|
// because it causes an 'unused function' exception upon buidling gn
|
|
|
|
// Ticket: https://crbug.com/1028617
|
|
|
|
#if !defined(V8_TARGET_OS_FUCHSIA)
|
2019-11-15 16:27:27 +00:00
|
|
|
SubstrWithPosition SubstrTester(const std::string& message, int line, int col) {
|
|
|
|
// Change line and column from 1-based to 0-based.
|
|
|
|
return {::testing::HasSubstr(message),
|
|
|
|
LineAndColumn{line + CountPreludeLines() - 1, col - 1}};
|
|
|
|
}
|
2019-12-05 13:30:12 +00:00
|
|
|
#endif
|
2019-11-15 16:27:27 +00:00
|
|
|
|
|
|
|
using SubstrVector = std::vector<SubstrWithPosition>;
|
|
|
|
|
2019-05-02 07:54:08 +00:00
|
|
|
} // namespace
|
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
TEST(Torque, Prelude) { ExpectSuccessfulCompilation(""); }
|
|
|
|
|
2018-10-15 22:31:24 +00:00
|
|
|
TEST(Torque, StackDeleteRange) {
|
|
|
|
Stack<int> stack = {1, 2, 3, 4, 5, 6, 7};
|
|
|
|
stack.DeleteRange(StackRange{BottomOffset{2}, BottomOffset{4}});
|
|
|
|
Stack<int> result = {1, 2, 5, 6, 7};
|
|
|
|
ASSERT_TRUE(stack == result);
|
|
|
|
}
|
|
|
|
|
2019-04-24 13:25:27 +00:00
|
|
|
using ::testing::HasSubstr;
|
|
|
|
TEST(Torque, TypeNamingConventionLintError) {
|
2019-05-23 20:23:36 +00:00
|
|
|
ExpectFailingCompilation(R"(
|
2019-04-24 13:25:27 +00:00
|
|
|
type foo generates 'TNode<Foo>';
|
2019-05-23 20:23:36 +00:00
|
|
|
)",
|
|
|
|
HasSubstr("\"foo\""));
|
|
|
|
}
|
2019-04-24 13:25:27 +00:00
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
TEST(Torque, StructNamingConventionLintError) {
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
struct foo {}
|
|
|
|
)",
|
|
|
|
HasSubstr("\"foo\""));
|
|
|
|
}
|
2019-05-02 07:54:08 +00:00
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
TEST(Torque, ClassDefinition) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
extern class TestClassWithAllTypes extends HeapObject {
|
|
|
|
a: int8;
|
|
|
|
b: uint8;
|
|
|
|
b2: uint8;
|
|
|
|
b3: uint8;
|
|
|
|
c: int16;
|
|
|
|
d: uint16;
|
|
|
|
e: int32;
|
|
|
|
f: uint32;
|
|
|
|
g: RawPtr;
|
|
|
|
h: intptr;
|
|
|
|
i: uintptr;
|
|
|
|
}
|
|
|
|
|
2019-06-11 12:26:32 +00:00
|
|
|
@export
|
2019-05-23 20:23:36 +00:00
|
|
|
macro TestClassWithAllTypesLoadsAndStores(
|
|
|
|
t: TestClassWithAllTypes, r: RawPtr, v1: int8, v2: uint8, v3: int16,
|
|
|
|
v4: uint16, v5: int32, v6: uint32, v7: intptr, v8: uintptr) {
|
|
|
|
t.a = v1;
|
|
|
|
t.b = v2;
|
|
|
|
t.c = v3;
|
|
|
|
t.d = v4;
|
|
|
|
t.e = v5;
|
|
|
|
t.f = v6;
|
|
|
|
t.g = r;
|
|
|
|
t.h = v7;
|
|
|
|
t.i = v8;
|
|
|
|
t.a = t.a;
|
|
|
|
t.b = t.b;
|
|
|
|
t.c = t.c;
|
|
|
|
t.d = t.d;
|
|
|
|
t.e = t.e;
|
|
|
|
t.f = t.f;
|
|
|
|
t.g = t.g;
|
|
|
|
t.h = t.h;
|
|
|
|
t.i = t.i;
|
|
|
|
}
|
|
|
|
)");
|
2019-05-02 07:54:08 +00:00
|
|
|
}
|
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
TEST(Torque, TypeDeclarationOrder) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
type Baztype = Foo | FooType;
|
2019-05-23 09:06:15 +00:00
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
@abstract
|
|
|
|
extern class Foo extends HeapObject {
|
|
|
|
fooField: FooType;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern class Bar extends Foo {
|
|
|
|
barField: Bartype;
|
|
|
|
bazfield: Baztype;
|
|
|
|
}
|
2019-05-23 20:23:36 +00:00
|
|
|
|
2019-05-23 20:23:36 +00:00
|
|
|
type Bartype = FooType;
|
|
|
|
|
|
|
|
type FooType = Smi | Bar;
|
|
|
|
)");
|
|
|
|
}
|
2019-04-24 13:25:27 +00:00
|
|
|
|
2019-12-05 13:30:12 +00:00
|
|
|
// TODO(almuthanna): These tests were skipped because they cause a crash when
|
|
|
|
// they are ran on Fuchsia. This issue should be solved later on
|
|
|
|
// Ticket: https://crbug.com/1028617
|
|
|
|
#if !defined(V8_TARGET_OS_FUCHSIA)
|
2019-05-23 20:23:36 +00:00
|
|
|
TEST(Torque, ConditionalFields) {
|
|
|
|
// This class should throw alignment errors if @if decorators aren't
|
|
|
|
// working.
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
extern class PreprocessingTest extends HeapObject {
|
|
|
|
@if(FALSE_FOR_TESTING) a: int8;
|
|
|
|
@if(TRUE_FOR_TESTING) a: int16;
|
|
|
|
b: int16;
|
|
|
|
d: int32;
|
|
|
|
@ifnot(TRUE_FOR_TESTING) e: int8;
|
|
|
|
@ifnot(FALSE_FOR_TESTING) f: int16;
|
|
|
|
g: int16;
|
|
|
|
h: int32;
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
extern class PreprocessingTest extends HeapObject {
|
|
|
|
@if(TRUE_FOR_TESTING) a: int8;
|
|
|
|
@if(FALSE_FOR_TESTING) a: int16;
|
|
|
|
b: int16;
|
|
|
|
d: int32;
|
|
|
|
@ifnot(FALSE_FOR_TESTING) e: int8;
|
|
|
|
@ifnot(TRUE_FOR_TESTING) f: int16;
|
|
|
|
g: int16;
|
|
|
|
h: int32;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("aligned"));
|
2019-04-24 13:25:27 +00:00
|
|
|
}
|
|
|
|
|
2019-05-24 11:29:51 +00:00
|
|
|
TEST(Torque, ConstexprLetBindingDoesNotCrash) {
|
|
|
|
ExpectFailingCompilation(
|
2019-06-11 12:26:32 +00:00
|
|
|
R"(@export macro FooBar() { let foo = 0; check(foo >= 0); })",
|
2019-05-24 11:29:51 +00:00
|
|
|
HasSubstr("Use 'const' instead of 'let' for variable 'foo'"));
|
|
|
|
}
|
|
|
|
|
2020-05-05 16:51:52 +00:00
|
|
|
TEST(Torque, FailedImplicitCastFromConstexprDoesNotCrash) {
|
|
|
|
ExpectFailingCompilation(
|
|
|
|
R"(
|
|
|
|
extern enum SomeEnum {
|
|
|
|
kValue,
|
|
|
|
...
|
|
|
|
}
|
|
|
|
macro Foo() {
|
|
|
|
Bar(SomeEnum::kValue);
|
|
|
|
}
|
|
|
|
macro Bar<T: type>(value: T) {}
|
|
|
|
)",
|
|
|
|
HasSubstr(
|
|
|
|
"Cannot find non-constexpr type corresponding to constexpr kValue"));
|
|
|
|
}
|
|
|
|
|
2019-06-06 12:41:27 +00:00
|
|
|
TEST(Torque, DoubleUnderScorePrefixIllegalForIdentifiers) {
|
|
|
|
ExpectFailingCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo() {
|
2019-06-06 12:41:27 +00:00
|
|
|
let __x;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("Lexer Error"));
|
|
|
|
}
|
2019-12-05 13:30:12 +00:00
|
|
|
#endif
|
2019-06-06 12:41:27 +00:00
|
|
|
|
|
|
|
TEST(Torque, UnusedLetBindingLintError) {
|
|
|
|
ExpectFailingCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(y: Smi) {
|
2019-06-06 12:41:27 +00:00
|
|
|
let x: Smi = y;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("Variable 'x' is never used."));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, UnderscorePrefixSilencesUnusedWarning) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(y: Smi) {
|
2019-06-06 12:41:27 +00:00
|
|
|
let _x: Smi = y;
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
}
|
|
|
|
|
2019-12-05 13:30:12 +00:00
|
|
|
// TODO(almuthanna): This test was skipped because it causes a crash when it is
|
|
|
|
// ran on Fuchsia. This issue should be solved later on
|
|
|
|
// Ticket: https://crbug.com/1028617
|
|
|
|
#if !defined(V8_TARGET_OS_FUCHSIA)
|
2019-06-06 12:41:27 +00:00
|
|
|
TEST(Torque, UsingUnderscorePrefixedIdentifierError) {
|
|
|
|
ExpectFailingCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(y: Smi) {
|
2019-06-06 12:41:27 +00:00
|
|
|
let _x: Smi = y;
|
|
|
|
check(_x == y);
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("Trying to reference '_x'"));
|
|
|
|
}
|
2019-12-05 13:30:12 +00:00
|
|
|
#endif
|
2019-06-06 12:41:27 +00:00
|
|
|
|
|
|
|
TEST(Torque, UnusedArgumentLintError) {
|
|
|
|
ExpectFailingCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(x: Smi) {}
|
2019-06-06 12:41:27 +00:00
|
|
|
)",
|
|
|
|
HasSubstr("Variable 'x' is never used."));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, UsingUnderscorePrefixedArgumentSilencesWarning) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(_y: Smi) {}
|
2019-06-06 12:41:27 +00:00
|
|
|
)");
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, UnusedLabelLintError) {
|
|
|
|
ExpectFailingCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo() labels Bar {}
|
2019-06-06 12:41:27 +00:00
|
|
|
)",
|
|
|
|
HasSubstr("Label 'Bar' is never used."));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, UsingUnderScorePrefixLabelSilencesWarning) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo() labels _Bar {}
|
2019-06-06 12:41:27 +00:00
|
|
|
)");
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, NoUnusedWarningForImplicitArguments) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(implicit c: Context, r: JSReceiver)() {}
|
2019-06-06 12:41:27 +00:00
|
|
|
)");
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, NoUnusedWarningForVariablesOnlyUsedInAsserts) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(x: bool) {
|
2019-06-06 12:41:27 +00:00
|
|
|
assert(x);
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
}
|
|
|
|
|
2019-12-05 13:30:12 +00:00
|
|
|
// TODO(almuthanna): This test was skipped because it causes a crash when it is
|
|
|
|
// ran on Fuchsia. This issue should be solved later on
|
|
|
|
// Ticket: https://crbug.com/1028617
|
|
|
|
#if !defined(V8_TARGET_OS_FUCHSIA)
|
2019-06-11 08:57:18 +00:00
|
|
|
TEST(Torque, ImportNonExistentFile) {
|
|
|
|
ExpectFailingCompilation(R"(import "foo/bar.tq")",
|
|
|
|
HasSubstr("File 'foo/bar.tq' not found."));
|
|
|
|
}
|
2019-12-05 13:30:12 +00:00
|
|
|
#endif
|
2019-06-11 08:57:18 +00:00
|
|
|
|
2019-06-11 10:56:10 +00:00
|
|
|
TEST(Torque, LetShouldBeConstLintError) {
|
|
|
|
ExpectFailingCompilation(R"(
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Foo(y: Smi): Smi {
|
2019-06-11 10:56:10 +00:00
|
|
|
let x: Smi = y;
|
|
|
|
return x;
|
|
|
|
})",
|
|
|
|
HasSubstr("Variable 'x' is never assigned to."));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, LetShouldBeConstIsSkippedForStructs) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
struct Foo{ a: Smi; }
|
2019-06-11 12:26:32 +00:00
|
|
|
@export macro Bar(x: Smi): Foo {
|
2019-06-11 10:56:10 +00:00
|
|
|
let foo = Foo{a: x};
|
|
|
|
return foo;
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
}
|
|
|
|
|
2019-12-05 13:30:12 +00:00
|
|
|
// TODO(almuthanna): These tests were skipped because they cause a crash when
|
|
|
|
// they are ran on Fuchsia. This issue should be solved later on
|
|
|
|
// Ticket: https://crbug.com/1028617
|
|
|
|
#if !defined(V8_TARGET_OS_FUCHSIA)
|
2019-10-24 07:51:58 +00:00
|
|
|
TEST(Torque, GenericAbstractType) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
type Foo<T: type> extends HeapObject;
|
|
|
|
extern macro F1(HeapObject);
|
|
|
|
macro F2<T: type>(x: Foo<T>) {
|
|
|
|
F1(x);
|
|
|
|
}
|
|
|
|
@export
|
|
|
|
macro F3(a: Foo<Smi>, b: Foo<HeapObject>){
|
|
|
|
F2(a);
|
|
|
|
F2(b);
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
type Foo<T: type> extends HeapObject;
|
|
|
|
macro F1<T: type>(x: Foo<T>) {}
|
|
|
|
@export
|
|
|
|
macro F2(a: Foo<Smi>) {
|
|
|
|
F1<HeapObject>(a);
|
|
|
|
})",
|
|
|
|
HasSubstr("cannot find suitable callable"));
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
type Foo<T: type> extends HeapObject;
|
|
|
|
extern macro F1(Foo<HeapObject>);
|
|
|
|
@export
|
|
|
|
macro F2(a: Foo<Smi>) {
|
|
|
|
F1(a);
|
|
|
|
})",
|
|
|
|
HasSubstr("cannot find suitable callable"));
|
|
|
|
}
|
|
|
|
|
2019-11-15 16:27:27 +00:00
|
|
|
TEST(Torque, SpecializationRequesters) {
|
|
|
|
ExpectFailingCompilation(
|
|
|
|
R"(
|
|
|
|
macro A<T: type extends HeapObject>() {}
|
|
|
|
macro B<T: type>() {
|
|
|
|
A<T>();
|
|
|
|
}
|
|
|
|
macro C<T: type>() {
|
|
|
|
B<T>();
|
|
|
|
}
|
|
|
|
macro D() {
|
|
|
|
C<Smi>();
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
SubstrVector{
|
|
|
|
SubstrTester("cannot find suitable callable", 4, 7),
|
|
|
|
SubstrTester("Note: in specialization B<Smi> requested here", 7, 7),
|
|
|
|
SubstrTester("Note: in specialization C<Smi> requested here", 10,
|
|
|
|
7)});
|
|
|
|
|
|
|
|
ExpectFailingCompilation(
|
|
|
|
R"(
|
|
|
|
extern macro RetVal(): Object;
|
|
|
|
builtin A<T: type extends HeapObject>(implicit context: Context)(): Object {
|
|
|
|
return RetVal();
|
|
|
|
}
|
|
|
|
builtin B<T: type>(implicit context: Context)(): Object {
|
|
|
|
return A<T>();
|
|
|
|
}
|
|
|
|
builtin C<T: type>(implicit context: Context)(): Object {
|
|
|
|
return B<T>();
|
|
|
|
}
|
|
|
|
builtin D(implicit context: Context)(): Object {
|
|
|
|
return C<Smi>();
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
SubstrVector{
|
|
|
|
SubstrTester("cannot find suitable callable", 7, 14),
|
|
|
|
SubstrTester("Note: in specialization B<Smi> requested here", 10, 14),
|
|
|
|
SubstrTester("Note: in specialization C<Smi> requested here", 13,
|
|
|
|
14)});
|
|
|
|
|
|
|
|
ExpectFailingCompilation(
|
|
|
|
R"(
|
|
|
|
struct A<T: type extends HeapObject> {}
|
|
|
|
struct B<T: type> {
|
|
|
|
a: A<T>;
|
|
|
|
}
|
|
|
|
struct C<T: type> {
|
|
|
|
b: B<T>;
|
|
|
|
}
|
|
|
|
struct D {
|
|
|
|
c: C<Smi>;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
SubstrVector{
|
|
|
|
SubstrTester("Could not instantiate generic", 4, 10),
|
|
|
|
SubstrTester("Note: in specialization B<Smi> requested here", 7, 10),
|
|
|
|
SubstrTester("Note: in specialization C<Smi> requested here", 10,
|
|
|
|
10)});
|
|
|
|
|
|
|
|
ExpectFailingCompilation(
|
|
|
|
R"(
|
|
|
|
macro A<T: type extends HeapObject>() {}
|
|
|
|
macro B<T: type>() {
|
|
|
|
A<T>();
|
|
|
|
}
|
|
|
|
struct C<T: type> {
|
2019-12-14 22:25:51 +00:00
|
|
|
macro Method() {
|
2019-11-15 16:27:27 +00:00
|
|
|
B<T>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
macro D(_b: C<Smi>) {}
|
|
|
|
)",
|
|
|
|
SubstrVector{
|
|
|
|
SubstrTester("cannot find suitable callable", 4, 7),
|
|
|
|
SubstrTester("Note: in specialization B<Smi> requested here", 8, 9),
|
|
|
|
SubstrTester("Note: in specialization C<Smi> requested here", 11,
|
|
|
|
5)});
|
|
|
|
}
|
2019-12-05 13:30:12 +00:00
|
|
|
#endif
|
2019-11-15 16:27:27 +00:00
|
|
|
|
2019-12-18 15:33:16 +00:00
|
|
|
TEST(Torque, Enums) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
extern enum MyEnum {
|
|
|
|
kValue0,
|
|
|
|
kValue1,
|
|
|
|
kValue2,
|
|
|
|
kValue3
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
extern enum MyEmptyEnum {
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("unexpected token \"}\""));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, EnumInTypeswitch) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
extern enum MyEnum extends Smi {
|
|
|
|
kA,
|
|
|
|
kB,
|
|
|
|
kC
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(v : MyEnum): Smi {
|
|
|
|
typeswitch(v) {
|
|
|
|
case (MyEnum::kA | MyEnum::kB): {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
case (MyEnum::kC): {
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
extern enum MyEnum extends Smi {
|
|
|
|
kA,
|
|
|
|
kB,
|
|
|
|
kC,
|
|
|
|
...
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(v : MyEnum): Smi {
|
|
|
|
typeswitch(v) {
|
|
|
|
case (MyEnum::kC): {
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
case (MyEnum::kA | MyEnum::kB): {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
case (MyEnum): {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
}
|
|
|
|
|
2020-03-17 18:53:51 +00:00
|
|
|
TEST(Torque, ConstClassFields) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
const x: int32;
|
|
|
|
y: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo, n: int32) {
|
|
|
|
const _x: int32 = o.x;
|
|
|
|
o.y = n;
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
const x: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo, n: int32) {
|
|
|
|
o.x = n;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("cannot assign to const value"));
|
|
|
|
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
s: Bar;
|
|
|
|
}
|
|
|
|
struct Bar {
|
|
|
|
const x: int32;
|
|
|
|
y: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo, n: int32) {
|
|
|
|
const _x: int32 = o.s.x;
|
|
|
|
// Assigning a struct as a value is OK, even when the struct contains
|
|
|
|
// const fields.
|
|
|
|
o.s = Bar{x: n, y: n};
|
|
|
|
o.s.y = n;
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
const s: Bar;
|
|
|
|
}
|
|
|
|
struct Bar {
|
|
|
|
const x: int32;
|
|
|
|
y: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo, n: int32) {
|
|
|
|
o.s.y = n;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("cannot assign to const value"));
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
s: Bar;
|
|
|
|
}
|
|
|
|
struct Bar {
|
|
|
|
const x: int32;
|
|
|
|
y: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo, n: int32) {
|
|
|
|
o.s.x = n;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("cannot assign to const value"));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Torque, References) {
|
|
|
|
ExpectSuccessfulCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
const x: int32;
|
|
|
|
y: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo, n: int32) {
|
|
|
|
const constRefX: const &int32 = &o.x;
|
|
|
|
const refY: &int32 = &o.y;
|
|
|
|
const constRefY: const &int32 = refY;
|
|
|
|
const _x: int32 = *constRefX;
|
|
|
|
const _y1: int32 = *refY;
|
|
|
|
const _y2: int32 = *constRefY;
|
|
|
|
*refY = n;
|
|
|
|
let r: const &int32 = constRefX;
|
|
|
|
r = constRefY;
|
|
|
|
}
|
|
|
|
)");
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
const x: int32;
|
|
|
|
y: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo) {
|
|
|
|
const _refX: &int32 = &o.x;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("cannot use expression of type const "
|
|
|
|
"&int32 as a value of type &int32"));
|
|
|
|
|
|
|
|
ExpectFailingCompilation(R"(
|
|
|
|
class Foo extends HeapObject {
|
|
|
|
const x: int32;
|
|
|
|
y: int32;
|
|
|
|
}
|
|
|
|
|
|
|
|
@export
|
|
|
|
macro Test(implicit context: Context)(o: Foo, n: int32) {
|
|
|
|
const constRefX: const &int32 = &o.x;
|
|
|
|
*constRefX = n;
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr("cannot assign to const value"));
|
|
|
|
}
|
|
|
|
|
2020-05-01 13:58:04 +00:00
|
|
|
TEST(Torque, CatchFirstHandler) {
|
|
|
|
ExpectFailingCompilation(
|
|
|
|
R"(
|
|
|
|
@export
|
|
|
|
macro Test() {
|
|
|
|
try {
|
|
|
|
} label Foo {
|
|
|
|
} catch (e) {}
|
|
|
|
}
|
|
|
|
)",
|
|
|
|
HasSubstr(
|
|
|
|
"catch handler always has to be first, before any label handler"));
|
|
|
|
}
|
|
|
|
|
2018-10-15 22:31:24 +00:00
|
|
|
} // namespace torque
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|