blob: 8ebd7ca2686451d6825d7c5387b60e66049dee29 [file] [log] [blame]
// Copyright 2023 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include "sim/kelvin_encoding.h"
#include <any>
#include <cstdint>
#include <type_traits>
#include "sim/kelvin_enums.h"
#include "sim/kelvin_state.h"
#include "googletest/include/gtest/gtest.h"
#include "riscv/riscv_register.h"
#include "riscv/riscv_state.h"
#include "mpact/sim/generic/register.h"
namespace {
using kelvin::sim::KelvinState;
using kelvin::sim::isa32::KelvinEncoding;
using SlotEnum = kelvin::sim::isa32::SlotEnum;
using OpcodeEnum = kelvin::sim::isa32::OpcodeEnum;
using SourceOpEnum = kelvin::sim::isa32::SourceOpEnum;
using RV32VectorSourceOperand = mpact::sim::riscv::RV32VectorSourceOperand;
using RV32SourceOperand = mpact::sim::generic::RegisterSourceOperand<uint32_t>;
using DestOpEnum = kelvin::sim::isa32::DestOpEnum;
using RV32VectorDestOperand = mpact::sim::riscv::RV32VectorDestinationOperand;
using RV32DestOperand =
mpact::sim::generic::RegisterDestinationOperand<uint32_t>;
// RV32I
constexpr uint32_t kLui = 0b0000000000000000000000000'0110111;
constexpr uint32_t kAuipc = 0b0000000000000000000000000'0010111;
constexpr uint32_t kJal = 0b00000000000000000000'00000'1101111;
constexpr uint32_t kJalr = 0b00000000000'00000'000'00000'1100111;
constexpr uint32_t kBeq = 0b0000000'00000'00000'000'00000'1100011;
constexpr uint32_t kBne = 0b0000000'00000'00000'001'00000'1100011;
constexpr uint32_t kBlt = 0b0000000'00000'00000'100'00000'1100011;
constexpr uint32_t kBge = 0b0000000'00000'00000'101'00000'1100011;
constexpr uint32_t kBltu = 0b0000000'00000'00000'110'00000'1100011;
constexpr uint32_t kBgeu = 0b0000000'00000'00000'111'00000'1100011;
constexpr uint32_t kLb = 0b000000000000'00000'000'00000'0000011;
constexpr uint32_t kLh = 0b000000000000'00000'001'00000'0000011;
constexpr uint32_t kLw = 0b000000000000'00000'010'00000'0000011;
constexpr uint32_t kLbu = 0b000000000000'00000'100'00000'0000011;
constexpr uint32_t kLhu = 0b000000000000'00000'101'00000'0000011;
constexpr uint32_t kSb = 0b0000000'00000'00000'000'00000'0100011;
constexpr uint32_t kSh = 0b0000000'00000'00000'001'00000'0100011;
constexpr uint32_t kSw = 0b0000000'00000'00000'010'00000'0100011;
constexpr uint32_t kAddi = 0b000000000000'00000'000'00000'0010011;
constexpr uint32_t kSlti = 0b000000000000'00000'010'00000'0010011;
constexpr uint32_t kSltiu = 0b000000000000'00000'011'00000'0010011;
constexpr uint32_t kXori = 0b000000000000'00000'100'00000'0010011;
constexpr uint32_t kOri = 0b000000000000'00000'110'00000'0010011;
constexpr uint32_t kAndi = 0b000000000000'00000'111'00000'0010011;
constexpr uint32_t kSlli = 0b0000000'00000'00000'001'00000'0010011;
constexpr uint32_t kSrli = 0b0000000'00000'00000'101'00000'0010011;
constexpr uint32_t kSrai = 0b0100000'00000'00000'101'00000'0010011;
constexpr uint32_t kAdd = 0b0000000'00000'00000'000'00000'0110011;
constexpr uint32_t kSub = 0b0100000'00000'00000'000'00000'0110011;
constexpr uint32_t kSll = 0b0000000'00000'00000'001'00000'0110011;
constexpr uint32_t kSlt = 0b0000000'00000'00000'010'00000'0110011;
constexpr uint32_t kSltu = 0b0000000'00000'00000'011'00000'0110011;
constexpr uint32_t kXor = 0b0000000'00000'00000'100'00000'0110011;
constexpr uint32_t kSrl = 0b0000000'00000'00000'101'00000'0110011;
constexpr uint32_t kSra = 0b0100000'00000'00000'101'00000'0110011;
constexpr uint32_t kOr = 0b0000000'00000'00000'110'00000'0110011;
constexpr uint32_t kAnd = 0b0000000'00000'00000'111'00000'0110011;
constexpr uint32_t kFence = 0b000000000000'00000'000'00000'0001111;
constexpr uint32_t kEcall = 0b000000000000'00000'000'00000'1110011;
constexpr uint32_t kEbreak = 0b000000000001'00000'000'00000'1110011;
constexpr uint32_t kMpause = 0b000010000000'00000'000'00000'1110011;
// Kelvin Memory ops
constexpr uint32_t kFlushall = 0b001001100000'00000'000'00000'1110111;
// RV32 Zifencei
constexpr uint32_t kFencei = 0b000000000000'00000'001'00000'0001111;
// RV32 Zicsr
constexpr uint32_t kCsrw = 0b000000000000'00000'001'00000'1110011;
constexpr uint32_t kCsrs = 0b000000000000'00000'010'00000'1110011;
constexpr uint32_t kCsrc = 0b000000000000'00000'011'00000'1110011;
constexpr uint32_t kCsrwi = 0b000000000000'00000'101'00000'1110011;
constexpr uint32_t kCsrsi = 0b000000000000'00000'110'00000'1110011;
constexpr uint32_t kCsrci = 0b000000000000'00000'111'00000'1110011;
// RV32M
constexpr uint32_t kMul = 0b0000001'00000'00000'000'00000'0110011;
constexpr uint32_t kMulh = 0b0000001'00000'00000'001'00000'0110011;
constexpr uint32_t kMulhsu = 0b0000001'00000'00000'010'00000'0110011;
constexpr uint32_t kMulhu = 0b0000001'00000'00000'011'00000'0110011;
constexpr uint32_t kDiv = 0b0000001'00000'00000'100'00000'0110011;
constexpr uint32_t kDivu = 0b0000001'00000'00000'101'00000'0110011;
constexpr uint32_t kRem = 0b0000001'00000'00000'110'00000'0110011;
constexpr uint32_t kRemu = 0b0000001'00000'00000'111'00000'0110011;
// Kelvin System Op
constexpr uint32_t kGetMaxVl = 0b0001'0'00'00000'00000'000'00000'111'0111;
// Kelvin Logging Op
constexpr uint32_t kFLog = 0b011'1100'00000'00000'000'00000'111'0111;
// Kelvin VLd
constexpr uint32_t kVld = 0b000000'000000'000000'00'000000'0'111'11;
// Kelvin vector ops
constexpr uint32_t kVAccsBase = 0b001010'000001'000000'00'001000'0'100'00;
constexpr uint32_t kVAddBase = 0b000000'000000'000001'00'000010'0'000'00;
constexpr uint32_t kAconvBase = 0b001000'000001'010000'10'110000'0'00'101;
constexpr uint32_t kVdwconvBase = 0b001000'000001'010000'10'110000'0'10'101;
class KelvinEncodingTest : public testing::Test {
protected:
KelvinEncodingTest() {
state_ = new KelvinState("test", mpact::sim::riscv::RiscVXlen::RV32);
enc_ = new KelvinEncoding(state_);
}
~KelvinEncodingTest() override {
delete enc_;
delete state_;
}
template <typename T>
T *EncodeOpHelper(uint32_t inst_word, OpcodeEnum opcode, std::any op) const {
enc_->ParseInstruction(inst_word);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), opcode);
if (std::is_same<T, RV32SourceOperand>::value ||
std::is_same<T, RV32VectorSourceOperand>::value) {
auto *source = enc_->GetSource(SlotEnum::kKelvin, 0, opcode,
std::any_cast<SourceOpEnum>(op), 0);
return reinterpret_cast<T *>(source);
}
auto *dest = enc_->GetDestination(SlotEnum::kKelvin, 0, opcode,
std::any_cast<DestOpEnum>(op), 0,
/*latency=*/0);
return reinterpret_cast<T *>(dest);
}
KelvinState *state_;
KelvinEncoding *enc_;
};
constexpr int kRdValue = 1;
constexpr int kSuccValue = 0xf;
constexpr int kPredValue = 0xf;
static uint32_t SetRd(uint32_t iword, uint32_t rdval) {
return (iword | ((rdval & 0x1f) << 7));
}
static uint32_t SetRs1(uint32_t iword, uint32_t rsval) {
return (iword | ((rsval & 0x1f) << 15));
}
static uint32_t SetRs2(uint32_t iword, uint32_t rsval) {
return (iword | ((rsval & 0x1f) << 20));
}
static uint32_t SetPred(uint32_t iword, uint32_t pred) {
return (iword | ((pred & 0xf) << 24));
}
static uint32_t SetSucc(uint32_t iword, uint32_t succ) {
return (iword | ((succ & 0xf) << 20));
}
static inline uint32_t SetSz(uint32_t iword, uint32_t sz) {
return (iword | ((sz & 0x3) << 12));
}
TEST_F(KelvinEncodingTest, RV32IOpcodes) {
enc_->ParseInstruction(SetRd(kLui, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kLui);
enc_->ParseInstruction(SetRd(kAuipc, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kAuipc);
enc_->ParseInstruction(SetRd(kJal, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kJal);
enc_->ParseInstruction(SetRs1(kJalr, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kRet);
enc_->ParseInstruction(SetRs1(kJalr, 0x2));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kJr);
enc_->ParseInstruction(SetRd(kJalr, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kJalr);
enc_->ParseInstruction(kBeq);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kBeq);
enc_->ParseInstruction(kBne);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kBne);
enc_->ParseInstruction(kBlt);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kBlt);
enc_->ParseInstruction(kBge);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kBge);
enc_->ParseInstruction(kBltu);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kBltu);
enc_->ParseInstruction(kBgeu);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kBgeu);
enc_->ParseInstruction(SetRd(kLb, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kLb);
enc_->ParseInstruction(SetRd(kLh, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kLh);
enc_->ParseInstruction(SetRd(kLw, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kLw);
enc_->ParseInstruction(SetRd(kLbu, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kLbu);
enc_->ParseInstruction(SetRd(kLhu, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kLhu);
enc_->ParseInstruction(SetRd(kSb, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSb);
enc_->ParseInstruction(SetRd(kSh, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSh);
enc_->ParseInstruction(SetRd(kSw, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSw);
enc_->ParseInstruction(kAddi);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kNop);
enc_->ParseInstruction(SetRd(kAddi, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kLi);
enc_->ParseInstruction(SetRs1(SetRd(kAddi, kRdValue), 0x2));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kMv);
enc_->ParseInstruction(SetRs1(SetRd(kAddi, kRdValue), 0x2) |
(0b1 << 20 /*imm12*/));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kAddi);
enc_->ParseInstruction(SetRd(kSlti, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSlti);
enc_->ParseInstruction(SetRd(kSltiu, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSltiu);
enc_->ParseInstruction(SetRd(kXori, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kXori);
enc_->ParseInstruction(SetRd(kOri, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kOri);
enc_->ParseInstruction(SetRd(kAndi, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kAndi);
enc_->ParseInstruction(SetRd(kSlli, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSlli);
enc_->ParseInstruction(SetRd(kSrli, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSrli);
enc_->ParseInstruction(SetRd(kSrai, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSrai);
enc_->ParseInstruction(SetRd(kAdd, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kAdd);
enc_->ParseInstruction(SetRd(kSub, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSub);
enc_->ParseInstruction(SetRd(kSll, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSll);
enc_->ParseInstruction(SetRd(kSlt, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSlt);
enc_->ParseInstruction(SetRd(kSltu, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSltu);
enc_->ParseInstruction(SetRd(kXor, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kXor);
enc_->ParseInstruction(SetRd(kSrl, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSrl);
enc_->ParseInstruction(SetRd(kSra, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSra);
enc_->ParseInstruction(SetRd(kOr, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kOr);
enc_->ParseInstruction(SetRd(kAnd, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kAnd);
enc_->ParseInstruction(SetSucc(SetPred(kFence, kPredValue), kSuccValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kFence);
enc_->ParseInstruction(kEcall);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kEcall);
enc_->ParseInstruction(kEbreak);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kEbreak);
enc_->ParseInstruction(kMpause);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kMpause);
}
TEST_F(KelvinEncodingTest, KelvinMemoryOpcodes) {
enc_->ParseInstruction(kFlushall);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kFlushall);
enc_->ParseInstruction(SetRs1(kFlushall, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kFlushat);
}
TEST_F(KelvinEncodingTest, KelvinSystemOpcodes) {
enc_->ParseInstruction(SetRd(kGetMaxVl, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kGetmaxvlB);
enc_->ParseInstruction(SetRd(SetRs1(kGetMaxVl, kRdValue), kRdValue) |
(0b1) << 25);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kGetvlHX);
enc_->ParseInstruction(
SetRd(SetRs1(SetRs2(kGetMaxVl, kRdValue), kRdValue), kRdValue) |
(0b10 << 25));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kGetvlWXx);
}
TEST_F(KelvinEncodingTest, KelvinLogOpcodes) {
enc_->ParseInstruction(SetRs1(kFLog, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kFlog);
enc_->ParseInstruction(SetRs1(kFLog, kRdValue) | (0b01 << 12));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kSlog);
enc_->ParseInstruction(SetRs1(kFLog, kRdValue) | (0b10 << 12));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kClog);
enc_->ParseInstruction(SetRs1(kFLog, kRdValue) | (0b11 << 12));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kKlog);
}
TEST_F(KelvinEncodingTest, ZifenceiOpcodes) {
// RV32 Zifencei
enc_->ParseInstruction(kFencei);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kFencei);
}
TEST_F(KelvinEncodingTest, ZicsrOpcodes) {
// RV32 Zicsr
enc_->ParseInstruction(SetRd(kCsrw, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrw);
enc_->ParseInstruction(SetRd(SetRs1(kCsrs, kRdValue), kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrs);
enc_->ParseInstruction(SetRd(SetRs1(kCsrc, kRdValue), kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrc);
enc_->ParseInstruction(kCsrw);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrwNr);
enc_->ParseInstruction(kCsrs);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrsNw);
enc_->ParseInstruction(kCsrc);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrcNw);
enc_->ParseInstruction(SetRd(kCsrwi, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrwi);
enc_->ParseInstruction(SetRd(SetRs1(kCsrsi, kRdValue), kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrsi);
enc_->ParseInstruction(SetRd(SetRs1(kCsrci, kRdValue), kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrci);
enc_->ParseInstruction(kCsrwi);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrwiNr);
enc_->ParseInstruction(kCsrsi);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrsiNw);
enc_->ParseInstruction(kCsrci);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kCsrrciNw);
}
TEST_F(KelvinEncodingTest, RV32MOpcodes) {
// RV32M
enc_->ParseInstruction(kMul);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kMul);
enc_->ParseInstruction(kMulh);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kMulh);
enc_->ParseInstruction(kMulhsu);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kMulhsu);
enc_->ParseInstruction(kMulhu);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kMulhu);
enc_->ParseInstruction(kDiv);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kDiv);
enc_->ParseInstruction(kDivu);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kDivu);
enc_->ParseInstruction(kRem);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kRem);
enc_->ParseInstruction(kRemu);
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kRemu);
}
TEST_F(KelvinEncodingTest, NoSourceDest) {
enc_->ParseInstruction(kVld);
auto *src = enc_->GetSource(SlotEnum::kKelvin, 0, OpcodeEnum::kVldBX,
SourceOpEnum::kNone, 0);
EXPECT_EQ(src, nullptr);
auto *src_op = enc_->GetSource(SlotEnum::kKelvin, 0, OpcodeEnum::kVldBX,
SourceOpEnum::kPastMaxValue, 0);
EXPECT_EQ(src_op, nullptr);
auto *dest = enc_->GetDestination(SlotEnum::kKelvin, 0, OpcodeEnum::kVldBX,
DestOpEnum::kNone, 0, /*latency=*/0);
EXPECT_EQ(dest, nullptr);
auto *dest_op =
enc_->GetDestination(SlotEnum::kKelvin, 0, OpcodeEnum::kVldBX,
DestOpEnum::kPastMaxValue, 0, /*latency=*/0);
EXPECT_EQ(dest_op, nullptr);
}
TEST_F(KelvinEncodingTest, KelvinVldEncodeXs1Xs2) {
enc_->ParseInstruction(SetRs1(kVld, kRdValue));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kVldBX);
enc_->ParseInstruction(SetSz(SetRs1(kVld, kRdValue), 0b1));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kVldHX);
// Test vld.b.x (x = x0)
auto *src = EncodeOpHelper<RV32SourceOperand>(kVld, OpcodeEnum::kVldBX,
SourceOpEnum::kVs1);
EXPECT_EQ(src->AsString(), "0");
delete src;
// Test vld.w.l.xx
src = EncodeOpHelper<RV32SourceOperand>(
SetSz(SetRs1(kVld, kRdValue), 0b10) | (0b10 << 20 /* xs2 */) |
(0b1 << 26 /* length */),
OpcodeEnum::kVldWLXx, SourceOpEnum::kVs1);
EXPECT_EQ(src->AsString(), "ra");
delete src;
src = EncodeOpHelper<RV32SourceOperand>(
SetSz(SetRs1(kVld, kRdValue), 0b10) | (0b10 << 20 /* xs2 */) |
(0b1 << 26 /* length */),
OpcodeEnum::kVldWLXx, SourceOpEnum::kVs2);
EXPECT_EQ(src->AsString(), "sp");
delete src;
}
TEST_F(KelvinEncodingTest, KelvinVstEncodeXs1Xs2Vd) {
constexpr uint32_t kVstBase = 0b001000'000000'000000'00'000000'0'111'11;
// Test vd in vst.b.x as source
auto *v_src = EncodeOpHelper<RV32VectorSourceOperand>(
SetRs1(kVstBase, kRdValue), OpcodeEnum::kVstBX, SourceOpEnum::kVd);
EXPECT_EQ(v_src->AsString(), "v0");
delete v_src;
// Test xs1 as x0
auto *dest = EncodeOpHelper<RV32DestOperand>(kVstBase, OpcodeEnum::kVstBX,
DestOpEnum::kVs1);
EXPECT_EQ(dest->AsString(), "zero");
delete dest;
// Test xs1 in vst.w.l.xx as destination
dest = EncodeOpHelper<RV32DestOperand>(
SetSz(SetRs1(kVstBase, kRdValue), 0b10) | (0b10 << 20 /* xs2 */) |
(1 << 26 /* length */),
OpcodeEnum::kVstWLXx, DestOpEnum::kVs1);
EXPECT_EQ(dest->AsString(), "ra");
delete dest;
// Test xs2 in vstq.b.s.xx as source
auto *src = EncodeOpHelper<RV32SourceOperand>(
SetRs1(kVstBase, kRdValue) | (1 << 30 /* vstq */) |
(0b10 << 20 /* xs2 */) | (1 << 27 /* stride */),
OpcodeEnum::kVstqBSXx, SourceOpEnum::kVs2);
EXPECT_EQ(src->AsString(), "sp");
delete src;
}
TEST_F(KelvinEncodingTest, KelvinWideningVs1) {
constexpr uint32_t kVSransBase = 0b010000'000001'000000'00'001000'0'010'00;
auto *v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kVSransBase, OpcodeEnum::kVsransBVv, SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 2);
delete v_src;
// Test vsrans.b.r.vv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kVSransBase | (1 << 27 /* vsrans.r */), OpcodeEnum::kVsransBRVv,
SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 2);
delete v_src;
// Test vsrans.h.vv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
SetSz(kVSransBase, 0b1), OpcodeEnum::kVsransHVv, SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 2);
delete v_src;
// Test vsraqs.b.vv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kVSransBase | (1 << 29 /* vsraqs */), OpcodeEnum::kVsraqsBVv,
SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 4);
delete v_src;
// Test vsraqs.b.r.vv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kVSransBase | (1 << 29) | (1 << 27), OpcodeEnum::kVsraqsBRVv,
SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 4);
delete v_src;
// Test illegal vsrans (vsrans.w.vv)
enc_->ParseInstruction(SetSz(kVSransBase, 0b10));
EXPECT_EQ(enc_->GetOpcode(SlotEnum::kKelvin, 0), OpcodeEnum::kNone);
// Test vacc.h.vv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
SetSz(kVAccsBase, 0b1), OpcodeEnum::kVaccHVv, SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 2);
delete v_src;
// Test vacc.w.u.vv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
SetSz(kVAccsBase, 0b10) | (1 << 26), OpcodeEnum::kVaccWUVv,
SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 2);
delete v_src;
// Test acset.v, actr.v, adwinit.v
constexpr uint32_t kACSetVBase = 0b010000'000000'010000'00'110000'0'001'10;
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kACSetVBase, OpcodeEnum::kAcset, SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 8);
delete v_src;
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kACSetVBase | (1 << 26 /* actr */), OpcodeEnum::kActr,
SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 8);
delete v_src;
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kACSetVBase | (1 << 27 /* adwinit */), OpcodeEnum::kAdwinit,
SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 4);
delete v_src;
// Test aconv.vxv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kAconvBase, OpcodeEnum::kAconvVxv, SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 8);
delete v_src;
// No widening for vadd.b.vv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kVAddBase, OpcodeEnum::kVaddBVv, SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 1);
delete v_src;
// Test vdwconv.vxv
v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kVdwconvBase, OpcodeEnum::kVdwconvVxv, SourceOpEnum::kVs1);
EXPECT_EQ(v_src->size(), 9);
delete v_src;
}
TEST_F(KelvinEncodingTest, KelvinWideningVd) {
// No widening for vld
auto *v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetRs1(kVld, kRdValue), OpcodeEnum::kVldBX, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 1);
delete v_dest;
// Test vaddw.h.vv
constexpr uint32_t kVAddwBase = 0b000100'000001'000000'00'001000'0'100'00;
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetSz(kVAddwBase, 0b1), OpcodeEnum::kVaddwHVv, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test vsubw.w.u.vv
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetSz(kVAddwBase, 0b10) | (0b11 << 26 /* vsubw.u */),
OpcodeEnum::kVsubwWUVv, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test vacc.h.vv
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetSz(kVAccsBase, 0b1), OpcodeEnum::kVaccHVv, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test vacc.w.u.vv
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetSz(kVAccsBase, 0b10) | (1 << 26), OpcodeEnum::kVaccWUVv,
DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test no-widening of vle(same func2 as vacc, func1 == 000)
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
kVAccsBase & ~(1 << 4), OpcodeEnum::kVleBVv, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 1);
delete v_dest;
// Test vmvp.vv
constexpr uint32_t kVMvpBase = 0b001101'000001'000000'00'001000'0'001'00;
v_dest = EncodeOpHelper<RV32VectorDestOperand>(kVMvpBase, OpcodeEnum::kVmvpVv,
DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test vmulw.h.vv
constexpr uint32_t kVMulwBase = 0b000100'000001'000000'00'001000'0'011'00;
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetSz(kVMulwBase, 0b1), OpcodeEnum::kVmulwHVv, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test vmulw.w.u.vv
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetSz(kVMulwBase, 0b10) | (1 << 26 /* vmulw.u */), OpcodeEnum::kVmulwWUVv,
DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test vevnodd.b.vv
constexpr uint32_t kVEvnoddBase = 0b011000'000001'000000'00'001000'0'110'00;
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
kVEvnoddBase | (0b10 << 26 /* vevenodd */), OpcodeEnum::kVevnoddBVv,
DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test vzip.h.vv
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
SetSz(kVEvnoddBase, 0b1) | (0b100 << 26 /* vzip */), OpcodeEnum::kVzipHVv,
DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 2);
delete v_dest;
// Test adwinit.v
constexpr uint32_t kAdwinitVBase = 0b010010'000000'010000'00'110000'0'001'10;
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
kAdwinitVBase, OpcodeEnum::kAdwinit, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 4);
delete v_dest;
// Test vcget
constexpr uint32_t kVCGet = 0b010100'000000'000000'00'110000'0'111'11;
v_dest = EncodeOpHelper<RV32VectorDestOperand>(kVCGet, OpcodeEnum::kVcget,
DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 8);
delete v_dest;
// Test vdwconv
v_dest = EncodeOpHelper<RV32VectorDestOperand>(
kVdwconvBase, OpcodeEnum::kVdwconvVxv, DestOpEnum::kVd);
EXPECT_EQ(v_dest->size(), 4);
delete v_dest;
}
TEST_F(KelvinEncodingTest, KelvinEncodeVs3) {
auto *v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kAconvBase, OpcodeEnum::kAconvVxv, SourceOpEnum::kVs3);
EXPECT_EQ(v_src->AsString(), "v8");
EXPECT_EQ(v_src->size(), 8);
delete v_src;
}
TEST_F(KelvinEncodingTest, KelvinEncodeVs2) {
auto *v_src = EncodeOpHelper<RV32VectorSourceOperand>(
kVAddBase, OpcodeEnum::kVaddBVv, SourceOpEnum::kVs2);
EXPECT_EQ(v_src->size(), 1);
EXPECT_EQ(v_src->AsString(), "v0");
delete v_src;
auto *src = EncodeOpHelper<RV32SourceOperand>(
kVAddBase | 0b10, OpcodeEnum::kVaddBVx, SourceOpEnum::kVs2);
EXPECT_EQ(src->AsString(), "0");
delete src;
}
} // namespace