mirror of
https://github.com/xenia-project/xenia.git
synced 2025-12-06 07:12:03 +01:00
283 lines
7.8 KiB
C++
283 lines
7.8 KiB
C++
/**
|
|
******************************************************************************
|
|
* Xenia : Xbox 360 Emulator Research Project *
|
|
******************************************************************************
|
|
* Copyright 2014 Ben Vanik. All rights reserved. *
|
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
|
******************************************************************************
|
|
*/
|
|
|
|
#include "xenia/cpu/testing/util.h"
|
|
|
|
using namespace xe;
|
|
using namespace xe::cpu;
|
|
using namespace xe::cpu::hir;
|
|
using namespace xe::cpu::testing;
|
|
using xe::cpu::ppc::PPCContext;
|
|
|
|
TEST_CASE("SHR_I8", "[instr]") {
|
|
TestFunction test([](HIRBuilder& b) {
|
|
StoreGPR(b, 3,
|
|
b.ZeroExtend(b.Shr(b.Truncate(LoadGPR(b, 4), INT8_TYPE),
|
|
b.Truncate(LoadGPR(b, 5), INT8_TYPE)),
|
|
INT64_TYPE));
|
|
b.Return();
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xF0;
|
|
ctx->r[5] = 4;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint8_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x0F);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFF;
|
|
ctx->r[5] = 0;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint8_t>(ctx->r[3]);
|
|
REQUIRE(result == 0xFF);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFF;
|
|
ctx->r[5] = 1;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint8_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x7F);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x80;
|
|
ctx->r[5] = 8;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint8_t>(ctx->r[3]);
|
|
REQUIRE(result == 0);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x7F;
|
|
ctx->r[5] = 7;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint8_t>(ctx->r[3]);
|
|
REQUIRE(result == 0);
|
|
});
|
|
}
|
|
|
|
TEST_CASE("SHR_I16", "[instr]") {
|
|
TestFunction test([](HIRBuilder& b) {
|
|
StoreGPR(b, 3,
|
|
b.ZeroExtend(b.Shr(b.Truncate(LoadGPR(b, 4), INT16_TYPE),
|
|
b.Truncate(LoadGPR(b, 5), INT8_TYPE)),
|
|
INT64_TYPE));
|
|
b.Return();
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFF00;
|
|
ctx->r[5] = 8;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint16_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x00FF);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFF;
|
|
ctx->r[5] = 0;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint16_t>(ctx->r[3]);
|
|
REQUIRE(result == 0xFFFF);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFE;
|
|
ctx->r[5] = 1;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint16_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x7FFF);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x8000;
|
|
ctx->r[5] = 16;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint16_t>(ctx->r[3]);
|
|
REQUIRE(result == 0);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x7FFF;
|
|
ctx->r[5] = 15;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint16_t>(ctx->r[3]);
|
|
REQUIRE(result == 0);
|
|
});
|
|
}
|
|
|
|
TEST_CASE("SHR_I32", "[instr]") {
|
|
TestFunction test([](HIRBuilder& b) {
|
|
StoreGPR(b, 3,
|
|
b.ZeroExtend(b.Shr(b.Truncate(LoadGPR(b, 4), INT32_TYPE),
|
|
b.Truncate(LoadGPR(b, 5), INT8_TYPE)),
|
|
INT64_TYPE));
|
|
b.Return();
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFF0000;
|
|
ctx->r[5] = 16;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint32_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x0000FFFF);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFFFFFF;
|
|
ctx->r[5] = 0;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint32_t>(ctx->r[3]);
|
|
REQUIRE(result == 0xFFFFFFFF);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFFFFFE;
|
|
ctx->r[5] = 1;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint32_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x7FFFFFFF);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x80000000;
|
|
ctx->r[5] = 32;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint32_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x80000000);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x7FFFFFFF;
|
|
ctx->r[5] = 31;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint32_t>(ctx->r[3]);
|
|
REQUIRE(result == 0);
|
|
});
|
|
}
|
|
|
|
TEST_CASE("SHR_I64", "[instr]") {
|
|
TestFunction test([](HIRBuilder& b) {
|
|
StoreGPR(b, 3,
|
|
b.Shr(b.Truncate(LoadGPR(b, 4), INT64_TYPE),
|
|
b.Truncate(LoadGPR(b, 5), INT8_TYPE)));
|
|
b.Return();
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFFFFFF00000000ull;
|
|
ctx->r[5] = 32;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint64_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x00000000FFFFFFFFull);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFFFFFFFFFFFFFFull;
|
|
ctx->r[5] = 0;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint64_t>(ctx->r[3]);
|
|
REQUIRE(result == 0xFFFFFFFFFFFFFFFFull);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0xFFFFFFFFFFFFFFFEull;
|
|
ctx->r[5] = 1;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint64_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x7FFFFFFFFFFFFFFFull);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x8000000000000000ull;
|
|
ctx->r[5] = 64;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint64_t>(ctx->r[3]);
|
|
REQUIRE(result == 0x8000000000000000ull);
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[4] = 0x7FFFFFFFFFFFFFFFull;
|
|
ctx->r[5] = 63;
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result = static_cast<uint64_t>(ctx->r[3]);
|
|
REQUIRE(result == 0);
|
|
});
|
|
}
|
|
|
|
TEST_CASE("SHR_V128", "[instr]") {
|
|
TestFunction test([](HIRBuilder& b) {
|
|
StoreVR(b, 3, b.Shr(LoadVR(b, 4), b.Truncate(LoadGPR(b, 1), INT8_TYPE)));
|
|
b.Return();
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[1] = 0;
|
|
ctx->v[4] = vec128i(0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF);
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result1 = ctx->v[3];
|
|
REQUIRE(result1 ==
|
|
vec128i(0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF));
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[1] = 1;
|
|
ctx->v[4] = vec128i(0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF);
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result1 = ctx->v[3];
|
|
REQUIRE(result1 ==
|
|
vec128i(0x7FFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF));
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[1] = 2;
|
|
ctx->v[4] = vec128i(0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF);
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result1 = ctx->v[3];
|
|
REQUIRE(result1 ==
|
|
vec128i(0x3FFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF));
|
|
});
|
|
test.Run(
|
|
[](PPCContext* ctx) {
|
|
ctx->r[1] = 8;
|
|
ctx->v[4] = vec128i(0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF);
|
|
},
|
|
[](PPCContext* ctx) {
|
|
auto result1 = ctx->v[3];
|
|
REQUIRE(result1 ==
|
|
vec128i(0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF, 0xFFFFFFFF));
|
|
});
|
|
}
|