ReC98/th04/hardware/egcrect.cpp

127 lines
2.7 KiB
C++

#pragma option -zCSHARED
#include "platform.h"
#include "x86real.h"
#include "pc98.h"
#include "planar.h"
#include "master.hpp"
#include "platform/x86real/flags.hpp"
#include "platform/x86real/pc98/egc.hpp"
#include "th01/hardware/egc.h"
inline void graph_accesspage_1(void) {
_outportb_(0xA6, 1);
}
inline void graph_accesspage_0(void) {
_outportb_(0xA6, (_AX ^= _AX));
}
extern vram_word_amount_t egcrect_w;
static void near egc_start_copy(void);
void DEFCONV egc_copy_rect_1_to_0_16(
screen_x_t left, vram_y_t top, pixel_t w, pixel_t h
)
{
#define vo_tmp _BX // vram_offset_t
#define first_bit _CX
#define stride static_cast<vram_byte_amount_t>(_BP)
#define w_tmp static_cast<vram_word_amount_t>(_AX)
#define rows_remaining static_cast<pixel_t>(_BX)
#define dots static_cast<dots16_t>(_DX)
#if (GAME == 4)
// TH04 wants to blit using a forward STOSW (DF = 0)
_asm { cld; }
#endif
egc_start_copy();
egc_setrop(EGC_COMPAREREAD | EGC_WS_ROP | EGC_RL_MEMREAD | 0xF0);
// Using inline assembly rather than register pseudovariables to prevent
// parameters from being moved to the SI register
asm { mov ax, left; }
asm { mov dx, top; }
vo_tmp = _AX;
static_cast<vram_offset_t>(vo_tmp) >>= EGC_REGISTER_BITS;
asm { shl bx, 1; }
_DX <<= 6;
vo_tmp += _DX;
_DX >>= 2;
vo_tmp += _DX;
_DI = vo_tmp;
_AX &= EGC_REGISTER_MASK;
first_bit = _AX;
w_tmp = ((_AX + w) >> EGC_REGISTER_BITS);
if(first_bit) {
w_tmp++;
}
egcrect_w = w_tmp;
_CX = (ROW_SIZE / EGC_REGISTER_SIZE);
_CX -= w_tmp;
asm { shl cx, 1; }
rows_remaining = h;
stride = _CX;
_ES = SEG_PLANE_B;
do {
_CX = egcrect_w;
put_loop: {
#if (GAME == 5)
_DI |= _DI;
if((!FLAGS_SIGN) && (_DI < PLANE_SIZE)) {
graph_accesspage_1(); dots = peek(_ES, _DI);
graph_accesspage_0(); _poke_(_ES, _DI, dots);
}
_DI += 2;
#else
graph_accesspage_1(); dots = peek(_ES, _DI);
graph_accesspage_0(); _AX = dots; asm { stosw; }
#endif
asm { loop put_loop; }
}
_DI += stride;
rows_remaining--;
} while(!FLAGS_SIGN);
egc_off();
}
#pragma codestring "\x90"
#pragma option -k-
static void near egc_start_copy(void)
{
_asm {
push es
push 0
pop es
pushf
cli
// The EGC does in fact require an active GRCG in TDW mode.
// (See PC-9801 Programmers' Bible, p. 456)
mov al, GC_TDW
out 0x7C, al // grcg_setmode()
// According to MEMSYS.TXT, GRCG mode changes should always be
// reflected in this memory location, for compatibility with interrupt
// routines?
mov byte ptr es:[0x495], al
popf
pop es
}
graph_egc_on();
outport(EGC_ACTIVEPLANEREG, 0xFFF0);
egc_selectpat();
outport(EGC_MASKREG, 0xFFFF);
_DX = EGC_ADDRRESSREG;
outport(_DX, (_AX - _AX)); // :zunpet:
outport(EGC_BITLENGTHREG, 0xF);
}
#pragma codestring "\x90"