mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-12 00:30:31 +00:00
65816: improve decimal calculations, posted IO addresses, read/write during redundant read-modify-write cycle.
This commit is contained in:
parent
ec98736bd7
commit
7dcfa9eb65
@ -115,6 +115,13 @@ template <typename BusHandler, bool uses_ready_line> void Processor<BusHandler,
|
|||||||
read(data_address_, data_buffer_.next_input());
|
read(data_address_, data_buffer_.next_input());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case CycleStoreOrFetchDataThrowaway:
|
||||||
|
if(registers_.emulation_flag) {
|
||||||
|
perform_bus(data_address_, data_buffer_.preview_output(), MOS6502Esque::InternalOperationWrite);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[fallthrough]];
|
||||||
case CycleFetchDataThrowaway:
|
case CycleFetchDataThrowaway:
|
||||||
perform_bus(data_address_, &bus_throwaway_, MOS6502Esque::InternalOperationRead);
|
perform_bus(data_address_, &bus_throwaway_, MOS6502Esque::InternalOperationRead);
|
||||||
break;
|
break;
|
||||||
@ -141,10 +148,6 @@ template <typename BusHandler, bool uses_ready_line> void Processor<BusHandler,
|
|||||||
write(data_address_, data_buffer_.next_output());
|
write(data_address_, data_buffer_.next_output());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CycleStoreDataThrowaway:
|
|
||||||
perform_bus(data_address_, data_buffer_.preview_output(), MOS6502Esque::InternalOperationWrite);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case CycleStoreIncrementData:
|
case CycleStoreIncrementData:
|
||||||
write(data_address_, data_buffer_.next_output());
|
write(data_address_, data_buffer_.next_output());
|
||||||
increment_data_address();
|
increment_data_address();
|
||||||
@ -906,35 +909,43 @@ template <typename BusHandler, bool uses_ready_line> void Processor<BusHandler,
|
|||||||
|
|
||||||
case SBC:
|
case SBC:
|
||||||
if(registers_.flags.decimal) {
|
if(registers_.flags.decimal) {
|
||||||
// I've yet to manage to find a rational way to map this to an ADC,
|
|
||||||
// hence the yucky repetition of code here.
|
|
||||||
const uint16_t a = registers_.a.full & registers_.m_masks[1];
|
const uint16_t a = registers_.a.full & registers_.m_masks[1];
|
||||||
unsigned int result = 0;
|
const uint16_t decimal_result = uint16_t(a - data_buffer_.value - (1 ^ registers_.flags.carry));
|
||||||
unsigned int borrow = registers_.flags.carry ^ 1;
|
data_buffer_.value = ~data_buffer_.value & registers_.m_masks[1];
|
||||||
const uint16_t decimal_result = uint16_t(a - data_buffer_.value - borrow);
|
|
||||||
|
#define begin_nibble(mask) \
|
||||||
|
result += (a & mask) + (data_buffer_.value & mask);
|
||||||
|
|
||||||
|
#define end_nibble(adjustment, carry) \
|
||||||
|
if(result < carry) result -= adjustment; \
|
||||||
|
result &= (carry | (carry - 1));
|
||||||
|
|
||||||
#define nibble(mask, adjustment, carry) \
|
#define nibble(mask, adjustment, carry) \
|
||||||
result += (a & mask) - (data_buffer_.value & mask) - borrow; \
|
begin_nibble(mask); \
|
||||||
if(result > mask) result -= adjustment; \
|
end_nibble(adjustment, carry)
|
||||||
borrow = (result > mask) ? carry : 0; \
|
|
||||||
result &= (carry - 1);
|
|
||||||
|
|
||||||
|
unsigned int result = registers_.flags.carry;
|
||||||
nibble(0x000f, 0x0006, 0x00010);
|
nibble(0x000f, 0x0006, 0x00010);
|
||||||
nibble(0x00f0, 0x0060, 0x00100);
|
nibble(0x00f0, 0x0060, 0x00100);
|
||||||
nibble(0x0f00, 0x0600, 0x01000);
|
nibble(0x0f00, 0x0600, 0x01000);
|
||||||
nibble(0xf000, 0x6000, 0x10000);
|
|
||||||
|
begin_nibble(0xf000);
|
||||||
|
registers_.flags.overflow = uint8_t((( (decimal_result ^ result) & (~decimal_result ^ result) ) >> (1 + registers_.m_shift))&0x40);
|
||||||
|
end_nibble(0x6000, 0x10000);
|
||||||
|
|
||||||
#undef nibble
|
#undef nibble
|
||||||
|
#undef begin_nibble
|
||||||
|
#undef end_nibble
|
||||||
|
|
||||||
registers_.flags.overflow = (( (decimal_result ^ a) & (~decimal_result ^ data_buffer_.value) ) >> (1 + registers_.m_shift))&0x40;
|
|
||||||
registers_.flags.set_nz(uint16_t(result), registers_.m_shift);
|
registers_.flags.set_nz(uint16_t(result), registers_.m_shift);
|
||||||
registers_.flags.carry = ((borrow >> 16)&1)^1;
|
registers_.flags.carry = (result >> (8 + registers_.m_shift))&1;
|
||||||
LDA(result);
|
LDA(result);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
data_buffer_.value = ~data_buffer_.value & registers_.m_masks[1];
|
data_buffer_.value = ~data_buffer_.value & registers_.m_masks[1];
|
||||||
|
|
||||||
[[fallthrough]];
|
[[fallthrough]];
|
||||||
|
|
||||||
case ADC: {
|
case ADC: {
|
||||||
@ -942,22 +953,28 @@ template <typename BusHandler, bool uses_ready_line> void Processor<BusHandler,
|
|||||||
const uint16_t a = registers_.a.full & registers_.m_masks[1];
|
const uint16_t a = registers_.a.full & registers_.m_masks[1];
|
||||||
|
|
||||||
if(registers_.flags.decimal) {
|
if(registers_.flags.decimal) {
|
||||||
uint16_t partials = 0;
|
#define begin_nibble(mask) \
|
||||||
result = registers_.flags.carry;
|
result += (a & mask) + (data_buffer_.value & mask);
|
||||||
|
|
||||||
|
#define end_nibble(limit, adjustment, carry) \
|
||||||
|
if(result >= limit) result = ((result + adjustment) & (carry - 1)) + carry;
|
||||||
|
|
||||||
#define nibble(mask, limit, adjustment, carry) \
|
#define nibble(mask, limit, adjustment, carry) \
|
||||||
result += (a & mask) + (data_buffer_.value & mask); \
|
begin_nibble(mask); \
|
||||||
partials += result & mask; \
|
end_nibble(limit, adjustment, carry)
|
||||||
if(result >= limit) result = ((result + (adjustment)) & (carry - 1)) + carry;
|
|
||||||
|
|
||||||
|
result = registers_.flags.carry;
|
||||||
nibble(0x000f, 0x000a, 0x0006, 0x00010);
|
nibble(0x000f, 0x000a, 0x0006, 0x00010);
|
||||||
nibble(0x00f0, 0x00a0, 0x0060, 0x00100);
|
nibble(0x00f0, 0x00a0, 0x0060, 0x00100);
|
||||||
nibble(0x0f00, 0x0a00, 0x0600, 0x01000);
|
nibble(0x0f00, 0x0a00, 0x0600, 0x01000);
|
||||||
nibble(0xf000, 0xa000, 0x6000, 0x10000);
|
|
||||||
|
begin_nibble(0xf000);
|
||||||
|
registers_.flags.overflow = (( (uint16_t(result) ^ registers_.a.full) & (uint16_t(result) ^ data_buffer_.value) ) >> (1 + registers_.m_shift))&0x40;
|
||||||
|
end_nibble(0xa000, 0x6000, 0x10000);
|
||||||
|
|
||||||
#undef nibble
|
#undef nibble
|
||||||
|
#undef begin_nibble
|
||||||
registers_.flags.overflow = (( (partials ^ registers_.a.full) & (partials ^ data_buffer_.value) ) >> (1 + registers_.m_shift))&0x40;
|
#undef end_nibble
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
result = int(a + data_buffer_.value + registers_.flags.carry);
|
result = int(a + data_buffer_.value + registers_.flags.carry);
|
||||||
|
@ -195,8 +195,8 @@ struct CPU::WDC65816::ProcessorStorageConstructor {
|
|||||||
if(!is8bit) target(CycleFetchIncrementData); // Data low.
|
if(!is8bit) target(CycleFetchIncrementData); // Data low.
|
||||||
target(CycleFetchData); // Data [high].
|
target(CycleFetchData); // Data [high].
|
||||||
|
|
||||||
if(!is8bit) target(CycleFetchDataThrowaway); // 16-bit: reread final byte of data.
|
target(CycleStoreOrFetchDataThrowaway); // Native mode: reread final byte of data.
|
||||||
else target(CycleStoreDataThrowaway); // 8-bit rewrite final byte of data.
|
// Emulated mode: rewrite final byte of data.
|
||||||
|
|
||||||
target(OperationPerform); // Perform operation within the data buffer.
|
target(OperationPerform); // Perform operation within the data buffer.
|
||||||
|
|
||||||
|
@ -39,8 +39,9 @@ enum MicroOp: uint8_t {
|
|||||||
|
|
||||||
/// Stores a byte from the data buffer.
|
/// Stores a byte from the data buffer.
|
||||||
CycleStoreData,
|
CycleStoreData,
|
||||||
/// Stores the most recent byte placed into the data buffer without removing it.
|
/// Emulated mode: stores the most recent byte placed into the data buffer without removing it;
|
||||||
CycleStoreDataThrowaway,
|
/// Native mode: performs CycleFetchDataThrowaway.
|
||||||
|
CycleStoreOrFetchDataThrowaway,
|
||||||
/// Stores a byte to the data address from the data buffer and increments the data address.
|
/// Stores a byte to the data address from the data buffer and increments the data address.
|
||||||
CycleStoreIncrementData,
|
CycleStoreIncrementData,
|
||||||
/// Stores a byte to the data address from the data buffer and decrements the data address.
|
/// Stores a byte to the data address from the data buffer and decrements the data address.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user