forked from eden-emu/eden
		
	
		
			
				
	
	
		
			149 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2021 yuzu Emulator Project
 | |
| // Licensed under GPLv2 or any later version
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <array>
 | |
| #include <bit>
 | |
| #include <memory>
 | |
| #include <string_view>
 | |
| 
 | |
| #include "common/common_types.h"
 | |
| #include "shader_recompiler/exception.h"
 | |
| #include "shader_recompiler/frontend/maxwell/decode.h"
 | |
| #include "shader_recompiler/frontend/maxwell/opcodes.h"
 | |
| 
 | |
| namespace Shader::Maxwell {
 | |
| namespace {
 | |
| struct MaskValue {
 | |
|     u64 mask;
 | |
|     u64 value;
 | |
| };
 | |
| 
 | |
| constexpr MaskValue MaskValueFromEncoding(const char* encoding) {
 | |
|     u64 mask{};
 | |
|     u64 value{};
 | |
|     u64 bit{u64(1) << 63};
 | |
|     while (*encoding) {
 | |
|         switch (*encoding) {
 | |
|         case '0':
 | |
|             mask |= bit;
 | |
|             break;
 | |
|         case '1':
 | |
|             mask |= bit;
 | |
|             value |= bit;
 | |
|             break;
 | |
|         case '-':
 | |
|             break;
 | |
|         case ' ':
 | |
|             break;
 | |
|         default:
 | |
|             throw LogicError("Invalid encoding character '{}'", *encoding);
 | |
|         }
 | |
|         ++encoding;
 | |
|         if (*encoding != ' ') {
 | |
|             bit >>= 1;
 | |
|         }
 | |
|     }
 | |
|     return MaskValue{.mask = mask, .value = value};
 | |
| }
 | |
| 
 | |
| struct InstEncoding {
 | |
|     MaskValue mask_value;
 | |
|     Opcode opcode;
 | |
| };
 | |
| constexpr std::array UNORDERED_ENCODINGS{
 | |
| #define INST(name, cute, encode)                                                                   \
 | |
|     InstEncoding{                                                                                  \
 | |
|         .mask_value{MaskValueFromEncoding(encode)},                                                \
 | |
|         .opcode = Opcode::name,                                                                    \
 | |
|     },
 | |
| #include "maxwell.inc"
 | |
| #undef INST
 | |
| };
 | |
| 
 | |
| constexpr auto SortedEncodings() {
 | |
|     std::array encodings{UNORDERED_ENCODINGS};
 | |
|     std::ranges::sort(encodings, [](const InstEncoding& lhs, const InstEncoding& rhs) {
 | |
|         return std::popcount(lhs.mask_value.mask) > std::popcount(rhs.mask_value.mask);
 | |
|     });
 | |
|     return encodings;
 | |
| }
 | |
| constexpr auto ENCODINGS{SortedEncodings()};
 | |
| 
 | |
| constexpr int WidestLeftBits() {
 | |
|     int bits{64};
 | |
|     for (const InstEncoding& encoding : ENCODINGS) {
 | |
|         bits = std::min(bits, std::countr_zero(encoding.mask_value.mask));
 | |
|     }
 | |
|     return 64 - bits;
 | |
| }
 | |
| constexpr int WIDEST_LEFT_BITS{WidestLeftBits()};
 | |
| constexpr int MASK_SHIFT{64 - WIDEST_LEFT_BITS};
 | |
| 
 | |
| constexpr size_t ToFastLookupIndex(u64 value) {
 | |
|     return static_cast<size_t>(value >> MASK_SHIFT);
 | |
| }
 | |
| 
 | |
| constexpr size_t FastLookupSize() {
 | |
|     size_t max_width{};
 | |
|     for (const InstEncoding& encoding : ENCODINGS) {
 | |
|         max_width = std::max(max_width, ToFastLookupIndex(encoding.mask_value.mask));
 | |
|     }
 | |
|     return max_width + 1;
 | |
| }
 | |
| constexpr size_t FAST_LOOKUP_SIZE{FastLookupSize()};
 | |
| 
 | |
| struct InstInfo {
 | |
|     [[nodiscard]] u64 Mask() const noexcept {
 | |
|         return static_cast<u64>(high_mask) << MASK_SHIFT;
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] u64 Value() const noexcept {
 | |
|         return static_cast<u64>(high_value) << MASK_SHIFT;
 | |
|     }
 | |
| 
 | |
|     u16 high_mask;
 | |
|     u16 high_value;
 | |
|     Opcode opcode;
 | |
| };
 | |
| 
 | |
| constexpr auto MakeFastLookupTableIndex(size_t index) {
 | |
|     std::array<InstInfo, 2> encodings{};
 | |
|     size_t element{};
 | |
|     for (const auto& encoding : ENCODINGS) {
 | |
|         const size_t mask{ToFastLookupIndex(encoding.mask_value.mask)};
 | |
|         const size_t value{ToFastLookupIndex(encoding.mask_value.value)};
 | |
|         if ((index & mask) == value) {
 | |
|             encodings.at(element) = InstInfo{
 | |
|                 .high_mask = static_cast<u16>(encoding.mask_value.mask >> MASK_SHIFT),
 | |
|                 .high_value = static_cast<u16>(encoding.mask_value.value >> MASK_SHIFT),
 | |
|                 .opcode = encoding.opcode,
 | |
|             };
 | |
|             ++element;
 | |
|         }
 | |
|     }
 | |
|     return encodings;
 | |
| }
 | |
| 
 | |
| /*constexpr*/ auto MakeFastLookupTable() {
 | |
|     auto encodings{std::make_unique<std::array<std::array<InstInfo, 2>, FAST_LOOKUP_SIZE>>()};
 | |
|     for (size_t index = 0; index < FAST_LOOKUP_SIZE; ++index) {
 | |
|         (*encodings)[index] = MakeFastLookupTableIndex(index);
 | |
|     }
 | |
|     return encodings;
 | |
| }
 | |
| const auto FAST_LOOKUP_TABLE{MakeFastLookupTable()};
 | |
| } // Anonymous namespace
 | |
| 
 | |
| Opcode Decode(u64 insn) {
 | |
|     const auto& table{(*FAST_LOOKUP_TABLE)[ToFastLookupIndex(insn)]};
 | |
|     const auto it{std::ranges::find_if(
 | |
|         table, [insn](const InstInfo& info) { return (insn & info.Mask()) == info.Value(); })};
 | |
|     if (it == table.end()) {
 | |
|         throw NotImplementedException("Instruction 0x{:016x} is unknown / unimplemented", insn);
 | |
|     }
 | |
|     return it->opcode;
 | |
| }
 | |
| 
 | |
| } // namespace Shader::Maxwell
 | 
