mirror of
https://github.com/holub/mame
synced 2025-10-04 16:34:53 +03:00
commit
5bb61a8965
@ -2415,3 +2415,19 @@ end
|
||||
if (CPUS["MB86901"]~=null or _OPTIONS["with-tools"]) then
|
||||
table.insert(disasm_files , MAME_DIR .. "src/devices/cpu/sparc/sparcdasm.cpp")
|
||||
end
|
||||
|
||||
--------------------------------------------------
|
||||
-- Intergraph CLIPPER (C100/C300/C400) series
|
||||
--@src/devices/cpu/clipper/clipper.h,CPUS["CLIPPER"] = true
|
||||
--------------------------------------------------
|
||||
|
||||
if (CPUS["CLIPPER"]~=null) then
|
||||
files {
|
||||
MAME_DIR .. "src/devices/cpu/clipper/clipper.cpp",
|
||||
MAME_DIR .. "src/devices/cpu/clipper/clipper.h",
|
||||
}
|
||||
end
|
||||
|
||||
if (CPUS["CLIPPER"]~=null or _OPTIONS["with-tools"]) then
|
||||
table.insert(disasm_files , MAME_DIR .. "src/devices/cpu/clipper/clipperd.cpp")
|
||||
end
|
@ -2073,6 +2073,22 @@ files {
|
||||
MAME_DIR .. "src/mame/drivers/tim100.cpp",
|
||||
}
|
||||
|
||||
createMESSProjects(_target, _subtarget, "interpro")
|
||||
files {
|
||||
MAME_DIR .. "src/mame/drivers/interpro.cpp",
|
||||
MAME_DIR .. "src/mame/includes/interpro.h",
|
||||
MAME_DIR .. "src/mame/machine/cammu.h",
|
||||
MAME_DIR .. "src/mame/machine/cammu.cpp",
|
||||
MAME_DIR .. "src/mame/machine/interpro_ioga.h",
|
||||
MAME_DIR .. "src/mame/machine/interpro_ioga.cpp",
|
||||
MAME_DIR .. "src/mame/machine/interpro_ioga.h",
|
||||
MAME_DIR .. "src/mame/machine/interpro_ioga.cpp",
|
||||
MAME_DIR .. "src/mame/machine/interpro_mcga.h",
|
||||
MAME_DIR .. "src/mame/machine/interpro_mcga.cpp",
|
||||
MAME_DIR .. "src/mame/machine/interpro_sga.h",
|
||||
MAME_DIR .. "src/mame/machine/interpro_sga.cpp",
|
||||
}
|
||||
|
||||
createMESSProjects(_target, _subtarget, "interton")
|
||||
files {
|
||||
MAME_DIR .. "src/mame/drivers/vc4000.cpp",
|
||||
|
1375
src/devices/cpu/clipper/clipper.cpp
Normal file
1375
src/devices/cpu/clipper/clipper.cpp
Normal file
File diff suppressed because it is too large
Load Diff
294
src/devices/cpu/clipper/clipper.h
Normal file
294
src/devices/cpu/clipper/clipper.h
Normal file
@ -0,0 +1,294 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
#pragma once
|
||||
|
||||
#ifndef __CLIPPER_H__
|
||||
#define __CLIPPER_H__
|
||||
|
||||
enum clipper_registers
|
||||
{
|
||||
CLIPPER_R0, CLIPPER_R1, CLIPPER_R2, CLIPPER_R3, CLIPPER_R4, CLIPPER_R5, CLIPPER_R6, CLIPPER_R7,
|
||||
CLIPPER_R8, CLIPPER_R9, CLIPPER_R10, CLIPPER_R11, CLIPPER_R12, CLIPPER_R13, CLIPPER_R14, CLIPPER_R15,
|
||||
|
||||
CLIPPER_F0, CLIPPER_F1, CLIPPER_F2, CLIPPER_F3, CLIPPER_F4, CLIPPER_F5, CLIPPER_F6, CLIPPER_F7,
|
||||
CLIPPER_F8, CLIPPER_F9, CLIPPER_F10, CLIPPER_F11, CLIPPER_F12, CLIPPER_F13, CLIPPER_F14, CLIPPER_F15,
|
||||
|
||||
CLIPPER_PSW,
|
||||
CLIPPER_SSW,
|
||||
CLIPPER_PC,
|
||||
};
|
||||
|
||||
enum clipper_addressing_modes
|
||||
{
|
||||
ADDR_MODE_PC32 = 0x10,
|
||||
ADDR_MODE_ABS32 = 0x30,
|
||||
ADDR_MODE_REL32 = 0x60,
|
||||
ADDR_MODE_PC16 = 0x90,
|
||||
ADDR_MODE_REL12 = 0xa0,
|
||||
ADDR_MODE_ABS16 = 0xb0,
|
||||
ADDR_MODE_PCX = 0xd0,
|
||||
ADDR_MODE_RELX = 0xe0,
|
||||
};
|
||||
|
||||
// branch conditions
|
||||
enum clipper_branch_conditions
|
||||
{
|
||||
BRANCH_T = 0x0,
|
||||
BRANCH_LT = 0x1,
|
||||
BRANCH_LE = 0x2,
|
||||
BRANCH_EQ = 0x3,
|
||||
BRANCH_GT = 0x4,
|
||||
BRANCH_GE = 0x5,
|
||||
BRANCH_NE = 0x6,
|
||||
BRANCH_LTU = 0x7,
|
||||
BRANCH_LEU = 0x8,
|
||||
BRANCH_GTU = 0x9,
|
||||
BRANCH_GEU = 0xa,
|
||||
BRANCH_V = 0xb,
|
||||
BRANCH_NV = 0xc,
|
||||
BRANCH_N = 0xd,
|
||||
BRANCH_NN = 0xe,
|
||||
BRANCH_FN = 0xf,
|
||||
};
|
||||
|
||||
enum clipper_psw
|
||||
{
|
||||
PSW_N = 0x00000001, // negative
|
||||
PSW_Z = 0x00000002, // zero
|
||||
PSW_V = 0x00000004, // overflow
|
||||
PSW_C = 0x00000008, // carry out or borrow in
|
||||
PSW_FX = 0x00000010, // floating inexact
|
||||
PSW_FU = 0x00000020, // floating underflow
|
||||
PSW_FD = 0x00000040, // floating divide by zero
|
||||
PSW_FV = 0x00000080, // floating overflow
|
||||
PSW_FI = 0x00000100, // floating invalid operation
|
||||
PSW_EFX = 0x00000200, // enable floating inexact trap
|
||||
PSW_EFU = 0x00000400, // enable floating underflow trap
|
||||
PSW_EFD = 0x00000800, // enable floating divide by zero trap
|
||||
PSW_EFV = 0x00001000, // enable floating overflow trap
|
||||
PSW_EFI = 0x00002000, // enable floating invalid operation trap
|
||||
PSW_EFT = 0x00004000, // enable floating trap
|
||||
PSW_FR = 0x00018000, // floating rounding mode (2 bits)
|
||||
// unused (3 bits)
|
||||
PSW_DSP = 0x00300000, // c400 - delay slot pointer (2 bits)
|
||||
PSW_BIG = 0x00400000, // c400 - big endian (hardware)
|
||||
PSW_T = 0x00800000, // trace trap
|
||||
PSW_CTS = 0x0f000000, // cpu trap status (4 bits)
|
||||
PSW_MTS = 0xf0000000, // memory trap status (4 bits)
|
||||
};
|
||||
|
||||
enum clipper_ssw
|
||||
{
|
||||
SSW_IN = 0x0000000f, // interrupt number (4 bits)
|
||||
SSW_IL = 0x000000f0, // interrupt level (4 bits)
|
||||
SSW_EI = 0x00000100, // enable interrupts
|
||||
SSW_ID = 0x0001fe00, // cpu rev # and type (8 bits)
|
||||
// unused (5 bits)
|
||||
SSW_FRD = 0x00400000, // floating registers dirty
|
||||
SSW_TP = 0x00800000, // trace trap pending
|
||||
SSW_ECM = 0x01000000, // enabled corrected memory error
|
||||
SSW_DF = 0x02000000, // fpu disabled
|
||||
SSW_M = 0x04000000, // mapped mode
|
||||
SSW_KU = 0x08000000, // user protect key
|
||||
SSW_UU = 0x10000000, // user data mode
|
||||
SSW_K = 0x20000000, // protect key
|
||||
SSW_U = 0x40000000, // user mode
|
||||
SSW_P = 0x80000000, // previous mode
|
||||
};
|
||||
|
||||
enum clipper_exception_vectors
|
||||
{
|
||||
// data memory trap group
|
||||
EXCEPTION_D_CORRECTED_MEMORY_ERROR = 0x108,
|
||||
EXCEPTION_D_UNCORRECTABLE_MEMORY_ERROR = 0x110,
|
||||
EXCEPTION_D_ALIGNMENT_FAULT = 0x120,
|
||||
EXCEPTION_D_PAGE_FAULT = 0x128,
|
||||
EXCEPTION_D_READ_PROTECT_FAULT = 0x130,
|
||||
EXCEPTION_D_WRITE_PROTECT_FAULT = 0x138,
|
||||
|
||||
// floating-point arithmetic trap group
|
||||
EXCEPTION_FLOATING_INEXACT = 0x180,
|
||||
EXCEPTION_FLOATING_UNDERFLOW = 0x188,
|
||||
EXCEPTION_FLOATING_DIVIDE_BY_ZERO = 0x190,
|
||||
EXCEPTION_FLOATING_OVERFLOW = 0x1a0,
|
||||
EXCEPTION_FLOATING_INVALID_OPERATION = 0x1c0,
|
||||
|
||||
// integer arithmetic trap group
|
||||
EXCEPTION_INTEGER_DIVIDE_BY_ZERO = 0x208,
|
||||
|
||||
// instruction memory trap group
|
||||
EXCEPTION_I_CORRECTED_MEMORY_ERROR = 0x288,
|
||||
EXCEPTION_I_UNCORRECTABLE_MEMORY_ERROR = 0x290,
|
||||
EXCEPTION_I_ALIGNMENT_FAULT = 0x2a0,
|
||||
EXCEPTION_I_PAGE_FAULT = 0x2a8,
|
||||
EXCEPTION_I_EXECUTE_PROTECT_FAULT = 0x2b0,
|
||||
|
||||
// illegal operation trap group
|
||||
EXCEPTION_ILLEGAL_OPERATION = 0x300,
|
||||
EXCEPTION_PRIVILEGED_INSTRUCTION = 0x308,
|
||||
|
||||
// diagnostic trap group
|
||||
EXCEPTION_TRACE = 0x380,
|
||||
|
||||
// supervisor calls (0x400-0x7f8)
|
||||
EXCEPTION_SUPERVISOR_CALL_BASE = 0x400,
|
||||
|
||||
// prioritized interrupts (0x800-0xff8)
|
||||
EXCEPTION_INTERRUPT_BASE = 0x800,
|
||||
};
|
||||
|
||||
// trap source values are shifted into the correct field in the psw
|
||||
enum clipper_cpu_trap_sources
|
||||
{
|
||||
CTS_NO_CPU_TRAP = 0 << 24,
|
||||
CTS_DIVIDE_BY_ZERO = 2 << 24,
|
||||
CTS_ILLEGAL_OPERATION = 4 << 24,
|
||||
CTS_PRIVILEGED_INSTRUCTION = 5 << 24,
|
||||
CTS_TRACE_TRAP = 7 << 24,
|
||||
};
|
||||
|
||||
enum clipper_memory_trap_sources
|
||||
{
|
||||
MTS_NO_MEMORY_TRAP = 0 << 28,
|
||||
MTS_CORRECTED_MEMORY_ERROR = 1 << 28,
|
||||
MTS_UNCORRECTABLE_MEMORY_ERROR = 2 << 28,
|
||||
MTS_ALIGNMENT_FAULT = 4 << 28,
|
||||
MTS_PAGE_FAULT = 5 << 28,
|
||||
MTS_READ_OR_EXECUTE_PROTECT_FAULT = 6 << 28,
|
||||
MTS_WRITE_PROTECT_FAULT = 7 << 28,
|
||||
};
|
||||
|
||||
// convenience macros for frequently used instruction fields
|
||||
#define R1 (m_info.r1)
|
||||
#define R2 (m_info.r2)
|
||||
|
||||
// convenience macros for dealing with the psw
|
||||
#define PSW(mask) (m_psw & PSW_##mask)
|
||||
#define SSW(mask) (m_ssw & SSW_##mask)
|
||||
|
||||
// macros for setting psw condition codes
|
||||
#define FLAGS(C,V,Z,N) \
|
||||
m_psw = (m_psw & ~(PSW_C | PSW_V | PSW_Z | PSW_N)) | (((C) << 3) | ((V) << 2) | ((Z) << 1) | ((N) << 0));
|
||||
#define FLAGS_CV(C,V) \
|
||||
m_psw = (m_psw & ~(PSW_C | PSW_V)) | (((C) << 3) | ((V) << 2));
|
||||
#define FLAGS_ZN(Z,N) \
|
||||
m_psw = (m_psw & ~(PSW_Z | PSW_N)) | (((Z) << 1) | ((N) << 0));
|
||||
|
||||
// over/underflow for addition/subtraction from here: http://stackoverflow.com/questions/199333/how-to-detect-integer-overflow-in-c-c
|
||||
#define OF_ADD(a, b) ((b > 0) && (a > INT_MAX - b))
|
||||
#define UF_ADD(a, b) ((b < 0) && (a < INT_MIN - b))
|
||||
#define OF_SUB(a, b) ((b < 0) && (a > INT_MAX + b))
|
||||
#define UF_SUB(a, b) ((b > 0) && (a < INT_MIN + b))
|
||||
|
||||
// CLIPPER logic for carry and overflow flags
|
||||
#define C_ADD(a, b) ((u32)a + (u32)b < (u32)a)
|
||||
#define V_ADD(a, b) (OF_ADD((s32)a, (s32)b) || UF_ADD((s32)a, (s32)b))
|
||||
#define C_SUB(a, b) ((u32)a < (u32)b)
|
||||
#define V_SUB(a, b) (OF_SUB((s32)a, (s32)b) || UF_SUB((s32)a, (s32)b))
|
||||
|
||||
class clipper_device : public cpu_device
|
||||
{
|
||||
public:
|
||||
clipper_device(const machine_config &mconfig, device_type type, const char *name, const char *tag, device_t *owner, u32 clock, const char *shortname, const char *source);
|
||||
|
||||
DECLARE_READ_LINE_MEMBER(ssw) { return m_ssw; }
|
||||
|
||||
protected:
|
||||
// device-level overrides
|
||||
virtual void device_start() override;
|
||||
virtual void device_reset() override;
|
||||
|
||||
// device_execute_interface overrides
|
||||
virtual u32 execute_min_cycles() const override { return 1; };
|
||||
virtual u32 execute_max_cycles() const override { return 1; }; // FIXME: don't know, especially macro instructions
|
||||
virtual u32 execute_input_lines() const override { return 2; }; // number of input/interrupt lines (irq/nmi)
|
||||
virtual void execute_run() override;
|
||||
virtual void execute_set_input(int inputnum, int state) override;
|
||||
|
||||
// device_memory_interface overrides
|
||||
virtual const address_space_config *memory_space_config(address_spacenum spacenum = AS_0) const override;
|
||||
|
||||
// device_state_interface overrides
|
||||
#if 0
|
||||
virtual void state_import(const device_state_entry &entry) override;
|
||||
virtual void state_export(const device_state_entry &entry) override;
|
||||
#endif
|
||||
virtual void state_string_export(const device_state_entry &entry, std::string &str) const override;
|
||||
|
||||
// device_disasm_interface overrides
|
||||
virtual uint32_t disasm_min_opcode_bytes() const override { return 2; } // smallest instruction
|
||||
virtual uint32_t disasm_max_opcode_bytes() const override { return 8; } // largest instruction
|
||||
virtual offs_t disasm_disassemble(std::ostream &stream, offs_t pc, const u8 *oprom, const u8 *opram, u32 options) override;
|
||||
|
||||
// core registers
|
||||
u32 m_pc;
|
||||
u32 m_psw;
|
||||
u32 m_ssw;
|
||||
|
||||
// integer registers
|
||||
s32 *m_r; // active registers
|
||||
s32 m_ru[16]; // user registers
|
||||
s32 m_rs[16]; // supervisor registers
|
||||
|
||||
// floating registers
|
||||
double m_f[16];
|
||||
|
||||
private:
|
||||
address_space_config m_insn_config;
|
||||
address_space_config m_data_config;
|
||||
|
||||
address_space *m_insn;
|
||||
address_space *m_data;
|
||||
|
||||
int m_icount;
|
||||
|
||||
int m_irq;
|
||||
int m_nmi;
|
||||
|
||||
// decoded instruction information
|
||||
struct
|
||||
{
|
||||
u8 opcode, subopcode;
|
||||
u8 r1, r2;
|
||||
|
||||
s32 imm;
|
||||
u16 macro;
|
||||
|
||||
// total size of instruction in bytes
|
||||
u32 size;
|
||||
|
||||
// computed effective address
|
||||
u32 address;
|
||||
} m_info;
|
||||
|
||||
void clipper_device::decode_instruction(u16 insn);
|
||||
int clipper_device::execute_instruction();
|
||||
bool clipper_device::evaluate_branch();
|
||||
|
||||
uint32_t clipper_device::intrap(u32 vector, u32 pc, u32 cts = CTS_NO_CPU_TRAP, u32 mts = MTS_NO_MEMORY_TRAP);
|
||||
};
|
||||
|
||||
class clipper_c100_device : public clipper_device
|
||||
{
|
||||
public:
|
||||
clipper_c100_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock);
|
||||
};
|
||||
|
||||
class clipper_c300_device : public clipper_device
|
||||
{
|
||||
public:
|
||||
clipper_c300_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock);
|
||||
};
|
||||
|
||||
class clipper_c400_device : public clipper_device
|
||||
{
|
||||
public:
|
||||
clipper_c400_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock);
|
||||
};
|
||||
|
||||
extern const device_type CLIPPER_C100;
|
||||
extern const device_type CLIPPER_C300;
|
||||
extern const device_type CLIPPER_C400;
|
||||
|
||||
extern CPU_DISASSEMBLE(clipper);
|
||||
#endif /* __CLIPPER_H__ */
|
351
src/devices/cpu/clipper/clipperd.cpp
Normal file
351
src/devices/cpu/clipper/clipperd.cpp
Normal file
@ -0,0 +1,351 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
/*
|
||||
* TODO
|
||||
* - dynamically switch between C400 and C100/C300 instruction sets
|
||||
* - handle failures of addressing mode decoding more elegantly
|
||||
* - improve address decoding to use streams
|
||||
* - detect various cases of illegal instruction encoding
|
||||
*/
|
||||
|
||||
// enable C400 instruction decoding
|
||||
#define C400_INSTRUCTIONS 1
|
||||
|
||||
// the CLIPPER addressing modes (unshifted)
|
||||
enum
|
||||
{
|
||||
ADDR_MODE_PC32 = 0x10,
|
||||
ADDR_MODE_ABS32 = 0x30,
|
||||
ADDR_MODE_REL32 = 0x60,
|
||||
ADDR_MODE_PC16 = 0x90,
|
||||
ADDR_MODE_REL12 = 0xa0,
|
||||
ADDR_MODE_ABS16 = 0xb0,
|
||||
ADDR_MODE_PCX = 0xd0,
|
||||
ADDR_MODE_RELX = 0xe0
|
||||
};
|
||||
|
||||
// macros for decoding various operand fields
|
||||
#define R1 ((insn[0] & 0x00f0) >> 4)
|
||||
#define R2 (insn[0] & 0x000f)
|
||||
|
||||
#define I16 ((int16_t)insn[1])
|
||||
#define I32 (*(int32_t *)&insn[1])
|
||||
#define IMM_VALUE (insn[0] & 0x0080 ? I16 : I32)
|
||||
#define IMM_SIZE (insn[0] & 0x0080 ? 2 : 4)
|
||||
|
||||
#define ADDR_MODE (insn[0] & 0x00f0)
|
||||
#define ADDR_R2 ((insn[0] & 0x0050) == 0x0010 ? (insn[0] & 0x000f) : (insn[1] & 0x000f))
|
||||
#define ADDR_SIZE (ADDR_MODE > ADDR_MODE_REL32 ? 2 : ADDR_MODE == ADDR_MODE_REL32 ? 6 : 4)
|
||||
#define ADDR_RX ((insn[1] & 0xf0) >> 4)
|
||||
#define ADDR_I12 (((int16_t)insn[1]) >> 4)
|
||||
|
||||
/*
|
||||
* Branch condition code mnemonics - the forms beginning with 'c' are
|
||||
* supposed to be used for branches following comparison instructions,
|
||||
* while those beginning with 'r' are for use after move or logical
|
||||
* instructions. We use the first form because we can't know which type
|
||||
* should be used without some kind of dynamic information.
|
||||
*/
|
||||
static const char *const cc[] =
|
||||
{
|
||||
"",
|
||||
"clt", // rgt
|
||||
"cle", // rge
|
||||
"ceq", // req
|
||||
"cgt", // rlt
|
||||
"cge", // rle
|
||||
"cne", // rne
|
||||
"cltu", // rgtu
|
||||
"cleu", // rgeu, nc
|
||||
"cgtu", // rltu, c
|
||||
"cgeu", // rleu
|
||||
"v",
|
||||
"nv",
|
||||
"n",
|
||||
"nn",
|
||||
"fn"
|
||||
};
|
||||
|
||||
/*
|
||||
* Decode an addressing mode into a string.
|
||||
*/
|
||||
char *address (offs_t pc, u16 *insn)
|
||||
{
|
||||
static char buffer[32];
|
||||
|
||||
switch (ADDR_MODE)
|
||||
{
|
||||
case ADDR_MODE_PC32: sprintf(buffer, "0x%x", pc + I32); break;
|
||||
case ADDR_MODE_ABS32: sprintf(buffer, "0x%x", I32); break;
|
||||
case ADDR_MODE_REL32: sprintf(buffer, "%d(r%d)", *(int32_t *)&insn[2], R2); break;
|
||||
case ADDR_MODE_PC16: sprintf(buffer, "0x%x", pc + I16); break;
|
||||
case ADDR_MODE_REL12: sprintf(buffer, "%d(r%d)", ADDR_I12, R2); break;
|
||||
case ADDR_MODE_ABS16: sprintf(buffer, "0x%x", I16); break;
|
||||
case ADDR_MODE_PCX: sprintf(buffer, "[r%d](pc)", ADDR_RX); break;
|
||||
case ADDR_MODE_RELX: sprintf(buffer, "[r%d](r%d)", ADDR_RX, R2); break;
|
||||
default: sprintf(buffer, "ERROR"); break;
|
||||
}
|
||||
|
||||
return buffer;
|
||||
}
|
||||
|
||||
/*
|
||||
* CLIPPER instructions are composed of 1, 2, 3 or 4 16-bit "parcels". The first parcel contains
|
||||
* the opcode and enough information to work out how many additional parcels might be required. The
|
||||
* instruction set is fairly typically RISC-ish, except for these variable length instructions, the
|
||||
* 8 addressing modes, and the "macro instructions", which are actually "subroutines" embedded in
|
||||
* an on-CPU macro instruction ROM. It appears at least some of these macro instructions were removed
|
||||
* from the C400 and generate traps which can be used to implement them in software instead.
|
||||
*/
|
||||
CPU_DISASSEMBLE(clipper)
|
||||
{
|
||||
u16 *insn = (u16 *)oprom;
|
||||
u32 flags = DASMFLAG_SUPPORTED;
|
||||
offs_t bytes;
|
||||
|
||||
switch (insn[0] >> 8)
|
||||
{
|
||||
case 0x00:
|
||||
if (oprom[0] == 0)
|
||||
util::stream_format(stream, "noop");
|
||||
else
|
||||
util::stream_format(stream, "noop $%d", oprom[0]);
|
||||
bytes = 2;
|
||||
break;
|
||||
|
||||
case 0x10: util::stream_format(stream, "movwp r%d,%s", R2, R1 == 0 ? "psw" : R1 == 1 ? "ssw" : "sswf"); bytes = 2; break;
|
||||
case 0x11: util::stream_format(stream, "movpw %s,r%d", R1 == 0 ? "psw" : "ssw", R2); bytes = 2; break;
|
||||
case 0x12: util::stream_format(stream, "calls $%d", insn[0] & 0x7F); bytes = 2; flags |= DASMFLAG_STEP_OVER; break;
|
||||
case 0x13: util::stream_format(stream, "ret r%d", R2); bytes = 2; flags |= DASMFLAG_STEP_OUT; break;
|
||||
case 0x14: util::stream_format(stream, "pushw r%d,r%d", R2, R1); bytes = 2; break;
|
||||
|
||||
case 0x16: util::stream_format(stream, "popw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0x20: util::stream_format(stream, "adds f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x21: util::stream_format(stream, "subs f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x22: util::stream_format(stream, "addd f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x23: util::stream_format(stream, "subd f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x24: util::stream_format(stream, "movs f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x25: util::stream_format(stream, "cmps f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x26: util::stream_format(stream, "movd f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x27: util::stream_format(stream, "cmpd f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x28: util::stream_format(stream, "muls f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x29: util::stream_format(stream, "divs f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x2a: util::stream_format(stream, "muld f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x2b: util::stream_format(stream, "divd f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x2c: util::stream_format(stream, "movsw f%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x2d: util::stream_format(stream, "movws r%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0x2e: util::stream_format(stream, "movdl f%d,r%d:%d", R1, R2 + 0, R2 + 1); bytes = 2; break;
|
||||
case 0x2f: util::stream_format(stream, "movld r%d:r%d,f%d", R1 + 0, R1 + 1, R2); bytes = 2; break;
|
||||
|
||||
case 0x30: util::stream_format(stream, "shaw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x31: util::stream_format(stream, "shal r%d,r%d:r%d", R1, R2 + 0, R2 + 1); bytes = 2; break;
|
||||
case 0x32: util::stream_format(stream, "shlw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x33: util::stream_format(stream, "shll r%d,r%d:r%d", R1, R2 + 0, R2 + 1); bytes = 2; break;
|
||||
case 0x34: util::stream_format(stream, "rotw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x35: util::stream_format(stream, "rotl r%d,r%d:r%d", R1, R2 + 0, R2 + 1); bytes = 2; break;
|
||||
|
||||
case 0x38: util::stream_format(stream, "shai $%d,r%d", I16, R2); bytes = 4; break;
|
||||
case 0x39: util::stream_format(stream, "shali $%d,r%d:r%d", I16, R2 + 0, R2 + 1); bytes = 4; break;
|
||||
case 0x3a: util::stream_format(stream, "shli $%d,r%d", I16, R2); bytes = 4; break;
|
||||
case 0x3b: util::stream_format(stream, "shlli $%d,r%d:r%d", I16, R2 + 0, R2 + 1); bytes = 4; break;
|
||||
case 0x3c: util::stream_format(stream, "roti $%d,r%d", I16, R2); bytes = 4; break;
|
||||
case 0x3d: util::stream_format(stream, "rotli $%d,r%d:r%d", I16, R2 + 0, R2 + 1); bytes = 4; break;
|
||||
|
||||
case 0x44: util::stream_format(stream, "call r%d,(r%d)", R2, R1); bytes = 2; flags |= DASMFLAG_STEP_OVER; break;
|
||||
case 0x45: util::stream_format(stream, "call r%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; flags |= DASMFLAG_STEP_OVER; break;
|
||||
#if C400_INSTRUCTIONS
|
||||
case 0x46: util::stream_format(stream, "loadd2 (r%d),f%d", R1, R2); bytes = 2; break;
|
||||
case 0x47: util::stream_format(stream, "loadd2 %s,f%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
#endif
|
||||
case 0x48: util::stream_format(stream, "b%-4s (r%d)", cc[R2], R1); bytes = 2; break;
|
||||
case 0x49: util::stream_format(stream, "b%-4s %s", cc[ADDR_R2], address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
#if C400_INSTRUCTIONS
|
||||
// delayed branches
|
||||
case 0x4a: util::stream_format(stream, "cdb r%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x4b: util::stream_format(stream, "cdb r%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x4c: util::stream_format(stream, "cdbeq r%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x4d: util::stream_format(stream, "cdbeq r%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x4e: util::stream_format(stream, "cdbne r%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x4f: util::stream_format(stream, "cdbne r%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x50: util::stream_format(stream, "db%-4s (r%d)", cc[R2], R1); bytes = 2; break;
|
||||
case 0x51: util::stream_format(stream, "db%-4s %s", cc[ADDR_R2], address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
#else
|
||||
// these instructions are in the C300 documentation, but appear to be replaced in the C400
|
||||
case 0x4c: util::stream_format(stream, "bf%s (r%d)", R2 == 0 ? "any" : "bad", R1); bytes = 2; break;
|
||||
case 0x4d: util::stream_format(stream, "bf%s %s", ADDR_R2 == 0 ? "any" : "bad", address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
#endif
|
||||
|
||||
case 0x60: util::stream_format(stream, "loadw (r%d),r%d", R1, R2); bytes = 2; break;
|
||||
case 0x61: util::stream_format(stream, "loadw %s,r%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x62: util::stream_format(stream, "loada (r%d),r%d", R1, R2); bytes = 2; break;
|
||||
case 0x63: util::stream_format(stream, "loada %s,r%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x64: util::stream_format(stream, "loads (r%d),f%d", R1, R2); bytes = 2; break;
|
||||
case 0x65: util::stream_format(stream, "loads %s,f%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x66: util::stream_format(stream, "loadd (r%d),f%d", R1, R2); bytes = 2; break;
|
||||
case 0x67: util::stream_format(stream, "loadd %s,f%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x68: util::stream_format(stream, "loadb (r%d),r%d", R1, R2); bytes = 2; break;
|
||||
case 0x69: util::stream_format(stream, "loadb %s,r%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x6a: util::stream_format(stream, "loadbu (r%d),r%d", R1, R2); bytes = 2; break;
|
||||
case 0x6b: util::stream_format(stream, "loadbu %s,r%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x6c: util::stream_format(stream, "loadh (r%d),r%d", R1, R2); bytes = 2; break;
|
||||
case 0x6d: util::stream_format(stream, "loadh %s,r%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x6e: util::stream_format(stream, "loadhu (r%d),r%d", R1, R2); bytes = 2; break;
|
||||
case 0x6f: util::stream_format(stream, "loadhu %s,r%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
|
||||
case 0x70: util::stream_format(stream, "storw r%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x71: util::stream_format(stream, "storw r%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x72: util::stream_format(stream, "tsts (r%d),r%d", R1, R2); bytes = 2; break;
|
||||
case 0x73: util::stream_format(stream, "tsts %s,r%d", address(pc, insn), ADDR_R2); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x74: util::stream_format(stream, "stors f%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x75: util::stream_format(stream, "stors f%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x76: util::stream_format(stream, "stord f%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x77: util::stream_format(stream, "stord f%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
case 0x78: util::stream_format(stream, "storb r%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x79: util::stream_format(stream, "storb r%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
|
||||
case 0x7c: util::stream_format(stream, "storh r%d,(r%d)", R2, R1); bytes = 2; break;
|
||||
case 0x7d: util::stream_format(stream, "storh r%d,%s", ADDR_R2, address(pc, insn)); bytes = 2 + ADDR_SIZE; break;
|
||||
|
||||
case 0x80: util::stream_format(stream, "addw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0x82: util::stream_format(stream, "addq $%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x83: util::stream_format(stream, "addi $%d,r%d", IMM_VALUE, R2); bytes = 2 + IMM_SIZE; break;
|
||||
case 0x84: util::stream_format(stream, "movw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0x86: util::stream_format(stream, "loadq $%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x87: util::stream_format(stream, "loadi $%d,r%d", IMM_VALUE, R2); bytes = 2 + IMM_SIZE; break;
|
||||
case 0x88: util::stream_format(stream, "andw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0x8b: util::stream_format(stream, "andi $%d,r%d", IMM_VALUE, R2); bytes = 2 + IMM_SIZE; break;
|
||||
case 0x8c: util::stream_format(stream, "orw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0x8f: util::stream_format(stream, "ori $%d,r%d", IMM_VALUE, R2); bytes = 2 + IMM_SIZE; break;
|
||||
|
||||
case 0x90: util::stream_format(stream, "addwc r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x91: util::stream_format(stream, "subwc r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0x93: util::stream_format(stream, "negw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0x98: util::stream_format(stream, "mulw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x99: util::stream_format(stream, "mulwx r%d,r%d:r%d", R1, R2 + 0, R2 + 1); bytes = 2; break;
|
||||
case 0x9a: util::stream_format(stream, "mulwu r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x9b: util::stream_format(stream, "mulwux r%d,r%d:r%d", R1, R2 + 0, R2 + 1); bytes = 2; break;
|
||||
case 0x9c: util::stream_format(stream, "divw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x9d: util::stream_format(stream, "modw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x9e: util::stream_format(stream, "divwu r%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0x9f: util::stream_format(stream, "modwu r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0xa0: util::stream_format(stream, "subw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0xa2: util::stream_format(stream, "subq $%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0xa3: util::stream_format(stream, "subi $%d,r%d", IMM_VALUE, R2); bytes = 2 + IMM_SIZE; break;
|
||||
case 0xa4: util::stream_format(stream, "cmpw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0xa6: util::stream_format(stream, "cmpq $%d,r%d", R1, R2); bytes = 2; break;
|
||||
case 0xa7: util::stream_format(stream, "cmpi $%d,r%d", IMM_VALUE, R2); bytes = 2 + IMM_SIZE; break;
|
||||
case 0xa8: util::stream_format(stream, "xorw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0xab: util::stream_format(stream, "xori $%d,r%d", IMM_VALUE, R2); bytes = 2 + IMM_SIZE; break;
|
||||
case 0xac: util::stream_format(stream, "notw r%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
case 0xae: util::stream_format(stream, "notq $%d,r%d", R1, R2); bytes = 2; break;
|
||||
|
||||
#if C400_INSTRUCTIONS
|
||||
case 0xb0: util::stream_format(stream, "abss f%d,f%d", R1, R2); bytes = 2; break;
|
||||
case 0xb2: util::stream_format(stream, "absd f%d,f%d", R1, R2); bytes = 2; break;
|
||||
#endif
|
||||
|
||||
case 0xb4:
|
||||
case 0xb5:
|
||||
// unprivileged macro instructions
|
||||
switch (insn[0] & 0xff)
|
||||
{
|
||||
case 0x00: case 0x01: case 0x02: case 0x03:
|
||||
case 0x04: case 0x05: case 0x06: case 0x07:
|
||||
case 0x08: case 0x09: case 0x0a: case 0x0b:
|
||||
case 0x0c:
|
||||
util::stream_format(stream, "savew%d", R2);
|
||||
break;
|
||||
|
||||
case 0x0d: util::stream_format(stream, "movc"); break;
|
||||
case 0x0e: util::stream_format(stream, "initc"); break;
|
||||
case 0x0f: util::stream_format(stream, "cmpc"); break;
|
||||
|
||||
case 0x10: case 0x11: case 0x12: case 0x13:
|
||||
case 0x14: case 0x15: case 0x16: case 0x17:
|
||||
case 0x18: case 0x19: case 0x1a: case 0x1b:
|
||||
case 0x1c:
|
||||
util::stream_format(stream, "restw%d", R2);
|
||||
break;
|
||||
|
||||
case 0x20: case 0x21: case 0x22: case 0x23:
|
||||
case 0x24: case 0x25: case 0x26: case 0x27:
|
||||
util::stream_format(stream, "saved%d", R2);
|
||||
break;
|
||||
|
||||
case 0x28: case 0x29: case 0x2a: case 0x2b:
|
||||
case 0x2c: case 0x2d: case 0x2e: case 0x2f:
|
||||
util::stream_format(stream, "restd%d", R2);
|
||||
break;
|
||||
|
||||
case 0x30: util::stream_format(stream, "cnvsw f%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x31: util::stream_format(stream, "cnvrsw f%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x32: util::stream_format(stream, "cnvtsw f%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x33: util::stream_format(stream, "cnvws r%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x34: util::stream_format(stream, "cnvdw f%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x35: util::stream_format(stream, "cnvrdw f%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x36: util::stream_format(stream, "cnvtdw f%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x37: util::stream_format(stream, "cnvwd r%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x38: util::stream_format(stream, "cnvsd f%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x39: util::stream_format(stream, "cnvds f%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x3a: util::stream_format(stream, "negs f%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x3b: util::stream_format(stream, "negd f%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x3c: util::stream_format(stream, "scalbs r%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x3d: util::stream_format(stream, "scalbd r%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x3e: util::stream_format(stream, "trapfn"); break;
|
||||
case 0x3f: util::stream_format(stream, "loadfs r%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
|
||||
default:
|
||||
util::stream_format(stream, "macro 0x%04x 0x%04x", insn[0], insn[1]);
|
||||
break;
|
||||
}
|
||||
bytes = 4;
|
||||
break;
|
||||
case 0xb6:
|
||||
case 0xb7:
|
||||
// privileged macro instructions
|
||||
switch (insn[0] & 0xff)
|
||||
{
|
||||
case 0x00: util::stream_format(stream, "movus r%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x01: util::stream_format(stream, "movsu r%d,r%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
case 0x02: util::stream_format(stream, "saveur r%d", (insn[1] & 0xf0) >> 4); break;
|
||||
case 0x03: util::stream_format(stream, "restur r%d", (insn[1] & 0xf0) >> 4); break;
|
||||
case 0x04: util::stream_format(stream, "reti r%d", (insn[1] & 0xf0) >> 4); flags |= DASMFLAG_STEP_OUT; break;
|
||||
case 0x05: util::stream_format(stream, "wait"); break;
|
||||
#if C400_INSTRUCTIONS
|
||||
case 0x07: util::stream_format(stream, "loadts r%d,f%d", (insn[1] & 0xf0) >> 4, insn[1] & 0xf); break;
|
||||
#endif
|
||||
default:
|
||||
util::stream_format(stream, "macro 0x%04x 0x%04x", insn[0], insn[1]);
|
||||
break;
|
||||
}
|
||||
bytes = 4;
|
||||
break;
|
||||
|
||||
#if C400_INSTRUCTIONS
|
||||
case 0xbc: util::stream_format(stream, "waitd"); bytes = 2; break;
|
||||
case 0xc0: util::stream_format(stream, "s%-4s r%d", cc[R2], R1); bytes = 2; break;
|
||||
#endif
|
||||
|
||||
default:
|
||||
util::stream_format(stream, ".word 0x%04x", insn[0]);
|
||||
bytes = 2;
|
||||
break;
|
||||
}
|
||||
|
||||
return bytes | flags;
|
||||
}
|
345
src/mame/drivers/interpro.cpp
Normal file
345
src/mame/drivers/interpro.cpp
Normal file
@ -0,0 +1,345 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
#include "includes/interpro.h"
|
||||
#include "debugger.h"
|
||||
|
||||
#define VERBOSE 0
|
||||
#if VERBOSE
|
||||
#define LOG_SYSTEM(...) logerror(__VA_ARGS__)
|
||||
#define LOG_IDPROM(...) logerror(__VA_ARGS__)
|
||||
#else
|
||||
#define LOG_SYSTEM(...) {}
|
||||
#define LOG_IDPROM(...) {}
|
||||
#endif
|
||||
|
||||
// machine start
|
||||
void interpro_state::machine_start()
|
||||
{
|
||||
m_system_reg[SREG_CTRL2] = CTRL2_COLDSTART | CTRL2_PWRENA | CTRL2_PWRUP;
|
||||
}
|
||||
|
||||
void interpro_state::machine_reset()
|
||||
{
|
||||
// flash rom requires the following values
|
||||
m_system_reg[SREG_ERROR] = 0x00;
|
||||
m_system_reg[SREG_STATUS] = 0x400;
|
||||
m_system_reg[SREG_CTRL1] = CTRL1_FLOPRDY;
|
||||
}
|
||||
|
||||
WRITE16_MEMBER(interpro_state::system_w)
|
||||
{
|
||||
switch (offset)
|
||||
{
|
||||
case SREG_LED:
|
||||
LOG_SYSTEM("LED value %d at pc 0x%08x\n", data, space.device().safe_pc());
|
||||
break;
|
||||
|
||||
case SREG_STATUS: // not sure if writable?
|
||||
break;
|
||||
|
||||
case SREG_CTRL1:
|
||||
LOG_SYSTEM("system control register 1 write data 0x%x pc 0x%08x\n", data, space.device().safe_pc());
|
||||
|
||||
if ((data ^ m_system_reg[offset]) & CTRL1_LEDDP)
|
||||
LOG_SYSTEM("LED decimal point %s\n", data & CTRL1_LEDDP ? "on" : "off");
|
||||
|
||||
m_system_reg[offset] = data;
|
||||
break;
|
||||
|
||||
case SREG_CTRL2:
|
||||
LOG_SYSTEM("system control register 2 write data 0x%x pc 0x%08x\n", data, space.device().safe_pc());
|
||||
if (data & CTRL2_RESET)
|
||||
{
|
||||
m_system_reg[SREG_CTRL2] &= ~CTRL2_COLDSTART;
|
||||
|
||||
machine().schedule_soft_reset();
|
||||
}
|
||||
else
|
||||
m_system_reg[offset] = data & 0x0f; // top four bits are not persistent
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
READ16_MEMBER(interpro_state::system_r)
|
||||
{
|
||||
LOG_SYSTEM("system register read offset %d pc 0x%08x\n", offset, space.device().safe_pc());
|
||||
switch (offset)
|
||||
{
|
||||
case SREG_ERROR:
|
||||
case SREG_STATUS:
|
||||
case SREG_CTRL1:
|
||||
case SREG_CTRL2:
|
||||
default:
|
||||
return m_system_reg[offset];
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
READ32_MEMBER(interpro_state::idprom_r)
|
||||
{
|
||||
LOG_IDPROM("idprom read offset 0x%x mask 0x%08x at 0x%08x\n", offset, mem_mask, space.device().safe_pc());
|
||||
|
||||
// abitrary fake number for now, not working properly
|
||||
uint32_t speed = 70000000;
|
||||
|
||||
static uint8_t idprom[] = {
|
||||
// module type id
|
||||
'M', 'P', 'C', 'B',
|
||||
'*', '*', '*', '*',
|
||||
|
||||
// ECO bytes
|
||||
0x87, 0x65, 0x43, 0x21,
|
||||
0xbb, 0xcc, 0xdd, 0xee,
|
||||
|
||||
// the following 8 bytes are "feature bytes"
|
||||
// the feature bytes contain a 32 bit word which is divided by 40000
|
||||
// if they're empty, a default value of 50 000 000 is used
|
||||
// perhaps this is a system speed (50MHz)?
|
||||
0x2, 0x34, 0x56, 0x78,
|
||||
(speed >> 24) & 0xff, (speed >> 16) & 0xff, (speed >> 8) & 0xff, (speed >> 0) & 0xff,
|
||||
|
||||
// reserved bytes
|
||||
0xff, 0xff,
|
||||
|
||||
// family
|
||||
// boot rom tests for family == 0x41 or 0x42
|
||||
// if so, speed read from feature bytes 2 & 3
|
||||
// if not, read speed from feature bytes 4-7
|
||||
//0x41, 0x00, // 2800-series CPU
|
||||
0x24, 0x00, // 2000-series system board
|
||||
|
||||
// footprint and checksum
|
||||
0x55, 0xaa, 0x55, 0x00
|
||||
};
|
||||
|
||||
switch (offset)
|
||||
{
|
||||
case 0x1f:
|
||||
{
|
||||
uint8_t sum = 0;
|
||||
|
||||
// compute the checksum (sum of all bytes must be == 0x00)
|
||||
for (int i = 0; i < 0x20; i++)
|
||||
sum += idprom[i];
|
||||
|
||||
return 0x100 - (sum & 0xff);
|
||||
}
|
||||
|
||||
default:
|
||||
return idprom[offset];
|
||||
}
|
||||
}
|
||||
|
||||
READ32_MEMBER(interpro_state::slot0_r)
|
||||
{
|
||||
// a known graphics board idprom
|
||||
static uint8_t slot0[] = {
|
||||
0x00, 0x00, 0x00, 0x00, '9', '6', '3', 'A', // board
|
||||
0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, // eco
|
||||
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, // features
|
||||
0xff, 0xff, // reserved
|
||||
0x22, 0x00, // family
|
||||
0x55, 0xaa, 0x55, 0x00
|
||||
};
|
||||
|
||||
return ((uint8_t *)&slot0)[offset % 32];
|
||||
}
|
||||
|
||||
WRITE8_MEMBER(interpro_state::rtc_w)
|
||||
{
|
||||
switch (offset)
|
||||
{
|
||||
case 0x00:
|
||||
// write to RTC register
|
||||
m_rtc->write(space, 1, data);
|
||||
break;
|
||||
|
||||
case 0x40:
|
||||
// set RTC read/write address
|
||||
m_rtc->write(space, 0, data);
|
||||
break;
|
||||
|
||||
default:
|
||||
logerror("rtc: write to unknown offset 0x%02x data 0x%02x at pc 0x%08x\n", offset, data, space.device().safe_pc());
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
READ8_MEMBER(interpro_state::rtc_r)
|
||||
{
|
||||
switch (offset)
|
||||
{
|
||||
case 0x00:
|
||||
// read from RTC register
|
||||
return m_rtc->read(space, 1);
|
||||
|
||||
// read from InterPro system ID PROM (contains MAC address)
|
||||
case 0x40: return 0x12;
|
||||
case 0x41: return 0x34;
|
||||
case 0x42: return 0x56;
|
||||
|
||||
default:
|
||||
logerror("rtc: read from unknown offset 0x%02x at pc 0x%08x\n", offset, space.device().safe_pc());
|
||||
return 0xff;
|
||||
}
|
||||
}
|
||||
|
||||
READ8_MEMBER(interpro_state::scsi_r)
|
||||
{
|
||||
return m_scsi->read(space, offset >> 6, mem_mask);
|
||||
}
|
||||
|
||||
WRITE8_MEMBER(interpro_state::scsi_w)
|
||||
{
|
||||
m_scsi->write(space, offset >> 6, data, mem_mask);
|
||||
}
|
||||
|
||||
// driver init
|
||||
DRIVER_INIT_MEMBER(interpro_state, ip2800)
|
||||
{
|
||||
address_space &as = m_mmu->space(AS_1);
|
||||
}
|
||||
|
||||
// these maps point the cpu virtual addresses to the mmu
|
||||
static ADDRESS_MAP_START(clipper_insn_map, AS_PROGRAM, 32, interpro_state)
|
||||
AM_RANGE(0x00000000, 0xffffffff) AM_DEVREAD32(INTERPRO_MMU_TAG, cammu_device, mmu_r, 0xffffffff)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
static ADDRESS_MAP_START(clipper_data_map, AS_DATA, 32, interpro_state)
|
||||
AM_RANGE(0x00000000, 0xffffffff) AM_DEVREADWRITE32(INTERPRO_MMU_TAG, cammu_device, mmu_r, mmu_w, 0xffffffff)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
// these maps represent the real main, i/o and boot spaces of the system
|
||||
static ADDRESS_MAP_START(interpro_main_map, AS_0, 32, interpro_state)
|
||||
AM_RANGE(0x00000000, 0x00ffffff) AM_RAM // 16M RAM
|
||||
|
||||
AM_RANGE(0x7f100000, 0x7f11ffff) AM_ROM AM_REGION(INTERPRO_ROM_TAG, 0)
|
||||
AM_RANGE(0x7f180000, 0x7f1bffff) AM_ROM AM_REGION(INTERPRO_EEPROM_TAG, 0)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
static ADDRESS_MAP_START(interpro_io_map, AS_1, 32, interpro_state)
|
||||
AM_RANGE(0x00000000, 0x00000fff) AM_DEVICE(INTERPRO_MMU_TAG, cammu_device, map)
|
||||
AM_RANGE(0x00001000, 0x00001fff) AM_RAM
|
||||
|
||||
AM_RANGE(0x40000000, 0x4000003f) AM_DEVICE16(INTERPRO_MCGA_TAG, interpro_mcga_device, map, 0xffff)
|
||||
AM_RANGE(0x4f007e00, 0x4f007eff) AM_DEVICE(INTERPRO_SGA_TAG, interpro_sga_device, map)
|
||||
|
||||
AM_RANGE(0x7f000100, 0x7f00011f) AM_DEVICE8(INTERPRO_FDC_TAG, n82077aa_device, map, 0xff)
|
||||
// this is probably the srx arbiter ga
|
||||
AM_RANGE(0x7f000200, 0x7f0002ff) AM_RAM
|
||||
AM_RANGE(0x7f000300, 0x7f00030f) AM_READWRITE16(system_r, system_w, 0xffff)
|
||||
AM_RANGE(0x7f000400, 0x7f00040f) AM_DEVREADWRITE8(INTERPRO_SCC1_TAG, scc85C30_device, ba_cd_inv_r, ba_cd_inv_w, 0xff)
|
||||
AM_RANGE(0x7f000410, 0x7f00041f) AM_DEVREADWRITE8(INTERPRO_SCC2_TAG, scc85230_device, ba_cd_inv_r, ba_cd_inv_w, 0xff)
|
||||
AM_RANGE(0x7f000500, 0x7f0006ff) AM_READWRITE8(rtc_r, rtc_w, 0xff)
|
||||
AM_RANGE(0x7f000700, 0x7f00077f) AM_READ(idprom_r)
|
||||
AM_RANGE(0x7f001000, 0x7f001fff) AM_READWRITE8(scsi_r, scsi_w, 0x0000ff00)
|
||||
|
||||
AM_RANGE(0x7f0fff00, 0x7f0fffff) AM_DEVICE(INTERPRO_IOGA_TAG, interpro_ioga_device, map)
|
||||
|
||||
AM_RANGE(0x08000000, 0x08000fff) AM_NOP // bogus
|
||||
AM_RANGE(0x8f000000, 0x8f0fffff) AM_READ(slot0_r)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
static ADDRESS_MAP_START(interpro_boot_map, AS_2, 32, interpro_state)
|
||||
AM_RANGE(0x00000000, 0x00001fff) AM_RAM
|
||||
ADDRESS_MAP_END
|
||||
|
||||
FLOPPY_FORMATS_MEMBER(interpro_state::floppy_formats)
|
||||
FLOPPY_PC_FORMAT
|
||||
FLOPPY_FORMATS_END
|
||||
|
||||
static SLOT_INTERFACE_START(interpro_floppies)
|
||||
SLOT_INTERFACE("525dd", FLOPPY_525_DD)
|
||||
SLOT_INTERFACE("35hd", FLOPPY_35_HD)
|
||||
SLOT_INTERFACE_END
|
||||
|
||||
// input ports
|
||||
static INPUT_PORTS_START(ip2800)
|
||||
INPUT_PORTS_END
|
||||
|
||||
static MACHINE_CONFIG_START(ip2800, interpro_state)
|
||||
MCFG_CPU_ADD(INTERPRO_CPU_TAG, CLIPPER_C400, XTAL_10MHz)
|
||||
MCFG_CPU_PROGRAM_MAP(clipper_insn_map)
|
||||
MCFG_CPU_DATA_MAP(clipper_data_map)
|
||||
MCFG_CPU_IRQ_ACKNOWLEDGE_DEVICE(INTERPRO_IOGA_TAG, interpro_ioga_device, inta_cb)
|
||||
|
||||
MCFG_DEVICE_ADD(INTERPRO_MMU_TAG, CAMMU_C4T, 0)
|
||||
MCFG_DEVICE_ADDRESS_MAP(AS_0, interpro_main_map)
|
||||
MCFG_DEVICE_ADDRESS_MAP(AS_1, interpro_io_map)
|
||||
MCFG_DEVICE_ADDRESS_MAP(AS_2, interpro_boot_map)
|
||||
MCFG_CAMMU_SSW_CB(DEVREADLINE(INTERPRO_CPU_TAG, clipper_device, ssw))
|
||||
|
||||
// serial controllers and rs232 bus
|
||||
MCFG_SCC85C30_ADD(INTERPRO_SCC1_TAG, XTAL_4_9152MHz, 0, 0, 0, 0)
|
||||
|
||||
MCFG_Z80SCC_OUT_TXDA_CB(DEVWRITELINE("rs232a", rs232_port_device, write_txd))
|
||||
MCFG_Z80SCC_OUT_TXDB_CB(DEVWRITELINE("rs232b", rs232_port_device, write_txd))
|
||||
MCFG_Z80SCC_OUT_INT_CB(DEVWRITELINE(INTERPRO_IOGA_TAG, interpro_ioga_device, ir11_w))
|
||||
|
||||
MCFG_RS232_PORT_ADD("rs232a", default_rs232_devices, nullptr)
|
||||
MCFG_RS232_RXD_HANDLER(DEVWRITELINE(INTERPRO_SCC1_TAG, z80scc_device, rxa_w))
|
||||
MCFG_RS232_DCD_HANDLER(DEVWRITELINE(INTERPRO_SCC1_TAG, z80scc_device, dcda_w))
|
||||
MCFG_RS232_CTS_HANDLER(DEVWRITELINE(INTERPRO_SCC1_TAG, z80scc_device, ctsa_w))
|
||||
|
||||
// the following port is known as "port 2"
|
||||
MCFG_RS232_PORT_ADD("rs232b", default_rs232_devices, "terminal")
|
||||
MCFG_RS232_RXD_HANDLER(DEVWRITELINE(INTERPRO_SCC1_TAG, z80scc_device, rxb_w))
|
||||
MCFG_RS232_DCD_HANDLER(DEVWRITELINE(INTERPRO_SCC1_TAG, z80scc_device, dcdb_w))
|
||||
MCFG_RS232_CTS_HANDLER(DEVWRITELINE(INTERPRO_SCC1_TAG, z80scc_device, ctsb_w))
|
||||
|
||||
MCFG_SCC85230_ADD(INTERPRO_SCC2_TAG, XTAL_4_9152MHz, 0, 0, 0, 0)
|
||||
|
||||
// real-time clock/non-volatile memory
|
||||
MCFG_MC146818_ADD(INTERPRO_RTC_TAG, XTAL_32_768kHz)
|
||||
MCFG_MC146818_UTC(true)
|
||||
MCFG_MC146818_IRQ_HANDLER(DEVWRITELINE(INTERPRO_IOGA_TAG, interpro_ioga_device, ir9_w))
|
||||
|
||||
// floppy
|
||||
MCFG_N82077AA_ADD(INTERPRO_FDC_TAG, n82077aa_device::MODE_PS2)
|
||||
MCFG_UPD765_INTRQ_CALLBACK(DEVWRITELINE(INTERPRO_IOGA_TAG, interpro_ioga_device, ir1_w))
|
||||
MCFG_UPD765_DRQ_CALLBACK(DEVWRITELINE(INTERPRO_IOGA_TAG, interpro_ioga_device, drq_floppy))
|
||||
MCFG_FLOPPY_DRIVE_ADD("fdc:0", interpro_floppies, "525dd", interpro_state::floppy_formats)
|
||||
MCFG_FLOPPY_DRIVE_ADD("fdc:1", interpro_floppies, "35hd", interpro_state::floppy_formats)
|
||||
MCFG_FLOPPY_DRIVE_SOUND(false)
|
||||
|
||||
// scsi
|
||||
MCFG_DEVICE_ADD(INTERPRO_SCSI_TAG, SCSI_PORT, 0)
|
||||
MCFG_SCSIDEV_ADD(INTERPRO_SCSI_TAG ":" SCSI_PORT_DEVICE1, "harddisk", SCSIHD, SCSI_ID_0)
|
||||
MCFG_SCSIDEV_ADD(INTERPRO_SCSI_TAG ":" SCSI_PORT_DEVICE2, "cdrom", SCSICD, SCSI_ID_3)
|
||||
|
||||
MCFG_DEVICE_ADD(INTERPRO_SCSI_ADAPTER_TAG, NCR539X, XTAL_12_5MHz)
|
||||
MCFG_LEGACY_SCSI_PORT(INTERPRO_SCSI_TAG)
|
||||
MCFG_NCR539X_OUT_IRQ_CB(DEVWRITELINE(INTERPRO_IOGA_TAG, interpro_ioga_device, ir0_w))
|
||||
MCFG_NCR539X_OUT_DRQ_CB(DEVWRITELINE(INTERPRO_IOGA_TAG, interpro_ioga_device, drq_scsi))
|
||||
|
||||
// i/o gate array
|
||||
MCFG_INTERPRO_IOGA_ADD(INTERPRO_IOGA_TAG)
|
||||
MCFG_INTERPRO_IOGA_NMI_CB(INPUTLINE(INTERPRO_CPU_TAG, INPUT_LINE_NMI))
|
||||
MCFG_INTERPRO_IOGA_IRQ_CB(INPUTLINE(INTERPRO_CPU_TAG, INPUT_LINE_IRQ0))
|
||||
//MCFG_INTERPRO_IOGA_DMA_CB(IOGA_DMA_CHANNEL_PLOTTER, unknown)
|
||||
//MCFG_INTERPRO_IOGA_DMA_CB(IOGA_DMA_SCSI, DEVREAD8(INTERPRO_SCSI_TAG, ncr539x_device, dma_read_data), DEVWRITE8(INTERPRO_SCSI_TAG, ncr539x_device, dma_write_data))
|
||||
MCFG_INTERPRO_IOGA_DMA_CB(IOGA_DMA_FLOPPY, DEVREAD8(INTERPRO_FDC_TAG, n82077aa_device, mdma_r), DEVWRITE8(INTERPRO_FDC_TAG, n82077aa_device, mdma_w))
|
||||
MCFG_INTERPRO_IOGA_DMA_CB(IOGA_DMA_SERIAL, DEVREAD8(INTERPRO_SCC1_TAG, z80scc_device, da_r), DEVWRITE8(INTERPRO_SCC1_TAG, z80scc_device, da_w))
|
||||
MCFG_INTERPRO_IOGA_FDCTC_CB(DEVWRITELINE(INTERPRO_FDC_TAG, n82077aa_device, tc_line_w))
|
||||
MCFG_INTERPRO_IOGA_DMA_BUS(INTERPRO_CAMMU_TAG, AS_0)
|
||||
|
||||
// memory control gate array
|
||||
MCFG_DEVICE_ADD(INTERPRO_MCGA_TAG, INTERPRO_MCGA, 0)
|
||||
|
||||
// srx gate array
|
||||
MCFG_DEVICE_ADD(INTERPRO_SGA_TAG, INTERPRO_SGA, 0)
|
||||
|
||||
MACHINE_CONFIG_END
|
||||
|
||||
ROM_START(ip2800)
|
||||
ROM_REGION(0x0020000, INTERPRO_ROM_TAG, 0)
|
||||
ROM_SYSTEM_BIOS(0, "IP2830", "IP2830")
|
||||
ROMX_LOAD("ip2830_rom.bin", 0x00000, 0x20000, CRC(467ce7bd) SHA1(53faee40d5df311f53b24c930e434cbf94a5c4aa), ROM_BIOS(1))
|
||||
|
||||
ROM_REGION(0x0040000, INTERPRO_EEPROM_TAG, 0)
|
||||
ROM_LOAD_OPTIONAL("ip2830_eeprom.bin", 0x00000, 0x40000, CRC(a0c0899f) SHA1(dda6fbca81f9885a1a76ca3c25e80463a83a0ef7))
|
||||
ROM_END
|
||||
|
||||
/* YEAR NAME PARENT COMPAT MACHINE INPUT CLASS INIT COMPANY FULLNAME FLAGS */
|
||||
COMP( 1990, ip2800, 0, 0, ip2800, ip2800, interpro_state, ip2800, "Intergraph", "InterPro 2800", MACHINE_NOT_WORKING | MACHINE_NO_SOUND)
|
131
src/mame/includes/interpro.h
Normal file
131
src/mame/includes/interpro.h
Normal file
@ -0,0 +1,131 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
#pragma once
|
||||
|
||||
#ifndef INTERPRO_H_
|
||||
#define INTERPRO_H_
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
#include "cpu/clipper/clipper.h"
|
||||
#include "machine/cammu.h"
|
||||
|
||||
#include "machine/interpro_ioga.h"
|
||||
#include "machine/interpro_mcga.h"
|
||||
#include "machine/interpro_sga.h"
|
||||
|
||||
#include "machine/z80scc.h"
|
||||
#include "machine/mc146818.h"
|
||||
#include "machine/upd765.h"
|
||||
#include "machine/ncr539x.h"
|
||||
|
||||
#include "bus/scsi/scsi.h"
|
||||
#include "bus/scsi/scsicd.h"
|
||||
#include "bus/scsi/scsihd.h"
|
||||
#include "bus/rs232/rs232.h"
|
||||
|
||||
#include "formats/pc_dsk.h"
|
||||
|
||||
#define INTERPRO_CPU_TAG "cpu"
|
||||
#define INTERPRO_MMU_TAG "mmu"
|
||||
|
||||
#define INTERPRO_RTC_TAG "rtc"
|
||||
#define INTERPRO_SCC1_TAG "scc1"
|
||||
#define INTERPRO_SCC2_TAG "scc2"
|
||||
#define INTERPRO_ROM_TAG "rom"
|
||||
#define INTERPRO_EEPROM_TAG "eeprom"
|
||||
#define INTERPRO_TERMINAL_TAG "terminal"
|
||||
#define INTERPRO_FDC_TAG "fdc"
|
||||
#define INTERPRO_SCSI_TAG "scsi"
|
||||
#define INTERPRO_IOGA_TAG "ioga"
|
||||
#define INTERPRO_MCGA_TAG "mcga"
|
||||
#define INTERPRO_SGA_TAG "sga"
|
||||
#define INTERPRO_SCSI_ADAPTER_TAG "adapter"
|
||||
|
||||
// system board register offsets
|
||||
#define SREG_LED 0
|
||||
#define SREG_ERROR 0
|
||||
#define SREG_STATUS 1
|
||||
#define SREG_CTRL1 2
|
||||
#define SREG_CTRL2 3
|
||||
|
||||
// control register 1
|
||||
#define CTRL1_FLOPLOW 0x0001
|
||||
#define CTRL1_FLOPRDY 0x0002
|
||||
#define CTRL1_LEDENA 0x0004
|
||||
#define CTRL1_LEDDP 0x0008
|
||||
#define CTRL1_ETHLOOP 0x0010
|
||||
#define CTRL1_ETHDTR 0x0020
|
||||
#define CTRL1_ETHRMOD 0x0040
|
||||
#define CTRL1_CLIPRESET 0x0040
|
||||
#define CTRL1_FIFOACTIVE 0x0080
|
||||
|
||||
// control register 2
|
||||
#define CTRL2_PWRUP 0x0001
|
||||
#define CTRL2_PWRENA 0x0002
|
||||
#define CTRL2_HOLDOFF 0x0004
|
||||
#define CTRL2_EXTNMIENA 0x0008
|
||||
#define CTRL2_COLDSTART 0x0010
|
||||
#define CTRL2_RESET 0x0020
|
||||
#define CTRL2_BUSENA 0x0040
|
||||
#define CTRL2_FRCPARITY 0x0080
|
||||
#define CTRL2_FLASHEN 0x0080
|
||||
#define CTRL2_WMASK 0x000f
|
||||
|
||||
class interpro_state : public driver_device
|
||||
{
|
||||
public:
|
||||
interpro_state(const machine_config &mconfig, device_type type, const char *tag)
|
||||
: driver_device(mconfig, type, tag),
|
||||
m_maincpu(*this, INTERPRO_CPU_TAG),
|
||||
m_mmu(*this, INTERPRO_MMU_TAG),
|
||||
m_scc1(*this, INTERPRO_SCC1_TAG),
|
||||
m_scc2(*this, INTERPRO_SCC2_TAG),
|
||||
m_rtc(*this, INTERPRO_RTC_TAG),
|
||||
m_fdc(*this, INTERPRO_FDC_TAG),
|
||||
m_scsi(*this, INTERPRO_SCSI_ADAPTER_TAG),
|
||||
m_ioga(*this, INTERPRO_IOGA_TAG),
|
||||
m_mcga(*this, INTERPRO_MCGA_TAG),
|
||||
m_sga(*this, INTERPRO_SGA_TAG)
|
||||
{ }
|
||||
|
||||
required_device<clipper_device> m_maincpu;
|
||||
required_device<cammu_device> m_mmu;
|
||||
|
||||
// FIXME: not sure which one is the escc
|
||||
required_device<z80scc_device> m_scc1;
|
||||
required_device<z80scc_device> m_scc2;
|
||||
required_device<mc146818_device> m_rtc;
|
||||
required_device<n82077aa_device> m_fdc;
|
||||
required_device<ncr539x_device> m_scsi;
|
||||
|
||||
required_device<interpro_ioga_device> m_ioga;
|
||||
required_device<interpro_mcga_device> m_mcga;
|
||||
required_device<interpro_sga_device> m_sga;
|
||||
|
||||
DECLARE_DRIVER_INIT(ip2800);
|
||||
|
||||
DECLARE_WRITE16_MEMBER(system_w);
|
||||
DECLARE_READ16_MEMBER(system_r);
|
||||
|
||||
DECLARE_WRITE8_MEMBER(rtc_w);
|
||||
DECLARE_READ8_MEMBER(rtc_r);
|
||||
|
||||
DECLARE_READ32_MEMBER(idprom_r);
|
||||
DECLARE_READ32_MEMBER(slot0_r);
|
||||
|
||||
DECLARE_READ8_MEMBER(scsi_r);
|
||||
DECLARE_WRITE8_MEMBER(scsi_w);
|
||||
|
||||
DECLARE_FLOPPY_FORMATS(floppy_formats);
|
||||
|
||||
protected:
|
||||
virtual void machine_start() override;
|
||||
virtual void machine_reset() override;
|
||||
|
||||
private:
|
||||
uint16_t m_system_reg[4];
|
||||
};
|
||||
|
||||
#endif
|
180
src/mame/machine/cammu.cpp
Normal file
180
src/mame/machine/cammu.cpp
Normal file
@ -0,0 +1,180 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
/*
|
||||
* An implementation of the Fairchild/Intergraph Cache and Memory Management Unit (CAMMU) designed for use with the CLIPPER CPU family.
|
||||
*
|
||||
* Primary reference: http://bitsavers.trailing-edge.com/pdf/fairchild/clipper/CLIPPER%20C300%2032-Bit%20Compute%20Engine.pdf
|
||||
* Another reference: http://www.eecs.berkeley.edu/Pubs/TechRpts/1986/CSD-86-289.pdf
|
||||
*
|
||||
* This implementation is currently at a very early stage, and is only sufficient to handle the bare minimum of boot/diagnostic code.
|
||||
*
|
||||
* TODO
|
||||
* - almost everything
|
||||
* - map registers
|
||||
* - refactor hardware tlb
|
||||
* - address translation
|
||||
* - faults
|
||||
* - tlb
|
||||
* - cache
|
||||
* - bus errors
|
||||
*/
|
||||
|
||||
#include "cammu.h"
|
||||
|
||||
#define VERBOSE 0
|
||||
|
||||
// each variant of the cammu has different registers and a different addressing map
|
||||
// TODO: decode the cammu registers properly
|
||||
DEVICE_ADDRESS_MAP_START(map, 32, cammu_c4t_device)
|
||||
AM_RANGE(0x000, 0xfff) AM_READWRITE(cammu_r, cammu_w)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
DEVICE_ADDRESS_MAP_START(map, 32, cammu_c4i_device)
|
||||
AM_RANGE(0x000, 0xfff) AM_READWRITE(cammu_r, cammu_w)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
DEVICE_ADDRESS_MAP_START(map, 32, cammu_c3_device)
|
||||
AM_RANGE(0x000, 0xfff) AM_READWRITE(cammu_r, cammu_w)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
const device_type CAMMU_C4T = &device_creator<cammu_c4t_device>;
|
||||
const device_type CAMMU_C4I = &device_creator<cammu_c4i_device>;
|
||||
const device_type CAMMU_C3 = &device_creator<cammu_c3_device>;
|
||||
|
||||
cammu_c4t_device::cammu_c4t_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
||||
: cammu_device(mconfig, CAMMU_C4T, "C4E/C4T CAMMU", tag, owner, clock, "C4T", __FILE__) { }
|
||||
|
||||
cammu_c4i_device::cammu_c4i_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
||||
: cammu_device(mconfig, CAMMU_C4I, "C4I CAMMU", tag, owner, clock, "C4I", __FILE__) { }
|
||||
|
||||
cammu_c3_device::cammu_c3_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
||||
: cammu_device(mconfig, CAMMU_C4T, "C1/C3 CAMMU", tag, owner, clock, "C3", __FILE__) { }
|
||||
|
||||
cammu_device::cammu_device(const machine_config &mconfig, device_type type, const char *name, const char *tag, device_t *owner, uint32_t clock, const char *shortname, const char *source)
|
||||
: device_t(mconfig, type, name, tag, owner, clock, shortname, source)
|
||||
, device_memory_interface(mconfig, *this),
|
||||
m_main_space_config("main", ENDIANNESS_LITTLE, 32, 32, 0),
|
||||
m_io_space_config("io", ENDIANNESS_LITTLE, 32, 32, 0),
|
||||
m_boot_space_config("boot", ENDIANNESS_LITTLE, 32, 32, 0),
|
||||
m_main_space(nullptr),
|
||||
m_io_space(nullptr),
|
||||
m_boot_space(nullptr),
|
||||
m_ssw_func(*this)
|
||||
{ }
|
||||
|
||||
void cammu_device::device_start()
|
||||
{
|
||||
m_ssw_func.resolve_safe(0);
|
||||
|
||||
m_main_space = &space(AS_0);
|
||||
m_io_space = &space(AS_1);
|
||||
m_boot_space = &space(AS_2);
|
||||
}
|
||||
|
||||
void cammu_device::device_reset()
|
||||
{
|
||||
}
|
||||
|
||||
const address_space_config *cammu_device::memory_space_config (address_spacenum spacenum) const
|
||||
{
|
||||
switch (spacenum)
|
||||
{
|
||||
case AS_0: return &m_main_space_config;
|
||||
case AS_1: return &m_io_space_config;
|
||||
case AS_2: return &m_boot_space_config;
|
||||
}
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
READ32_MEMBER(cammu_device::mmu_r)
|
||||
{
|
||||
u32 ssw = m_ssw_func();
|
||||
u32 address = offset << 2;
|
||||
|
||||
// in supervisor mode, the first 8 pages are always mapped via the hard-wired tlb
|
||||
if ((ssw & 0x40000000) == 0 && (address & ~0x7fff) == 0)
|
||||
{
|
||||
switch (address & 0xf000)
|
||||
{
|
||||
case 0x0000:
|
||||
case 0x1000:
|
||||
case 0x2000:
|
||||
case 0x3000:
|
||||
// pages 0-3: main space pages 0-3
|
||||
return m_main_space->read_dword(address, mem_mask);
|
||||
|
||||
case 0x4000:
|
||||
case 0x5000:
|
||||
// pages 4-5: i/o space pages 0-1
|
||||
return m_io_space->read_dword(address & 0x1fff, mem_mask);
|
||||
|
||||
case 0x6000:
|
||||
case 0x7000:
|
||||
// pages 6-7: boot space pages 0-1
|
||||
return m_boot_space->read_dword(address & 0x1fff, mem_mask);
|
||||
}
|
||||
}
|
||||
|
||||
// FIXME: currently maps addresses with upper bits 0x00 or 0x7f1 to main memory and everything else to I/O
|
||||
if ((address & 0xff000000) == 0x00000000 || (address & 0xfff00000) == 0x7f100000)
|
||||
{
|
||||
#ifdef ICACHE_ENTRIES
|
||||
// if this is an instruction fetch, check the cache first
|
||||
if (space.spacenum() == AS_PROGRAM)
|
||||
{
|
||||
if (m_icache[offset & (ICACHE_ENTRIES-1)].offset != offset)
|
||||
{
|
||||
m_icache[offset & (ICACHE_ENTRIES - 1)].offset = offset;
|
||||
m_icache[offset & (ICACHE_ENTRIES - 1)].data = m_main_space->read_dword(address, mem_mask);
|
||||
}
|
||||
|
||||
return m_icache[offset & (ICACHE_ENTRIES - 1)].data;
|
||||
}
|
||||
else
|
||||
#endif
|
||||
return m_main_space->read_dword(address, mem_mask);
|
||||
}
|
||||
else
|
||||
return m_io_space->read_dword(address, mem_mask);
|
||||
}
|
||||
|
||||
WRITE32_MEMBER(cammu_device::mmu_w)
|
||||
{
|
||||
u32 ssw = m_ssw_func();
|
||||
u32 address = offset << 2;
|
||||
|
||||
// in supervisor mode, the first 8 pages are always mapped via the hard-wired tlb
|
||||
if ((ssw & 0x40000000) == 0 && (address & ~0x7fff) == 0)
|
||||
{
|
||||
switch (address & 0xf000)
|
||||
{
|
||||
case 0x0000:
|
||||
case 0x1000:
|
||||
case 0x2000:
|
||||
case 0x3000:
|
||||
// pages 0-3: main space pages 0-3
|
||||
m_main_space->write_dword(address, data, mem_mask);
|
||||
return;
|
||||
|
||||
case 0x4000:
|
||||
case 0x5000:
|
||||
// pages 4-5: i/o space pages 0-1
|
||||
m_io_space->write_dword(address & 0x1fff, data, mem_mask);
|
||||
return;
|
||||
|
||||
case 0x6000:
|
||||
case 0x7000:
|
||||
// pages 6-7: boot space pages 0-1
|
||||
m_boot_space->write_dword(address & 0x1fff, data, mem_mask);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// FIXME: currently maps addresses with upper bits 0x00 or 0x7f1 to main memory and everything else to I/O
|
||||
if ((address & 0xff000000) == 0x00000000 || (address & 0xfff00000) == 0x7f100000)
|
||||
m_main_space->write_dword(address, data, mem_mask);
|
||||
else
|
||||
m_io_space->write_dword(address, data, mem_mask);
|
||||
}
|
93
src/mame/machine/cammu.h
Normal file
93
src/mame/machine/cammu.h
Normal file
@ -0,0 +1,93 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
#pragma once
|
||||
|
||||
#ifndef CAMMU_H_
|
||||
#define CAMMU_H_
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
// the following enables a very crude instruction cache - it has known (future)
|
||||
// problems, but speeds up cpu execution quite noticeably in the short term by
|
||||
// avoiding some of the delays in the mame memory subsystem
|
||||
#define ICACHE_ENTRIES 32768
|
||||
|
||||
#define MCFG_CAMMU_SSW_CB(_sswcb) \
|
||||
devcb = &cammu_device::static_set_ssw_callback(*device, DEVCB_##_sswcb);
|
||||
|
||||
class cammu_device : public device_t, public device_memory_interface
|
||||
{
|
||||
public:
|
||||
cammu_device(const machine_config &mconfig, device_type type, const char *name, const char *tag, device_t *owner, uint32_t clock, const char *shortname, const char *source);
|
||||
|
||||
template<class _Object> static devcb_base &static_set_ssw_callback(device_t &device, _Object object) { return downcast<cammu_device &>(device).m_ssw_func.set_callback(object); }
|
||||
|
||||
virtual DECLARE_ADDRESS_MAP(map, 32) = 0;
|
||||
|
||||
DECLARE_READ32_MEMBER(mmu_r);
|
||||
DECLARE_WRITE32_MEMBER(mmu_w);
|
||||
|
||||
DECLARE_READ32_MEMBER(cammu_r) { return m_cammu[offset]; }
|
||||
DECLARE_WRITE32_MEMBER(cammu_w) { m_cammu[offset] = data; }
|
||||
|
||||
protected:
|
||||
// device-level overrides
|
||||
virtual void device_start() override;
|
||||
virtual void device_reset() override;
|
||||
|
||||
// device_memory_interface overrides
|
||||
virtual const address_space_config *memory_space_config (address_spacenum spacenum) const override;
|
||||
|
||||
private:
|
||||
devcb_read32 m_ssw_func;
|
||||
|
||||
address_space_config m_main_space_config;
|
||||
address_space_config m_io_space_config;
|
||||
address_space_config m_boot_space_config;
|
||||
|
||||
address_space *m_main_space;
|
||||
address_space *m_io_space;
|
||||
address_space *m_boot_space;
|
||||
|
||||
u32 m_cammu[1024];
|
||||
|
||||
#ifdef ICACHE_ENTRIES
|
||||
struct icache
|
||||
{
|
||||
u32 offset;
|
||||
u32 data;
|
||||
} m_icache[ICACHE_ENTRIES];
|
||||
#endif
|
||||
};
|
||||
|
||||
class cammu_c4t_device : public cammu_device
|
||||
{
|
||||
public:
|
||||
cammu_c4t_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock);
|
||||
|
||||
virtual DECLARE_ADDRESS_MAP(map, 32) override;
|
||||
};
|
||||
|
||||
class cammu_c4i_device : public cammu_device
|
||||
{
|
||||
public:
|
||||
cammu_c4i_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock);
|
||||
|
||||
virtual DECLARE_ADDRESS_MAP(map, 32) override;
|
||||
};
|
||||
|
||||
class cammu_c3_device : public cammu_device
|
||||
{
|
||||
public:
|
||||
cammu_c3_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock);
|
||||
|
||||
virtual DECLARE_ADDRESS_MAP(map, 32) override;
|
||||
};
|
||||
|
||||
// device type definitions
|
||||
extern const device_type CAMMU_C4T;
|
||||
extern const device_type CAMMU_C4I;
|
||||
extern const device_type CAMMU_C3;
|
||||
|
||||
#endif
|
691
src/mame/machine/interpro_ioga.cpp
Normal file
691
src/mame/machine/interpro_ioga.cpp
Normal file
@ -0,0 +1,691 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
/*
|
||||
* An implementation of the IOGA device found on Intergraph InterPro family workstations. There is no
|
||||
* public documentation on this device, so the implementation is being built to follow the logic of the
|
||||
* system boot ROM and its diagnostic tests.
|
||||
*
|
||||
* The device handles most of the I/O for the system, including timers, interrupts, DMA and target device
|
||||
* interfacing. There remains a significant amount of work to be completed before the boot diagnostics will
|
||||
* pass without errors, let alone successfully booting CLIX.
|
||||
*
|
||||
* Please be aware that code in here is not only broken, it's likely wrong in many cases.
|
||||
*
|
||||
* TODO
|
||||
* - too long to list
|
||||
*/
|
||||
#include "interpro_ioga.h"
|
||||
|
||||
#define VERBOSE 0
|
||||
#if VERBOSE
|
||||
#define LOG_TIMER_MASK 0xff
|
||||
#define LOG_TIMER(timer, ...) if (LOG_TIMER_MASK & (1 << timer)) logerror(__VA_ARGS__)
|
||||
#define LOG_INTERRUPT(...) logerror(__VA_ARGS__)
|
||||
#define LOG_IOGA(...) logerror(__VA_ARGS__)
|
||||
#define LOG_DMA(...) logerror(__VA_ARGS__)
|
||||
#else
|
||||
#define LOG_TIMER_MASK 0x00
|
||||
#define LOG_TIMER(timer, ...)
|
||||
#define LOG_INTERRUPT(...)
|
||||
#define LOG_IOGA(...)
|
||||
#define LOG_DMA(...)
|
||||
#endif
|
||||
|
||||
DEVICE_ADDRESS_MAP_START(map, 32, interpro_ioga_device)
|
||||
AM_RANGE(0x0c, 0x1b) AM_READWRITE(dma_plotter_r, dma_plotter_w)
|
||||
AM_RANGE(0x1c, 0x1f) AM_READWRITE(dma_plotter_eosl_r, dma_plotter_eosl_w)
|
||||
AM_RANGE(0x20, 0x2f) AM_READWRITE(dma_scsi_r, dma_scsi_w)
|
||||
AM_RANGE(0x30, 0x3f) AM_READWRITE(dma_floppy_r, dma_floppy_w)
|
||||
|
||||
AM_RANGE(0x5c, 0x7f) AM_READWRITE16(icr_r, icr_w, 0xffffffff)
|
||||
AM_RANGE(0x80, 0x83) AM_READWRITE16(icr18_r, icr18_w, 0x0000ffff)
|
||||
AM_RANGE(0x80, 0x83) AM_READWRITE8(softint_r, softint_w, 0x00ff0000)
|
||||
AM_RANGE(0x80, 0x83) AM_READWRITE8(nmictrl_r, nmictrl_w, 0xff000000)
|
||||
|
||||
AM_RANGE(0x88, 0x8b) AM_READWRITE(timer_prescaler_r, timer_prescaler_w)
|
||||
AM_RANGE(0x8c, 0x8f) AM_READWRITE(timer0_r, timer0_w)
|
||||
AM_RANGE(0x90, 0x93) AM_READWRITE(timer1_r, timer1_w)
|
||||
AM_RANGE(0x94, 0x97) AM_READ(error_address_r)
|
||||
AM_RANGE(0x98, 0x9b) AM_READ(error_businfo_r)
|
||||
AM_RANGE(0x9c, 0x9f) AM_READWRITE16(arbctl_r, arbctl_w, 0x0000ffff)
|
||||
|
||||
AM_RANGE(0xa8, 0xab) AM_READWRITE(timer3_r, timer3_w)
|
||||
|
||||
AM_RANGE(0xb0, 0xbf) AM_READWRITE16(softint_vector_r, softint_vector_w, 0xffffffff)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
const device_type INTERPRO_IOGA = &device_creator<interpro_ioga_device>;
|
||||
|
||||
interpro_ioga_device::interpro_ioga_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
||||
: device_t(mconfig, INTERPRO_IOGA, "InterPro IOGA", tag, owner, clock, "ioga", __FILE__),
|
||||
m_out_nmi_func(*this),
|
||||
m_out_int_func(*this),
|
||||
m_memory_space(nullptr),
|
||||
m_dma_channel{
|
||||
{ 0,0,0,0,false, 0, {*this}, {*this} },
|
||||
{ 0,0,0,0,false, 0, {*this}, {*this} },
|
||||
{ 0,0,0,0,false, 0, {*this}, {*this} },
|
||||
{ 0,0,0,0,false, 0, {*this}, {*this} } },
|
||||
m_fdc_tc_func(*this)
|
||||
{
|
||||
}
|
||||
|
||||
void interpro_ioga_device::device_start()
|
||||
{
|
||||
// resolve callbacks
|
||||
m_out_nmi_func.resolve();
|
||||
m_out_int_func.resolve();
|
||||
|
||||
// TODO: parameterise the cammu name and space number
|
||||
// grab the main memory space from the mmu so we can do DMA to/from it
|
||||
device_memory_interface *mmu;
|
||||
siblingdevice("mmu")->interface(mmu);
|
||||
m_memory_space = &mmu->space(AS_0);
|
||||
|
||||
for (int i = 0; i < IOGA_DMA_CHANNELS; i++)
|
||||
{
|
||||
m_dma_channel[i].device_r.resolve_safe(0xff);
|
||||
m_dma_channel[i].device_w.resolve();
|
||||
}
|
||||
|
||||
m_fdc_tc_func.resolve();
|
||||
|
||||
// allocate ioga timers
|
||||
m_timer[0] = timer_alloc(IOGA_TIMER_0);
|
||||
m_timer[1] = timer_alloc(IOGA_TIMER_1);
|
||||
m_timer[2] = timer_alloc(IOGA_TIMER_2);
|
||||
m_timer[3] = timer_alloc(IOGA_TIMER_3);
|
||||
|
||||
for (auto & elem : m_timer)
|
||||
elem->enable(false);
|
||||
|
||||
// allocate timer for DMA controller
|
||||
m_dma_timer = timer_alloc(IOGA_TIMER_DMA);
|
||||
m_dma_timer->adjust(attotime::never);
|
||||
}
|
||||
|
||||
void interpro_ioga_device::device_reset()
|
||||
{
|
||||
m_nmi_pending = false;
|
||||
|
||||
m_interrupt_active = 0;
|
||||
m_irq_forced = 0;
|
||||
|
||||
// configure timer 0 at 60Hz
|
||||
m_timer_reg[0] = 0;
|
||||
m_timer[0]->adjust(attotime::zero, IOGA_TIMER_0, attotime::from_hz(60));
|
||||
}
|
||||
|
||||
/******************************************************************************
|
||||
Timers
|
||||
******************************************************************************/
|
||||
READ32_MEMBER(interpro_ioga_device::timer1_r)
|
||||
{
|
||||
uint32_t result = m_timer1_count & IOGA_TIMER1_VMASK;
|
||||
|
||||
// set the start bit if the timer is currently enabled
|
||||
if (m_timer[1]->enabled())
|
||||
result |= IOGA_TIMER1_START;
|
||||
else if (m_timer[1]->param())
|
||||
result |= IOGA_TIMER1_EXPIRED;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
READ32_MEMBER(interpro_ioga_device::timer3_r)
|
||||
{
|
||||
uint32_t result = m_timer3_count & IOGA_TIMER3_VMASK;
|
||||
|
||||
if (m_timer[3]->enabled())
|
||||
result |= IOGA_TIMER3_START;
|
||||
else if (m_timer[3]->param())
|
||||
result |= IOGA_TIMER3_EXPIRED;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
void interpro_ioga_device::write_timer(int timer, uint32_t value, device_timer_id id)
|
||||
{
|
||||
switch (id)
|
||||
{
|
||||
case IOGA_TIMER_1:
|
||||
// disable the timer
|
||||
m_timer[timer]->enable(false);
|
||||
|
||||
// store the timer count value
|
||||
m_timer1_count = value;
|
||||
|
||||
// start the timer if necessary
|
||||
if (value & IOGA_TIMER1_START)
|
||||
{
|
||||
LOG_TIMER(1, "timer 1: started prescaler %d value %d\n", m_prescaler & 0x7fff, value & IOGA_TIMER1_VMASK);
|
||||
|
||||
// FIXME: this division by 50 is sufficient to pass iogadiag timer 1 tests
|
||||
m_timer[timer]->adjust(attotime::zero, false, attotime::from_usec((m_prescaler & 0x7fff) / 50));
|
||||
}
|
||||
break;
|
||||
|
||||
case IOGA_TIMER_3:
|
||||
// stop the timer so it won't trigger while we're fiddling with it
|
||||
m_timer[timer]->enable(false);
|
||||
|
||||
// write the new value to the timer register
|
||||
m_timer3_count = value & IOGA_TIMER3_VMASK;
|
||||
|
||||
// start the timer if necessary
|
||||
if (value & IOGA_TIMER3_START)
|
||||
{
|
||||
LOG_TIMER(3, "timer 3: started value %d\n", value & IOGA_TIMER3_VMASK);
|
||||
|
||||
m_timer[timer]->adjust(attotime::zero, false, attotime::from_hz(XTAL_25MHz));
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
// save the value
|
||||
m_timer_reg[timer] = value;
|
||||
|
||||
// timer_set(attotime::from_usec(500), id);
|
||||
|
||||
LOG_TIMER(0xf, "timer %d: set to 0x%x (%d)\n", timer, m_timer_reg[timer], m_timer_reg[timer]);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void interpro_ioga_device::device_timer(emu_timer &timer, device_timer_id id, int param, void *ptr)
|
||||
{
|
||||
switch (id)
|
||||
{
|
||||
case IOGA_TIMER_0:
|
||||
m_timer_reg[0]++;
|
||||
set_irq_line(IOGA_TIMER0_IRQ, ASSERT_LINE);
|
||||
break;
|
||||
|
||||
case IOGA_TIMER_1:
|
||||
// decrement timer count value
|
||||
m_timer1_count--;
|
||||
|
||||
// check if timer has expired
|
||||
if (m_timer1_count == 0)
|
||||
{
|
||||
LOG_TIMER(1, "timer 1: stopped\n");
|
||||
|
||||
// disable timer and set the zero flag
|
||||
timer.enable(false);
|
||||
timer.set_param(true);
|
||||
|
||||
// throw an interrupt
|
||||
set_irq_line(IOGA_TIMER1_IRQ, ASSERT_LINE);
|
||||
}
|
||||
break;
|
||||
|
||||
case IOGA_TIMER_3:
|
||||
// decrement timer count value
|
||||
m_timer3_count--;
|
||||
|
||||
// check for expiry
|
||||
if (m_timer3_count == 0)
|
||||
{
|
||||
LOG_TIMER(3, "timer 3: stopped\n");
|
||||
|
||||
// disable timer and set the zero flag
|
||||
timer.enable(false);
|
||||
timer.set_param(true);
|
||||
|
||||
// throw an interrupt
|
||||
set_irq_line(IOGA_TIMER3_IRQ, ASSERT_LINE);
|
||||
}
|
||||
break;
|
||||
|
||||
case IOGA_TIMER_DMA:
|
||||
// transfer data between device and main memory
|
||||
|
||||
// TODO: figure out what indicates dma write (memory -> device)
|
||||
// TODO: implement multiple dma channels
|
||||
// TODO: virtual memory?
|
||||
|
||||
if (!m_dma_channel[param].dma_active)
|
||||
{
|
||||
LOG_DMA("dma: transfer started, channel = %d, control 0x%08x, real address 0x%08x count 0x%08x\n",
|
||||
param, m_dma_channel[param].control, m_dma_channel[param].real_address, m_dma_channel[param].transfer_count);
|
||||
m_dma_channel[param].dma_active = true;
|
||||
}
|
||||
|
||||
// while the device is requesting a data transfer and the transfer count is not zero
|
||||
while (m_dma_channel[param].drq_state && m_dma_channel[param].transfer_count)
|
||||
{
|
||||
// transfer a byte between device and memory
|
||||
if (true)
|
||||
m_memory_space->write_byte(m_dma_channel[param].real_address, m_dma_channel[param].device_r());
|
||||
else
|
||||
m_dma_channel[param].device_w(m_memory_space->read_byte(m_dma_channel[param].real_address));
|
||||
|
||||
// increment addresses and decrement count
|
||||
m_dma_channel[param].real_address++;
|
||||
m_dma_channel[param].virtual_address++;
|
||||
m_dma_channel[param].transfer_count--;
|
||||
}
|
||||
|
||||
// if there are no more bytes remaining, terminate the transfer
|
||||
if (m_dma_channel[param].transfer_count == 0)
|
||||
{
|
||||
LOG_DMA("dma: transfer stopped, control 0x%08x, real address 0x%08x count 0x%08x\n",
|
||||
m_dma_channel[param].control, m_dma_channel[param].fdc_real_address, m_dma_channel[param].transfer_count);
|
||||
|
||||
if (param == IOGA_DMA_FLOPPY)
|
||||
{
|
||||
LOG_DMA("dma: asserting fdc terminal count line\n");
|
||||
|
||||
m_fdc_tc_func(ASSERT_LINE);
|
||||
m_fdc_tc_func(CLEAR_LINE);
|
||||
}
|
||||
|
||||
m_dma_channel[param].dma_active = false;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/******************************************************************************
|
||||
Interrupts
|
||||
******************************************************************************/
|
||||
|
||||
static const uint16_t irq_enable_mask[IOGA_INTERRUPT_COUNT] =
|
||||
{
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL | IOGA_INTERRUPT_ENABLE_INTERNAL, // external interrupt 0: SCSI
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL | IOGA_INTERRUPT_ENABLE_INTERNAL, // external interrupt 1: floppy
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL | IOGA_INTERRUPT_ENABLE_INTERNAL, // external interrupt 2: plotter
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
|
||||
// internal interrupt 5: serial DMA - one interrupt enable per DMA channel
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL << 0 | IOGA_INTERRUPT_ENABLE_EXTERNAL << 1 | IOGA_INTERRUPT_ENABLE_EXTERNAL << 2,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL,
|
||||
IOGA_INTERRUPT_ENABLE_EXTERNAL | IOGA_INTERRUPT_ENABLE_INTERNAL // external interrupt 12: Ethernet
|
||||
};
|
||||
|
||||
void interpro_ioga_device::set_nmi_line(int state)
|
||||
{
|
||||
switch (state)
|
||||
{
|
||||
case ASSERT_LINE:
|
||||
|
||||
LOG_INTERRUPT("nmi: ctrl = 0x%02x\n", m_nmictrl);
|
||||
|
||||
if ((m_nmictrl & IOGA_NMI_ENABLE) == IOGA_NMI_ENABLE)
|
||||
{
|
||||
// if edge triggered mode, clear enable in
|
||||
if (m_nmictrl & IOGA_NMI_EDGE)
|
||||
m_nmictrl &= ~IOGA_NMI_ENABLE_IN;
|
||||
|
||||
m_nmi_pending = true;
|
||||
update_interrupt(ASSERT_LINE);
|
||||
}
|
||||
break;
|
||||
|
||||
case CLEAR_LINE:
|
||||
m_nmi_pending = false;
|
||||
update_interrupt(ASSERT_LINE);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void interpro_ioga_device::set_irq_line(int irq, int state)
|
||||
{
|
||||
LOG_INTERRUPT("set_irq_line(%d, %d)\n", irq, state);
|
||||
switch (state)
|
||||
{
|
||||
case ASSERT_LINE:
|
||||
if (m_int_vector[irq] & irq_enable_mask[irq])
|
||||
{
|
||||
// set pending bit
|
||||
m_int_vector[irq] |= IOGA_INTERRUPT_PENDING;
|
||||
|
||||
// update irq line state
|
||||
update_interrupt(state);
|
||||
}
|
||||
else
|
||||
LOG_INTERRUPT("received disabled interrupt irq %d vector 0x%04x\n", irq, m_int_vector[irq]);
|
||||
break;
|
||||
|
||||
case CLEAR_LINE:
|
||||
// clear pending bit
|
||||
m_int_vector[irq] &= ~IOGA_INTERRUPT_PENDING;
|
||||
|
||||
// update irq line state
|
||||
update_interrupt(state);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void interpro_ioga_device::set_irq_soft(int irq, int state)
|
||||
{
|
||||
LOG_INTERRUPT("set_irq_soft(%d, %d)\n", irq, state);
|
||||
switch (state)
|
||||
{
|
||||
case ASSERT_LINE:
|
||||
// set pending bit
|
||||
if (irq < 8)
|
||||
m_softint |= 1 << irq;
|
||||
else
|
||||
m_softint_vector[irq - 8] |= IOGA_INTERRUPT_PENDING;
|
||||
|
||||
update_interrupt(state);
|
||||
break;
|
||||
|
||||
case CLEAR_LINE:
|
||||
// clear pending bit
|
||||
if (irq < 8)
|
||||
m_softint &= ~(1 << irq);
|
||||
else
|
||||
m_softint_vector[irq - 8] &= ~IOGA_INTERRUPT_PENDING;
|
||||
|
||||
// update irq line state
|
||||
update_interrupt(state);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
IRQ_CALLBACK_MEMBER(interpro_ioga_device::inta_cb)
|
||||
{
|
||||
switch (irqline)
|
||||
{
|
||||
case INPUT_LINE_IRQ0:
|
||||
// FIXME: clear pending bit - can't rely on device callbacks
|
||||
switch (m_interrupt_active)
|
||||
{
|
||||
case IOGA_INTERRUPT_INTERNAL:
|
||||
case IOGA_INTERRUPT_EXTERNAL:
|
||||
m_int_vector[m_irq_current] &= ~IOGA_INTERRUPT_PENDING;
|
||||
break;
|
||||
|
||||
case IOGA_INTERRUPT_SOFT_LO:
|
||||
m_softint &= ~(1 << m_irq_current);
|
||||
break;
|
||||
|
||||
case IOGA_INTERRUPT_SOFT_HI:
|
||||
m_softint_vector[m_irq_current] &= ~IOGA_INTERRUPT_PENDING;
|
||||
break;
|
||||
}
|
||||
|
||||
// clear irq line
|
||||
update_interrupt(CLEAR_LINE);
|
||||
|
||||
// fall through to return interrupt vector
|
||||
case -1:
|
||||
// return vector for current interrupt without clearing irq line
|
||||
switch (m_interrupt_active)
|
||||
{
|
||||
case IOGA_INTERRUPT_EXTERNAL:
|
||||
case IOGA_INTERRUPT_INTERNAL:
|
||||
return m_int_vector[m_irq_current] & 0xff;
|
||||
|
||||
case IOGA_INTERRUPT_SOFT_LO:
|
||||
return 0x8f + m_irq_current * 0x10;
|
||||
|
||||
case IOGA_INTERRUPT_SOFT_HI:
|
||||
return m_softint_vector[m_irq_current] & 0xff;
|
||||
}
|
||||
break;
|
||||
|
||||
case INPUT_LINE_NMI:
|
||||
// clear pending flag
|
||||
m_nmi_pending = false;
|
||||
|
||||
// clear line
|
||||
update_interrupt(CLEAR_LINE);
|
||||
|
||||
// return vector
|
||||
return 0;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void interpro_ioga_device::update_interrupt(int state)
|
||||
{
|
||||
switch (state)
|
||||
{
|
||||
case CLEAR_LINE:
|
||||
if (m_interrupt_active)
|
||||
{
|
||||
// the cpu has acknowledged the active interrupt, deassert the nmi/irq line
|
||||
m_interrupt_active == IOGA_INTERRUPT_NMI ? m_out_nmi_func(CLEAR_LINE) : m_out_int_func(CLEAR_LINE);
|
||||
|
||||
// clear the active status
|
||||
m_interrupt_active = 0;
|
||||
}
|
||||
// fall through to handle any pending interrupts
|
||||
|
||||
case ASSERT_LINE:
|
||||
// if an interrupt is currently active, don't do anything
|
||||
if (m_interrupt_active == 0)
|
||||
{
|
||||
// check for pending nmi
|
||||
if (m_nmi_pending)
|
||||
{
|
||||
m_interrupt_active = IOGA_INTERRUPT_NMI;
|
||||
|
||||
m_out_nmi_func(ASSERT_LINE);
|
||||
return;
|
||||
}
|
||||
|
||||
// check for any pending irq
|
||||
for (int i = 0; i < IOGA_INTERRUPT_COUNT; i++)
|
||||
{
|
||||
if (m_int_vector[i] & IOGA_INTERRUPT_PENDING)
|
||||
{
|
||||
m_interrupt_active = IOGA_INTERRUPT_INTERNAL; // TODO: flag internal/external
|
||||
m_irq_current = i;
|
||||
|
||||
m_out_int_func(ASSERT_LINE);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// check for any pending soft interrupts (low type)
|
||||
for (int i = 0; i < 8; i++)
|
||||
{
|
||||
if (m_softint & (1 << i))
|
||||
{
|
||||
m_interrupt_active = IOGA_INTERRUPT_SOFT_LO;
|
||||
m_irq_current = i;
|
||||
|
||||
m_out_int_func(ASSERT_LINE);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// check for any pending soft interrupts (high type)
|
||||
for (int i = 0; i < 8; i++)
|
||||
{
|
||||
if (m_softint_vector[i] & IOGA_INTERRUPT_PENDING)
|
||||
{
|
||||
m_interrupt_active = IOGA_INTERRUPT_SOFT_HI;
|
||||
m_irq_current = i;
|
||||
|
||||
m_out_int_func(ASSERT_LINE);
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
WRITE16_MEMBER(interpro_ioga_device::icr_w)
|
||||
{
|
||||
LOG_INTERRUPT("interrupt vector %d set to 0x%04x at pc 0x%08x\n", offset, data, space.device().safe_pc());
|
||||
|
||||
// FIXME: now that the interrupt handling only depends on IOGA_INTERRUPT_PENDING, we might be able
|
||||
// to avoid this hack
|
||||
if (data & IOGA_INTERRUPT_PENDING)
|
||||
{
|
||||
m_irq_forced |= 1 << offset;
|
||||
m_int_vector[offset] = data & ~IOGA_INTERRUPT_PENDING;
|
||||
}
|
||||
else if (m_irq_forced & 1 << offset)
|
||||
{
|
||||
m_int_vector[offset] = data;
|
||||
|
||||
// clear forced flag
|
||||
m_irq_forced &= ~(1 << offset);
|
||||
|
||||
// force an interrupt
|
||||
set_irq_line(offset, ASSERT_LINE);
|
||||
}
|
||||
else
|
||||
m_int_vector[offset] = data;
|
||||
}
|
||||
|
||||
WRITE8_MEMBER(interpro_ioga_device::softint_w)
|
||||
{
|
||||
// save the existing value
|
||||
uint8_t previous = m_softint;
|
||||
|
||||
// store the written value
|
||||
m_softint = data;
|
||||
|
||||
// force soft interrupt for any bit written from 1 to 0
|
||||
for (int i = 0; i < 8; i++)
|
||||
{
|
||||
uint8_t mask = 1 << i;
|
||||
|
||||
// check for transition from 1 to 0 and force a soft interrupt
|
||||
if (previous & mask && !(data & mask))
|
||||
set_irq_soft(i, ASSERT_LINE);
|
||||
}
|
||||
}
|
||||
|
||||
WRITE8_MEMBER(interpro_ioga_device::nmictrl_w)
|
||||
{
|
||||
// save the existing value
|
||||
uint8_t previous = m_nmictrl;
|
||||
|
||||
// store the written value
|
||||
m_nmictrl = data;
|
||||
|
||||
// force an nmi when pending bit is written low
|
||||
if (previous & IOGA_NMI_PENDING && !(data & IOGA_NMI_PENDING))
|
||||
set_nmi_line(ASSERT_LINE);
|
||||
}
|
||||
|
||||
WRITE16_MEMBER(interpro_ioga_device::softint_vector_w)
|
||||
{
|
||||
// save the existing value
|
||||
uint16_t previous = m_softint_vector[offset];
|
||||
|
||||
// store the written value
|
||||
m_softint_vector[offset] = data;
|
||||
|
||||
// check for transition from 1 to 0 and force a soft interrupt
|
||||
if (previous & IOGA_INTERRUPT_PENDING && !(data & IOGA_INTERRUPT_PENDING))
|
||||
set_irq_soft(offset + 8, ASSERT_LINE);
|
||||
}
|
||||
|
||||
/******************************************************************************
|
||||
DMA
|
||||
******************************************************************************/
|
||||
void interpro_ioga_device::drq(int state, int channel)
|
||||
{
|
||||
// this member is called when the device has data ready for reading via dma
|
||||
m_dma_channel[channel].drq_state = state;
|
||||
|
||||
if (state)
|
||||
{
|
||||
// TODO: check if dma is enabled
|
||||
m_dma_timer->adjust(attotime::zero, channel);
|
||||
}
|
||||
}
|
||||
/*
|
||||
0x94: error address reg: expect 0x7f200000 after bus error (from dma virtual address)
|
||||
0x98: error cycle type: expect 0x52f0 (after failed dma?)
|
||||
0x5331 - forced berr with nmi/interrupts disabled?
|
||||
0xc2f0
|
||||
0x62f0
|
||||
*/
|
||||
// TODO: 7.0266 - forced BERR not working
|
||||
|
||||
uint32_t interpro_ioga_device::dma_r(address_space &space, offs_t offset, uint32_t mem_mask, int channel)
|
||||
{
|
||||
switch (offset)
|
||||
{
|
||||
case 0:
|
||||
return m_dma_channel[channel].real_address;
|
||||
|
||||
case 1:
|
||||
return m_dma_channel[channel].virtual_address;
|
||||
|
||||
case 2:
|
||||
return m_dma_channel[channel].transfer_count;
|
||||
|
||||
case 3:
|
||||
return m_dma_channel[channel].control;
|
||||
}
|
||||
|
||||
logerror("dma_r: unknown channel %d\n", channel);
|
||||
return 0;
|
||||
}
|
||||
|
||||
void interpro_ioga_device::dma_w(address_space &space, offs_t offset, uint32_t data, uint32_t mem_mask, int channel)
|
||||
{
|
||||
switch (offset)
|
||||
{
|
||||
case 0:
|
||||
m_dma_channel[channel].real_address = data;
|
||||
break;
|
||||
|
||||
case 1:
|
||||
m_dma_channel[channel].virtual_address = data & ~0x3;
|
||||
break;
|
||||
|
||||
case 2:
|
||||
m_dma_channel[channel].transfer_count = data;
|
||||
break;
|
||||
|
||||
case 3:
|
||||
m_dma_channel[channel].control = data & IOGA_DMA_CTRL_WMASK;
|
||||
|
||||
logerror("dma: channel = %d, control = 0x%08x, ra = 0x%08x, va = 0x%08x, tc = 0x%08x\n",
|
||||
channel, data, m_dma_channel[channel].real_address, m_dma_channel[channel].virtual_address, m_dma_channel[channel].transfer_count);
|
||||
|
||||
// iogadiag test 7.0265
|
||||
if (data == IOGA_DMA_CTRL_START)
|
||||
{
|
||||
uint32_t mask = 0;
|
||||
|
||||
switch (channel)
|
||||
{
|
||||
case IOGA_DMA_PLOTTER:
|
||||
mask = IOGA_ARBCTL_BGR_PLOT;
|
||||
break;
|
||||
|
||||
case IOGA_DMA_SCSI:
|
||||
mask = IOGA_ARBCTL_BGR_SCSI;
|
||||
break;
|
||||
|
||||
case IOGA_DMA_FLOPPY:
|
||||
mask = IOGA_ARBCTL_BGR_FDC;
|
||||
break;
|
||||
}
|
||||
|
||||
// if bus grant is not enabled, set the busy flag
|
||||
if (!(m_arbctl & mask))
|
||||
m_dma_channel[channel].control |= IOGA_DMA_CTRL_BUSY;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
256
src/mame/machine/interpro_ioga.h
Normal file
256
src/mame/machine/interpro_ioga.h
Normal file
@ -0,0 +1,256 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
#pragma once
|
||||
|
||||
#ifndef INTERPRO_IOGA_H_
|
||||
#define INTERPRO_IOGA_H_
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
#define MCFG_INTERPRO_IOGA_ADD(_tag) \
|
||||
MCFG_DEVICE_ADD(_tag, INTERPRO_IOGA, 0)
|
||||
|
||||
#define MCFG_INTERPRO_IOGA_NMI_CB(_out_nmi) \
|
||||
devcb = &interpro_ioga_device::static_set_out_nmi_callback(*device, DEVCB_##_out_nmi);
|
||||
|
||||
#define MCFG_INTERPRO_IOGA_IRQ_CB(_out_int) \
|
||||
devcb = &interpro_ioga_device::static_set_out_int_callback(*device, DEVCB_##_out_int);
|
||||
|
||||
#define MCFG_INTERPRO_IOGA_DMA_CB(_channel, _dma_r, _dma_w) \
|
||||
devcb = &interpro_ioga_device::static_set_dma_r_callback(*device, _channel, DEVCB_##_dma_r); \
|
||||
devcb = &interpro_ioga_device::static_set_dma_w_callback(*device, _channel, DEVCB_##_dma_w);
|
||||
|
||||
#define MCFG_INTERPRO_IOGA_FDCTC_CB(_tc) \
|
||||
devcb = &interpro_ioga_device::static_set_fdc_tc_callback(*device, DEVCB_##_tc);
|
||||
|
||||
#define MCFG_INTERPRO_IOGA_DMA_BUS(_mmu, _space)
|
||||
|
||||
// timer 0 seem to be a 60Hz cycle
|
||||
#define IOGA_TIMER0_IRQ 14
|
||||
|
||||
// best guess for timer 1 is 10MHz based on typical prescaler value of 1000 and timer value of 100 for a delay of 100ms
|
||||
#define IOGA_TIMER1_IRQ 15
|
||||
#define IOGA_TIMER1_VMASK 0xffff
|
||||
#define IOGA_TIMER1_START 0x10000
|
||||
#define IOGA_TIMER1_EXPIRED 0x20000
|
||||
|
||||
// best guess for timer 3 is 12.5MHz based on typical value of 12500 for a delay of 1ms
|
||||
#define IOGA_TIMER3_CLOCK XTAL_12_5MHz
|
||||
#define IOGA_TIMER3_IRQ 1
|
||||
#define IOGA_TIMER3_VMASK 0x3fffffff
|
||||
#define IOGA_TIMER3_START 0x40000000
|
||||
#define IOGA_TIMER3_EXPIRED 0x80000000
|
||||
|
||||
#define IOGA_INTERRUPT_COUNT 19
|
||||
#define IOGA_INTERRUPT_PENDING 0x0100
|
||||
#define IOGA_INTERRUPT_ENABLE_EXTERNAL 0x0200
|
||||
#define IOGA_INTERRUPT_EDGE 0x0400
|
||||
#define IOGA_INTERRUPT_NEGPOL 0x0800
|
||||
#define IOGA_INTERRUPT_ENABLE_INTERNAL 0x1000
|
||||
|
||||
#define IOGA_NMI_EDGE 0x02
|
||||
#define IOGA_NMI_PENDING 0x08
|
||||
#define IOGA_NMI_ENABLE_IN 0x10
|
||||
#define IOGA_NMI_ENABLE (IOGA_NMI_EDGE | IOGA_NMI_ENABLE_IN)
|
||||
|
||||
#define IOGA_INTERRUPT_NMI 1
|
||||
#define IOGA_INTERRUPT_INTERNAL 2
|
||||
#define IOGA_INTERRUPT_EXTERNAL 3
|
||||
#define IOGA_INTERRUPT_SOFT_LO 4
|
||||
#define IOGA_INTERRUPT_SOFT_HI 5
|
||||
|
||||
#define IOGA_DMA_CHANNELS 4
|
||||
#define IOGA_DMA_PLOTTER 0
|
||||
#define IOGA_DMA_SCSI 1
|
||||
#define IOGA_DMA_FLOPPY 2
|
||||
#define IOGA_DMA_SERIAL 3
|
||||
|
||||
// dma write values
|
||||
#define IOGA_DMA_CTRL_WMASK 0xfd000e00
|
||||
#define IOGA_DMA_CTRL_RESET_L 0x61000000 // do not clear bus error bit
|
||||
#define IOGA_DMA_CTRL_RESET 0x60400000 // clear bus error bit
|
||||
|
||||
#define IOGA_DMA_CTRL_START 0x63000800 // perhaps start a transfer? - maybe the 8 is the channel?
|
||||
#define IOGA_DMA_CTRL_UNK1 0x60000000 // don't know yet
|
||||
#define IOGA_DMA_CTRL_UNK2 0x67000600 // forced berr with nmi and interrupts disabled
|
||||
|
||||
// read values
|
||||
#define IOGA_DMA_CTRL_BUSY 0x02000000
|
||||
#define IOGA_DMA_CTRL_BERR 0x00400000 // iogadiag code expects 0x60400000 on bus error
|
||||
// iogadiag expects 0x64400800 after forced berr with nmi/interrupts disabled
|
||||
|
||||
|
||||
// bus arbitration bus grant bits
|
||||
#define IOGA_ARBCTL_BGR_ETHC 0x0001
|
||||
#define IOGA_ARBCTL_BGR_SCSI 0x0002
|
||||
#define IOGA_ARBCTL_BGR_PLOT 0x0004
|
||||
#define IOGA_ARBCTL_BGR_FDC 0x0008
|
||||
#define IOGA_ARBCTL_BGR_SER0 0x0010
|
||||
#define IOGA_ARBCTL_BGR_SER1 0x0020
|
||||
#define IOGA_ARBCTL_BGR_SER2 0x0040
|
||||
#define IOGA_ARBCTL_BGR_ETHB 0x0080
|
||||
#define IOGA_ARBCTL_BGR_ETHA 0x0100
|
||||
|
||||
class interpro_ioga_device : public device_t
|
||||
{
|
||||
public:
|
||||
interpro_ioga_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock);
|
||||
|
||||
template<class _Object> static devcb_base &static_set_out_nmi_callback(device_t &device, _Object object) { return downcast<interpro_ioga_device &>(device).m_out_nmi_func.set_callback(object); }
|
||||
template<class _Object> static devcb_base &static_set_out_int_callback(device_t &device, _Object object) { return downcast<interpro_ioga_device &>(device).m_out_int_func.set_callback(object); }
|
||||
|
||||
template<class _Object> static devcb_base &static_set_dma_r_callback(device_t &device, int channel, _Object object) { return downcast<interpro_ioga_device &>(device).m_dma_channel[channel].device_r.set_callback(object); }
|
||||
template<class _Object> static devcb_base &static_set_dma_w_callback(device_t &device, int channel, _Object object) { return downcast<interpro_ioga_device &>(device).m_dma_channel[channel].device_w.set_callback(object); }
|
||||
|
||||
template<class _Object> static devcb_base &static_set_fdc_tc_callback(device_t &device, _Object object) { return downcast<interpro_ioga_device &>(device).m_fdc_tc_func.set_callback(object); }
|
||||
|
||||
virtual DECLARE_ADDRESS_MAP(map, 32);
|
||||
|
||||
// external interrupt lines
|
||||
DECLARE_WRITE_LINE_MEMBER(ir0_w) { set_irq_line(2, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir1_w) { set_irq_line(3, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir2_w) { set_irq_line(4, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir3_w) { set_irq_line(5, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir4_w) { set_irq_line(6, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir5_w) { set_irq_line(7, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir6_w) { set_irq_line(8, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir7_w) { set_irq_line(9, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir8_w) { set_irq_line(10, state); }
|
||||
// FIXME: this is a workaround for the mc146818 code which inverts the normal irq state convention
|
||||
DECLARE_WRITE_LINE_MEMBER(ir9_w) { set_irq_line(11, !state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir10_w) { set_irq_line(12, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir11_w) { set_irq_line(17, state); }
|
||||
DECLARE_WRITE_LINE_MEMBER(ir12_w) { set_irq_line(18, state); }
|
||||
|
||||
IRQ_CALLBACK_MEMBER(inta_cb);
|
||||
|
||||
DECLARE_WRITE_LINE_MEMBER(drq_plotter) { drq(state, IOGA_DMA_PLOTTER); }
|
||||
DECLARE_WRITE_LINE_MEMBER(drq_scsi) { drq(state, IOGA_DMA_SCSI); }
|
||||
DECLARE_WRITE_LINE_MEMBER(drq_floppy) { drq(state, IOGA_DMA_FLOPPY); }
|
||||
|
||||
DECLARE_READ32_MEMBER(timer_prescaler_r) { return m_prescaler; }
|
||||
DECLARE_READ32_MEMBER(timer0_r) { return m_timer_reg[0]; }
|
||||
DECLARE_READ32_MEMBER(timer1_r);
|
||||
DECLARE_READ16_MEMBER(arbctl_r) { return m_arbctl; }
|
||||
DECLARE_WRITE16_MEMBER(arbctl_w) { m_arbctl = data; }
|
||||
DECLARE_READ32_MEMBER(timer2_r) { return m_timer_reg[2]; }
|
||||
DECLARE_READ32_MEMBER(timer3_r);
|
||||
|
||||
DECLARE_WRITE32_MEMBER(timer_prescaler_w) {
|
||||
// this logic satisfies prescaler tests, but fails timer prescaler tests
|
||||
if ((data & 0x7fff) < 0x100 && (data & 0x7fff) != 0)
|
||||
m_prescaler = (data ^ 0xffff0000);
|
||||
else
|
||||
m_prescaler = (data ^ 0xffff0000) - 0x10000;
|
||||
|
||||
//logerror("prescaler: input 0x%08x output 0x%08x\n", data, m_prescaler);
|
||||
}
|
||||
DECLARE_WRITE32_MEMBER(timer0_w) { write_timer(0, data, IOGA_TIMER_0); }
|
||||
DECLARE_WRITE32_MEMBER(timer1_w) { write_timer(1, data, IOGA_TIMER_1); }
|
||||
DECLARE_WRITE32_MEMBER(timer2_w) { write_timer(2, data, IOGA_TIMER_2); }
|
||||
DECLARE_WRITE32_MEMBER(timer3_w) { write_timer(3, data, IOGA_TIMER_3); }
|
||||
|
||||
DECLARE_READ16_MEMBER(icr_r) { return m_int_vector[offset]; }
|
||||
DECLARE_WRITE16_MEMBER(icr_w);
|
||||
DECLARE_READ16_MEMBER(icr18_r) { return icr_r(space, 18, mem_mask); }
|
||||
DECLARE_WRITE16_MEMBER(icr18_w) { icr_w(space, 18, data, mem_mask); }
|
||||
|
||||
DECLARE_READ8_MEMBER(softint_r) { return m_softint; }
|
||||
DECLARE_WRITE8_MEMBER(softint_w);
|
||||
DECLARE_READ8_MEMBER(nmictrl_r) { return m_nmictrl; }
|
||||
DECLARE_WRITE8_MEMBER(nmictrl_w);
|
||||
|
||||
DECLARE_READ16_MEMBER(softint_vector_r) { return m_softint_vector[offset]; }
|
||||
DECLARE_WRITE16_MEMBER(softint_vector_w);
|
||||
|
||||
DECLARE_READ32_MEMBER(dma_plotter_r) { return dma_r(space, offset, mem_mask, IOGA_DMA_PLOTTER); }
|
||||
DECLARE_WRITE32_MEMBER(dma_plotter_w) { dma_w(space, offset, data, mem_mask, IOGA_DMA_PLOTTER); }
|
||||
DECLARE_READ32_MEMBER(dma_scsi_r) { return dma_r(space, offset, mem_mask, IOGA_DMA_SCSI); }
|
||||
DECLARE_WRITE32_MEMBER(dma_scsi_w) { dma_w(space, offset, data, mem_mask, IOGA_DMA_SCSI); }
|
||||
DECLARE_READ32_MEMBER(dma_floppy_r) { return dma_r(space, offset, mem_mask, IOGA_DMA_FLOPPY); }
|
||||
DECLARE_WRITE32_MEMBER(dma_floppy_w) { dma_w(space, offset, data, mem_mask, IOGA_DMA_FLOPPY); }
|
||||
|
||||
DECLARE_READ32_MEMBER(dma_plotter_eosl_r) { return m_dma_plotter_eosl; }
|
||||
DECLARE_WRITE32_MEMBER(dma_plotter_eosl_w) { m_dma_plotter_eosl = data; }
|
||||
|
||||
DECLARE_READ32_MEMBER(error_address_r) { return m_error_address; }
|
||||
DECLARE_READ32_MEMBER(error_businfo_r) { return m_error_businfo; }
|
||||
void bus_error(uint32_t address, uint32_t cycle_type) { m_error_address = address; m_error_businfo = cycle_type; }
|
||||
|
||||
protected:
|
||||
// device-level overrides
|
||||
virtual void device_start() override;
|
||||
virtual void device_reset() override;
|
||||
virtual void device_timer(emu_timer &timer, device_timer_id id, int param, void *ptr) override;
|
||||
|
||||
private:
|
||||
static const device_timer_id IOGA_TIMER_0 = 0;
|
||||
static const device_timer_id IOGA_TIMER_1 = 1;
|
||||
static const device_timer_id IOGA_TIMER_2 = 2;
|
||||
static const device_timer_id IOGA_TIMER_3 = 3;
|
||||
|
||||
static const device_timer_id IOGA_TIMER_DMA = 4;
|
||||
|
||||
void set_nmi_line(int state);
|
||||
void set_irq_line(int irq, int state);
|
||||
void set_irq_soft(int irq, int state);
|
||||
void write_timer(int timer, uint32_t value, device_timer_id id);
|
||||
|
||||
void update_interrupt(int state);
|
||||
|
||||
void drq(int state, int channel);
|
||||
|
||||
devcb_write_line m_out_nmi_func;
|
||||
devcb_write_line m_out_int_func;
|
||||
|
||||
devcb_write_line m_fdc_tc_func;
|
||||
|
||||
bool m_nmi_pending;
|
||||
uint32_t m_interrupt_active;
|
||||
uint32_t m_irq_current;
|
||||
uint32_t m_irq_forced;
|
||||
|
||||
uint16_t m_int_vector[IOGA_INTERRUPT_COUNT];
|
||||
uint8_t m_softint;
|
||||
uint8_t m_nmictrl;
|
||||
uint16_t m_softint_vector[8];
|
||||
|
||||
uint32_t m_prescaler;
|
||||
uint32_t m_timer_reg[3];
|
||||
uint16_t m_timer1_count;
|
||||
uint32_t m_timer3_count;
|
||||
emu_timer *m_timer[4];
|
||||
|
||||
// dma state
|
||||
address_space *m_memory_space;
|
||||
emu_timer *m_dma_timer;
|
||||
|
||||
// dma channels
|
||||
struct dma
|
||||
{
|
||||
uint32_t real_address;
|
||||
uint32_t virtual_address;
|
||||
uint32_t transfer_count;
|
||||
uint32_t control;
|
||||
|
||||
bool dma_active;
|
||||
int drq_state;
|
||||
devcb_read8 device_r;
|
||||
devcb_write8 device_w;
|
||||
} m_dma_channel[IOGA_DMA_CHANNELS];
|
||||
uint32_t m_dma_plotter_eosl;
|
||||
|
||||
uint32_t dma_r(address_space &space, offs_t offset, uint32_t mem_mask, int channel);
|
||||
void dma_w(address_space &space, offs_t offset, uint32_t data, uint32_t mem_mask, int channel);
|
||||
|
||||
uint16_t m_arbctl;
|
||||
|
||||
uint32_t m_error_address;
|
||||
uint32_t m_error_businfo;
|
||||
};
|
||||
|
||||
// device type definition
|
||||
extern const device_type INTERPRO_IOGA;
|
||||
|
||||
#endif
|
89
src/mame/machine/interpro_mcga.cpp
Normal file
89
src/mame/machine/interpro_mcga.cpp
Normal file
@ -0,0 +1,89 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
/*
|
||||
* An implementation of the MCGA device found on Intergraph InterPro family workstations. There is no
|
||||
* public documentation on this device, so the implementation is being built to follow the logic of the
|
||||
* system boot ROM and its diagnostic tests.
|
||||
*
|
||||
* Please be aware that code in here is not only broken, it's likely wrong in many cases.
|
||||
*
|
||||
* TODO
|
||||
* - too long to list
|
||||
*/
|
||||
#include "interpro_mcga.h"
|
||||
|
||||
#define VERBOSE 0
|
||||
#if VERBOSE
|
||||
#define LOG_MCGA(...) logerror(__VA_ARGS__)
|
||||
#else
|
||||
#define LOG_MCGA(...) {}
|
||||
#endif
|
||||
|
||||
DEVICE_ADDRESS_MAP_START(map, 16, interpro_mcga_device)
|
||||
AM_RANGE(0x00, 0x3f) AM_READWRITE16(read, write, 0xffff)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
const device_type INTERPRO_MCGA = &device_creator<interpro_mcga_device>;
|
||||
|
||||
interpro_mcga_device::interpro_mcga_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
||||
: device_t(mconfig, INTERPRO_MCGA, "InterPro MCGA", tag, owner, clock, "mcga", __FILE__)
|
||||
{
|
||||
}
|
||||
|
||||
void interpro_mcga_device::device_start()
|
||||
{
|
||||
}
|
||||
|
||||
void interpro_mcga_device::device_reset()
|
||||
{
|
||||
m_reg[0] = 0x00ff; // 0x00
|
||||
m_reg[2] = MCGA_CTRL_ENREFRESH | MCGA_CTRL_CBITFRCSUB | MCGA_CTRL_CBITFRCRD; // 0x08 ctrl
|
||||
//m_mcga[4] = 0x8000; // 0x10 error
|
||||
m_reg[10] = 0x00ff; // 0x28
|
||||
m_reg[14] = 0x0340; // 0x38 memsize
|
||||
}
|
||||
|
||||
WRITE16_MEMBER(interpro_mcga_device::write)
|
||||
{
|
||||
/*
|
||||
read MEMSIZE 0x38 mask 0xffff
|
||||
read 0x00 mask 0x0000
|
||||
write CBSUB 0x20 mask 0x00ff data 0
|
||||
write FRCRD 0x18 mask 0x00ff data 0
|
||||
read ERROR 0x10 mask 0xffff
|
||||
read 0x00 mask 0xffff
|
||||
|
||||
(0x38 >> 8) & 0xF == 3?
|
||||
|
||||
if (0x00 != 0xFF) -> register reset error
|
||||
|
||||
0x00 = 0x0055 (test value & 0xff)
|
||||
r7 = 0x00 & 0xff
|
||||
*/
|
||||
LOG_MCGA("mcga write offset = 0x%08x, mask = 0x%08x, data = 0x%08x, pc = 0x%08x\n", offset, mem_mask, data, space.device().safe_pc());
|
||||
switch (offset)
|
||||
{
|
||||
case 0x02: // MCGA_CTRL
|
||||
// HACK: set or clear error status depending on ENMMBE bit
|
||||
if (data & MCGA_CTRL_ENMMBE)
|
||||
m_reg[4] |= MCGA_ERROR_VALID;
|
||||
// else
|
||||
// m_reg[4] &= ~MCGA_ERROR_VALID;
|
||||
|
||||
default:
|
||||
m_reg[offset] = data;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
READ16_MEMBER(interpro_mcga_device::read)
|
||||
{
|
||||
LOG_MCGA("mcga read offset = 0x%08x, mask = 0x%08x, pc = 0x%08x\n", offset, mem_mask, space.device().safe_pc());
|
||||
|
||||
switch (offset)
|
||||
{
|
||||
default:
|
||||
return m_reg[offset];
|
||||
}
|
||||
}
|
45
src/mame/machine/interpro_mcga.h
Normal file
45
src/mame/machine/interpro_mcga.h
Normal file
@ -0,0 +1,45 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
#pragma once
|
||||
|
||||
#ifndef INTERPRO_MCGA_H_
|
||||
#define INTERPRO_MCGA_H_
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
// mcga control register
|
||||
#define MCGA_CTRL_OPTMASK 0x00000003
|
||||
#define MCGA_CTRL_CBITFRCRD 0x00000004
|
||||
#define MCGA_CTRL_CBITFRCSUB 0x00000008
|
||||
#define MCGA_CTRL_ENREFRESH 0x00000010
|
||||
#define MCGA_CTRL_ENMSBE 0x00000100
|
||||
#define MCGA_CTRL_ENMMBE 0x00000200
|
||||
#define MCGA_CTRL_ENECC 0x00000400
|
||||
#define MCGA_CTRL_WRPROT 0x00008000
|
||||
|
||||
// mcga error register
|
||||
#define MCGA_ERROR_VALID 0x00008000
|
||||
|
||||
class interpro_mcga_device : public device_t
|
||||
{
|
||||
public:
|
||||
interpro_mcga_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock);
|
||||
|
||||
virtual DECLARE_ADDRESS_MAP(map, 32);
|
||||
|
||||
DECLARE_WRITE16_MEMBER(write);
|
||||
DECLARE_READ16_MEMBER(read);
|
||||
|
||||
protected:
|
||||
virtual void device_start() override;
|
||||
virtual void device_reset() override;
|
||||
|
||||
private:
|
||||
uint16_t m_reg[32];
|
||||
};
|
||||
|
||||
// device type definition
|
||||
extern const device_type INTERPRO_MCGA;
|
||||
|
||||
#endif
|
88
src/mame/machine/interpro_sga.cpp
Normal file
88
src/mame/machine/interpro_sga.cpp
Normal file
@ -0,0 +1,88 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
/*
|
||||
* An implementation of the SGA device found on Intergraph InterPro family workstations. There is no
|
||||
* public documentation on this device, so the implementation is being built to follow the logic of the
|
||||
* system boot ROM and its diagnostic tests.
|
||||
*
|
||||
* Please be aware that code in here is not only broken, it's likely wrong in many cases.
|
||||
*
|
||||
* TODO
|
||||
* - too long to list
|
||||
*/
|
||||
#include "interpro_sga.h"
|
||||
|
||||
#define VERBOSE 0
|
||||
|
||||
DEVICE_ADDRESS_MAP_START(map, 32, interpro_sga_device)
|
||||
AM_RANGE(0x00, 0x03) AM_READWRITE(gcs_r, gcs_w)
|
||||
AM_RANGE(0x04, 0x07) AM_READWRITE(ipoll_r, ipoll_w)
|
||||
AM_RANGE(0x08, 0x0b) AM_READWRITE(imask_r, imask_w)
|
||||
AM_RANGE(0x0c, 0x0f) AM_READWRITE(range_base_r, range_base_w)
|
||||
AM_RANGE(0x10, 0x13) AM_READWRITE(range_end_r, range_end_w)
|
||||
AM_RANGE(0x14, 0x17) AM_READWRITE(cttag_r, cttag_w)
|
||||
AM_RANGE(0x18, 0x1b) AM_READWRITE(address_r, address_w)
|
||||
AM_RANGE(0x1c, 0x1f) AM_READWRITE(dmacs_r, dmacs_w)
|
||||
AM_RANGE(0x20, 0x23) AM_READWRITE(edmacs_r, edmacs_w)
|
||||
|
||||
AM_RANGE(0xa4, 0xa7) AM_READWRITE(dspad1_r, dspad1_w)
|
||||
AM_RANGE(0xa8, 0xab) AM_READWRITE(dsoff1_r, dsoff1_w)
|
||||
|
||||
AM_RANGE(0xb4, 0xb7) AM_READWRITE(unknown1_r, unknown1_w)
|
||||
AM_RANGE(0xb8, 0xbb) AM_READWRITE(unknown2_r, unknown2_w)
|
||||
AM_RANGE(0xbc, 0xbf) AM_READWRITE(ddtc1_r, ddtc1_w)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
const device_type INTERPRO_SGA = &device_creator<interpro_sga_device>;
|
||||
|
||||
interpro_sga_device::interpro_sga_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
||||
: device_t(mconfig, INTERPRO_SGA, "InterPro SGA", tag, owner, clock, "sga", __FILE__)
|
||||
{
|
||||
}
|
||||
|
||||
void interpro_sga_device::device_start()
|
||||
{
|
||||
}
|
||||
|
||||
void interpro_sga_device::device_reset()
|
||||
{
|
||||
}
|
||||
|
||||
WRITE32_MEMBER(interpro_sga_device::ddtc1_w)
|
||||
{
|
||||
// we assume that when this register is written, we should start a
|
||||
// memory to memory dma transfer
|
||||
|
||||
logerror(" gcs = 0x%08x dmacs = 0x%08x\n", m_gcs, m_dmacs);
|
||||
logerror(" ipoll = 0x%08x imask = 0x%08x\n", m_ipoll, m_imask);
|
||||
logerror("dspad1 = 0x%08x dsoff1 = 0x%08x\n", m_dspad1, m_dsoff1);
|
||||
logerror(" unk1 = 0x%08x unk2 = 0x%08x\n", m_unknown1, m_unknown2);
|
||||
logerror(" ddtc1 = 0x%08x\n", data);
|
||||
|
||||
m_ddtc1 = data;
|
||||
|
||||
// when complete, we indicate by setting DMAEND(2) - 2 is probably the channel
|
||||
// we also turn off the INTBERR and INTMMBE flags
|
||||
m_ipoll &= ~(0x20000 | 0x10000);
|
||||
m_ipoll |= 0x200;
|
||||
|
||||
// if the address is invalid, fake a bus error
|
||||
if (m_dspad1 == 0x40000000 || m_unknown1 == 0x40000000
|
||||
|| m_dspad1 == 0x40000200 || m_unknown1 == 0x40000200)
|
||||
{
|
||||
m_ipoll |= 0x10000;
|
||||
|
||||
// error cycle - bit 0x10 indicates source address error (dspad1)
|
||||
// now expecting 0x5463?
|
||||
#if 0
|
||||
if ((m_dspad1 & 0xfffff000) == 0x40000000)
|
||||
m_ioga->bus_error(m_dspad1, 0x5433);
|
||||
else
|
||||
m_ioga->bus_error(m_unknown1, 0x5423);
|
||||
#endif
|
||||
// 0x5423 = BERR|SNAPOK | BG(ICAMMU)? | CT(23)
|
||||
// 0x5433 = BERR|SNAPOK | BG(ICAMMU)? | CT(33)
|
||||
// 0x5463 = BERR|SNAPOK | BG(ICAMMU)? | TAG(1) | CT(23)
|
||||
}
|
||||
}
|
71
src/mame/machine/interpro_sga.h
Normal file
71
src/mame/machine/interpro_sga.h
Normal file
@ -0,0 +1,71 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:Patrick Mackinlay
|
||||
|
||||
#pragma once
|
||||
|
||||
#ifndef INTERPRO_SGA_H_
|
||||
#define INTERPRO_SGA_H_
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
class interpro_sga_device : public device_t
|
||||
{
|
||||
public:
|
||||
interpro_sga_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock);
|
||||
|
||||
virtual DECLARE_ADDRESS_MAP(map, 32);
|
||||
|
||||
DECLARE_READ32_MEMBER(gcs_r) { return m_gcs; }
|
||||
DECLARE_WRITE32_MEMBER(gcs_w) { m_gcs = data; }
|
||||
DECLARE_READ32_MEMBER(ipoll_r) { return m_ipoll; }
|
||||
DECLARE_WRITE32_MEMBER(ipoll_w) { m_ipoll = data; }
|
||||
DECLARE_READ32_MEMBER(imask_r) { return m_imask; }
|
||||
DECLARE_WRITE32_MEMBER(imask_w) { m_imask = data; }
|
||||
DECLARE_READ32_MEMBER(range_base_r) { return m_range_base; }
|
||||
DECLARE_WRITE32_MEMBER(range_base_w) { m_range_base = data; }
|
||||
DECLARE_READ32_MEMBER(range_end_r) { return m_range_end; }
|
||||
DECLARE_WRITE32_MEMBER(range_end_w) { m_range_end = data; }
|
||||
DECLARE_READ32_MEMBER(cttag_r) { return m_cttag; }
|
||||
DECLARE_WRITE32_MEMBER(cttag_w) { m_cttag = data; }
|
||||
DECLARE_READ32_MEMBER(address_r) { return m_address; }
|
||||
DECLARE_WRITE32_MEMBER(address_w) { m_address = data; }
|
||||
DECLARE_READ32_MEMBER(dmacs_r) { return m_dmacs; }
|
||||
DECLARE_WRITE32_MEMBER(dmacs_w) { m_dmacs = data; }
|
||||
DECLARE_READ32_MEMBER(edmacs_r) { return m_edmacs; }
|
||||
DECLARE_WRITE32_MEMBER(edmacs_w) { m_edmacs = data; }
|
||||
DECLARE_READ32_MEMBER(dspad1_r) { return m_dspad1; }
|
||||
DECLARE_WRITE32_MEMBER(dspad1_w) { m_dspad1 = data; }
|
||||
DECLARE_READ32_MEMBER(dsoff1_r) { return m_dsoff1; }
|
||||
DECLARE_WRITE32_MEMBER(dsoff1_w) { m_dsoff1 = data; }
|
||||
DECLARE_READ32_MEMBER(unknown1_r) { return m_unknown1; }
|
||||
DECLARE_WRITE32_MEMBER(unknown1_w) { m_unknown1 = data; }
|
||||
DECLARE_READ32_MEMBER(unknown2_r) { return m_unknown2; }
|
||||
DECLARE_WRITE32_MEMBER(unknown2_w) { m_unknown2 = data; }
|
||||
DECLARE_READ32_MEMBER(ddtc1_r) { return m_ddtc1; }
|
||||
DECLARE_WRITE32_MEMBER(ddtc1_w);
|
||||
|
||||
protected:
|
||||
virtual void device_start() override;
|
||||
virtual void device_reset() override;
|
||||
|
||||
private:
|
||||
uint32_t m_gcs; // general control/status
|
||||
uint32_t m_ipoll; // interrupt poll
|
||||
uint32_t m_imask; // interrupt mask
|
||||
uint32_t m_range_base;
|
||||
uint32_t m_range_end;
|
||||
uint32_t m_cttag; // error cycletype/tag
|
||||
uint32_t m_address;
|
||||
uint32_t m_dmacs; // dma control/status
|
||||
uint32_t m_edmacs; // extended dma control/status
|
||||
uint32_t m_dspad1;
|
||||
uint32_t m_dsoff1;
|
||||
uint32_t m_unknown1;
|
||||
uint32_t m_unknown2;
|
||||
uint32_t m_ddtc1;
|
||||
};
|
||||
|
||||
// device type definition
|
||||
extern const device_type INTERPRO_SGA;
|
||||
|
||||
#endif
|
@ -14781,6 +14781,9 @@ instruct //
|
||||
hector1 //
|
||||
interact //
|
||||
|
||||
@source:interpro.cpp
|
||||
ip2800 // Intergraph InterPro 28xx workstation
|
||||
|
||||
@source:intrscti.cpp
|
||||
intrscti // ????
|
||||
|
||||
|
@ -79,6 +79,7 @@ CPU_DISASSEMBLE( avr8 );
|
||||
CPU_DISASSEMBLE( ccpu );
|
||||
CPU_DISASSEMBLE( cdp1801 );
|
||||
CPU_DISASSEMBLE( cdp1802 );
|
||||
CPU_DISASSEMBLE( clipper );
|
||||
CPU_DISASSEMBLE( coldfire );
|
||||
CPU_DISASSEMBLE( cop410 );
|
||||
CPU_DISASSEMBLE( cop420 );
|
||||
@ -248,6 +249,7 @@ static const dasm_table_entry dasm_table[] =
|
||||
{ "ccpu", _8bit, 0, CPU_DISASSEMBLE_NAME(ccpu) },
|
||||
{ "cdp1801", _8bit, 0, CPU_DISASSEMBLE_NAME(cdp1801) },
|
||||
{ "cdp1802", _8bit, 0, CPU_DISASSEMBLE_NAME(cdp1802) },
|
||||
{ "clipper", _16le, 0, CPU_DISASSEMBLE_NAME(clipper) },
|
||||
{ "coldfire", _16be, 0, CPU_DISASSEMBLE_NAME(coldfire) },
|
||||
{ "cop410", _8bit, 0, CPU_DISASSEMBLE_NAME(cop410) },
|
||||
{ "cop420", _8bit, 0, CPU_DISASSEMBLE_NAME(cop420) },
|
||||
|
Loading…
Reference in New Issue
Block a user