mirror of
https://github.com/FranLMSP/rultra64.git
synced 2026-01-01 07:51:34 -05:00
SRLV and SLLV
This commit is contained in:
55
src/cpu.rs
55
src/cpu.rs
@@ -7,6 +7,13 @@ pub fn params_rd_rs_rt(opcode: u32) -> (usize, usize, usize) {
|
||||
(rd as usize, rs as usize, rt as usize)
|
||||
}
|
||||
|
||||
pub fn params_rd_rt_rs(opcode: u32) -> (usize, usize, usize) {
|
||||
let rd = (opcode >> 11) & 0b11111;
|
||||
let rt = (opcode >> 16) & 0b11111;
|
||||
let rs = (opcode >> 21) & 0b11111;
|
||||
(rd as usize, rt as usize, rs as usize)
|
||||
}
|
||||
|
||||
pub fn params_rt_rs_immediate(opcode: u32) -> (usize, usize, i16) {
|
||||
let rt = (opcode >> 11) & 0b11111;
|
||||
let rs = (opcode >> 21) & 0b11111;
|
||||
@@ -183,6 +190,16 @@ impl CPU {
|
||||
let (rd, rt, sa) = params_rd_rt_sa(opcode);
|
||||
self.sra(rd, rt, sa);
|
||||
},
|
||||
// SLLV
|
||||
0b000_0000_0100 => {
|
||||
let (rd, rt, rs) = params_rd_rt_rs(opcode);
|
||||
self.sllv(rd, rt, rs);
|
||||
},
|
||||
// SRLV
|
||||
0b000_0000_0110 => {
|
||||
let (rd, rt, rs) = params_rd_rt_rs(opcode);
|
||||
self.sllv(rd, rt, rs);
|
||||
},
|
||||
_ => unimplemented!(),
|
||||
};
|
||||
},
|
||||
@@ -516,6 +533,20 @@ impl CPU {
|
||||
let result = ((t >> sa) as u32) & 0xEFFFFFFF;
|
||||
self.registers.set_by_number(rd, ((result | sign) as i32) as i64);
|
||||
}
|
||||
|
||||
pub fn sllv(&mut self, rd: usize, rt: usize, rs: usize) {
|
||||
let t = self.registers.get_by_number(rt);
|
||||
let s = (self.registers.get_by_number(rs) & 0b11111) as usize;
|
||||
let result = t << s;
|
||||
self.registers.set_by_number(rd, result as i64);
|
||||
}
|
||||
|
||||
pub fn srlv(&mut self, rd: usize, rt: usize, rs: usize) {
|
||||
let t = self.registers.get_by_number(rt);
|
||||
let s = (self.registers.get_by_number(rs) & 0b11111) as usize;
|
||||
let result = t >> s;
|
||||
self.registers.set_by_number(rd, result as i64);
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
@@ -913,4 +944,28 @@ mod cpu_instructions_tests {
|
||||
cpu.sra(rd, rt, 3);
|
||||
assert_eq!(cpu.registers.get_by_number(rd), 0b111);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_sllv() {
|
||||
let mut cpu = CPU::new();
|
||||
let rd = 15;
|
||||
let rt = 20;
|
||||
let rs = 25;
|
||||
cpu.registers.set_by_number(rt, 0b111);
|
||||
cpu.registers.set_by_number(rs, 0b11);
|
||||
cpu.sllv(rd, rt, rs);
|
||||
assert_eq!(cpu.registers.get_by_number(rd), 0b111000);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_srlv() {
|
||||
let mut cpu = CPU::new();
|
||||
let rd = 15;
|
||||
let rt = 20;
|
||||
let rs = 25;
|
||||
cpu.registers.set_by_number(rt, 0b111000);
|
||||
cpu.registers.set_by_number(rs, 0b11);
|
||||
cpu.srlv(rd, rt, rs);
|
||||
assert_eq!(cpu.registers.get_by_number(rd), 0b111);
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user