#include #include #include #include #include #include #include "emu.h" #define BO4 (1<<0) #define BO3 (1<<1) #define BO2 (1<<2) #define BO1 (1<<3) #define BO0 (1<<4) #define XER_SO (1<<31) #define XER_OV (1<<30) #define XER_CA (1<<29) cpu_t cpu; static inline bool carry(void) { fatal("carry() not supported yet"); } /* Byte-swaps a 16-bit value. */ static inline uint16_t swb16(uint16_t n) { return (((n >> 8) & 0xff) | ((n & 0xff) << 8)); } /* Byte-swaps a 32-bit value. */ static inline uint32_t swb32(uint32_t n) { return (((n & 0xff000000) >> 24) | ((n & 0x00ff0000) >> 8) | ((n & 0x0000ff00) << 8) | ((n & 0x000000ff) << 24)); } /* Returns the state of a carry flag after a three-way add. */ static inline bool carry_3(uint32_t a, uint32_t b, uint32_t c) { if ((a+b) < a) return true; if ((a+b+c) < c) return true; return false; } static inline uint32_t reg(uint8_t n) { return cpu.gpr[n]; } static inline uint32_t reg0(uint8_t n) { if (n == 0) return 0; return cpu.gpr[n]; } /* Double to bytes */ static inline uint64_t d2b(double n) { return *(uint64_t*)&n; } /* Float to bytes */ static inline uint32_t f2b(float n) { return *(uint32_t*)&n; } /* Bytes to double */ static inline double b2d(uint64_t n) { return *(double*)&n; } /* Bytes to float */ static inline float b2f(uint32_t n) { return *(float*)&n; } static inline double fpr(uint8_t n) { return b2d(cpu.fpr[n]); } static inline uint32_t ext8(int8_t n) { return (n << 24) >> 24; } static inline uint32_t ext16(int16_t n) { return (n << 16) >> 16; } static inline uint32_t ext26(int32_t n) { return (n << 6) >> 6; } static bool getcr(uint8_t bit) { bit = 31 - bit; /* note PowerPC bit numbering */ return cpu.cr & (1< 0); setcr(2, value == 0); setcr(3, cpu.xer & (1<<31)); } } static void setcr1(bool setcr1, uint64_t value) { if (setcr1) fatal("setcr1 not implemented yet"); } static void mcrf(uint8_t destfield, uint8_t srcfield) { fatal("mcrf not supported yet"); } static void branch(uint8_t bo, uint8_t bi, uint32_t dest, bool a_bit, bool l_bit) { bool bo0 = bo & BO0; bool bo1 = bo & BO1; bool bo2 = bo & BO2; bool bo3 = bo & BO3; bool ctr_ok; bool cond_ok; if (!bo2) cpu.ctr--; ctr_ok = bo2 || (!!cpu.ctr ^ bo3); cond_ok = bo0 || (!!(cpu.cr & (1<<(31-bi))) == bo1); if (ctr_ok && cond_ok) { if (a_bit) cpu.nia = dest; else cpu.nia = dest + cpu.cia; } if (l_bit) cpu.lr = cpu.cia + 4; } static void read_multiple(uint32_t address, uint8_t reg) { while (reg != 32) { cpu.gpr[reg] = read_long(address); reg++; address += 4; } } static void write_multiple(uint32_t address, uint8_t reg) { while (reg != 32) { write_long(address, cpu.gpr[reg]); reg++; address += 4; } } static void read_string(uint32_t address, uint8_t reg, uint8_t bytes) { fatal("read_string not supported yet"); } static void write_string(uint32_t address, uint8_t reg, uint8_t bytes) { fatal("write_string not supported yet"); } static uint32_t addo(uint32_t a, uint32_t b, uint32_t c, bool set_o, bool set_c) { if (set_o) fatal("can't use O bit in add yet"); if (set_c) { cpu.xer = cpu.xer & ~XER_CA; if (carry_3(a, b, c)) cpu.xer = cpu.xer | XER_CA; } return a + b + c; } static uint32_t mulo(uint32_t a, uint32_t b, bool set_o) { if (set_o) fatal("can't use O bit in mul yet"); return a * b; } static int32_t divo(int32_t a, int32_t b, bool set_o) { if (set_o) fatal("can't use O bit in div yet"); if (b == 0) return 0; return a / b; } static uint32_t divuo(uint32_t a, uint32_t b, bool set_o) { if (set_o) fatal("can't use O bit in divu yet"); if (b == 0) return 0; return a / b; } static void compares(int32_t a, int32_t b, uint8_t field) { uint8_t bit = field*4; setcr(bit+0, ab); setcr(bit+2, a==b); setcr(bit+3, cpu.xer & (1<<31)); } static void compareu(uint32_t a, uint32_t b, uint8_t field) { uint8_t bit = field*4; setcr(bit+0, ab); setcr(bit+2, a==b); setcr(bit+3, cpu.xer & (1<<31)); } static void comparef(double a, double b, uint8_t field) { uint8_t c; if (isnan(a) || isnan(b)) c = 0x1; else if (a < b) c = 0x8; else if (a > b) c = 0x4; else c = 0x2; uint8_t bit = 28 - field*4; /* note PowerPC bit numbering */ cpu.cr = cpu.cr & ~(0xf<> (32-shift)); } static uint32_t rlwnm(uint32_t source, uint8_t shift, uint8_t mb, uint8_t me) { uint8_t masksize = 1 + me - mb; /* me and mb are inclusive */ uint32_t mask = (((uint64_t)1<>26); } void dump_state(FILE* stream) { int i; fprintf(stream, "\n"); fprintf(stream, "pc=0x%08x lr=0x%08x ctr=0x%08x xer=0x%08x cr=0x%08x", cpu.cia, cpu.lr, cpu.ctr, cpu.xer, cpu.cr); for (i=0; i<32; i++) { if ((i % 4) == 0) fprintf(stream, "\n"); fprintf(stream, "gpr%02d=0x%08x ", i, cpu.gpr[i]); } fprintf(stderr, "\n"); /* This might fail and cause a reentrant trap if cia is invalid, so * do it last. */ fprintf(stream, "insn=0x%08x\n", read_long(cpu.cia)); } void single_step(void) { uint32_t value = read_long(cpu.cia); cpu.nia = cpu.cia + 4; dispatch(value); cpu.cia = cpu.nia; }