2014-07-30 13:54:45 +00:00
|
|
|
// Copyright 2014 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.
|
|
|
|
|
|
|
|
#ifndef V8_CCTEST_COMPILER_GRAPH_BUILDER_TESTER_H_
|
|
|
|
#define V8_CCTEST_COMPILER_GRAPH_BUILDER_TESTER_H_
|
|
|
|
|
2016-08-31 08:49:14 +00:00
|
|
|
#include "src/compilation-info.h"
|
2014-07-30 13:54:45 +00:00
|
|
|
#include "src/compiler/common-operator.h"
|
2015-10-11 19:59:23 +00:00
|
|
|
#include "src/compiler/instruction-selector.h"
|
2015-04-30 14:51:50 +00:00
|
|
|
#include "src/compiler/linkage.h"
|
2014-07-30 13:54:45 +00:00
|
|
|
#include "src/compiler/machine-operator.h"
|
2015-07-22 18:28:23 +00:00
|
|
|
#include "src/compiler/operator-properties.h"
|
2015-04-30 14:51:50 +00:00
|
|
|
#include "src/compiler/pipeline.h"
|
2014-07-30 13:54:45 +00:00
|
|
|
#include "src/compiler/simplified-operator.h"
|
2015-10-28 13:32:17 +00:00
|
|
|
#include "test/cctest/cctest.h"
|
2014-07-30 13:54:45 +00:00
|
|
|
#include "test/cctest/compiler/call-tester.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace compiler {
|
|
|
|
|
|
|
|
class GraphAndBuilders {
|
|
|
|
public:
|
|
|
|
explicit GraphAndBuilders(Zone* zone)
|
|
|
|
: main_graph_(new (zone) Graph(zone)),
|
|
|
|
main_common_(zone),
|
2015-12-10 09:03:30 +00:00
|
|
|
main_machine_(zone, MachineType::PointerRepresentation(),
|
2016-07-22 20:55:03 +00:00
|
|
|
InstructionSelector::SupportedMachineOperatorFlags(),
|
|
|
|
InstructionSelector::AlignmentRequirements()),
|
2014-07-30 13:54:45 +00:00
|
|
|
main_simplified_(zone) {}
|
|
|
|
|
2015-08-11 15:23:04 +00:00
|
|
|
Graph* graph() const { return main_graph_; }
|
|
|
|
Zone* zone() const { return graph()->zone(); }
|
|
|
|
CommonOperatorBuilder* common() { return &main_common_; }
|
|
|
|
MachineOperatorBuilder* machine() { return &main_machine_; }
|
|
|
|
SimplifiedOperatorBuilder* simplified() { return &main_simplified_; }
|
|
|
|
|
2014-07-30 13:54:45 +00:00
|
|
|
protected:
|
2015-04-30 14:51:50 +00:00
|
|
|
// Prefixed with main_ to avoid naming conflicts.
|
2014-08-11 09:40:02 +00:00
|
|
|
Graph* main_graph_;
|
2014-07-30 13:54:45 +00:00
|
|
|
CommonOperatorBuilder main_common_;
|
|
|
|
MachineOperatorBuilder main_machine_;
|
|
|
|
SimplifiedOperatorBuilder main_simplified_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
template <typename ReturnType>
|
2015-05-29 14:05:39 +00:00
|
|
|
class GraphBuilderTester : public HandleAndZoneScope,
|
2015-08-11 15:23:04 +00:00
|
|
|
public GraphAndBuilders,
|
2015-07-23 08:25:25 +00:00
|
|
|
public CallHelper<ReturnType> {
|
2014-07-30 13:54:45 +00:00
|
|
|
public:
|
2015-12-10 09:03:30 +00:00
|
|
|
explicit GraphBuilderTester(MachineType p0 = MachineType::None(),
|
|
|
|
MachineType p1 = MachineType::None(),
|
|
|
|
MachineType p2 = MachineType::None(),
|
|
|
|
MachineType p3 = MachineType::None(),
|
|
|
|
MachineType p4 = MachineType::None())
|
2014-07-30 13:54:45 +00:00
|
|
|
: GraphAndBuilders(main_zone()),
|
2015-05-29 14:05:39 +00:00
|
|
|
CallHelper<ReturnType>(
|
2015-01-23 15:19:34 +00:00
|
|
|
main_isolate(),
|
2015-06-02 15:00:55 +00:00
|
|
|
CSignature::New(main_zone(), MachineTypeForC<ReturnType>(), p0, p1,
|
|
|
|
p2, p3, p4)),
|
2017-10-13 16:33:03 +00:00
|
|
|
effect_(nullptr),
|
|
|
|
return_(nullptr),
|
2015-04-30 14:51:50 +00:00
|
|
|
parameters_(main_zone()->template NewArray<Node*>(parameter_count())) {
|
2014-09-03 10:13:21 +00:00
|
|
|
Begin(static_cast<int>(parameter_count()));
|
2015-04-30 14:51:50 +00:00
|
|
|
InitParameters();
|
2014-07-30 13:54:45 +00:00
|
|
|
}
|
|
|
|
virtual ~GraphBuilderTester() {}
|
|
|
|
|
2015-04-30 14:51:50 +00:00
|
|
|
void GenerateCode() { Generate(); }
|
|
|
|
Node* Parameter(size_t index) {
|
2015-12-07 05:36:41 +00:00
|
|
|
CHECK_LT(index, parameter_count());
|
2015-04-30 14:51:50 +00:00
|
|
|
return parameters_[index];
|
|
|
|
}
|
|
|
|
|
2015-07-23 08:25:25 +00:00
|
|
|
Isolate* isolate() { return main_isolate(); }
|
|
|
|
Factory* factory() { return isolate()->factory(); }
|
2015-07-22 18:28:23 +00:00
|
|
|
|
|
|
|
// Initialize graph and builder.
|
|
|
|
void Begin(int num_parameters) {
|
2015-12-07 05:36:41 +00:00
|
|
|
CHECK_NULL(graph()->start());
|
2015-07-22 18:28:23 +00:00
|
|
|
Node* start = graph()->NewNode(common()->Start(num_parameters + 3));
|
|
|
|
graph()->SetStart(start);
|
|
|
|
effect_ = start;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Return(Node* value) {
|
2016-11-02 13:15:39 +00:00
|
|
|
Node* zero = graph()->NewNode(common()->Int32Constant(0));
|
|
|
|
return_ = graph()->NewNode(common()->Return(), zero, value, effect_,
|
|
|
|
graph()->start());
|
2017-10-13 16:33:03 +00:00
|
|
|
effect_ = nullptr;
|
2015-07-22 18:28:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close the graph.
|
|
|
|
void End() {
|
|
|
|
Node* end = graph()->NewNode(common()->End(1), return_);
|
|
|
|
graph()->SetEnd(end);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* PointerConstant(void* value) {
|
|
|
|
intptr_t intptr_value = reinterpret_cast<intptr_t>(value);
|
|
|
|
return kPointerSize == 8 ? NewNode(common()->Int64Constant(intptr_value))
|
|
|
|
: Int32Constant(static_cast<int>(intptr_value));
|
|
|
|
}
|
|
|
|
Node* Int32Constant(int32_t value) {
|
|
|
|
return NewNode(common()->Int32Constant(value));
|
|
|
|
}
|
|
|
|
Node* HeapConstant(Handle<HeapObject> object) {
|
2015-08-31 08:24:52 +00:00
|
|
|
return NewNode(common()->HeapConstant(object));
|
2015-07-22 18:28:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* BooleanNot(Node* a) { return NewNode(simplified()->BooleanNot(), a); }
|
|
|
|
|
|
|
|
Node* NumberEqual(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberEqual(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberLessThan(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberLessThan(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberLessThanOrEqual(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberLessThanOrEqual(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberAdd(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberAdd(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberSubtract(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberSubtract(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberMultiply(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberMultiply(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberDivide(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberDivide(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberModulus(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->NumberModulus(), a, b);
|
|
|
|
}
|
|
|
|
Node* NumberToInt32(Node* a) {
|
|
|
|
return NewNode(simplified()->NumberToInt32(), a);
|
|
|
|
}
|
|
|
|
Node* NumberToUint32(Node* a) {
|
|
|
|
return NewNode(simplified()->NumberToUint32(), a);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* StringEqual(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->StringEqual(), a, b);
|
|
|
|
}
|
|
|
|
Node* StringLessThan(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->StringLessThan(), a, b);
|
|
|
|
}
|
|
|
|
Node* StringLessThanOrEqual(Node* a, Node* b) {
|
|
|
|
return NewNode(simplified()->StringLessThanOrEqual(), a, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* ChangeTaggedToInt32(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeTaggedToInt32(), a);
|
|
|
|
}
|
|
|
|
Node* ChangeTaggedToUint32(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeTaggedToUint32(), a);
|
|
|
|
}
|
|
|
|
Node* ChangeTaggedToFloat64(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeTaggedToFloat64(), a);
|
|
|
|
}
|
|
|
|
Node* ChangeInt32ToTagged(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeInt32ToTagged(), a);
|
|
|
|
}
|
|
|
|
Node* ChangeUint32ToTagged(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeUint32ToTagged(), a);
|
|
|
|
}
|
|
|
|
Node* ChangeFloat64ToTagged(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeFloat64ToTagged(), a);
|
|
|
|
}
|
2016-05-02 10:23:02 +00:00
|
|
|
Node* ChangeTaggedToBit(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeTaggedToBit(), a);
|
2015-07-22 18:28:23 +00:00
|
|
|
}
|
2016-05-02 10:23:02 +00:00
|
|
|
Node* ChangeBitToTagged(Node* a) {
|
|
|
|
return NewNode(simplified()->ChangeBitToTagged(), a);
|
2015-07-22 18:28:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* LoadField(const FieldAccess& access, Node* object) {
|
|
|
|
return NewNode(simplified()->LoadField(access), object);
|
|
|
|
}
|
|
|
|
Node* StoreField(const FieldAccess& access, Node* object, Node* value) {
|
|
|
|
return NewNode(simplified()->StoreField(access), object, value);
|
|
|
|
}
|
|
|
|
Node* LoadElement(const ElementAccess& access, Node* object, Node* index) {
|
|
|
|
return NewNode(simplified()->LoadElement(access), object, index);
|
|
|
|
}
|
|
|
|
Node* StoreElement(const ElementAccess& access, Node* object, Node* index,
|
|
|
|
Node* value) {
|
|
|
|
return NewNode(simplified()->StoreElement(access), object, index, value);
|
|
|
|
}
|
2015-04-30 14:51:50 +00:00
|
|
|
|
2015-07-23 08:25:25 +00:00
|
|
|
Node* NewNode(const Operator* op) {
|
2017-10-13 16:33:03 +00:00
|
|
|
return MakeNode(op, 0, static_cast<Node**>(nullptr));
|
2015-07-23 08:25:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Node* NewNode(const Operator* op, Node* n1) { return MakeNode(op, 1, &n1); }
|
|
|
|
|
|
|
|
Node* NewNode(const Operator* op, Node* n1, Node* n2) {
|
|
|
|
Node* buffer[] = {n1, n2};
|
|
|
|
return MakeNode(op, arraysize(buffer), buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* NewNode(const Operator* op, Node* n1, Node* n2, Node* n3) {
|
|
|
|
Node* buffer[] = {n1, n2, n3};
|
|
|
|
return MakeNode(op, arraysize(buffer), buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* NewNode(const Operator* op, Node* n1, Node* n2, Node* n3, Node* n4) {
|
|
|
|
Node* buffer[] = {n1, n2, n3, n4};
|
|
|
|
return MakeNode(op, arraysize(buffer), buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* NewNode(const Operator* op, Node* n1, Node* n2, Node* n3, Node* n4,
|
|
|
|
Node* n5) {
|
|
|
|
Node* buffer[] = {n1, n2, n3, n4, n5};
|
|
|
|
return MakeNode(op, arraysize(buffer), buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* NewNode(const Operator* op, Node* n1, Node* n2, Node* n3, Node* n4,
|
|
|
|
Node* n5, Node* n6) {
|
|
|
|
Node* nodes[] = {n1, n2, n3, n4, n5, n6};
|
|
|
|
return MakeNode(op, arraysize(nodes), nodes);
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* NewNode(const Operator* op, int value_input_count,
|
|
|
|
Node** value_inputs) {
|
|
|
|
return MakeNode(op, value_input_count, value_inputs);
|
|
|
|
}
|
|
|
|
|
2015-10-27 00:37:27 +00:00
|
|
|
Handle<Code> GetCode() {
|
|
|
|
Generate();
|
|
|
|
return code_.ToHandleChecked();
|
|
|
|
}
|
|
|
|
|
2015-04-30 14:51:50 +00:00
|
|
|
protected:
|
2015-07-23 08:25:25 +00:00
|
|
|
Node* MakeNode(const Operator* op, int value_input_count,
|
|
|
|
Node** value_inputs) {
|
2015-12-07 05:36:41 +00:00
|
|
|
CHECK_EQ(op->ValueInputCount(), value_input_count);
|
2015-07-22 18:28:23 +00:00
|
|
|
|
2015-12-07 05:36:41 +00:00
|
|
|
CHECK(!OperatorProperties::HasContextInput(op));
|
2016-08-03 10:38:25 +00:00
|
|
|
CHECK(!OperatorProperties::HasFrameStateInput(op));
|
2015-07-22 18:28:23 +00:00
|
|
|
bool has_control = op->ControlInputCount() == 1;
|
|
|
|
bool has_effect = op->EffectInputCount() == 1;
|
|
|
|
|
2015-12-07 05:36:41 +00:00
|
|
|
CHECK_LT(op->ControlInputCount(), 2);
|
|
|
|
CHECK_LT(op->EffectInputCount(), 2);
|
2015-07-22 18:28:23 +00:00
|
|
|
|
2017-10-13 16:33:03 +00:00
|
|
|
Node* result = nullptr;
|
2015-07-22 18:28:23 +00:00
|
|
|
if (!has_control && !has_effect) {
|
2015-07-23 08:25:25 +00:00
|
|
|
result = graph()->NewNode(op, value_input_count, value_inputs);
|
2015-07-22 18:28:23 +00:00
|
|
|
} else {
|
|
|
|
int input_count_with_deps = value_input_count;
|
|
|
|
if (has_control) ++input_count_with_deps;
|
|
|
|
if (has_effect) ++input_count_with_deps;
|
|
|
|
Node** buffer = zone()->template NewArray<Node*>(input_count_with_deps);
|
|
|
|
memcpy(buffer, value_inputs, kPointerSize * value_input_count);
|
|
|
|
Node** current_input = buffer + value_input_count;
|
|
|
|
if (has_effect) {
|
|
|
|
*current_input++ = effect_;
|
|
|
|
}
|
|
|
|
if (has_control) {
|
|
|
|
*current_input++ = graph()->start();
|
|
|
|
}
|
2015-07-23 08:25:25 +00:00
|
|
|
result = graph()->NewNode(op, input_count_with_deps, buffer);
|
2015-07-22 18:28:23 +00:00
|
|
|
if (has_effect) {
|
|
|
|
effect_ = result;
|
|
|
|
}
|
|
|
|
// This graph builder does not support control flow.
|
|
|
|
CHECK_EQ(0, op->ControlOutputCount());
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-04-30 14:51:50 +00:00
|
|
|
virtual byte* Generate() {
|
|
|
|
if (code_.is_null()) {
|
|
|
|
Zone* zone = graph()->zone();
|
|
|
|
CallDescriptor* desc =
|
2015-06-02 15:00:55 +00:00
|
|
|
Linkage::GetSimplifiedCDescriptor(zone, this->csig_);
|
2017-11-15 14:36:57 +00:00
|
|
|
CompilationInfo info(ArrayVector("testing"), main_zone(), Code::STUB);
|
|
|
|
code_ = Pipeline::GenerateCodeForTesting(&info, main_isolate(), desc,
|
|
|
|
graph());
|
2015-07-31 09:12:29 +00:00
|
|
|
#ifdef ENABLE_DISASSEMBLER
|
|
|
|
if (!code_.is_null() && FLAG_print_opt_code) {
|
|
|
|
OFStream os(stdout);
|
|
|
|
code_.ToHandleChecked()->Disassemble("test code", os);
|
|
|
|
}
|
|
|
|
#endif
|
2015-04-30 14:51:50 +00:00
|
|
|
}
|
|
|
|
return code_.ToHandleChecked()->entry();
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitParameters() {
|
|
|
|
int param_count = static_cast<int>(parameter_count());
|
|
|
|
for (int i = 0; i < param_count; ++i) {
|
|
|
|
parameters_[i] = this->NewNode(common()->Parameter(i), graph()->start());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-02 15:00:55 +00:00
|
|
|
size_t parameter_count() const { return this->csig_->parameter_count(); }
|
2015-04-30 14:51:50 +00:00
|
|
|
|
|
|
|
private:
|
2015-07-22 18:28:23 +00:00
|
|
|
Node* effect_;
|
|
|
|
Node* return_;
|
2015-04-30 14:51:50 +00:00
|
|
|
Node** parameters_;
|
|
|
|
MaybeHandle<Code> code_;
|
2014-07-30 13:54:45 +00:00
|
|
|
};
|
2015-04-30 14:51:50 +00:00
|
|
|
|
2014-07-30 13:54:45 +00:00
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|
|
|
|
|
|
|
|
#endif // V8_CCTEST_COMPILER_GRAPH_BUILDER_TESTER_H_
|