/* This file is part of the sirit project. * Copyright (c) 2018 ReinUsesLisp * This software may be used and distributed according to the terms of the GNU * Lesser General Public License version 3 or any later version. */ #include #include #include "common_types.h" #include "literal_number.h" #include "literal_string.h" #include "op.h" #include "operand.h" namespace Sirit { Op::Op(spv::Op opcode, std::optional id, Id result_type) : opcode(opcode), result_type(result_type), id(id) { operand_type = OperandType::Op; } Op::~Op() = default; void Op::Fetch(Stream& stream) const { assert(id.has_value()); stream.Write(id.value()); } u16 Op::GetWordCount() const { return 1; } bool Op::operator==(const Operand& other) const { if (operand_type != other.GetType()) { return false; } const Op& op = dynamic_cast(other); if (op.opcode == opcode && result_type == op.result_type && operands.size() == op.operands.size()) { for (std::size_t i = 0; i < operands.size(); i++) { if (*operands[i] != *op.operands[i]) { return false; } } return true; } return false; } std::size_t Op::Hash() const { std::size_t hash = Operand::Hash(); hash ^= static_cast(opcode) << 20; if (result_type) { hash ^= result_type->Hash() << 16; } hash ^= static_cast(id.value_or(0)) << 8; std::size_t wrap = 32; for (const auto operand : operands) { wrap = (wrap + 7) % (sizeof(std::size_t) * CHAR_BIT); hash ^= operand->Hash() << wrap; } return hash; } void Op::Write(Stream& stream) const { stream.Write(static_cast(opcode)); stream.Write(WordCount()); if (result_type) { result_type->Fetch(stream); } if (id.has_value()) { stream.Write(id.value()); } for (const auto* operand : operands) { operand->Fetch(stream); } } void Op::Sink(Operand* operand) { Add(static_cast(operand)); operand_store.push_back(std::unique_ptr(operand)); } void Op::Sink(const std::vector& operands_) { for (auto* operand : operands_) { Sink(operand); } } void Op::Add(const Literal& literal) { Operand* operand = [&]() { switch (literal.index()) { case 0: return LiteralNumber::Create(std::get<0>(literal)); case 1: return LiteralNumber::Create(std::get<1>(literal)); case 2: return LiteralNumber::Create(std::get<2>(literal)); case 3: return LiteralNumber::Create(std::get<3>(literal)); case 4: return LiteralNumber::Create(std::get<4>(literal)); case 5: return LiteralNumber::Create(std::get<5>(literal)); default: assert(!"Invalid literal type"); abort(); } }(); Sink(operand); } void Op::Add(const std::vector& literals) { for (const auto& literal : literals) { Add(literal); } } void Op::Add(const Operand* operand) { operands.push_back(operand); } void Op::Add(u32 integer) { Sink(LiteralNumber::Create(integer)); } void Op::Add(std::string string) { Sink(new LiteralString(std::move(string))); } void Op::Add(const std::vector& ids) { operands.insert(operands.end(), ids.begin(), ids.end()); } u16 Op::WordCount() const { u16 count = 1; if (result_type) { count++; } if (id.has_value()) { count++; } for (const Operand* operand : operands) { count += operand->GetWordCount(); } return count; } } // namespace Sirit