Compare commits
	
		
			2 Commits
		
	
	
		
			dcff3fd588
			...
			580e7baca9
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 580e7baca9 | |||
| aad3bdc9ea | 
							
								
								
									
										25
									
								
								src/arm.zig
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								src/arm.zig
									
									
									
									
									
								
							| @@ -142,7 +142,7 @@ pub fn Arm32(comptime isa: Architecture) type { | |||||||
|                 return (idx * 2) + if (kind == .R14) @as(usize, 1) else 0; |                 return (idx * 2) + if (kind == .R14) @as(usize, 1) else 0; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             inline fn spsrIdx(mode: Mode) usize { |             pub inline fn spsrIdx(mode: Mode) usize { | ||||||
|                 return switch (mode) { |                 return switch (mode) { | ||||||
|                     .Supervisor => 0, |                     .Supervisor => 0, | ||||||
|                     .Abort => 1, |                     .Abort => 1, | ||||||
| @@ -409,6 +409,29 @@ pub fn Arm32(comptime isa: Architecture) type { | |||||||
|             std.debug.panic(format, args); |             std.debug.panic(format, args); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         // TODO: Rename | ||||||
|  |         pub fn undefinedInstructionTrap(self: *Self) void { | ||||||
|  |             // Copy Values from Current Mode | ||||||
|  |             const ret_addr = self.r[15] - @as(u32, if (self.cpsr.t.read()) 2 else 4); | ||||||
|  |             const cpsr = self.cpsr.raw; | ||||||
|  |  | ||||||
|  |             // Switch Mode | ||||||
|  |             self.changeMode(.Undefined); | ||||||
|  |             self.cpsr.t.write(false); // Force ARM Mode | ||||||
|  |             self.cpsr.i.write(true); // Disable normal interrupts | ||||||
|  |  | ||||||
|  |             self.r[14] = ret_addr; // Resume Execution | ||||||
|  |             self.spsr.raw = cpsr; // Previous mode CPSR | ||||||
|  |             self.r[15] = switch (Self.arch) { | ||||||
|  |                 .v4t => 0x0000_0004, | ||||||
|  |                 .v5te => blk: { | ||||||
|  |                     const ctrl = self.cp15.read(0, 1, 0, 0); | ||||||
|  |                     break :blk if (ctrl >> 13 & 1 == 1) 0xFFFF_0004 else 0x0000_0004; | ||||||
|  |                 }, | ||||||
|  |             }; | ||||||
|  |             self.pipe.reload(self); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         pub fn interface(self: *Self) Interpreter { |         pub fn interface(self: *Self) Interpreter { | ||||||
|             return switch (isa) { |             return switch (isa) { | ||||||
|                 .v4t => .{ .v4t = self }, |                 .v4t => .{ .v4t = self }, | ||||||
|   | |||||||
| @@ -51,14 +51,13 @@ pub fn dataTransfer( | |||||||
|             // TODO: Increment address + 4 (and perform op) until coprocessor says stop |             // TODO: Increment address + 4 (and perform op) until coprocessor says stop | ||||||
|  |  | ||||||
|             if (L) { |             if (L) { | ||||||
|                 log.debug("TODO: ldc{s} p{}, c{}, 0x{X:0>8}", .{ [_]u8{if (N) 'l' else ' '}, cp_num, crd, start_address }); |                 cpu.panic("TODO: ldc{s} p{}, c{}, 0x{X:0>8}", .{ [_]u8{if (N) 'l' else ' '}, cp_num, crd, start_address }); | ||||||
|             } else { |             } else { | ||||||
|                 log.debug("TODO: stc{s} p{}, c{}, 0x{X:0>8}", .{ [_]u8{if (N) 'l' else ' '}, cp_num, crd, start_address }); |                 cpu.panic("TODO: stc{s} p{}, c{}, 0x{X:0>8}", .{ [_]u8{if (N) 'l' else ' '}, cp_num, crd, start_address }); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         fn copExt(cpu: *Arm32, opcode: u32) void { |         fn copExt(cpu: *Arm32, opcode: u32) void { | ||||||
|             _ = cpu; |  | ||||||
|             const cp_num = opcode >> 8 & 0xF; |             const cp_num = opcode >> 8 & 0xF; | ||||||
|             const rd = opcode >> 12 & 0xF; |             const rd = opcode >> 12 & 0xF; | ||||||
|             const rn = opcode >> 16 & 0xF; |             const rn = opcode >> 16 & 0xF; | ||||||
| @@ -71,10 +70,10 @@ pub fn dataTransfer( | |||||||
|  |  | ||||||
|             if (L) { |             if (L) { | ||||||
|                 // MRRC |                 // MRRC | ||||||
|                 log.debug("TODO: mrrc p{}, {}, r{}, r{}, c{}", .{ cp_num, cp_opcode, rd, rn, crm }); |                 cpu.panic("TODO: mrrc p{}, {}, r{}, r{}, c{}", .{ cp_num, cp_opcode, rd, rn, crm }); | ||||||
|             } else { |             } else { | ||||||
|                 // MCRR |                 // MCRR | ||||||
|                 log.debug("TODO: mcrr p{}, {}, r{}, r{}, c{}", .{ cp_num, cp_opcode, rd, rn, crm }); |                 cpu.panic("TODO: mcrr p{}, {}, r{}, r{}, c{}", .{ cp_num, cp_opcode, rd, rn, crm }); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     }.inner; |     }.inner; | ||||||
| @@ -90,7 +89,11 @@ pub fn registerTransfer(comptime InstrFn: type, comptime opcode1: u3, comptime L | |||||||
|             const cp_num = opcode >> 8 & 0xF; |             const cp_num = opcode >> 8 & 0xF; | ||||||
|             const crm: u4 = @intCast(opcode & 0xF); |             const crm: u4 = @intCast(opcode & 0xF); | ||||||
|  |  | ||||||
|             std.debug.assert(cp_num == 0xF); // There's no other coprocessor on NDS9; |             switch (cp_num) { | ||||||
|  |                 14 => return, | ||||||
|  |                 15 => if (Arm32.arch == .v4t) return cpu.undefinedInstructionTrap(), | ||||||
|  |                 else => cpu.panic("MRC: unexpected coprocessor #: {}", .{cp_num}), | ||||||
|  |             } | ||||||
|  |  | ||||||
|             if (L) { |             if (L) { | ||||||
|                 // MRC |                 // MRC | ||||||
| @@ -148,9 +151,7 @@ pub fn dataProcessing(comptime InstrFn: type, comptime opcode1: u4, comptime opc | |||||||
|  |  | ||||||
|     return struct { |     return struct { | ||||||
|         fn inner(cpu: *Arm32, opcode: u32) void { |         fn inner(cpu: *Arm32, opcode: u32) void { | ||||||
|             _ = cpu; |             cpu.panic("TODO: handle 0x{X:0>8} which is a coprocessor data processing instr", .{opcode}); | ||||||
|  |  | ||||||
|             log.err("TODO: handle 0x{X:0>8} which is a coprocessor data processing instr", .{opcode}); |  | ||||||
|         } |         } | ||||||
|     }.inner; |     }.inner; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,9 @@ | |||||||
|  | const std = @import("std"); | ||||||
| const sext = @import("zba-util").sext; | const sext = @import("zba-util").sext; | ||||||
| const rotr = @import("zba-util").rotr; | const rotr = @import("zba-util").rotr; | ||||||
|  |  | ||||||
|  | const log = std.log.scoped(.half_and_signed_data_transfer); | ||||||
|  |  | ||||||
| pub fn halfAndSignedDataTransfer(comptime InstrFn: type, comptime P: bool, comptime U: bool, comptime I: bool, comptime W: bool, comptime L: bool) InstrFn { | pub fn halfAndSignedDataTransfer(comptime InstrFn: type, comptime P: bool, comptime U: bool, comptime I: bool, comptime W: bool, comptime L: bool) InstrFn { | ||||||
|     const Arm32 = @typeInfo(@typeInfo(@typeInfo(InstrFn).Pointer.child).Fn.params[0].type.?).Pointer.child; |     const Arm32 = @typeInfo(@typeInfo(@typeInfo(InstrFn).Pointer.child).Fn.params[0].type.?).Pointer.child; | ||||||
|  |  | ||||||
| @@ -66,9 +69,9 @@ pub fn halfAndSignedDataTransfer(comptime InstrFn: type, comptime P: bool, compt | |||||||
|                         // FIXME: I shouldn't have to use @as(u16, ...) here |                         // FIXME: I shouldn't have to use @as(u16, ...) here | ||||||
|                         cpu.write(u16, address, @as(u16, @truncate(cpu.r[rd]))); |                         cpu.write(u16, address, @as(u16, @truncate(cpu.r[rd]))); | ||||||
|                     }, |                     }, | ||||||
|                     0b10 => { |                     0b10 => blk: { | ||||||
|                         // LDRD |                         // LDRD | ||||||
|                         if (Arm32.arch != .v5te) cpu.panic("LDRD: unsupported on arm{s}", .{@tagName(Arm32.arch)}); |                         if (Arm32.arch == .v4t) break :blk; | ||||||
|                         if (rd & 0 != 0) cpu.panic("LDRD: UNDEFINED behaviour when Rd is not even", .{}); |                         if (rd & 0 != 0) cpu.panic("LDRD: UNDEFINED behaviour when Rd is not even", .{}); | ||||||
|                         if (rd == 0xE) cpu.panic("LDRD: UNPREDICTABLE behaviour when rd == 14", .{}); |                         if (rd == 0xE) cpu.panic("LDRD: UNPREDICTABLE behaviour when rd == 14", .{}); | ||||||
|                         if (address & 0x7 != 0b000) cpu.panic("LDRD: UNPREDICTABLE when address (0x{X:0>8} is not double (64-bit) aligned", .{address}); |                         if (address & 0x7 != 0b000) cpu.panic("LDRD: UNPREDICTABLE when address (0x{X:0>8} is not double (64-bit) aligned", .{address}); | ||||||
|   | |||||||
| @@ -30,6 +30,7 @@ pub fn control(comptime InstrFn: type, comptime I: bool, comptime op: u6) InstrF | |||||||
|                 }, |                 }, | ||||||
|                 0b01_0001 => cpu.panic("TODO: implement v5TE BX", .{}), |                 0b01_0001 => cpu.panic("TODO: implement v5TE BX", .{}), | ||||||
|                 0b11_0001 => { // CLZ |                 0b11_0001 => { // CLZ | ||||||
|  |                     if (Arm32.arch == .v4t) return cpu.undefinedInstructionTrap(); | ||||||
|                     const rd = opcode >> 12 & 0xF; |                     const rd = opcode >> 12 & 0xF; | ||||||
|                     const rm = opcode & 0xF; |                     const rm = opcode & 0xF; | ||||||
|  |  | ||||||
| @@ -50,6 +51,7 @@ pub fn control(comptime InstrFn: type, comptime I: bool, comptime op: u6) InstrF | |||||||
|                     cpu.pipe.reload(cpu); |                     cpu.pipe.reload(cpu); | ||||||
|                 }, |                 }, | ||||||
|                 0b00_0101, 0b01_0101, 0b10_0101, 0b11_0101 => { // QADD / QDADD / QSUB / QDSUB |                 0b00_0101, 0b01_0101, 0b10_0101, 0b11_0101 => { // QADD / QDADD / QSUB / QDSUB | ||||||
|  |                     if (Arm32.arch == .v4t) return cpu.undefinedInstructionTrap(); | ||||||
|                     const U = op >> 4 & 1 == 1; |                     const U = op >> 4 & 1 == 1; | ||||||
|                     const D = op >> 5 & 1 == 1; |                     const D = op >> 5 & 1 == 1; | ||||||
|  |  | ||||||
| @@ -84,6 +86,7 @@ pub fn control(comptime InstrFn: type, comptime I: bool, comptime op: u6) InstrF | |||||||
|                 }, |                 }, | ||||||
|                 0b01_0111 => cpu.panic("TODO: handle BKPT", .{}), |                 0b01_0111 => cpu.panic("TODO: handle BKPT", .{}), | ||||||
|                 0b00_1000, 0b00_1010, 0b00_1100, 0b00_1110 => { // SMLA<x><y> |                 0b00_1000, 0b00_1010, 0b00_1100, 0b00_1110 => { // SMLA<x><y> | ||||||
|  |                     if (Arm32.arch == .v4t) return; // no-op | ||||||
|                     const X = op >> 1 & 1; |                     const X = op >> 1 & 1; | ||||||
|                     const Y = op >> 2 & 1; |                     const Y = op >> 2 & 1; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,6 +20,8 @@ pub const arm = struct { | |||||||
|     /// Arithmetic Instruction Extension Space |     /// Arithmetic Instruction Extension Space | ||||||
|     const multiplyExt = @import("cpu/arm/multiply.zig").multiply; |     const multiplyExt = @import("cpu/arm/multiply.zig").multiply; | ||||||
|  |  | ||||||
|  |     const cop = @import("cpu/arm/coprocessor.zig"); | ||||||
|  |  | ||||||
|     /// Determine index into ARM InstrFn LUT |     /// Determine index into ARM InstrFn LUT | ||||||
|     pub fn idx(opcode: u32) u12 { |     pub fn idx(opcode: u32) u12 { | ||||||
|         // FIXME: omit these? |         // FIXME: omit these? | ||||||
| @@ -87,8 +89,29 @@ pub const arm = struct { | |||||||
|                             const L = i >> 8 & 1 == 1; |                             const L = i >> 8 & 1 == 1; | ||||||
|                             break :blk branch(InstrFn, L); |                             break :blk branch(InstrFn, L); | ||||||
|                         }, |                         }, | ||||||
|                         0b10 => und, // COP Data Transfer |                         0b10 => blk: { | ||||||
|                         0b11 => if (i >> 8 & 1 == 1) swi(InstrFn) else und, // COP Data Operation + Register Transfer |                             const P = i >> 8 & 1 == 1; | ||||||
|  |                             const U = i >> 7 & 1 == 1; | ||||||
|  |                             const N = i >> 6 & 1 == 1; | ||||||
|  |                             const W = i >> 5 & 1 == 1; | ||||||
|  |                             const L = i >> 4 & 1 == 1; | ||||||
|  |  | ||||||
|  |                             break :blk cop.dataTransfer(InstrFn, P, U, N, W, L); | ||||||
|  |                         }, | ||||||
|  |                         0b11 => blk: { | ||||||
|  |                             if (i >> 8 & 1 == 1) break :blk swi(InstrFn); | ||||||
|  |  | ||||||
|  |                             const data_opcode1 = i >> 4 & 0xF; // bits 20 -> 23 | ||||||
|  |                             const reg_opcode1 = i >> 5 & 0x7; // bits 21 -> 23 | ||||||
|  |                             const opcode2 = i >> 1 & 0x7; // bits 5 -> 7 | ||||||
|  |                             const L = i >> 4 & 1 == 1; // bit 20 | ||||||
|  |  | ||||||
|  |                             // Bit 4 (index pos of 0) distinguishes between these classes of instructions | ||||||
|  |                             break :blk switch (i & 1 == 1) { | ||||||
|  |                                 true => cop.registerTransfer(InstrFn, reg_opcode1, L, opcode2), | ||||||
|  |                                 false => cop.dataProcessing(InstrFn, data_opcode1, opcode2), | ||||||
|  |                             }; | ||||||
|  |                         }, | ||||||
|                     }, |                     }, | ||||||
|                 }; |                 }; | ||||||
|             } |             } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user