early-access version 1755
This commit is contained in:
parent
59785630ec
commit
0e52ad1522
23 changed files with 325 additions and 82 deletions
|
@ -1,7 +1,7 @@
|
||||||
yuzu emulator early access
|
yuzu emulator early access
|
||||||
=============
|
=============
|
||||||
|
|
||||||
This is the source code for early-access 1753.
|
This is the source code for early-access 1755.
|
||||||
|
|
||||||
## Legal Notice
|
## Legal Notice
|
||||||
|
|
||||||
|
|
70
externals/dynarmic/.github/workflows/build-and-test.yml
vendored
Executable file
70
externals/dynarmic/.github/workflows/build-and-test.yml
vendored
Executable file
|
@ -0,0 +1,70 @@
|
||||||
|
name: Build and Test
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
env:
|
||||||
|
BUILD_TYPE: Release
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
os: [ubuntu-latest, macos-latest]
|
||||||
|
cpu_detection: [0, 1]
|
||||||
|
fail-fast: false
|
||||||
|
|
||||||
|
runs-on: ${{matrix.os}}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
|
||||||
|
- name: Install build dependencies
|
||||||
|
if: ${{matrix.os == 'ubuntu-latest'}}
|
||||||
|
run: sudo apt-get install llvm ninja-build
|
||||||
|
|
||||||
|
- name: Install build dependencies
|
||||||
|
if: ${{matrix.os == 'macos-latest'}}
|
||||||
|
run: |
|
||||||
|
brew install llvm ninja
|
||||||
|
echo "/usr/local/opt/llvm/bin" >> $GITHUB_PATH
|
||||||
|
|
||||||
|
- name: Checkout dynarmic repo
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Checkout ext-boost repo
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
repository: MerryMage/ext-boost
|
||||||
|
path: externals/ext-boost
|
||||||
|
|
||||||
|
- name: Checkout unicorn repo
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
repository: MerryMage/unicorn
|
||||||
|
path: externals/unicorn
|
||||||
|
|
||||||
|
- name: Build unicorn
|
||||||
|
working-directory: externals/unicorn
|
||||||
|
run: UNICORN_ARCHS=aarch64,arm ./make.sh
|
||||||
|
|
||||||
|
- name: Configure CMake
|
||||||
|
run: >
|
||||||
|
cmake
|
||||||
|
-B ${{github.workspace}}/build
|
||||||
|
-DBoost_INCLUDE_DIRS=${{github.workspace}}/externals/ext-boost
|
||||||
|
-DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}}
|
||||||
|
-DDYNARMIC_ENABLE_CPU_FEATURE_DETECTION=${{matrix.cpu_detection}}
|
||||||
|
-DDYNARMIC_TESTS_USE_UNICORN=1
|
||||||
|
-DDYNARMIC_USE_LLVM=1
|
||||||
|
-DLIBUNICORN_INCLUDE_DIR=${{github.workspace}}/externals/unicorn/include
|
||||||
|
-DLIBUNICORN_LIBRARY=${{github.workspace}}/externals/unicorn/libunicorn.a
|
||||||
|
-G Ninja
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
working-directory: ${{github.workspace}}/build
|
||||||
|
run: ninja
|
||||||
|
|
||||||
|
- name: Test
|
||||||
|
env:
|
||||||
|
DYLD_FALLBACK_LIBRARY_PATH: ${{github.workspace}}/externals/unicorn
|
||||||
|
working-directory: ${{github.workspace}}/build
|
||||||
|
run: ctest --extra-verbose -C ${{env.BUILD_TYPE}}
|
18
externals/dynarmic/.github/workflows/clang-format.yml
vendored
Executable file
18
externals/dynarmic/.github/workflows/clang-format.yml
vendored
Executable file
|
@ -0,0 +1,18 @@
|
||||||
|
name: clang-format
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
clang-format:
|
||||||
|
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
|
||||||
|
- name: Checkout dynarmic repo
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- uses: DoozyX/clang-format-lint-action@v0.12
|
||||||
|
with:
|
||||||
|
source: 'src tests'
|
||||||
|
clangFormatVersion: 12
|
2
externals/dynarmic/README.md
vendored
2
externals/dynarmic/README.md
vendored
|
@ -1,7 +1,7 @@
|
||||||
Dynarmic
|
Dynarmic
|
||||||
========
|
========
|
||||||
|
|
||||||
[![Travis CI Build Status](https://api.travis-ci.org/MerryMage/dynarmic.svg?branch=master)](https://travis-ci.org/MerryMage/dynarmic/branches) [![Appveyor CI Build status](https://ci.appveyor.com/api/projects/status/maeiqr41rgm1innm/branch/master?svg=true)](https://ci.appveyor.com/project/MerryMage/dynarmic/branch/master)
|
[![Github Actions Build Status](https://github.com/MerryMage/dynarmic/actions/workflows/build-and-test.yml/badge.svg)](https://github.com/MerryMage/dynarmic/actions/workflows/build-and-test.yml) [![Appveyor CI Build status](https://ci.appveyor.com/api/projects/status/maeiqr41rgm1innm/branch/master?svg=true)](https://ci.appveyor.com/project/MerryMage/dynarmic/branch/master)
|
||||||
|
|
||||||
A dynamic recompiler for ARM.
|
A dynamic recompiler for ARM.
|
||||||
|
|
||||||
|
|
4
externals/dynarmic/externals/CMakeLists.txt
vendored
4
externals/dynarmic/externals/CMakeLists.txt
vendored
|
@ -28,7 +28,7 @@ target_include_directories(robin_map SYSTEM INTERFACE "$<BUILD_INTERFACE:${CMAKE
|
||||||
# vixl
|
# vixl
|
||||||
|
|
||||||
if (ARCHITECTURE STREQUAL "arm64")
|
if (ARCHITECTURE STREQUAL "arm64")
|
||||||
add_subdirectory(vixl)
|
add_subdirectory(vixl EXCLUDE_FROM_ALL)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# xbyak
|
# xbyak
|
||||||
|
@ -46,4 +46,4 @@ endif()
|
||||||
option(ZYDIS_BUILD_TOOLS "" OFF)
|
option(ZYDIS_BUILD_TOOLS "" OFF)
|
||||||
option(ZYDIS_BUILD_EXAMPLES "" OFF)
|
option(ZYDIS_BUILD_EXAMPLES "" OFF)
|
||||||
set(ZYDIS_ZYCORE_PATH "${CMAKE_CURRENT_LIST_DIR}/zycore" CACHE PATH "")
|
set(ZYDIS_ZYCORE_PATH "${CMAKE_CURRENT_LIST_DIR}/zycore" CACHE PATH "")
|
||||||
add_subdirectory(zydis)
|
add_subdirectory(zydis EXCLUDE_FROM_ALL)
|
||||||
|
|
|
@ -1680,7 +1680,9 @@ void A32EmitX64::EmitPatchMovRcx(CodePtr target_code_ptr) {
|
||||||
void A32EmitX64::Unpatch(const IR::LocationDescriptor& location) {
|
void A32EmitX64::Unpatch(const IR::LocationDescriptor& location) {
|
||||||
EmitX64::Unpatch(location);
|
EmitX64::Unpatch(location);
|
||||||
if (conf.HasOptimization(OptimizationFlag::FastDispatch)) {
|
if (conf.HasOptimization(OptimizationFlag::FastDispatch)) {
|
||||||
|
code.DisableWriting();
|
||||||
(*fast_dispatch_table_lookup)(location.Value()) = {};
|
(*fast_dispatch_table_lookup)(location.Value()) = {};
|
||||||
|
code.EnableWriting();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -62,7 +62,7 @@ A64EmitX64::A64EmitX64(BlockOfCode& code, A64::UserConfig conf, A64::Jit* jit_in
|
||||||
code.PreludeComplete();
|
code.PreludeComplete();
|
||||||
ClearFastDispatchTable();
|
ClearFastDispatchTable();
|
||||||
|
|
||||||
exception_handler.SetFastmemCallback([this](u64 rip_){
|
exception_handler.SetFastmemCallback([this](u64 rip_) {
|
||||||
return FastmemCallback(rip_);
|
return FastmemCallback(rip_);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -990,8 +990,7 @@ void A64EmitX64::EmitMemoryRead(A64EmitContext& ctx, IR::Inst* inst) {
|
||||||
Common::BitCast<u64>(code.getCurr()),
|
Common::BitCast<u64>(code.getCurr()),
|
||||||
Common::BitCast<u64>(wrapped_fn),
|
Common::BitCast<u64>(wrapped_fn),
|
||||||
*fastmem_marker,
|
*fastmem_marker,
|
||||||
}
|
});
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
// Use page table
|
// Use page table
|
||||||
ASSERT(conf.page_table);
|
ASSERT(conf.page_table);
|
||||||
|
@ -1045,8 +1044,7 @@ void A64EmitX64::EmitMemoryWrite(A64EmitContext& ctx, IR::Inst* inst) {
|
||||||
Common::BitCast<u64>(code.getCurr()),
|
Common::BitCast<u64>(code.getCurr()),
|
||||||
Common::BitCast<u64>(wrapped_fn),
|
Common::BitCast<u64>(wrapped_fn),
|
||||||
*fastmem_marker,
|
*fastmem_marker,
|
||||||
}
|
});
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
// Use page table
|
// Use page table
|
||||||
ASSERT(conf.page_table);
|
ASSERT(conf.page_table);
|
||||||
|
@ -1114,8 +1112,7 @@ void A64EmitX64::EmitA64ReadMemory128(A64EmitContext& ctx, IR::Inst* inst) {
|
||||||
Common::BitCast<u64>(code.getCurr()),
|
Common::BitCast<u64>(code.getCurr()),
|
||||||
Common::BitCast<u64>(wrapped_fn),
|
Common::BitCast<u64>(wrapped_fn),
|
||||||
*fastmem_marker,
|
*fastmem_marker,
|
||||||
}
|
});
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
// Use page table
|
// Use page table
|
||||||
ASSERT(conf.page_table);
|
ASSERT(conf.page_table);
|
||||||
|
@ -1187,8 +1184,7 @@ void A64EmitX64::EmitA64WriteMemory128(A64EmitContext& ctx, IR::Inst* inst) {
|
||||||
Common::BitCast<u64>(code.getCurr()),
|
Common::BitCast<u64>(code.getCurr()),
|
||||||
Common::BitCast<u64>(wrapped_fn),
|
Common::BitCast<u64>(wrapped_fn),
|
||||||
*fastmem_marker,
|
*fastmem_marker,
|
||||||
}
|
});
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
// Use page table
|
// Use page table
|
||||||
ASSERT(conf.page_table);
|
ASSERT(conf.page_table);
|
||||||
|
@ -1481,7 +1477,9 @@ void A64EmitX64::EmitPatchMovRcx(CodePtr target_code_ptr) {
|
||||||
void A64EmitX64::Unpatch(const IR::LocationDescriptor& location) {
|
void A64EmitX64::Unpatch(const IR::LocationDescriptor& location) {
|
||||||
EmitX64::Unpatch(location);
|
EmitX64::Unpatch(location);
|
||||||
if (conf.HasOptimization(OptimizationFlag::FastDispatch)) {
|
if (conf.HasOptimization(OptimizationFlag::FastDispatch)) {
|
||||||
|
code.DisableWriting();
|
||||||
(*fast_dispatch_table_lookup)(location.Value()) = {};
|
(*fast_dispatch_table_lookup)(location.Value()) = {};
|
||||||
|
code.EnableWriting();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,11 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <optional>
|
||||||
|
|
||||||
|
#include "dynarmic/common/bit_util.h"
|
||||||
#include "dynarmic/common/common_types.h"
|
#include "dynarmic/common/common_types.h"
|
||||||
|
#include "dynarmic/common/fp/rounding_mode.h"
|
||||||
|
|
||||||
namespace Dynarmic::Backend::X64 {
|
namespace Dynarmic::Backend::X64 {
|
||||||
|
|
||||||
|
@ -42,4 +46,60 @@ constexpr u8 b = 0b11001100;
|
||||||
constexpr u8 c = 0b10101010;
|
constexpr u8 c = 0b10101010;
|
||||||
} // namespace Tern
|
} // namespace Tern
|
||||||
|
|
||||||
|
// Opcodes for use with vfixupimm
|
||||||
|
enum class FpFixup : u8 {
|
||||||
|
A = 0b0000, // A
|
||||||
|
B = 0b0001, // B
|
||||||
|
QNaN_B = 0b0010, // QNaN with sign of B
|
||||||
|
IndefNaN = 0b0011, // Indefinite QNaN (Negative QNaN with no payload on x86)
|
||||||
|
NegInf = 0b0100, // -Infinity
|
||||||
|
PosInf = 0b0101, // +Infinity
|
||||||
|
Inf_B = 0b0110, // Infinity with sign of B
|
||||||
|
NegZero = 0b0111, // -0.0
|
||||||
|
PosZero = 0b1000, // +0.0
|
||||||
|
NegOne = 0b1001, // -1.0
|
||||||
|
PosOne = 0b1010, // +1.0
|
||||||
|
Half = 0b1011, // 0.5
|
||||||
|
Ninety = 0b1100, // 90.0
|
||||||
|
HalfPi = 0b1101, // PI/2
|
||||||
|
PosMax = 0b1110, // +{FLT_MAX,DBL_MAX}
|
||||||
|
NegMax = 0b1111, // -{FLT_MAX,DBL_MAX}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Generates 32-bit LUT for vfixupimm instruction
|
||||||
|
constexpr u32 FixupLUT(FpFixup src_qnan = FpFixup::A,
|
||||||
|
FpFixup src_snan = FpFixup::A,
|
||||||
|
FpFixup src_zero = FpFixup::A,
|
||||||
|
FpFixup src_posone = FpFixup::A,
|
||||||
|
FpFixup src_neginf = FpFixup::A,
|
||||||
|
FpFixup src_posinf = FpFixup::A,
|
||||||
|
FpFixup src_pos = FpFixup::A,
|
||||||
|
FpFixup src_neg = FpFixup::A) {
|
||||||
|
u32 fixup_lut = 0;
|
||||||
|
fixup_lut = Common::ModifyBits<0, 3, u32>(fixup_lut, static_cast<u32>(src_qnan));
|
||||||
|
fixup_lut = Common::ModifyBits<4, 7, u32>(fixup_lut, static_cast<u32>(src_snan));
|
||||||
|
fixup_lut = Common::ModifyBits<8, 11, u32>(fixup_lut, static_cast<u32>(src_zero));
|
||||||
|
fixup_lut = Common::ModifyBits<12, 15, u32>(fixup_lut, static_cast<u32>(src_posone));
|
||||||
|
fixup_lut = Common::ModifyBits<16, 19, u32>(fixup_lut, static_cast<u32>(src_neginf));
|
||||||
|
fixup_lut = Common::ModifyBits<20, 23, u32>(fixup_lut, static_cast<u32>(src_posinf));
|
||||||
|
fixup_lut = Common::ModifyBits<24, 27, u32>(fixup_lut, static_cast<u32>(src_pos));
|
||||||
|
fixup_lut = Common::ModifyBits<28, 31, u32>(fixup_lut, static_cast<u32>(src_neg));
|
||||||
|
return fixup_lut;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr std::optional<int> ConvertRoundingModeToX64Immediate(FP::RoundingMode rounding_mode) {
|
||||||
|
switch (rounding_mode) {
|
||||||
|
case FP::RoundingMode::ToNearest_TieEven:
|
||||||
|
return 0b00;
|
||||||
|
case FP::RoundingMode::TowardsPlusInfinity:
|
||||||
|
return 0b10;
|
||||||
|
case FP::RoundingMode::TowardsMinusInfinity:
|
||||||
|
return 0b01;
|
||||||
|
case FP::RoundingMode::TowardsZero:
|
||||||
|
return 0b11;
|
||||||
|
default:
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Dynarmic::Backend::X64
|
} // namespace Dynarmic::Backend::X64
|
||||||
|
|
|
@ -235,7 +235,7 @@ static void EmitExtractRegister(BlockOfCode& code, EmitContext& ctx, IR::Inst* i
|
||||||
auto args = ctx.reg_alloc.GetArgumentInfo(inst);
|
auto args = ctx.reg_alloc.GetArgumentInfo(inst);
|
||||||
|
|
||||||
const Xbyak::Reg result = ctx.reg_alloc.UseScratchGpr(args[0]).changeBit(bit_size);
|
const Xbyak::Reg result = ctx.reg_alloc.UseScratchGpr(args[0]).changeBit(bit_size);
|
||||||
const Xbyak::Reg operand = ctx.reg_alloc.UseScratchGpr(args[1]).changeBit(bit_size);
|
const Xbyak::Reg operand = ctx.reg_alloc.UseGpr(args[1]).changeBit(bit_size);
|
||||||
const u8 lsb = args[2].GetImmediateU8();
|
const u8 lsb = args[2].GetImmediateU8();
|
||||||
|
|
||||||
code.shrd(result, operand, lsb);
|
code.shrd(result, operand, lsb);
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
|
|
||||||
#include "dynarmic/backend/x64/abi.h"
|
#include "dynarmic/backend/x64/abi.h"
|
||||||
#include "dynarmic/backend/x64/block_of_code.h"
|
#include "dynarmic/backend/x64/block_of_code.h"
|
||||||
|
#include "dynarmic/backend/x64/constants.h"
|
||||||
#include "dynarmic/backend/x64/emit_x64.h"
|
#include "dynarmic/backend/x64/emit_x64.h"
|
||||||
#include "dynarmic/common/assert.h"
|
#include "dynarmic/common/assert.h"
|
||||||
#include "dynarmic/common/cast_util.h"
|
#include "dynarmic/common/cast_util.h"
|
||||||
|
@ -79,21 +80,6 @@ constexpr u64 f64_max_u64_lim = 0x43f0000000000000u; // 2^64 as a double (actua
|
||||||
} \
|
} \
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<int> ConvertRoundingModeToX64Immediate(FP::RoundingMode rounding_mode) {
|
|
||||||
switch (rounding_mode) {
|
|
||||||
case FP::RoundingMode::ToNearest_TieEven:
|
|
||||||
return 0b00;
|
|
||||||
case FP::RoundingMode::TowardsPlusInfinity:
|
|
||||||
return 0b10;
|
|
||||||
case FP::RoundingMode::TowardsMinusInfinity:
|
|
||||||
return 0b01;
|
|
||||||
case FP::RoundingMode::TowardsZero:
|
|
||||||
return 0b11;
|
|
||||||
default:
|
|
||||||
return std::nullopt;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template<size_t fsize>
|
template<size_t fsize>
|
||||||
void DenormalsAreZero(BlockOfCode& code, EmitContext& ctx, std::initializer_list<Xbyak::Xmm> to_daz) {
|
void DenormalsAreZero(BlockOfCode& code, EmitContext& ctx, std::initializer_list<Xbyak::Xmm> to_daz) {
|
||||||
if (ctx.FPCR().FZ()) {
|
if (ctx.FPCR().FZ()) {
|
||||||
|
@ -116,9 +102,18 @@ void DenormalsAreZero(BlockOfCode& code, EmitContext& ctx, std::initializer_list
|
||||||
|
|
||||||
template<size_t fsize>
|
template<size_t fsize>
|
||||||
void ZeroIfNaN(BlockOfCode& code, Xbyak::Xmm xmm_value, Xbyak::Xmm xmm_scratch) {
|
void ZeroIfNaN(BlockOfCode& code, Xbyak::Xmm xmm_value, Xbyak::Xmm xmm_scratch) {
|
||||||
|
if (code.HasHostFeature(HostFeature::AVX512_OrthoFloat)) {
|
||||||
|
constexpr u32 nan_to_zero = FixupLUT(FpFixup::PosZero,
|
||||||
|
FpFixup::PosZero);
|
||||||
|
FCODE(vfixupimms)(xmm_value, xmm_value, code.MConst(ptr, u64(nan_to_zero)), u8(0));
|
||||||
|
} else if (code.HasHostFeature(HostFeature::AVX)) {
|
||||||
|
FCODE(vcmpords)(xmm_scratch, xmm_value, xmm_value);
|
||||||
|
FCODE(vandp)(xmm_value, xmm_value, xmm_scratch);
|
||||||
|
} else {
|
||||||
code.xorps(xmm_scratch, xmm_scratch);
|
code.xorps(xmm_scratch, xmm_scratch);
|
||||||
FCODE(cmpords)(xmm_scratch, xmm_value); // true mask when ordered (i.e.: when not an NaN)
|
FCODE(cmpords)(xmm_scratch, xmm_value); // true mask when ordered (i.e.: when not an NaN)
|
||||||
code.pand(xmm_value, xmm_scratch);
|
code.pand(xmm_value, xmm_scratch);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template<size_t fsize>
|
template<size_t fsize>
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
#include "dynarmic/backend/x64/abi.h"
|
#include "dynarmic/backend/x64/abi.h"
|
||||||
#include "dynarmic/backend/x64/block_of_code.h"
|
#include "dynarmic/backend/x64/block_of_code.h"
|
||||||
|
#include "dynarmic/backend/x64/constants.h"
|
||||||
#include "dynarmic/backend/x64/emit_x64.h"
|
#include "dynarmic/backend/x64/emit_x64.h"
|
||||||
#include "dynarmic/common/assert.h"
|
#include "dynarmic/common/assert.h"
|
||||||
#include "dynarmic/common/fp/fpcr.h"
|
#include "dynarmic/common/fp/fpcr.h"
|
||||||
|
@ -203,7 +204,11 @@ void ForceToDefaultNaN(BlockOfCode& code, FP::FPCR fpcr, Xbyak::Xmm result) {
|
||||||
template<size_t fsize>
|
template<size_t fsize>
|
||||||
void ZeroIfNaN(BlockOfCode& code, Xbyak::Xmm result) {
|
void ZeroIfNaN(BlockOfCode& code, Xbyak::Xmm result) {
|
||||||
const Xbyak::Xmm nan_mask = xmm0;
|
const Xbyak::Xmm nan_mask = xmm0;
|
||||||
if (code.HasHostFeature(HostFeature::AVX)) {
|
if (code.HasHostFeature(HostFeature::AVX512_OrthoFloat)) {
|
||||||
|
constexpr u32 nan_to_zero = FixupLUT(FpFixup::PosZero,
|
||||||
|
FpFixup::PosZero);
|
||||||
|
FCODE(vfixupimmp)(result, result, code.MConst(ptr_b, u64(nan_to_zero)), u8(0));
|
||||||
|
} else if (code.HasHostFeature(HostFeature::AVX)) {
|
||||||
FCODE(vcmpordp)(nan_mask, result, result);
|
FCODE(vcmpordp)(nan_mask, result, result);
|
||||||
FCODE(vandp)(result, result, nan_mask);
|
FCODE(vandp)(result, result, nan_mask);
|
||||||
} else {
|
} else {
|
||||||
|
@ -637,6 +642,49 @@ void EmitX64::EmitFPVectorEqual64(EmitContext& ctx, IR::Inst* inst) {
|
||||||
ctx.reg_alloc.DefineValue(inst, a);
|
ctx.reg_alloc.DefineValue(inst, a);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EmitX64::EmitFPVectorFromHalf32(EmitContext& ctx, IR::Inst* inst) {
|
||||||
|
const auto rounding_mode = static_cast<FP::RoundingMode>(inst->GetArg(1).GetU8());
|
||||||
|
const bool fpcr_controlled = inst->GetArg(2).GetU1();
|
||||||
|
|
||||||
|
if (code.HasHostFeature(HostFeature::F16C) && !ctx.FPCR().AHP() && !ctx.FPCR().FZ16()) {
|
||||||
|
auto args = ctx.reg_alloc.GetArgumentInfo(inst);
|
||||||
|
|
||||||
|
const Xbyak::Xmm result = ctx.reg_alloc.ScratchXmm();
|
||||||
|
const Xbyak::Xmm value = ctx.reg_alloc.UseXmm(args[0]);
|
||||||
|
|
||||||
|
code.vcvtph2ps(result, value);
|
||||||
|
ForceToDefaultNaN<32>(code, ctx.FPCR(fpcr_controlled), result);
|
||||||
|
|
||||||
|
ctx.reg_alloc.DefineValue(inst, result);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
using rounding_list = mp::list<
|
||||||
|
mp::lift_value<FP::RoundingMode::ToNearest_TieEven>,
|
||||||
|
mp::lift_value<FP::RoundingMode::TowardsPlusInfinity>,
|
||||||
|
mp::lift_value<FP::RoundingMode::TowardsMinusInfinity>,
|
||||||
|
mp::lift_value<FP::RoundingMode::TowardsZero>,
|
||||||
|
mp::lift_value<FP::RoundingMode::ToNearest_TieAwayFromZero>>;
|
||||||
|
|
||||||
|
static const auto lut = Common::GenerateLookupTableFromList(
|
||||||
|
[](auto arg) {
|
||||||
|
return std::pair{
|
||||||
|
mp::lower_to_tuple_v<decltype(arg)>,
|
||||||
|
Common::FptrCast(
|
||||||
|
[](VectorArray<u32>& output, const VectorArray<u16>& input, FP::FPCR fpcr, FP::FPSR& fpsr) {
|
||||||
|
constexpr auto t = mp::lower_to_tuple_v<decltype(arg)>;
|
||||||
|
constexpr FP::RoundingMode rounding_mode = std::get<0>(t);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < output.size(); ++i) {
|
||||||
|
output[i] = FP::FPConvert<u32, u16>(input[i], fpcr, rounding_mode, fpsr);
|
||||||
|
}
|
||||||
|
})};
|
||||||
|
},
|
||||||
|
mp::cartesian_product<rounding_list>{});
|
||||||
|
|
||||||
|
EmitTwoOpFallback<2>(code, ctx, inst, lut.at(std::make_tuple(rounding_mode)));
|
||||||
|
}
|
||||||
|
|
||||||
void EmitX64::EmitFPVectorFromSignedFixed32(EmitContext& ctx, IR::Inst* inst) {
|
void EmitX64::EmitFPVectorFromSignedFixed32(EmitContext& ctx, IR::Inst* inst) {
|
||||||
auto args = ctx.reg_alloc.GetArgumentInfo(inst);
|
auto args = ctx.reg_alloc.GetArgumentInfo(inst);
|
||||||
const Xbyak::Xmm xmm = ctx.reg_alloc.UseScratchXmm(args[0]);
|
const Xbyak::Xmm xmm = ctx.reg_alloc.UseScratchXmm(args[0]);
|
||||||
|
@ -1602,6 +1650,53 @@ void EmitX64::EmitFPVectorSub64(EmitContext& ctx, IR::Inst* inst) {
|
||||||
EmitThreeOpVectorOperation<64, DefaultIndexer>(code, ctx, inst, &Xbyak::CodeGenerator::subpd);
|
EmitThreeOpVectorOperation<64, DefaultIndexer>(code, ctx, inst, &Xbyak::CodeGenerator::subpd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EmitX64::EmitFPVectorToHalf32(EmitContext& ctx, IR::Inst* inst) {
|
||||||
|
const auto rounding_mode = static_cast<FP::RoundingMode>(inst->GetArg(1).GetU8());
|
||||||
|
const bool fpcr_controlled = inst->GetArg(2).GetU1();
|
||||||
|
|
||||||
|
if (code.HasHostFeature(HostFeature::F16C) && !ctx.FPCR().AHP() && !ctx.FPCR().FZ16()) {
|
||||||
|
auto args = ctx.reg_alloc.GetArgumentInfo(inst);
|
||||||
|
const auto round_imm = ConvertRoundingModeToX64Immediate(rounding_mode);
|
||||||
|
|
||||||
|
const Xbyak::Xmm result = ctx.reg_alloc.UseScratchXmm(args[0]);
|
||||||
|
|
||||||
|
ForceToDefaultNaN<32>(code, ctx.FPCR(fpcr_controlled), result);
|
||||||
|
code.vcvtps2ph(result, result, static_cast<u8>(*round_imm));
|
||||||
|
|
||||||
|
ctx.reg_alloc.DefineValue(inst, result);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
using rounding_list = mp::list<
|
||||||
|
mp::lift_value<FP::RoundingMode::ToNearest_TieEven>,
|
||||||
|
mp::lift_value<FP::RoundingMode::TowardsPlusInfinity>,
|
||||||
|
mp::lift_value<FP::RoundingMode::TowardsMinusInfinity>,
|
||||||
|
mp::lift_value<FP::RoundingMode::TowardsZero>,
|
||||||
|
mp::lift_value<FP::RoundingMode::ToNearest_TieAwayFromZero>>;
|
||||||
|
|
||||||
|
static const auto lut = Common::GenerateLookupTableFromList(
|
||||||
|
[](auto arg) {
|
||||||
|
return std::pair{
|
||||||
|
mp::lower_to_tuple_v<decltype(arg)>,
|
||||||
|
Common::FptrCast(
|
||||||
|
[](VectorArray<u16>& output, const VectorArray<u32>& input, FP::FPCR fpcr, FP::FPSR& fpsr) {
|
||||||
|
constexpr auto t = mp::lower_to_tuple_v<decltype(arg)>;
|
||||||
|
constexpr FP::RoundingMode rounding_mode = std::get<0>(t);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < output.size(); ++i) {
|
||||||
|
if (i < input.size()) {
|
||||||
|
output[i] = FP::FPConvert<u16, u32>(input[i], fpcr, rounding_mode, fpsr);
|
||||||
|
} else {
|
||||||
|
output[i] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})};
|
||||||
|
},
|
||||||
|
mp::cartesian_product<rounding_list>{});
|
||||||
|
|
||||||
|
EmitTwoOpFallback<2>(code, ctx, inst, lut.at(std::make_tuple(rounding_mode)));
|
||||||
|
}
|
||||||
|
|
||||||
template<size_t fsize, bool unsigned_>
|
template<size_t fsize, bool unsigned_>
|
||||||
void EmitFPVectorToFixed(BlockOfCode& code, EmitContext& ctx, IR::Inst* inst) {
|
void EmitFPVectorToFixed(BlockOfCode& code, EmitContext& ctx, IR::Inst* inst) {
|
||||||
using FPT = mp::unsigned_integer_of_size<fsize>;
|
using FPT = mp::unsigned_integer_of_size<fsize>;
|
||||||
|
|
|
@ -113,7 +113,7 @@ kern_return_t MachHandler::HandleRequest(x86_thread_state64_t* ts) {
|
||||||
|
|
||||||
const auto iter = FindCodeBlockInfo(ts->__rip);
|
const auto iter = FindCodeBlockInfo(ts->__rip);
|
||||||
if (iter == code_block_infos.end()) {
|
if (iter == code_block_infos.end()) {
|
||||||
fmt::print(stderr, "dynarmic: macOS MachHandler: Exception was not in registered code blocks (rip {:#016x})\n", ts->__rip);
|
fmt::print(stderr, "Unhandled EXC_BAD_ACCESS at rip {:#016x}\n", ts->__rip);
|
||||||
return KERN_FAILURE;
|
return KERN_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -149,7 +149,7 @@ void SigHandler::SigAction(int sig, siginfo_t* info, void* raw_context) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt::print(stderr, "dynarmic: POSIX SigHandler: Exception was not in registered code blocks (rip {:#016x})\n", CTX_RIP);
|
fmt::print(stderr, "Unhandled {} at rip {:#016x}\n", sig == SIGSEGV ? "SIGSEGV" : "SIGBUS", CTX_RIP);
|
||||||
|
|
||||||
struct sigaction* retry_sa = sig == SIGSEGV ? &sig_handler.old_sa_segv : &sig_handler.old_sa_bus;
|
struct sigaction* retry_sa = sig == SIGSEGV ? &sig_handler.old_sa_segv : &sig_handler.old_sa_bus;
|
||||||
if (retry_sa->sa_flags & SA_SIGINFO) {
|
if (retry_sa->sa_flags & SA_SIGINFO) {
|
||||||
|
|
|
@ -620,24 +620,13 @@ bool TranslatorVisitor::asimd_VCVT_half(bool D, size_t sz, size_t Vd, bool half_
|
||||||
}
|
}
|
||||||
|
|
||||||
const size_t esize = 8U << sz;
|
const size_t esize = 8U << sz;
|
||||||
const size_t num_elements = 4;
|
|
||||||
const auto rounding_mode = FP::RoundingMode::ToNearest_TieEven; // StandardFPSCRValue().RMode
|
const auto rounding_mode = FP::RoundingMode::ToNearest_TieEven; // StandardFPSCRValue().RMode
|
||||||
const auto d = ToVector(half_to_single, Vd, D);
|
const auto d = ToVector(half_to_single, Vd, D);
|
||||||
const auto m = ToVector(!half_to_single, Vm, M);
|
const auto m = ToVector(!half_to_single, Vm, M);
|
||||||
|
|
||||||
const auto operand = ir.GetVector(m);
|
const auto operand = ir.GetVector(m);
|
||||||
IR::U128 result = ir.ZeroVector();
|
const IR::U128 result = half_to_single ? ir.FPVectorFromHalf(esize * 2, operand, rounding_mode, false)
|
||||||
for (size_t i = 0; i < num_elements; i++) {
|
: ir.FPVectorToHalf(esize * 2, operand, rounding_mode, false);
|
||||||
if (half_to_single) {
|
|
||||||
const IR::U16 old_element = ir.VectorGetElement(esize, operand, i);
|
|
||||||
const IR::U32 new_element = ir.FPHalfToSingle(old_element, rounding_mode);
|
|
||||||
result = ir.VectorSetElement(esize * 2, result, i, new_element);
|
|
||||||
} else {
|
|
||||||
const IR::U32 old_element = ir.VectorGetElement(esize * 2, operand, i);
|
|
||||||
const IR::U16 new_element = ir.FPSingleToHalf(old_element, rounding_mode);
|
|
||||||
result = ir.VectorSetElement(esize, result, i, new_element);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ir.SetVector(d, result);
|
ir.SetVector(d, result);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2404,6 +2404,11 @@ U128 IREmitter::FPVectorEqual(size_t esize, const U128& a, const U128& b, bool f
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
U128 IREmitter::FPVectorFromHalf(size_t esize, const U128& a, FP::RoundingMode rounding, bool fpcr_controlled) {
|
||||||
|
ASSERT(esize == 32);
|
||||||
|
return Inst<U128>(Opcode::FPVectorFromHalf32, a, Imm8(static_cast<u8>(rounding)), Imm1(fpcr_controlled));
|
||||||
|
}
|
||||||
|
|
||||||
U128 IREmitter::FPVectorFromSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled) {
|
U128 IREmitter::FPVectorFromSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled) {
|
||||||
ASSERT(fbits <= esize);
|
ASSERT(fbits <= esize);
|
||||||
switch (esize) {
|
switch (esize) {
|
||||||
|
@ -2613,6 +2618,11 @@ U128 IREmitter::FPVectorSub(size_t esize, const U128& a, const U128& b, bool fpc
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
U128 IREmitter::FPVectorToHalf(size_t esize, const U128& a, FP::RoundingMode rounding, bool fpcr_controlled) {
|
||||||
|
ASSERT(esize == 32);
|
||||||
|
return Inst<U128>(Opcode::FPVectorToHalf32, a, Imm8(static_cast<u8>(rounding)), Imm1(fpcr_controlled));
|
||||||
|
}
|
||||||
|
|
||||||
U128 IREmitter::FPVectorToSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled) {
|
U128 IREmitter::FPVectorToSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled) {
|
||||||
ASSERT(fbits <= esize);
|
ASSERT(fbits <= esize);
|
||||||
|
|
||||||
|
|
|
@ -370,6 +370,7 @@ public:
|
||||||
U128 FPVectorAdd(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
U128 FPVectorAdd(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
||||||
U128 FPVectorDiv(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
U128 FPVectorDiv(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
||||||
U128 FPVectorEqual(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
U128 FPVectorEqual(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
||||||
|
U128 FPVectorFromHalf(size_t esize, const U128& a, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
||||||
U128 FPVectorFromSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
U128 FPVectorFromSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
||||||
U128 FPVectorFromUnsignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
U128 FPVectorFromUnsignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
||||||
U128 FPVectorGreater(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
U128 FPVectorGreater(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
||||||
|
@ -389,6 +390,7 @@ public:
|
||||||
U128 FPVectorRSqrtStepFused(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
U128 FPVectorRSqrtStepFused(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
||||||
U128 FPVectorSqrt(size_t esize, const U128& a, bool fpcr_controlled = true);
|
U128 FPVectorSqrt(size_t esize, const U128& a, bool fpcr_controlled = true);
|
||||||
U128 FPVectorSub(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
U128 FPVectorSub(size_t esize, const U128& a, const U128& b, bool fpcr_controlled = true);
|
||||||
|
U128 FPVectorToHalf(size_t esize, const U128& a, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
||||||
U128 FPVectorToSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
U128 FPVectorToSignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
||||||
U128 FPVectorToUnsignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
U128 FPVectorToUnsignedFixed(size_t esize, const U128& a, size_t fbits, FP::RoundingMode rounding, bool fpcr_controlled = true);
|
||||||
|
|
||||||
|
|
|
@ -613,6 +613,7 @@ OPCODE(FPVectorDiv64, U128, U128
|
||||||
OPCODE(FPVectorEqual16, U128, U128, U128, U1 )
|
OPCODE(FPVectorEqual16, U128, U128, U128, U1 )
|
||||||
OPCODE(FPVectorEqual32, U128, U128, U128, U1 )
|
OPCODE(FPVectorEqual32, U128, U128, U128, U1 )
|
||||||
OPCODE(FPVectorEqual64, U128, U128, U128, U1 )
|
OPCODE(FPVectorEqual64, U128, U128, U128, U1 )
|
||||||
|
OPCODE(FPVectorFromHalf32, U128, U128, U8, U1 )
|
||||||
OPCODE(FPVectorFromSignedFixed32, U128, U128, U8, U8, U1 )
|
OPCODE(FPVectorFromSignedFixed32, U128, U128, U8, U8, U1 )
|
||||||
OPCODE(FPVectorFromSignedFixed64, U128, U128, U8, U8, U1 )
|
OPCODE(FPVectorFromSignedFixed64, U128, U128, U8, U8, U1 )
|
||||||
OPCODE(FPVectorFromUnsignedFixed32, U128, U128, U8, U8, U1 )
|
OPCODE(FPVectorFromUnsignedFixed32, U128, U128, U8, U8, U1 )
|
||||||
|
@ -658,6 +659,7 @@ OPCODE(FPVectorSqrt32, U128, U128
|
||||||
OPCODE(FPVectorSqrt64, U128, U128, U1 )
|
OPCODE(FPVectorSqrt64, U128, U128, U1 )
|
||||||
OPCODE(FPVectorSub32, U128, U128, U128, U1 )
|
OPCODE(FPVectorSub32, U128, U128, U128, U1 )
|
||||||
OPCODE(FPVectorSub64, U128, U128, U128, U1 )
|
OPCODE(FPVectorSub64, U128, U128, U128, U1 )
|
||||||
|
OPCODE(FPVectorToHalf32, U128, U128, U8, U1 )
|
||||||
OPCODE(FPVectorToSignedFixed16, U128, U128, U8, U8, U1 )
|
OPCODE(FPVectorToSignedFixed16, U128, U128, U8, U8, U1 )
|
||||||
OPCODE(FPVectorToSignedFixed32, U128, U128, U8, U8, U1 )
|
OPCODE(FPVectorToSignedFixed32, U128, U128, U8, U8, U1 )
|
||||||
OPCODE(FPVectorToSignedFixed64, U128, U128, U8, U8, U1 )
|
OPCODE(FPVectorToSignedFixed64, U128, U128, U8, U8, U1 )
|
||||||
|
|
3
externals/dynarmic/tests/A32/testenv.h
vendored
3
externals/dynarmic/tests/A32/testenv.h
vendored
|
@ -120,7 +120,8 @@ public:
|
||||||
u64 ticks_left = 0;
|
u64 ticks_left = 0;
|
||||||
char* backing_memory = nullptr;
|
char* backing_memory = nullptr;
|
||||||
|
|
||||||
explicit A32FastmemTestEnv(char* addr) : backing_memory(addr) {}
|
explicit A32FastmemTestEnv(char* addr)
|
||||||
|
: backing_memory(addr) {}
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
T read(std::uint32_t vaddr) {
|
T read(std::uint32_t vaddr) {
|
||||||
|
|
3
externals/dynarmic/tests/A64/testenv.h
vendored
3
externals/dynarmic/tests/A64/testenv.h
vendored
|
@ -130,7 +130,8 @@ public:
|
||||||
u64 ticks_left = 0;
|
u64 ticks_left = 0;
|
||||||
char* backing_memory = nullptr;
|
char* backing_memory = nullptr;
|
||||||
|
|
||||||
explicit A64FastmemTestEnv(char* addr) : backing_memory(addr) {}
|
explicit A64FastmemTestEnv(char* addr)
|
||||||
|
: backing_memory(addr) {}
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
T read(u64 vaddr) {
|
T read(u64 vaddr) {
|
||||||
|
|
2
externals/dynarmic/tests/CMakeLists.txt
vendored
2
externals/dynarmic/tests/CMakeLists.txt
vendored
|
@ -61,4 +61,4 @@ target_include_directories(dynarmic_print_info PRIVATE . ../src)
|
||||||
target_compile_options(dynarmic_print_info PRIVATE ${DYNARMIC_CXX_FLAGS})
|
target_compile_options(dynarmic_print_info PRIVATE ${DYNARMIC_CXX_FLAGS})
|
||||||
target_compile_definitions(dynarmic_print_info PRIVATE FMT_USE_USER_DEFINED_LITERALS=0)
|
target_compile_definitions(dynarmic_print_info PRIVATE FMT_USE_USER_DEFINED_LITERALS=0)
|
||||||
|
|
||||||
add_test(dynarmic_tests dynarmic_tests)
|
add_test(dynarmic_tests dynarmic_tests --durations yes)
|
||||||
|
|
|
@ -32,7 +32,7 @@
|
||||||
|
|
||||||
class SDLGLContext : public Core::Frontend::GraphicsContext {
|
class SDLGLContext : public Core::Frontend::GraphicsContext {
|
||||||
public:
|
public:
|
||||||
explicit SDLGLContext(SDL_Window* window) : window(window) {
|
explicit SDLGLContext(SDL_Window* window_) : window{window_} {
|
||||||
context = SDL_GL_CreateContext(window);
|
context = SDL_GL_CreateContext(window);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue