Index: src/interpreter/mkpeephole.cc |
diff --git a/src/interpreter/mkpeephole.cc b/src/interpreter/mkpeephole.cc |
deleted file mode 100644 |
index d89390d45d125f9ca2d6dfaf619ce911794bdcb6..0000000000000000000000000000000000000000 |
--- a/src/interpreter/mkpeephole.cc |
+++ /dev/null |
@@ -1,387 +0,0 @@ |
-// Copyright 2016 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. |
- |
-#include <array> |
-#include <fstream> |
-#include <map> |
-#include <string> |
-#include <vector> |
- |
-#include "src/globals.h" |
-#include "src/interpreter/bytecode-peephole-table.h" |
-#include "src/interpreter/bytecodes.h" |
- |
-namespace v8 { |
-namespace internal { |
- |
-namespace interpreter { |
- |
-const char* ActionName(PeepholeAction action) { |
- switch (action) { |
-#define CASE(Name) \ |
- case PeepholeAction::k##Name: \ |
- return "PeepholeAction::k" #Name; |
- PEEPHOLE_ACTION_LIST(CASE) |
-#undef CASE |
- default: |
- UNREACHABLE(); |
- return ""; |
- } |
-} |
- |
-std::string BytecodeName(Bytecode bytecode) { |
- return "Bytecode::k" + std::string(Bytecodes::ToString(bytecode)); |
-} |
- |
-class PeepholeActionTableWriter final { |
- public: |
- static const size_t kNumberOfBytecodes = |
- static_cast<size_t>(Bytecode::kLast) + 1; |
- typedef std::array<PeepholeActionAndData, kNumberOfBytecodes> Row; |
- |
- void BuildTable(); |
- void Write(std::ostream& os); |
- |
- private: |
- static const char* kIndent; |
- static const char* kNamespaceElements[]; |
- |
- void WriteHeader(std::ostream& os); |
- void WriteIncludeFiles(std::ostream& os); |
- void WriteClassMethods(std::ostream& os); |
- void WriteUniqueRows(std::ostream& os); |
- void WriteRowMap(std::ostream& os); |
- void WriteRow(std::ostream& os, size_t row_index); |
- void WriteOpenNamespace(std::ostream& os); |
- void WriteCloseNamespace(std::ostream& os); |
- |
- PeepholeActionAndData LookupActionAndData(Bytecode last, Bytecode current); |
- void BuildRow(Bytecode last, Row* row); |
- size_t HashRow(const Row* row); |
- void InsertRow(size_t row_index, const Row* const row, size_t row_hash, |
- std::map<size_t, size_t>* hash_to_row_map); |
- bool RowsEqual(const Row* const first, const Row* const second); |
- |
- std::vector<Row>* table() { return &table_; } |
- |
- // Table of unique rows. |
- std::vector<Row> table_; |
- |
- // Mapping of row index to unique row index. |
- std::array<size_t, kNumberOfBytecodes> row_map_; |
-}; |
- |
-const char* PeepholeActionTableWriter::kIndent = " "; |
-const char* PeepholeActionTableWriter::kNamespaceElements[] = {"v8", "internal", |
- "interpreter"}; |
- |
-// static |
-PeepholeActionAndData PeepholeActionTableWriter::LookupActionAndData( |
- Bytecode last, Bytecode current) { |
- // Optimize various accumulator loads followed by store accumulator |
- // to an equivalent register load and loading the accumulator with |
- // the register. The latter accumulator load can often be elided as |
- // it is side-effect free and often followed by another accumulator |
- // load so can be elided. |
- if (current == Bytecode::kStar) { |
- switch (last) { |
- case Bytecode::kLdaNamedProperty: |
- return {PeepholeAction::kTransformLdaStarToLdrLdarAction, |
- Bytecode::kLdrNamedProperty}; |
- case Bytecode::kLdaKeyedProperty: |
- return {PeepholeAction::kTransformLdaStarToLdrLdarAction, |
- Bytecode::kLdrKeyedProperty}; |
- case Bytecode::kLdaGlobal: |
- return {PeepholeAction::kTransformLdaStarToLdrLdarAction, |
- Bytecode::kLdrGlobal}; |
- case Bytecode::kLdaContextSlot: |
- return {PeepholeAction::kTransformLdaStarToLdrLdarAction, |
- Bytecode::kLdrContextSlot}; |
- case Bytecode::kLdaUndefined: |
- return {PeepholeAction::kTransformLdaStarToLdrLdarAction, |
- Bytecode::kLdrUndefined}; |
- default: |
- break; |
- } |
- } |
- |
- // ToName optimizations: remove unnecessary ToName bytecodes. |
- if (current == Bytecode::kToName) { |
- if (last == Bytecode::kLdaConstant) { |
- return {PeepholeAction::kElideCurrentIfLoadingNameConstantAction, |
- Bytecode::kIllegal}; |
- } else if (Bytecodes::PutsNameInAccumulator(last)) { |
- return {PeepholeAction::kElideCurrentAction, Bytecode::kIllegal}; |
- } |
- } |
- |
- // Nop are placeholders for holding source position information and can be |
- // elided if there is no source information. |
- if (last == Bytecode::kNop) { |
- if (Bytecodes::IsJump(current)) { |
- return {PeepholeAction::kElideLastBeforeJumpAction, Bytecode::kIllegal}; |
- } else { |
- return {PeepholeAction::kElideLastAction, Bytecode::kIllegal}; |
- } |
- } |
- |
- // The accumulator is invisible to the debugger. If there is a sequence |
- // of consecutive accumulator loads (that don't have side effects) then |
- // only the final load is potentially visible. |
- if (Bytecodes::IsAccumulatorLoadWithoutEffects(last) && |
- Bytecodes::IsAccumulatorLoadWithoutEffects(current)) { |
- return {PeepholeAction::kElideLastAction, Bytecode::kIllegal}; |
- } |
- |
- // The current instruction clobbers the accumulator without reading |
- // it. The load in the last instruction can be elided as it has no |
- // effect. |
- if (Bytecodes::IsAccumulatorLoadWithoutEffects(last) && |
- Bytecodes::GetAccumulatorUse(current) == AccumulatorUse::kWrite) { |
- return {PeepholeAction::kElideLastAction, Bytecode::kIllegal}; |
- } |
- |
- // Ldar and Star make the accumulator and register hold equivalent |
- // values. Only the first bytecode is needed if there's a sequence |
- // of back-to-back Ldar and Star bytecodes with the same operand. |
- if (Bytecodes::IsLdarOrStar(last) && Bytecodes::IsLdarOrStar(current)) { |
- return {PeepholeAction::kElideCurrentIfOperand0MatchesAction, |
- Bytecode::kIllegal}; |
- } |
- |
- // Remove ToBoolean coercion from conditional jumps where possible. |
- if (Bytecodes::WritesBooleanToAccumulator(last)) { |
- if (Bytecodes::IsJumpIfToBoolean(current)) { |
- return {PeepholeAction::kChangeJumpBytecodeAction, |
- Bytecodes::GetJumpWithoutToBoolean(current)}; |
- } else if (current == Bytecode::kToBooleanLogicalNot) { |
- return {PeepholeAction::kChangeBytecodeAction, Bytecode::kLogicalNot}; |
- } |
- } |
- |
- // Fuse LdaSmi followed by binary op to produce binary op with a |
- // immediate integer argument. This savaes on dispatches and size. |
- if (last == Bytecode::kLdaSmi) { |
- switch (current) { |
- case Bytecode::kAdd: |
- return {PeepholeAction::kTransformLdaSmiBinaryOpToBinaryOpWithSmiAction, |
- Bytecode::kAddSmi}; |
- case Bytecode::kSub: |
- return {PeepholeAction::kTransformLdaSmiBinaryOpToBinaryOpWithSmiAction, |
- Bytecode::kSubSmi}; |
- case Bytecode::kBitwiseAnd: |
- return {PeepholeAction::kTransformLdaSmiBinaryOpToBinaryOpWithSmiAction, |
- Bytecode::kBitwiseAndSmi}; |
- case Bytecode::kBitwiseOr: |
- return {PeepholeAction::kTransformLdaSmiBinaryOpToBinaryOpWithSmiAction, |
- Bytecode::kBitwiseOrSmi}; |
- case Bytecode::kShiftLeft: |
- return {PeepholeAction::kTransformLdaSmiBinaryOpToBinaryOpWithSmiAction, |
- Bytecode::kShiftLeftSmi}; |
- case Bytecode::kShiftRight: |
- return {PeepholeAction::kTransformLdaSmiBinaryOpToBinaryOpWithSmiAction, |
- Bytecode::kShiftRightSmi}; |
- default: |
- break; |
- } |
- } |
- |
- // Fuse LdaZero followed by binary op to produce binary op with a |
- // zero immediate argument. This saves dispatches, but not size. |
- if (last == Bytecode::kLdaZero) { |
- switch (current) { |
- case Bytecode::kAdd: |
- return { |
- PeepholeAction::kTransformLdaZeroBinaryOpToBinaryOpWithZeroAction, |
- Bytecode::kAddSmi}; |
- case Bytecode::kSub: |
- return { |
- PeepholeAction::kTransformLdaZeroBinaryOpToBinaryOpWithZeroAction, |
- Bytecode::kSubSmi}; |
- case Bytecode::kBitwiseAnd: |
- return { |
- PeepholeAction::kTransformLdaZeroBinaryOpToBinaryOpWithZeroAction, |
- Bytecode::kBitwiseAndSmi}; |
- case Bytecode::kBitwiseOr: |
- return { |
- PeepholeAction::kTransformLdaZeroBinaryOpToBinaryOpWithZeroAction, |
- Bytecode::kBitwiseOrSmi}; |
- case Bytecode::kShiftLeft: |
- return { |
- PeepholeAction::kTransformLdaZeroBinaryOpToBinaryOpWithZeroAction, |
- Bytecode::kShiftLeftSmi}; |
- case Bytecode::kShiftRight: |
- return { |
- PeepholeAction::kTransformLdaZeroBinaryOpToBinaryOpWithZeroAction, |
- Bytecode::kShiftRightSmi}; |
- default: |
- break; |
- } |
- } |
- |
- // If there is no last bytecode to optimize against, store the incoming |
- // bytecode or for jumps emit incoming bytecode immediately. |
- if (last == Bytecode::kIllegal) { |
- if (Bytecodes::IsJump(current)) { |
- return {PeepholeAction::kUpdateLastJumpAction, Bytecode::kIllegal}; |
- } else if (current == Bytecode::kNop) { |
- return {PeepholeAction::kUpdateLastIfSourceInfoPresentAction, |
- Bytecode::kIllegal}; |
- } else { |
- return {PeepholeAction::kUpdateLastAction, Bytecode::kIllegal}; |
- } |
- } |
- |
- // No matches, take the default action. |
- if (Bytecodes::IsJump(current)) { |
- return {PeepholeAction::kDefaultJumpAction, Bytecode::kIllegal}; |
- } else { |
- return {PeepholeAction::kDefaultAction, Bytecode::kIllegal}; |
- } |
-} |
- |
-void PeepholeActionTableWriter::Write(std::ostream& os) { |
- WriteHeader(os); |
- WriteIncludeFiles(os); |
- WriteOpenNamespace(os); |
- WriteUniqueRows(os); |
- WriteRowMap(os); |
- WriteClassMethods(os); |
- WriteCloseNamespace(os); |
-} |
- |
-void PeepholeActionTableWriter::WriteHeader(std::ostream& os) { |
- os << "// Copyright 2016 the V8 project authors. All rights reserved.\n" |
- << "// Use of this source code is governed by a BSD-style license that\n" |
- << "// can be found in the LICENSE file.\n\n" |
- << "// Autogenerated by " __FILE__ ". Do not edit.\n\n"; |
-} |
- |
-void PeepholeActionTableWriter::WriteIncludeFiles(std::ostream& os) { |
- os << "#include \"src/interpreter/bytecode-peephole-table.h\"\n\n"; |
-} |
- |
-void PeepholeActionTableWriter::WriteUniqueRows(std::ostream& os) { |
- os << "const PeepholeActionAndData PeepholeActionTable::row_data_[" |
- << table_.size() << "][" << kNumberOfBytecodes << "] = {\n"; |
- for (size_t i = 0; i < table_.size(); ++i) { |
- os << "{\n"; |
- WriteRow(os, i); |
- os << "},\n"; |
- } |
- os << "};\n\n"; |
-} |
- |
-void PeepholeActionTableWriter::WriteRowMap(std::ostream& os) { |
- os << "const PeepholeActionAndData* const PeepholeActionTable::row_[" |
- << kNumberOfBytecodes << "] = {\n"; |
- for (size_t i = 0; i < kNumberOfBytecodes; ++i) { |
- os << kIndent << " PeepholeActionTable::row_data_[" << row_map_[i] |
- << "], \n"; |
- } |
- os << "};\n\n"; |
-} |
- |
-void PeepholeActionTableWriter::WriteRow(std::ostream& os, size_t row_index) { |
- const Row row = table_.at(row_index); |
- for (PeepholeActionAndData action_data : row) { |
- os << kIndent << "{" << ActionName(action_data.action) << "," |
- << BytecodeName(action_data.bytecode) << "},\n"; |
- } |
-} |
- |
-void PeepholeActionTableWriter::WriteOpenNamespace(std::ostream& os) { |
- for (auto element : kNamespaceElements) { |
- os << "namespace " << element << " {\n"; |
- } |
- os << "\n"; |
-} |
- |
-void PeepholeActionTableWriter::WriteCloseNamespace(std::ostream& os) { |
- for (auto element : kNamespaceElements) { |
- os << "} // namespace " << element << "\n"; |
- } |
-} |
- |
-void PeepholeActionTableWriter::WriteClassMethods(std::ostream& os) { |
- os << "// static\n" |
- << "const PeepholeActionAndData*\n" |
- << "PeepholeActionTable::Lookup(Bytecode last, Bytecode current) {\n" |
- << kIndent |
- << "return &row_[Bytecodes::ToByte(last)][Bytecodes::ToByte(current)];\n" |
- << "}\n\n"; |
-} |
- |
-void PeepholeActionTableWriter::BuildTable() { |
- std::map<size_t, size_t> hash_to_row_map; |
- Row row; |
- for (size_t i = 0; i < kNumberOfBytecodes; ++i) { |
- uint8_t byte_value = static_cast<uint8_t>(i); |
- Bytecode last = Bytecodes::FromByte(byte_value); |
- BuildRow(last, &row); |
- size_t row_hash = HashRow(&row); |
- InsertRow(i, &row, row_hash, &hash_to_row_map); |
- } |
-} |
- |
-void PeepholeActionTableWriter::BuildRow(Bytecode last, Row* row) { |
- for (size_t i = 0; i < kNumberOfBytecodes; ++i) { |
- uint8_t byte_value = static_cast<uint8_t>(i); |
- Bytecode current = Bytecodes::FromByte(byte_value); |
- PeepholeActionAndData action_data = LookupActionAndData(last, current); |
- row->at(i) = action_data; |
- } |
-} |
- |
-// static |
-bool PeepholeActionTableWriter::RowsEqual(const Row* const first, |
- const Row* const second) { |
- return memcmp(first, second, sizeof(*first)) == 0; |
-} |
- |
-// static |
-void PeepholeActionTableWriter::InsertRow( |
- size_t row_index, const Row* const row, size_t row_hash, |
- std::map<size_t, size_t>* hash_to_row_map) { |
- // Insert row if no existing row matches, otherwise use existing row. |
- auto iter = hash_to_row_map->find(row_hash); |
- if (iter == hash_to_row_map->end()) { |
- row_map_[row_index] = table()->size(); |
- table()->push_back(*row); |
- } else { |
- row_map_[row_index] = iter->second; |
- |
- // If the following DCHECK fails, the HashRow() is not adequate. |
- DCHECK(RowsEqual(&table()->at(iter->second), row)); |
- } |
-} |
- |
-// static |
-size_t PeepholeActionTableWriter::HashRow(const Row* row) { |
- static const size_t kHashShift = 3; |
- std::size_t result = (1u << 31) - 1u; |
- const uint8_t* raw_data = reinterpret_cast<const uint8_t*>(row); |
- for (size_t i = 0; i < sizeof(*row); ++i) { |
- size_t top_bits = result >> (kBitsPerByte * sizeof(size_t) - kHashShift); |
- result = (result << kHashShift) ^ top_bits ^ raw_data[i]; |
- } |
- return result; |
-} |
- |
-} // namespace interpreter |
-} // namespace internal |
-} // namespace v8 |
- |
-int main(int argc, const char* argv[]) { |
- CHECK_EQ(argc, 2); |
- |
- std::ofstream ofs(argv[1], std::ofstream::trunc); |
- v8::internal::interpreter::PeepholeActionTableWriter writer; |
- writer.BuildTable(); |
- writer.Write(ofs); |
- ofs.flush(); |
- ofs.close(); |
- |
- return 0; |
-} |