dynarmic/src/frontend/A32/decoder/arm.h

54 lines
1.6 KiB
C
Raw Normal View History

2016-07-01 15:01:06 +02:00
/* This file is part of the dynarmic project.
* Copyright (c) 2016 MerryMage
2020-04-23 16:25:11 +02:00
* SPDX-License-Identifier: 0BSD
2016-07-01 15:01:06 +02:00
*
* Original version of table by Lioncash.
*/
#pragma once
#include <algorithm>
#include <functional>
#include <optional>
#include <vector>
2016-07-01 15:01:06 +02:00
#include "common/bit_util.h"
2016-07-01 15:01:06 +02:00
#include "common/common_types.h"
#include "frontend/decoder/decoder_detail.h"
#include "frontend/decoder/matcher.h"
2016-07-01 15:01:06 +02:00
namespace Dynarmic::A32 {
2016-07-01 15:01:06 +02:00
template <typename Visitor>
2018-01-01 16:23:56 +01:00
using ArmMatcher = Decoder::Matcher<Visitor, u32>;
2016-07-01 15:01:06 +02:00
template <typename V>
std::vector<ArmMatcher<V>> GetArmDecodeTable() {
std::vector<ArmMatcher<V>> table = {
2016-07-01 15:01:06 +02:00
#define INST(fn, name, bitstring) Decoder::detail::detail<ArmMatcher<V>>::GetMatcher(&V::fn, name, bitstring),
#include "arm.inc"
2016-07-01 15:01:06 +02:00
#undef INST
};
2016-07-01 15:01:06 +02:00
// If a matcher has more bits in its mask it is more specific, so it should come first.
std::stable_sort(table.begin(), table.end(), [](const auto& matcher1, const auto& matcher2) {
return Common::BitCount(matcher1.GetMask()) > Common::BitCount(matcher2.GetMask());
});
return table;
}
template<typename V>
std::optional<std::reference_wrapper<const ArmMatcher<V>>> DecodeArm(u32 instruction) {
static const auto table = GetArmDecodeTable<V>();
const auto matches_instruction = [instruction](const auto& matcher) { return matcher.Matches(instruction); };
2016-07-01 15:01:06 +02:00
auto iter = std::find_if(table.begin(), table.end(), matches_instruction);
return iter != table.end() ? std::optional<std::reference_wrapper<const ArmMatcher<V>>>(*iter) : std::nullopt;
2016-07-01 15:01:06 +02:00
}
} // namespace Dynarmic::A32