redquark-amiberry-rb/src/drawing.cpp
2017-01-23 15:45:57 +01:00

2212 lines
62 KiB
C++

/*
* UAE - The Un*x Amiga Emulator
*
* Screen drawing functions
*
* Copyright 1995-2000 Bernd Schmidt
* Copyright 1995 Alessandro Bissacco
* Copyright 2000-2008 Toni Wilen
*/
/* There are a couple of concepts of "coordinates" in this file.
- DIW coordinates
- DDF coordinates (essentially cycles, resolution lower than lores by a factor of 2)
- Pixel coordinates
* in the Amiga's resolution as determined by BPLCON0 ("Amiga coordinates")
* in the window resolution as determined by the preferences ("window coordinates").
* in the window resolution, and with the origin being the topmost left corner of
the window ("native coordinates")
One note about window coordinates. The visible area depends on the width of the
window, and the centering code. The first visible horizontal window coordinate is
often _not_ 0, but the value of VISIBLE_LEFT_BORDER instead.
One important thing to remember: DIW coordinates are in the lowest possible
resolution.
To prevent extremely bad things (think pixels cut in half by window borders) from
happening, all ports should restrict window widths to be multiples of 16 pixels. */
#include "sysconfig.h"
#include "sysdeps.h"
#include <ctype.h>
#include <assert.h>
#include "options.h"
#include "threaddep/thread.h"
#include "uae.h"
#include "memory.h"
#include "custom.h"
#include "newcpu.h"
#include "xwin.h"
#include "autoconf.h"
#include "gui.h"
#include "picasso96.h"
#include "drawing.h"
#include "savestate.h"
#include "statusline.h"
#include <sys/time.h>
#include <time.h>
extern int sprite_buffer_res;
static void lores_reset(void)
{
sprite_buffer_res = (aga_mode) ? RES_SUPERHIRES : RES_LORES;
if (sprite_buffer_res > currprefs.gfx_resolution)
sprite_buffer_res = currprefs.gfx_resolution;
}
bool aga_mode; /* mirror of chipset_mask & CSMASK_AGA */
#ifdef PANDORA
#define OFFSET_Y_ADJUST 15
#else
#define OFFSET_Y_ADJUST 0
#endif
/* The shift factor to apply when converting between Amiga coordinates and window
coordinates. Zero if the resolution is the same, positive if window coordinates
have a higher resolution (i.e. we're stretching the image), negative if window
coordinates have a lower resolution (i.e. we're shrinking the image). */
static int res_shift;
/* Lookup tables for dual playfields. The dblpf_*1 versions are for the case
that playfield 1 has the priority, dbplpf_*2 are used if playfield 2 has
priority. If we need an array for non-dual playfield mode, it has no number. */
/* The dbplpf_ms? arrays contain a shift value. plf_spritemask is initialized
to contain two 16 bit words, with the appropriate mask if pf1 is in the
foreground being at bit offset 0, the one used if pf2 is in front being at
offset 16. */
static int dblpf_ms1[256], dblpf_ms2[256];
static int dblpf_ind1[256], dblpf_ind2[256];
static int dblpf_2nd1[256], dblpf_2nd2[256];
static const int dblpfofs[] = { 0, 2, 4, 8, 16, 32, 64, 128 };
static int sprite_col_nat[65536];
static int sprite_col_at[65536];
static int sprite_bit[65536];
static uae_u32 clxtab[256];
/* Video buffer description structure. Filled in by the graphics system
* dependent code. */
struct vidbuf_description gfxvidinfo;
/* OCS/ECS color lookup table. */
xcolnr xcolors[4096];
static uae_u8 spritepixels[MAX_PIXELS_PER_LINE * 5]; /* used when sprite resolution > lores */
static int sprite_first_x, sprite_last_x;
/* AGA mode color lookup tables */
unsigned int xredcolors[256], xgreencolors[256], xbluecolors[256];
static int dblpf_ind1_aga[256], dblpf_ind2_aga[256];
struct color_entry colors_for_drawing;
/* The size of these arrays is pretty arbitrary; it was chosen to be "more
than enough". The coordinates used for indexing into these arrays are
almost, but not quite, Amiga coordinates (there's a constant offset). */
static union pixdata_u {
uae_u8 apixels[MAX_PIXELS_PER_LINE * 2];
uae_u16 apixels_w[MAX_PIXELS_PER_LINE * 2 / sizeof(uae_u16)];
uae_u32 apixels_l[MAX_PIXELS_PER_LINE * 2 / sizeof(uae_u32)];
} pixdata;
uae_u16 spixels[2 * MAX_SPR_PIXELS];
/* Eight bits for every pixel. */
union sps_union spixstate;
static uae_u16 ham_linebuf[MAX_PIXELS_PER_LINE * 2];
static uae_u8 *xlinebuffer;
static int *native2amiga_line_map;
static uae_u8 **row_map;
static uae_u8 row_tmp[MAX_PIXELS_PER_LINE * 32 / 8];
/* line_draw_funcs: pfield_do_linetoscr, pfield_do_fill_line, decode_ham */
typedef void(*line_draw_func)(int, int);
#define LINE_UNDECIDED 1
#define LINE_DECIDED 2
#define LINE_DECIDED_DOUBLE 3
#define LINE_AS_PREVIOUS 4
#define LINE_BLACK 5
#define LINE_REMEMBERED_AS_BLACK 6
#define LINE_DONE 7
#define LINE_DONE_AS_PREVIOUS 8
#define LINE_REMEMBERED_AS_PREVIOUS 9
static int linestate_first_undecided = 0;
uae_u8 line_data[(MAXVPOS + 2) * 2][MAX_PLANES * MAX_WORDS_PER_LINE * 2];
/* The visible window: VISIBLE_LEFT_BORDER contains the left border of the visible
area, VISIBLE_RIGHT_BORDER the right border. These are in window coordinates. */
static int visible_left_border, visible_right_border;
static int linetoscr_x_adjust_bytes;
static int thisframe_y_adjust;
static int thisframe_y_adjust_real, max_ypos_thisframe;
/* These are generated by the drawing code from the line_decisions array for
each line that needs to be drawn. These are basically extracted out of
bit fields in the hardware registers. */
static int bplehb, bplham, bpldualpf, bpldualpfpri, bpldualpf2of, bplplanecnt;
static int bplres;
static int plf1pri, plf2pri, bplxor;
static uae_u32 plf_sprite_mask;
static uae_u32 plf_sprite_mask_n16;
static int sbasecol[2] = { 16, 16 };
bool picasso_requested_on;
bool picasso_on;
int inhibit_frame;
int framecnt = 0;
/* Calculate idle time (time to wait for vsync) */
int idletime_frames = 0;
unsigned long idletime_time = 0;
int idletime_percent = 0;
#define IDLETIME_FRAMES 20
long time_per_frame = 20000; // Default for PAL (50 Hz): 20000 microsecs
// Temporary define
#define OFFSET_FREE_CPU_CYCLES 1500
#define STEP_CYCLES 250
void adjust_idletime(long ms_waited)
{
if (ms_waited == -1)
{
// Frame miss...
idletime_percent = 0;
ms_waited = 0;
}
idletime_frames++;
idletime_time += ms_waited;
if (idletime_frames >= IDLETIME_FRAMES)
{
unsigned long ms_for_frames = time_per_frame * idletime_frames * (1 + currprefs.gfx_framerate);
idletime_percent = idletime_time * 100 / ms_for_frames;
if (idletime_percent < 0)
idletime_percent = 0;
else if (idletime_percent > 100)
idletime_percent = 100;
idletime_time = 0;
idletime_frames = 0;
}
if (currprefs.m68k_speed < 0) {
if (ms_waited < (500 + OFFSET_FREE_CPU_CYCLES) && speedup_timelimit > -10000)
speedup_timelimit -= STEP_CYCLES;
else
if (ms_waited > (1400 + OFFSET_FREE_CPU_CYCLES) && speedup_timelimit < -1000)
speedup_timelimit += STEP_CYCLES;
}
}
void reset_decision_table(void)
{
linestate_first_undecided = 0;
}
STATIC_INLINE void count_frame(void)
{
framecnt++;
if (framecnt > currprefs.gfx_framerate)
framecnt = 0;
if (inhibit_frame)
framecnt = 1;
}
STATIC_INLINE int xshift(int x, int shift)
{
if (shift < 0)
return x >> (-shift);
else
return x << shift;
}
int coord_native_to_amiga_x(int x)
{
if (gfxvidinfo.outwidth > 600)
{
x += (visible_left_border << 1);
return x + 2*DISPLAY_LEFT_SHIFT - 2*DIW_DDF_OFFSET;
}
else
{
x += visible_left_border;
x <<= (1 - lores_shift);
return x + 2*DISPLAY_LEFT_SHIFT - 2*DIW_DDF_OFFSET;
}
}
int coord_native_to_amiga_y(int y)
{
return native2amiga_line_map[y];
}
STATIC_INLINE int res_shift_from_window(int x)
{
if (res_shift >= 0)
return x >> res_shift;
return x << -res_shift;
}
STATIC_INLINE int res_shift_from_amiga(int x)
{
if (res_shift >= 0)
return x >> res_shift;
return x << -res_shift;
}
static struct decision *dp_for_drawing;
static struct draw_info *dip_for_drawing;
/*
* Screen update macros/functions
*/
/* The important positions in the line: where do we start drawing the left border,
where do we start drawing the playfield, where do we start drawing the right border.
All of these are forced into the visible window (VISIBLE_LEFT_BORDER .. VISIBLE_RIGHT_BORDER).
PLAYFIELD_START and PLAYFIELD_END are in window coordinates. */
static int playfield_start, playfield_end;
static int pixels_offset;
static int src_pixel, ham_src_pixel;
/* How many pixels in window coordinates which are to the left of the left border. */
static int unpainted;
#include "linetoscr.c"
static void pfield_do_linetoscr_0_640(int start, int stop)
{
int local_res_shift = 1 - bplres; // stretch LORES, nothing for HIRES, shrink for SUPERHIRES
start = start << 1;
stop = stop << 1;
if (local_res_shift == 0)
src_pixel = linetoscr_16(src_pixel, start, stop);
else if (local_res_shift == 1)
src_pixel = linetoscr_16_stretch1(src_pixel, start, stop);
else //if (local_res_shift == -1)
src_pixel = linetoscr_16_shrink1(src_pixel, start, stop);
}
static void pfield_do_linetoscr_0_640_AGA(int start, int stop)
{
int local_res_shift = 1 - bplres; // stretch LORES, nothing for HIRES, shrink for SUPERHIRES
start = start << 1;
stop = stop << 1;
if (local_res_shift == 0)
src_pixel = linetoscr_16_aga(src_pixel, start, stop);
else if (local_res_shift == 1)
src_pixel = linetoscr_16_stretch1_aga(src_pixel, start, stop);
else //if (local_res_shift == -1)
src_pixel = linetoscr_16_shrink1_aga(src_pixel, start, stop);
}
static void pfield_do_linetoscr_0(int start, int stop)
{
if (res_shift == 0)
src_pixel = linetoscr_16(src_pixel, start, stop);
//else if (res_shift == 1) //can't happen, target is lores
// src_pixel = linetoscr_16_stretch1 (src_pixel, start, stop);
else //if (res_shift == -1)
src_pixel = linetoscr_16_shrink1(src_pixel, start, stop);
}
static void pfield_do_linetoscr_0_AGA(int start, int stop)
{
if (res_shift == 0)
src_pixel = linetoscr_16_aga(src_pixel, start, stop);
//else if (res_shift == 1) //can't happen, target is lores
// src_pixel = linetoscr_16_stretch1_aga (src_pixel, start, stop);
else //if (res_shift == -1)
src_pixel = linetoscr_16_shrink1_aga(src_pixel, start, stop);
}
STATIC_INLINE xcolnr getbgc(bool blank)
{
return (blank || colors_for_drawing.borderblank) ? 0 : colors_for_drawing.acolors[0];
}
static void pfield_do_fill_line_0_640(int start, int stop)
{
register uae_u16 *b = &(((uae_u16 *)xlinebuffer)[start << 1]);
register xcolnr col = getbgc(false);
register int i;
register int max = (stop - start) << 1;
for (i = 0; i < max; i++, b++)
*b = col;
}
static void pfield_do_fill_line_0(int start, int stop)
{
register uae_u16 *b = &(((uae_u16 *)xlinebuffer)[start]);
register xcolnr col = getbgc(false);
register int i;
register int max = (stop - start);
for (i = 0; i < max; i++, b++)
*b = col;
}
static line_draw_func pfield_do_linetoscr = (line_draw_func)pfield_do_linetoscr_0;
static line_draw_func pfield_do_fill_line = (line_draw_func)pfield_do_fill_line_0;
/* Initialize the variables necessary for drawing a line.
* This involves setting up start/stop positions and display window
* borders. */
static void pfield_init_linetoscr(void)
{
/* First, get data fetch start/stop in DIW coordinates. */
int ddf_left = dp_for_drawing->plfleft * 2 + DIW_DDF_OFFSET;
int ddf_right = dp_for_drawing->plfright * 2 + DIW_DDF_OFFSET;
int leftborderhidden;
/* Compute datafetch start/stop in pixels; native display coordinates. */
int native_ddf_left = coord_hw_to_window_x(ddf_left);
int native_ddf_right = coord_hw_to_window_x(ddf_right);
int linetoscr_diw_start = dp_for_drawing->diwfirstword;
int linetoscr_diw_end = dp_for_drawing->diwlastword;
if (dip_for_drawing->nr_sprites == 0) {
if (linetoscr_diw_start < native_ddf_left)
linetoscr_diw_start = native_ddf_left;
if (linetoscr_diw_end > native_ddf_right)
linetoscr_diw_end = native_ddf_right;
}
/* Perverse cases happen. */
if (linetoscr_diw_end < linetoscr_diw_start)
linetoscr_diw_end = linetoscr_diw_start;
res_shift = lores_shift - bplres;
playfield_start = linetoscr_diw_start;
playfield_end = linetoscr_diw_end;
if (playfield_start < visible_left_border)
playfield_start = visible_left_border;
if (playfield_start > visible_right_border)
playfield_start = visible_right_border;
if (playfield_end < visible_left_border)
playfield_end = visible_left_border;
if (playfield_end > visible_right_border)
playfield_end = visible_right_border;
if (dp_for_drawing->bordersprite_seen && !colors_for_drawing.borderblank && dip_for_drawing->nr_sprites) {
int min = visible_right_border, max = visible_left_border, i;
for (i = 0; i < dip_for_drawing->nr_sprites; i++) {
int x;
x = curr_sprite_entries[dip_for_drawing->first_sprite_entry + i].pos;
if (x < min)
min = x;
x = curr_sprite_entries[dip_for_drawing->first_sprite_entry + i].max;
if (x > max)
max = x;
}
min = coord_hw_to_window_x(min >> sprite_buffer_res) + (DIW_DDF_OFFSET << lores_shift);
max = coord_hw_to_window_x(max >> sprite_buffer_res) + (DIW_DDF_OFFSET << lores_shift);
if (min < playfield_start)
playfield_start = min;
if (playfield_start < visible_left_border)
playfield_start = visible_left_border;
if (max > playfield_end)
playfield_end = max;
if (playfield_end > visible_right_border)
playfield_end = visible_right_border;
}
unpainted = visible_left_border < playfield_start ? 0 : visible_left_border - playfield_start;
ham_src_pixel = MAX_PIXELS_PER_LINE + res_shift_from_window(playfield_start - native_ddf_left);
unpainted = res_shift_from_window(unpainted);
if (sprite_first_x < sprite_last_x) {
uae_u8 *p = spritepixels + sprite_first_x;
int len = sprite_last_x - sprite_first_x + 1;
int i;
/* clear previous sprite data storage line */
memset(p, 0, len);
sprite_last_x = 0;
sprite_first_x = MAX_PIXELS_PER_LINE - 1;
}
/* Now, compute some offsets. */
ddf_left -= DISPLAY_LEFT_SHIFT;
ddf_left <<= bplres;
pixels_offset = MAX_PIXELS_PER_LINE - ddf_left;
leftborderhidden = playfield_start - native_ddf_left;
src_pixel = MAX_PIXELS_PER_LINE + res_shift_from_window(leftborderhidden);
if (dip_for_drawing->nr_sprites == 0)
return;
/* We need to clear parts of apixels. */
if (linetoscr_diw_start < native_ddf_left) {
int size = res_shift_from_window(native_ddf_left - linetoscr_diw_start);
linetoscr_diw_start = native_ddf_left;
memset(pixdata.apixels + MAX_PIXELS_PER_LINE - size, 0, size);
}
if (linetoscr_diw_end > native_ddf_right) {
int pos = res_shift_from_window(native_ddf_right - native_ddf_left);
int size = res_shift_from_window(linetoscr_diw_end - native_ddf_right);
linetoscr_diw_start = native_ddf_left;
memset(pixdata.apixels + MAX_PIXELS_PER_LINE + pos, 0, size);
}
}
STATIC_INLINE void fill_line(void)
{
int nints;
int *start;
xcolnr val;
nints = gfxvidinfo.outwidth >> 1;
if (gfxvidinfo.outwidth > 600)
start = (int *)(((uae_u8*)xlinebuffer) + (visible_left_border << 2));
else
start = (int *)(((uae_u8*)xlinebuffer) + (visible_left_border << 1));
val = getbgc(false);
val |= val << 16;
for (; nints > 0; nints -= 8, start += 8) {
*start = val;
*(start + 1) = val;
*(start + 2) = val;
*(start + 3) = val;
*(start + 4) = val;
*(start + 5) = val;
*(start + 6) = val;
*(start + 7) = val;
}
}
static void dummy_worker(int start, int stop)
{
}
#ifdef ARMV6T2
STATIC_INLINE int DECODE_HAM8_1(int col, int pv)
{
__asm__(
"ubfx %[pv], %[pv], #3, #5 \n\t"
"bfi %[col], %[pv], #0, #5 \n\t"
: [col] "+r" (col),
[pv] "+r" (pv));
return col;
}
STATIC_INLINE int DECODE_HAM8_2(int col, int pv)
{
__asm__(
"ubfx %[pv], %[pv], #3, #5 \n\t"
"bfi %[col], %[pv], #11, #5 \n\t"
: [col] "+r" (col),
[pv] "+r" (pv));
return col;
}
STATIC_INLINE int DECODE_HAM8_3(int col, int pv)
{
__asm__(
"ubfx %[pv], %[pv], #2, #6 \n\t"
"bfi %[col], %[pv], #5, #6 \n\t"
: [col] "+r" (col),
[pv] "+r" (pv));
return col;
}
STATIC_INLINE int DECODE_HAM6_1(int col, int pv)
{
__asm__(
"lsl %[pv], %[pv], #1 \n\t"
"bfi %[col], %[pv], #0, #5 \n\t"
: [col] "+r" (col),
[pv] "+r" (pv));
return (col);
}
STATIC_INLINE int DECODE_HAM6_2(int col, int pv)
{
__asm__(
"lsl %[pv], %[pv], #1 \n\t"
"bfi %[col], %[pv], #11, #5 \n\t"
: [col] "+r" (col),
[pv] "+r" (pv));
return (col);
}
STATIC_INLINE int DECODE_HAM6_3(int col, int pv)
{
__asm__(
"lsl %[pv], %[pv], #2 \n\t"
"bfi %[col], %[pv], #5, #6 \n\t"
: [col] "+r" (col),
[pv] "+r" (pv));
return (col);
}
#endif
static int ham_decode_pixel;
static uae_u16 ham_lastcolor;
/* Decode HAM in the invisible portion of the display (left of VISIBLE_LEFT_BORDER),
* but don't draw anything in. This is done to prepare HAM_LASTCOLOR for later,
* when decode_ham runs.
*
*/
static void init_ham_decoding(void)
{
int unpainted_amiga = unpainted;
ham_decode_pixel = ham_src_pixel;
ham_lastcolor = colors_for_drawing.acolors[0];
if (!bplham) {
if (unpainted_amiga > 0) {
int pv = pixdata.apixels[ham_decode_pixel + unpainted_amiga - 1];
if (aga_mode)
ham_lastcolor = colors_for_drawing.acolors[pv ^ bplxor];
else
ham_lastcolor = colors_for_drawing.acolors[pv];
}
}
else if (aga_mode) {
if (bplplanecnt >= 7) { /* AGA mode HAM8 */
while (unpainted_amiga-- > 0) {
int pv = pixdata.apixels[ham_decode_pixel++] ^ bplxor;
switch (pv & 0x3)
{
case 0x0: ham_lastcolor = colors_for_drawing.acolors[pv >> 2]; break;
#ifdef ARMV6T2
case 0x1: ham_lastcolor = DECODE_HAM8_1(ham_lastcolor, pv); break;
case 0x2: ham_lastcolor = DECODE_HAM8_2(ham_lastcolor, pv); break;
case 0x3: ham_lastcolor = DECODE_HAM8_3(ham_lastcolor, pv); break;
#else
case 0x1: ham_lastcolor &= 0xFFFF03; ham_lastcolor |= (pv & 0xFC); break;
case 0x2: ham_lastcolor &= 0x03FFFF; ham_lastcolor |= (pv & 0xFC) << 16; break;
case 0x3: ham_lastcolor &= 0xFF03FF; ham_lastcolor |= (pv & 0xFC) << 8; break;
#endif
}
}
}
else { /* AGA mode HAM6 */
while (unpainted_amiga-- > 0) {
int pv = pixdata.apixels[ham_decode_pixel++] ^ bplxor;
switch (pv & 0x30)
{
case 0x00: ham_lastcolor = colors_for_drawing.acolors[pv]; break;
#ifdef ARMV6T2
case 0x10: ham_lastcolor = DECODE_HAM6_1(ham_lastcolor, pv); break;
case 0x20: ham_lastcolor = DECODE_HAM6_2(ham_lastcolor, pv); break;
case 0x30: ham_lastcolor = DECODE_HAM6_3(ham_lastcolor, pv); break;
#else
case 0x10: ham_lastcolor &= 0xFFFF00; ham_lastcolor |= (pv & 0xF) << 4; break;
case 0x20: ham_lastcolor &= 0x00FFFF; ham_lastcolor |= (pv & 0xF) << 20; break;
case 0x30: ham_lastcolor &= 0xFF00FF; ham_lastcolor |= (pv & 0xF) << 12; break;
#endif
}
}
}
}
else {
/* OCS/ECS mode HAM6 */
while (unpainted_amiga-- > 0) {
int pv = pixdata.apixels[ham_decode_pixel++];
switch (pv & 0x30)
{
case 0x00: ham_lastcolor = colors_for_drawing.acolors[pv]; break;
#ifdef ARMV6T2
case 0x10: ham_lastcolor = DECODE_HAM6_1(ham_lastcolor, pv); break;
case 0x20: ham_lastcolor = DECODE_HAM6_2(ham_lastcolor, pv); break;
case 0x30: ham_lastcolor = DECODE_HAM6_3(ham_lastcolor, pv); break;
#else
case 0x10: ham_lastcolor &= 0xFF0; ham_lastcolor |= (pv & 0xF); break;
case 0x20: ham_lastcolor &= 0x0FF; ham_lastcolor |= (pv & 0xF) << 8; break;
case 0x30: ham_lastcolor &= 0xF0F; ham_lastcolor |= (pv & 0xF) << 4; break;
#endif
}
}
}
}
static void decode_ham(int pix, int stoppos)
{
int todraw_amiga = res_shift_from_window(stoppos - pix);
if (!bplham) {
while (todraw_amiga-- > 0) {
int pv = pixdata.apixels[ham_decode_pixel];
if (aga_mode)
ham_lastcolor = colors_for_drawing.acolors[pv ^ bplxor];
else
ham_lastcolor = colors_for_drawing.acolors[pv];
ham_linebuf[ham_decode_pixel++] = ham_lastcolor;
}
}
else if (aga_mode) {
if (bplplanecnt >= 7) { /* AGA mode HAM8 */
while (todraw_amiga-- > 0) {
int pv = pixdata.apixels[ham_decode_pixel] ^ bplxor;
switch (pv & 0x3)
{
case 0x0: ham_lastcolor = colors_for_drawing.acolors[pv >> 2]; break;
#ifdef ARMV6T2
case 0x1: ham_lastcolor = DECODE_HAM8_1(ham_lastcolor, pv); break;
case 0x2: ham_lastcolor = DECODE_HAM8_2(ham_lastcolor, pv); break;
case 0x3: ham_lastcolor = DECODE_HAM8_3(ham_lastcolor, pv); break;
#else
case 0x1: ham_lastcolor &= 0xFFFF03; ham_lastcolor |= (pv & 0xFC); break;
case 0x2: ham_lastcolor &= 0x03FFFF; ham_lastcolor |= (pv & 0xFC) << 16; break;
case 0x3: ham_lastcolor &= 0xFF03FF; ham_lastcolor |= (pv & 0xFC) << 8; break;
#endif
}
ham_linebuf[ham_decode_pixel++] = ham_lastcolor;
}
}
else { /* AGA mode HAM6 */
while (todraw_amiga-- > 0) {
int pv = pixdata.apixels[ham_decode_pixel] ^ bplxor;
switch (pv & 0x30)
{
case 0x00: ham_lastcolor = colors_for_drawing.acolors[pv]; break;
#ifdef ARMV6T2
case 0x10: ham_lastcolor = DECODE_HAM6_1(ham_lastcolor, pv); break;
case 0x20: ham_lastcolor = DECODE_HAM6_2(ham_lastcolor, pv); break;
case 0x30: ham_lastcolor = DECODE_HAM6_3(ham_lastcolor, pv); break;
#else
case 0x10: ham_lastcolor &= 0xFFFF00; ham_lastcolor |= (pv & 0xF) << 4; break;
case 0x20: ham_lastcolor &= 0x00FFFF; ham_lastcolor |= (pv & 0xF) << 20; break;
case 0x30: ham_lastcolor &= 0xFF00FF; ham_lastcolor |= (pv & 0xF) << 12; break;
#endif
}
ham_linebuf[ham_decode_pixel++] = ham_lastcolor;
}
}
}
else {
/* OCS/ECS mode HAM6 */
while (todraw_amiga-- > 0) {
int pv = pixdata.apixels[ham_decode_pixel];
switch (pv & 0x30)
{
case 0x00: ham_lastcolor = colors_for_drawing.acolors[pv]; break;
#ifdef ARMV6T2
case 0x10: ham_lastcolor = DECODE_HAM6_1(ham_lastcolor, pv); break;
case 0x20: ham_lastcolor = DECODE_HAM6_2(ham_lastcolor, pv); break;
case 0x30: ham_lastcolor = DECODE_HAM6_3(ham_lastcolor, pv); break;
#else
case 0x10: ham_lastcolor &= 0xFF0; ham_lastcolor |= (pv & 0xF); break;
case 0x20: ham_lastcolor &= 0x0FF; ham_lastcolor |= (pv & 0xF) << 8; break;
case 0x30: ham_lastcolor &= 0xF0F; ham_lastcolor |= (pv & 0xF) << 4; break;
#endif
}
ham_linebuf[ham_decode_pixel++] = ham_lastcolor;
}
}
}
static void gen_pfield_tables(void)
{
int i;
for (i = 0; i < 256; i++) {
int plane1 = ((i >> 0) & 1) | ((i >> 1) & 2) | ((i >> 2) & 4) | ((i >> 3) & 8);
int plane2 = ((i >> 1) & 1) | ((i >> 2) & 2) | ((i >> 3) & 4) | ((i >> 4) & 8);
dblpf_2nd1[i] = plane1 == 0 ? (plane2 == 0 ? 0 : 2) : 1;
dblpf_2nd2[i] = plane2 == 0 ? (plane1 == 0 ? 0 : 1) : 2;
dblpf_ind1_aga[i] = plane1 == 0 ? plane2 : plane1;
dblpf_ind2_aga[i] = plane2 == 0 ? plane1 : plane2;
dblpf_ms1[i] = plane1 == 0 ? (plane2 == 0 ? 16 : 8) : 0;
dblpf_ms2[i] = plane2 == 0 ? (plane1 == 0 ? 16 : 0) : 8;
if (plane2 > 0)
plane2 += 8;
dblpf_ind1[i] = i >= 128 ? i & 0x7F : (plane1 == 0 ? plane2 : plane1);
dblpf_ind2[i] = i >= 128 ? i & 0x7F : (plane2 == 0 ? plane1 : plane2);
clxtab[i] = ((((i & 3) && (i & 12)) << 9)
| (((i & 3) && (i & 48)) << 10)
| (((i & 3) && (i & 192)) << 11)
| (((i & 12) && (i & 48)) << 12)
| (((i & 12) && (i & 192)) << 13)
| (((i & 48) && (i & 192)) << 14));
}
for (i = 0; i < 65536; ++i)
{
sprite_col_nat[i] =
(i & 0x0003) ? ((i >> 0) & 3) + 0 :
(i & 0x000C) ? ((i >> 2) & 3) + 0 :
(i & 0x0030) ? ((i >> 4) & 3) + 4 :
(i & 0x00C0) ? ((i >> 6) & 3) + 4 :
(i & 0x0300) ? ((i >> 8) & 3) + 8 :
(i & 0x0C00) ? ((i >> 10) & 3) + 8 :
(i & 0x3000) ? ((i >> 12) & 3) + 12 :
(i & 0xC000) ? ((i >> 14) & 3) + 12 : 0;
sprite_col_at[i] =
(i & 0x000F) ? ((i >> 0) & 0x000F) :
(i & 0x00F0) ? ((i >> 4) & 0x000F) :
(i & 0x0F00) ? ((i >> 8) & 0x000F) :
(i & 0xF000) ? ((i >> 12) & 0x000F) : 0;
sprite_bit[i] =
(i & 0x0003) ? 0x01 :
(i & 0x000C) ? 0x02 :
(i & 0x0030) ? 0x04 :
(i & 0x00C0) ? 0x08 :
(i & 0x0300) ? 0x10 :
(i & 0x0C00) ? 0x20 :
(i & 0x3000) ? 0x40 :
(i & 0xC000) ? 0x80 : 0;
}
}
static void draw_sprites_normal_sp_lo_nat(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
col = sprite_col_nat[v] + 16;
pixdata.apixels[window_pos] = col;
}
window_pos++;
}
}
static void draw_sprites_normal_ham_lo_nat(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
col = sprite_col_nat[v] + 16;
ham_linebuf[window_pos] = colors_for_drawing.acolors[col];
}
window_pos++;
}
}
static void draw_sprites_normal_dp_lo_nat(struct sprite_entry *e)
{
int *shift_lookup = (bpldualpfpri ? dblpf_ms2 : dblpf_ms1);
uae_u16 *buf = spixels + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
int maskshift, plfmask;
unsigned int v = buf[pos];
maskshift = shift_lookup[pixdata.apixels[window_pos]];
plfmask = (plf_sprite_mask >> maskshift) >> maskshift;
v &= ~plfmask;
if (v != 0) {
unsigned int col;
col = sprite_col_nat[v] + 16 + 128;
pixdata.apixels[window_pos] = col;
}
window_pos++;
}
}
static void draw_sprites_normal_sp_lo_at(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (stbuf[pos] & offs) {
col = sprite_col_at[v] + 16;
}
else {
col = sprite_col_nat[v] + 16;
}
pixdata.apixels[window_pos] = col;
}
window_pos++;
}
}
static void draw_sprites_normal_ham_lo_at(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (stbuf[pos] & offs) {
col = sprite_col_at[v] + 16;
}
else {
col = sprite_col_nat[v] + 16;
}
ham_linebuf[window_pos] = colors_for_drawing.acolors[col];
}
window_pos++;
}
}
static void draw_sprites_normal_dp_lo_at(struct sprite_entry *e)
{
int *shift_lookup = (bpldualpfpri ? dblpf_ms2 : dblpf_ms1);
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
int maskshift, plfmask;
unsigned int v = buf[pos];
maskshift = shift_lookup[pixdata.apixels[window_pos]];
plfmask = (plf_sprite_mask >> maskshift) >> maskshift;
v &= ~plfmask;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (stbuf[pos] & offs) {
col = sprite_col_at[v] + 16 + 128;
}
else {
col = sprite_col_nat[v] + 16 + 128;
}
pixdata.apixels[window_pos] = col;
}
window_pos++;
}
}
static void draw_sprites_normal_sp_hi_nat(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos <<= 1;
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
col = sprite_col_nat[v] + 16;
pixdata.apixels_w[window_pos >> 1] = col | (col << 8);
}
window_pos += 2;
}
}
static void draw_sprites_normal_ham_hi_nat(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos <<= 1;
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
col = sprite_col_nat[v] + 16;
col = col | (col << 8);
ham_linebuf[window_pos >> 1] = colors_for_drawing.acolors[col];
}
window_pos += 2;
}
}
static void draw_sprites_normal_dp_hi_nat(struct sprite_entry *e)
{
int *shift_lookup = (bpldualpfpri ? dblpf_ms2 : dblpf_ms1);
uae_u16 *buf = spixels + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos <<= 1;
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
int maskshift, plfmask;
unsigned int v = buf[pos];
maskshift = shift_lookup[pixdata.apixels[window_pos]];
plfmask = (plf_sprite_mask >> maskshift) >> maskshift;
v &= ~plfmask;
if (v != 0) {
unsigned int col;
col = sprite_col_nat[v] + 16 + 128;
pixdata.apixels_w[window_pos >> 1] = col | (col << 8);
}
window_pos += 2;
}
}
static void draw_sprites_normal_sp_hi_at(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos <<= 1;
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (stbuf[pos] & offs) {
col = sprite_col_at[v] + 16;
}
else {
col = sprite_col_nat[v] + 16;
}
pixdata.apixels_w[window_pos >> 1] = col | (col << 8);
}
window_pos += 2;
}
}
static void draw_sprites_normal_ham_hi_at(struct sprite_entry *e)
{
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos <<= 1;
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
unsigned int v = buf[pos];
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (stbuf[pos] & offs) {
col = sprite_col_at[v] + 16;
}
else {
col = sprite_col_nat[v] + 16;
}
col = col | (col << 8);
ham_linebuf[window_pos >> 1] = colors_for_drawing.acolors[col];
}
window_pos += 2;
}
}
static void draw_sprites_normal_dp_hi_at(struct sprite_entry *e)
{
int *shift_lookup = (bpldualpfpri ? dblpf_ms2 : dblpf_ms1);
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT));
window_pos <<= 1;
window_pos += pixels_offset;
unsigned max = e->max;
for (pos = e->pos; pos < max; pos++) {
int maskshift, plfmask;
unsigned int v = buf[pos];
maskshift = shift_lookup[pixdata.apixels[window_pos]];
plfmask = (plf_sprite_mask >> maskshift) >> maskshift;
v &= ~plfmask;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (stbuf[pos] & offs) {
col = sprite_col_at[v] + 16 + 128;
}
else {
col = sprite_col_nat[v] + 16 + 128;
}
pixdata.apixels_w[window_pos >> 1] = col | (col << 8);
}
window_pos += 2;
}
}
typedef void(*draw_sprites_func)(struct sprite_entry *e);
static draw_sprites_func draw_sprites_dp_hi[2] = {
draw_sprites_normal_dp_hi_nat,
draw_sprites_normal_dp_hi_at
};
static draw_sprites_func draw_sprites_sp_hi[2] = {
draw_sprites_normal_sp_hi_nat,
draw_sprites_normal_sp_hi_at
};
static draw_sprites_func draw_sprites_ham_hi[2] = {
draw_sprites_normal_ham_hi_nat,
draw_sprites_normal_ham_hi_at
};
static draw_sprites_func draw_sprites_dp_lo[2] = {
draw_sprites_normal_dp_lo_nat,
draw_sprites_normal_dp_lo_at
};
static draw_sprites_func draw_sprites_sp_lo[2] = {
draw_sprites_normal_sp_lo_nat,
draw_sprites_normal_sp_lo_at
};
static draw_sprites_func draw_sprites_ham_lo[2] = {
draw_sprites_normal_ham_lo_nat,
draw_sprites_normal_ham_lo_at
};
static draw_sprites_func *draw_sprites_punt = draw_sprites_sp_lo;
/* When looking at this function and the ones that inline it, bear in mind
what an optimizing compiler will do with this code. All callers of this
function only pass in constant arguments (except for E). This means
that many of the if statements will go away completely after inlining. */
/* NOTE: This function is called for AGA modes *only* */
STATIC_INLINE void draw_sprites_aga_ham(struct sprite_entry *e, const int doubling, const int skip, const int has_attach)
{
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT) << sprite_buffer_res);
if (skip)
window_pos >>= 1;
else if (doubling)
window_pos <<= 1;
window_pos += pixels_offset;
for (pos = e->pos; pos < e->max; pos += 1 << skip) {
unsigned int v = buf[pos];
if (v) {
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (has_attach && (stbuf[pos] & offs)) {
col = sprite_col_at[v] + sbasecol[1];
}
else {
if (offs & 0x55)
col = sprite_col_nat[v] + sbasecol[0];
else
col = sprite_col_nat[v] + sbasecol[1];
}
col = colors_for_drawing.acolors[col ^ bplxor];
ham_linebuf[window_pos] = col;
if (doubling)
ham_linebuf[window_pos + 1] = col;
}
}
window_pos += 1 << doubling;
}
}
STATIC_INLINE void draw_sprites_aga_dp(struct sprite_entry *e, const int doubling, const int skip, const int has_attach)
{
int *shift_lookup = (bpldualpfpri ? dblpf_ms2 : dblpf_ms1);
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT) << sprite_buffer_res);
if (skip)
window_pos >>= 1;
else if (doubling)
window_pos <<= 1;
window_pos += pixels_offset;
if (window_pos < sprite_first_x)
sprite_first_x = window_pos;
for (pos = e->pos; pos < e->max; pos += 1 << skip) {
int maskshift, plfmask;
unsigned int v = buf[pos];
if (v) {
/* The value in the shift lookup table is _half_ the shift count we
need. This is because we can't shift 32 bits at once (undefined
behaviour in C). */
maskshift = shift_lookup[pixdata.apixels[window_pos]];
plfmask = (plf_sprite_mask >> maskshift) >> maskshift;
v &= ~plfmask;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (has_attach && (stbuf[pos] & offs)) {
col = sprite_col_at[v] + sbasecol[1];
}
else {
if (offs & 0x55)
col = sprite_col_nat[v] + sbasecol[0];
else
col = sprite_col_nat[v] + sbasecol[1];
}
spritepixels[window_pos] = col;
if (doubling)
spritepixels[window_pos + 1] = col;
}
}
window_pos += 1 << doubling;
}
if (window_pos > sprite_last_x)
sprite_last_x = window_pos;
}
STATIC_INLINE void draw_sprites_aga_sp(struct sprite_entry *e, const int doubling, const int skip, const int has_attach)
{
uae_u16 *buf = spixels + e->first_pixel;
uae_u8 *stbuf = spixstate.bytes + e->first_pixel;
int pos, window_pos;
buf -= e->pos;
stbuf -= e->pos;
window_pos = e->pos + ((DIW_DDF_OFFSET - DISPLAY_LEFT_SHIFT) << sprite_buffer_res);
if (skip)
window_pos >>= 1;
else if (doubling)
window_pos <<= 1;
window_pos += pixels_offset;
for (pos = e->pos; pos < e->max; pos += 1 << skip) {
unsigned int v = buf[pos];
if (v) {
if (pixdata.apixels[window_pos])
v &= plf_sprite_mask_n16;
if (v != 0) {
unsigned int col;
int offs;
offs = sprite_bit[v];
if (has_attach && (stbuf[pos] & offs)) {
col = sprite_col_at[v] + sbasecol[1];
}
else {
if (offs & 0x55)
col = sprite_col_nat[v] + sbasecol[0];
else
col = sprite_col_nat[v] + sbasecol[1];
}
col ^= bplxor;
if (doubling)
pixdata.apixels_w[window_pos >> 1] = col | (col << 8);
else
pixdata.apixels[window_pos] = col;
}
}
window_pos += 1 << doubling;
}
}
// ENTRY, doubling, skip, has_attach
static void draw_sprites_aga_sp_lo_nat(struct sprite_entry *e) { draw_sprites_aga_sp(e, 0, 1, 0); }
static void draw_sprites_aga_dp_lo_nat(struct sprite_entry *e) { draw_sprites_aga_dp(e, 0, 1, 0); }
static void draw_sprites_aga_ham_lo_nat(struct sprite_entry *e) { draw_sprites_aga_ham(e, 0, 1, 0); }
static void draw_sprites_aga_sp_lo_at(struct sprite_entry *e) { draw_sprites_aga_sp(e, 0, 1, 1); }
static void draw_sprites_aga_dp_lo_at(struct sprite_entry *e) { draw_sprites_aga_dp(e, 0, 1, 1); }
static void draw_sprites_aga_ham_lo_at(struct sprite_entry *e) { draw_sprites_aga_ham(e, 0, 1, 1); }
static void draw_sprites_aga_sp_hi_nat(struct sprite_entry *e) { draw_sprites_aga_sp(e, 0, 0, 0); }
static void draw_sprites_aga_dp_hi_nat(struct sprite_entry *e) { draw_sprites_aga_dp(e, 0, 0, 0); }
static void draw_sprites_aga_ham_hi_nat(struct sprite_entry *e) { draw_sprites_aga_ham(e, 0, 0, 0); }
static void draw_sprites_aga_sp_hi_at(struct sprite_entry *e) { draw_sprites_aga_sp(e, 0, 0, 1); }
static void draw_sprites_aga_dp_hi_at(struct sprite_entry *e) { draw_sprites_aga_dp(e, 0, 0, 1); }
static void draw_sprites_aga_ham_hi_at(struct sprite_entry *e) { draw_sprites_aga_ham(e, 0, 0, 1); }
static void draw_sprites_aga_sp_shi_nat(struct sprite_entry *e) { draw_sprites_aga_sp(e, 1, 0, 0); }
static void draw_sprites_aga_dp_shi_nat(struct sprite_entry *e) { draw_sprites_aga_dp(e, 1, 0, 0); }
static void draw_sprites_aga_ham_shi_nat(struct sprite_entry *e) { draw_sprites_aga_ham(e, 1, 0, 0); }
static void draw_sprites_aga_sp_shi_at(struct sprite_entry *e) { draw_sprites_aga_sp(e, 1, 0, 1); }
static void draw_sprites_aga_dp_shi_at(struct sprite_entry *e) { draw_sprites_aga_dp(e, 1, 0, 1); }
static void draw_sprites_aga_ham_shi_at(struct sprite_entry *e) { draw_sprites_aga_ham(e, 1, 0, 1); }
static draw_sprites_func draw_sprites_aga_sp_lo[2] = {
draw_sprites_aga_sp_lo_nat,
draw_sprites_aga_sp_lo_at
};
static draw_sprites_func draw_sprites_aga_sp_hi[2] = {
draw_sprites_aga_sp_hi_nat,
draw_sprites_aga_sp_hi_at
};
static draw_sprites_func draw_sprites_aga_sp_shi[2] = {
draw_sprites_aga_sp_shi_nat,
draw_sprites_aga_sp_shi_at
};
static draw_sprites_func draw_sprites_aga_dp_lo[2] = {
draw_sprites_aga_dp_lo_nat,
draw_sprites_aga_dp_lo_at
};
static draw_sprites_func draw_sprites_aga_dp_hi[2] = {
draw_sprites_aga_dp_hi_nat,
draw_sprites_aga_dp_hi_at
};
static draw_sprites_func draw_sprites_aga_dp_shi[2] = {
draw_sprites_aga_dp_shi_nat,
draw_sprites_aga_dp_shi_at
};
static draw_sprites_func draw_sprites_aga_ham_lo[2] = {
draw_sprites_aga_ham_lo_nat,
draw_sprites_aga_ham_lo_at
};
static draw_sprites_func draw_sprites_aga_ham_hi[2] = {
draw_sprites_aga_ham_hi_nat,
draw_sprites_aga_ham_hi_at
};
static draw_sprites_func draw_sprites_aga_ham_shi[2] = {
draw_sprites_aga_ham_shi_nat,
draw_sprites_aga_ham_shi_at
};
static __inline__ void decide_draw_sprites(void)
{
if (aga_mode)
{
int diff = sprite_buffer_res - bplres;
if (diff > 0)
{ // doubling = 0, skip = 1
if (bpldualpf)
{ // ham = 0, dualpf = 1
draw_sprites_punt = draw_sprites_aga_dp_lo;
}
else if (dp_for_drawing->ham_seen)
{ // ham = 1, dualpf = 0
draw_sprites_punt = draw_sprites_aga_ham_lo;
}
else
{ // ham = 0, dualpf = 0
draw_sprites_punt = draw_sprites_aga_sp_lo;
}
}
else if (diff == 0)
{ // doubling = 0, skip = 0
if (bpldualpf)
{ // ham = 0, dualpf = 1
draw_sprites_punt = draw_sprites_aga_dp_hi;
}
else if (dp_for_drawing->ham_seen)
{ // ham = 1, dualpf = 0
draw_sprites_punt = draw_sprites_aga_ham_hi;
}
else
{ // ham = 0, dualpf = 0
draw_sprites_punt = draw_sprites_aga_sp_hi;
}
}
else
{ // doubling = 1, skip = 0
if (bpldualpf)
{ // ham = 0, dualpf = 1
draw_sprites_punt = draw_sprites_aga_dp_shi;
}
else if (dp_for_drawing->ham_seen)
{ // ham = 1, dualpf = 0
draw_sprites_punt = draw_sprites_aga_ham_shi;
}
else
{ // ham = 0, dualpf = 0
draw_sprites_punt = draw_sprites_aga_sp_shi;
}
}
}
else
{
if (bplres == RES_LORES)
if (bpldualpf)
draw_sprites_punt = draw_sprites_dp_lo;
else if (dp_for_drawing->ham_seen)
draw_sprites_punt = draw_sprites_ham_lo;
else
draw_sprites_punt = draw_sprites_sp_lo;
else
if (bpldualpf)
draw_sprites_punt = draw_sprites_dp_hi;
else if (dp_for_drawing->ham_seen)
draw_sprites_punt = draw_sprites_ham_hi;
else
draw_sprites_punt = draw_sprites_sp_hi;
}
}
#define MERGE(a,b,mask,shift) do {\
uae_u32 tmp = mask & (a ^ (b >> shift)); \
a ^= tmp; \
b ^= (tmp << shift); \
} while (0)
#define GETLONG(P) (*(uae_u32 *)P)
#define DATA_POINTER(n) (line_data[lineno] + (n) * MAX_WORDS_PER_LINE * 2)
#ifdef USE_ARMNEON
#ifdef __cplusplus
extern "C" {
#endif
void ARM_doline_n1(uae_u32 *pixels, int wordcount, int lineno);
void NEON_doline_n2(uae_u32 *pixels, int wordcount, int lineno);
void NEON_doline_n3(uae_u32 *pixels, int wordcount, int lineno);
void NEON_doline_n4(uae_u32 *pixels, int wordcount, int lineno);
void NEON_doline_n6(uae_u32 *pixels, int wordcount, int lineno);
void NEON_doline_n8(uae_u32 *pixels, int wordcount, int lineno);
#ifdef __cplusplus
}
#endif
static void pfield_doline_n0(uae_u32 *pixels, int wordcount, int lineno)
{
memset(pixels, 0, wordcount << 5);
}
#define MERGE_0(a,b,mask,shift) {\
uae_u32 tmp = mask & (b>>shift); \
a = tmp; \
b ^= (tmp << shift); \
}
static void pfield_doline_n5(uae_u32 *pixels, int wordcount, int lineno)
{
uae_u8 *real_bplpt[5];
real_bplpt[0] = DATA_POINTER(0);
real_bplpt[1] = DATA_POINTER(1);
real_bplpt[2] = DATA_POINTER(2);
real_bplpt[3] = DATA_POINTER(3);
real_bplpt[4] = DATA_POINTER(4);
while (wordcount-- > 0) {
uae_u32 b0, b1, b2, b3, b4, b5, b6, b7;
b3 = GETLONG((uae_u32 *)real_bplpt[4]); real_bplpt[4] += 4;
b4 = GETLONG((uae_u32 *)real_bplpt[3]); real_bplpt[3] += 4;
b5 = GETLONG((uae_u32 *)real_bplpt[2]); real_bplpt[2] += 4;
b6 = GETLONG((uae_u32 *)real_bplpt[1]); real_bplpt[1] += 4;
b7 = GETLONG((uae_u32 *)real_bplpt[0]); real_bplpt[0] += 4;
MERGE_0(b2, b3, 0x55555555, 1);
MERGE(b4, b5, 0x55555555, 1);
MERGE(b6, b7, 0x55555555, 1);
MERGE_0(b0, b2, 0x33333333, 2);
MERGE_0(b1, b3, 0x33333333, 2);
MERGE(b4, b6, 0x33333333, 2);
MERGE(b5, b7, 0x33333333, 2);
MERGE(b0, b4, 0x0f0f0f0f, 4);
MERGE(b1, b5, 0x0f0f0f0f, 4);
MERGE(b2, b6, 0x0f0f0f0f, 4);
MERGE(b3, b7, 0x0f0f0f0f, 4);
MERGE(b0, b1, 0x00ff00ff, 8);
MERGE(b2, b3, 0x00ff00ff, 8);
MERGE(b4, b5, 0x00ff00ff, 8);
MERGE(b6, b7, 0x00ff00ff, 8);
MERGE(b0, b2, 0x0000ffff, 16);
do_put_mem_long(pixels, b0);
do_put_mem_long(pixels + 4, b2);
MERGE(b1, b3, 0x0000ffff, 16);
do_put_mem_long(pixels + 2, b1);
do_put_mem_long(pixels + 6, b3);
MERGE(b4, b6, 0x0000ffff, 16);
do_put_mem_long(pixels + 1, b4);
do_put_mem_long(pixels + 5, b6);
MERGE(b5, b7, 0x0000ffff, 16);
do_put_mem_long(pixels + 3, b5);
do_put_mem_long(pixels + 7, b7);
pixels += 8;
}
}
static void pfield_doline_n7(uae_u32 *pixels, int wordcount, int lineno)
{
uae_u8 *real_bplpt[7];
real_bplpt[0] = DATA_POINTER(0);
real_bplpt[1] = DATA_POINTER(1);
real_bplpt[2] = DATA_POINTER(2);
real_bplpt[3] = DATA_POINTER(3);
real_bplpt[4] = DATA_POINTER(4);
real_bplpt[5] = DATA_POINTER(5);
real_bplpt[6] = DATA_POINTER(6);
while (wordcount-- > 0) {
uae_u32 b0, b1, b2, b3, b4, b5, b6, b7;
b1 = GETLONG((uae_u32 *)real_bplpt[6]); real_bplpt[6] += 4;
b2 = GETLONG((uae_u32 *)real_bplpt[5]); real_bplpt[5] += 4;
b3 = GETLONG((uae_u32 *)real_bplpt[4]); real_bplpt[4] += 4;
b4 = GETLONG((uae_u32 *)real_bplpt[3]); real_bplpt[3] += 4;
b5 = GETLONG((uae_u32 *)real_bplpt[2]); real_bplpt[2] += 4;
b6 = GETLONG((uae_u32 *)real_bplpt[1]); real_bplpt[1] += 4;
b7 = GETLONG((uae_u32 *)real_bplpt[0]); real_bplpt[0] += 4;
MERGE_0(b0, b1, 0x55555555, 1);
MERGE(b2, b3, 0x55555555, 1);
MERGE(b4, b5, 0x55555555, 1);
MERGE(b6, b7, 0x55555555, 1);
MERGE(b0, b2, 0x33333333, 2);
MERGE(b1, b3, 0x33333333, 2);
MERGE(b4, b6, 0x33333333, 2);
MERGE(b5, b7, 0x33333333, 2);
MERGE(b0, b4, 0x0f0f0f0f, 4);
MERGE(b1, b5, 0x0f0f0f0f, 4);
MERGE(b2, b6, 0x0f0f0f0f, 4);
MERGE(b3, b7, 0x0f0f0f0f, 4);
MERGE(b0, b1, 0x00ff00ff, 8);
MERGE(b2, b3, 0x00ff00ff, 8);
MERGE(b4, b5, 0x00ff00ff, 8);
MERGE(b6, b7, 0x00ff00ff, 8);
MERGE(b0, b2, 0x0000ffff, 16);
do_put_mem_long(pixels, b0);
do_put_mem_long(pixels + 4, b2);
MERGE(b1, b3, 0x0000ffff, 16);
do_put_mem_long(pixels + 2, b1);
do_put_mem_long(pixels + 6, b3);
MERGE(b4, b6, 0x0000ffff, 16);
do_put_mem_long(pixels + 1, b4);
do_put_mem_long(pixels + 5, b6);
MERGE(b5, b7, 0x0000ffff, 16);
do_put_mem_long(pixels + 3, b5);
do_put_mem_long(pixels + 7, b7);
pixels += 8;
}
}
typedef void(*pfield_doline_func)(uae_u32 *, int, int);
static pfield_doline_func pfield_doline_n[9] = {
pfield_doline_n0,
ARM_doline_n1,
NEON_doline_n2,
NEON_doline_n3,
NEON_doline_n4,
pfield_doline_n5,
NEON_doline_n6,
pfield_doline_n7,
NEON_doline_n8
};
#else
static uae_u8 *real_bplpt[8];
/* We use the compiler's inlining ability to ensure that PLANES is in effect a compile time
constant. That will cause some unnecessary code to be optimized away.
Don't touch this if you don't know what you are doing. */
STATIC_INLINE void pfield_doline_1(uae_u32 *pixels, int wordcount, int planes)
{
while (wordcount-- > 0) {
uae_u32 b0, b1, b2, b3, b4, b5, b6, b7;
b0 = 0, b1 = 0, b2 = 0, b3 = 0, b4 = 0, b5 = 0, b6 = 0, b7 = 0;
switch (planes) {
case 8: b0 = GETLONG(real_bplpt[7]); real_bplpt[7] += 4;
case 7: b1 = GETLONG(real_bplpt[6]); real_bplpt[6] += 4;
case 6: b2 = GETLONG(real_bplpt[5]); real_bplpt[5] += 4;
case 5: b3 = GETLONG(real_bplpt[4]); real_bplpt[4] += 4;
case 4: b4 = GETLONG(real_bplpt[3]); real_bplpt[3] += 4;
case 3: b5 = GETLONG(real_bplpt[2]); real_bplpt[2] += 4;
case 2: b6 = GETLONG(real_bplpt[1]); real_bplpt[1] += 4;
case 1: b7 = GETLONG(real_bplpt[0]); real_bplpt[0] += 4;
}
MERGE(b0, b1, 0x55555555, 1);
MERGE(b2, b3, 0x55555555, 1);
MERGE(b4, b5, 0x55555555, 1);
MERGE(b6, b7, 0x55555555, 1);
MERGE(b0, b2, 0x33333333, 2);
MERGE(b1, b3, 0x33333333, 2);
MERGE(b4, b6, 0x33333333, 2);
MERGE(b5, b7, 0x33333333, 2);
MERGE(b0, b4, 0x0f0f0f0f, 4);
MERGE(b1, b5, 0x0f0f0f0f, 4);
MERGE(b2, b6, 0x0f0f0f0f, 4);
MERGE(b3, b7, 0x0f0f0f0f, 4);
MERGE(b0, b1, 0x00ff00ff, 8);
MERGE(b2, b3, 0x00ff00ff, 8);
MERGE(b4, b5, 0x00ff00ff, 8);
MERGE(b6, b7, 0x00ff00ff, 8);
MERGE(b0, b2, 0x0000ffff, 16);
do_put_mem_long(pixels, b0);
do_put_mem_long(pixels + 4, b2);
MERGE(b1, b3, 0x0000ffff, 16);
do_put_mem_long(pixels + 2, b1);
do_put_mem_long(pixels + 6, b3);
MERGE(b4, b6, 0x0000ffff, 16);
do_put_mem_long(pixels + 1, b4);
do_put_mem_long(pixels + 5, b6);
MERGE(b5, b7, 0x0000ffff, 16);
do_put_mem_long(pixels + 3, b5);
do_put_mem_long(pixels + 7, b7);
pixels += 8;
}
}
/* See above for comments on inlining. These functions should _not_
be inlined themselves. */
static void NOINLINE pfield_doline_n1(uae_u32 *data, int count) { pfield_doline_1(data, count, 1); }
static void NOINLINE pfield_doline_n2(uae_u32 *data, int count) { pfield_doline_1(data, count, 2); }
static void NOINLINE pfield_doline_n3(uae_u32 *data, int count) { pfield_doline_1(data, count, 3); }
static void NOINLINE pfield_doline_n4(uae_u32 *data, int count) { pfield_doline_1(data, count, 4); }
static void NOINLINE pfield_doline_n5(uae_u32 *data, int count) { pfield_doline_1(data, count, 5); }
static void NOINLINE pfield_doline_n6(uae_u32 *data, int count) { pfield_doline_1(data, count, 6); }
static void NOINLINE pfield_doline_n7(uae_u32 *data, int count) { pfield_doline_1(data, count, 7); }
static void NOINLINE pfield_doline_n8(uae_u32 *data, int count) { pfield_doline_1(data, count, 8); }
#endif /* USE_ARMNEON */
static void pfield_doline(int lineno)
{
int wordcount = dp_for_drawing->plflinelen;
uae_u32 *data = pixdata.apixels_l + MAX_PIXELS_PER_LINE / 4;
#ifdef USE_ARMNEON
pfield_doline_n[bplplanecnt](data, wordcount, lineno);
#else
real_bplpt[0] = DATA_POINTER(0);
real_bplpt[1] = DATA_POINTER(1);
real_bplpt[2] = DATA_POINTER(2);
real_bplpt[3] = DATA_POINTER(3);
real_bplpt[4] = DATA_POINTER(4);
real_bplpt[5] = DATA_POINTER(5);
real_bplpt[6] = DATA_POINTER(6);
real_bplpt[7] = DATA_POINTER(7);
switch (bplplanecnt) {
default: break;
case 0: memset(data, 0, wordcount * 32); break;
case 1: pfield_doline_n1(data, wordcount); break;
case 2: pfield_doline_n2(data, wordcount); break;
case 3: pfield_doline_n3(data, wordcount); break;
case 4: pfield_doline_n4(data, wordcount); break;
case 5: pfield_doline_n5(data, wordcount); break;
case 6: pfield_doline_n6(data, wordcount); break;
case 7: pfield_doline_n7(data, wordcount); break;
case 8: pfield_doline_n8(data, wordcount); break;
}
#endif /* USE_ARMNEON */
}
void init_row_map(void)
{
static uae_u8 *oldbufmem;
static int oldheight, oldpitch;
static bool oldgenlock;
int i, j;
if (gfxvidinfo.outheight > max_uae_height) {
write_log(_T("Resolution too high, aborting\n"));
abort();
}
if (!row_map) {
row_map = xmalloc(uae_u8*, max_uae_height + 1);
}
if (oldbufmem && oldbufmem == gfxvidinfo.bufmem &&
oldheight == gfxvidinfo.outheight &&
oldpitch == gfxvidinfo.rowbytes)
return;
j = oldheight == 0 ? max_uae_height : oldheight;
for (i = gfxvidinfo.outheight; i < max_uae_height + 1 && i < j + 1; i++)
row_map[i] = row_tmp;
for (i = 0, j = 0; i < gfxvidinfo.outheight; i++, j += gfxvidinfo.rowbytes)
row_map[i] = gfxvidinfo.bufmem + j;
}
static void init_aspect_maps(void)
{
int i, maxl, h;
h = gfxvidinfo.outheight;
if (h == 0)
/* Do nothing if the gfx driver hasn't initialized the screen yet */
return;
if (native2amiga_line_map)
xfree(native2amiga_line_map);
native2amiga_line_map = xmalloc(int, h);
maxl = (MAXVPOS + 1);
for (i = 0; i < h; i++)
native2amiga_line_map[i] = -1;
for (i = maxl - 1; i >= minfirstline; i--) {
int j;
for (j = i - minfirstline; j < h && native2amiga_line_map[j] == -1; j++)
native2amiga_line_map[j] = i + currprefs.pandora_vertical_offset + OFFSET_Y_ADJUST;
}
}
/*
* One drawing frame has been finished. Tell the graphics code about it.
*/
STATIC_INLINE void do_flush_screen()
{
unlockscr();
flush_screen(); /* vsync mode */
}
/* We only save hardware registers during the hardware frame. Now, when
* drawing the frame, we expand the data into a slightly more useful
* form. */
static void pfield_expand_dp_bplcon(void)
{
bplres = dp_for_drawing->bplres;
bplplanecnt = dp_for_drawing->nr_planes;
bplham = dp_for_drawing->ham_seen;
if (aga_mode) {
bplehb = (dp_for_drawing->bplcon0 & 0x7010) == 0x6000;
bpldualpf2of = (dp_for_drawing->bplcon3 >> 10) & 7;
sbasecol[0] = ((dp_for_drawing->bplcon4 >> 4) & 15) << 4;
sbasecol[1] = ((dp_for_drawing->bplcon4 >> 0) & 15) << 4;
bplxor = dp_for_drawing->bplcon4 >> 8;
}
else
bplehb = (dp_for_drawing->bplcon0 & 0xFC00) == 0x6000 || (dp_for_drawing->bplcon0 & 0xFC00) == 0x7000;
if ((currprefs.chipset_mask & CSMASK_ECS_DENISE) && (dp_for_drawing->bplcon2 & 0x0200))
bplehb = 0;
plf1pri = dp_for_drawing->bplcon2 & 7;
plf2pri = (dp_for_drawing->bplcon2 >> 3) & 7;
plf_sprite_mask = 0xFFFF0000 << (4 * plf2pri);
plf_sprite_mask |= (0x0000FFFF << (4 * plf1pri)) & 0xFFFF;
plf_sprite_mask_n16 = ~(plf_sprite_mask >> 16);
bpldualpf = (dp_for_drawing->bplcon0 & 0x400) == 0x400;
bpldualpfpri = (dp_for_drawing->bplcon2 & 0x40) == 0x40;
}
static bool isham(uae_u16 bplcon0)
{
int p = GET_PLANES(bplcon0);
if (!(bplcon0 & 0x800))
return 0;
if (aga_mode) {
// AGA only has 6 or 8 plane HAM
if (p == 6 || p == 8)
return 1;
}
else {
// OCS/ECS also supports 5 plane HAM
if (GET_RES_DENISE(bplcon0) > 0)
return 0;
if (p >= 5)
return 1;
}
return 0;
}
static void pfield_expand_dp_bplconx(int regno, int v)
{
regno -= 0x1000;
switch (regno)
{
case 0x100: // BPLCON0
dp_for_drawing->bplcon0 = v;
dp_for_drawing->bplres = GET_RES_DENISE(v);
dp_for_drawing->nr_planes = GET_PLANES(v);
dp_for_drawing->ham_seen = isham(v);
break;
case 0x104: // BPLCON2
dp_for_drawing->bplcon2 = v;
break;
case 0x106: // BPLCON3
dp_for_drawing->bplcon3 = v;
break;
case 0x10c: // BPLCON4
dp_for_drawing->bplcon4 = v;
break;
}
pfield_expand_dp_bplcon();
// res_shift = lores_shift - bplres;
}
static int drawing_color_matches;
static enum { color_match_acolors, color_match_full } color_match_type;
/* Set up colors_for_drawing to the state at the beginning of the currently drawn
line. Try to avoid copying color tables around whenever possible. */
static void adjust_drawing_colors(int ctable, int need_full)
{
if (drawing_color_matches != ctable || need_full < 0) {
if (need_full) {
color_reg_cpy(&colors_for_drawing, curr_color_tables + ctable);
color_match_type = color_match_full;
}
else {
memcpy(colors_for_drawing.acolors,
curr_color_tables[ctable].acolors,
sizeof colors_for_drawing.acolors);
colors_for_drawing.borderblank = curr_color_tables[ctable].borderblank;
colors_for_drawing.bordersprite = curr_color_tables[ctable].bordersprite;
color_match_type = color_match_acolors;
}
drawing_color_matches = ctable;
}
else if (need_full && color_match_type != color_match_full) {
color_reg_cpy(&colors_for_drawing, &curr_color_tables[ctable]);
color_match_type = color_match_full;
}
}
STATIC_INLINE void do_color_changes(line_draw_func worker_border, line_draw_func worker_pfield)
{
int i;
int lastpos = visible_left_border;
int endpos = visible_right_border;
for (i = dip_for_drawing->first_color_change; i <= dip_for_drawing->last_color_change; i++) {
int nextpos, nextpos_in_range;
if (i == dip_for_drawing->last_color_change)
nextpos = endpos;
else
nextpos = coord_hw_to_window_x(curr_color_changes[i].linepos);
nextpos_in_range = nextpos;
if (nextpos > endpos)
nextpos_in_range = endpos;
// left border (hblank end to playfield start)
if (nextpos_in_range > lastpos && lastpos < playfield_start) {
int t = nextpos_in_range <= playfield_start ? nextpos_in_range : playfield_start;
(*worker_border)(lastpos, t);
lastpos = t;
}
// playfield
if (nextpos_in_range > lastpos && lastpos >= playfield_start && lastpos < playfield_end) {
int t = nextpos_in_range <= playfield_end ? nextpos_in_range : playfield_end;
(*worker_pfield)(lastpos, t);
lastpos = t;
}
// right border (playfield end to hblank start)
if (nextpos_in_range > lastpos && lastpos >= playfield_end) {
(*worker_border)(lastpos, nextpos_in_range);
lastpos = nextpos_in_range;
}
if (i != dip_for_drawing->last_color_change) {
int regno = curr_color_changes[i].regno;
unsigned int value = curr_color_changes[i].value;
if (regno >= 0x1000) {
pfield_expand_dp_bplconx(regno, value);
}
else if (regno >= 0) {
if (regno == 0 && (value & COLOR_CHANGE_BRDBLANK)) {
colors_for_drawing.borderblank = (value & 1) != 0;
colors_for_drawing.bordersprite = (value & 3) == 2;
}
else {
color_reg_set(&colors_for_drawing, regno, value);
colors_for_drawing.acolors[regno] = getxcolor(value);
}
}
}
if (lastpos >= endpos)
break;
}
}
static void pfield_draw_line(int lineno, int gfx_ypos)
{
int border = 0;
dp_for_drawing = line_decisions + lineno;
dip_for_drawing = curr_drawinfo + lineno;
if (dp_for_drawing->plfleft < 0)
border = 1;
xlinebuffer = row_map[gfx_ypos];
xlinebuffer -= linetoscr_x_adjust_bytes;
if (border == 0) {
pfield_expand_dp_bplcon();
pfield_init_linetoscr();
pfield_doline(lineno);
adjust_drawing_colors(dp_for_drawing->ctable, dp_for_drawing->ham_seen || bplehb);
/* The problem is that we must call decode_ham() BEFORE we do the sprites. */
if (dp_for_drawing->ham_seen) {
init_ham_decoding();
if (dip_for_drawing->nr_color_changes == 0) {
/* The easy case: need to do HAM decoding only once for the
* full line. */
decode_ham(visible_left_border, visible_right_border);
}
else /* Argh. */ {
do_color_changes(dummy_worker, decode_ham);
// reset colors to state before above do_color_changes()
adjust_drawing_colors(dp_for_drawing->ctable, (dp_for_drawing->ham_seen || bplehb) ? -1 : 0);
pfield_expand_dp_bplcon();
}
bplham = dp_for_drawing->ham_at_start;
}
if (dip_for_drawing->nr_sprites) {
int i;
decide_draw_sprites();
for (i = 0; i < dip_for_drawing->nr_sprites; i++) {
struct sprite_entry *e = curr_sprite_entries + dip_for_drawing->first_sprite_entry + i;
draw_sprites_punt[e->has_attached](e);
}
}
do_color_changes(pfield_do_fill_line, pfield_do_linetoscr);
}
else { // border > 0: top or bottom border
bool dosprites = false;
adjust_drawing_colors(dp_for_drawing->ctable, 0);
/* this makes things complex.. */
if (dp_for_drawing->bordersprite_seen && !colors_for_drawing.borderblank && dip_for_drawing->nr_sprites) {
dosprites = true;
pfield_expand_dp_bplcon();
pfield_init_linetoscr();
}
if (!dosprites && dip_for_drawing->nr_color_changes == 0) {
fill_line();
return;
}
if (dosprites) {
int i;
uae_u16 oxor = bplxor;
memset(pixdata.apixels, 0, sizeof pixdata);
decide_draw_sprites();
for (i = 0; i < dip_for_drawing->nr_sprites; i++) {
struct sprite_entry *e = curr_sprite_entries + dip_for_drawing->first_sprite_entry + i;
draw_sprites_punt[e->has_attached](e);
}
if (dp_for_drawing->ham_seen) {
int todraw_amiga = res_shift_from_window(visible_right_border - visible_left_border);
init_ham_decoding();
memset(ham_linebuf + ham_decode_pixel, 0, todraw_amiga * sizeof(uae_u32));
}
bplxor = 0;
do_color_changes(pfield_do_fill_line, pfield_do_linetoscr);
bplxor = oxor;
}
else {
playfield_start = visible_right_border;
playfield_end = visible_right_border;
do_color_changes(pfield_do_fill_line, pfield_do_fill_line);
}
}
}
static void center_image(void)
{
int deltaToBorder;
deltaToBorder = (gfxvidinfo.outwidth >> currprefs.gfx_resolution) - 320;
visible_left_border = 73 - (deltaToBorder >> 1);
visible_right_border = 393 + (deltaToBorder >> 1);
if (gfxvidinfo.outwidth > 600)
linetoscr_x_adjust_bytes = (visible_left_border * gfxvidinfo.pixbytes) << 1;
else
linetoscr_x_adjust_bytes = visible_left_border * gfxvidinfo.pixbytes;
thisframe_y_adjust = minfirstline;
thisframe_y_adjust_real = thisframe_y_adjust + currprefs.pandora_vertical_offset + OFFSET_Y_ADJUST;
max_ypos_thisframe = (maxvpos_display - thisframe_y_adjust);
}
static void init_drawing_frame(void)
{
init_hardware_for_drawing_frame();
linestate_first_undecided = 0;
center_image();
drawing_color_matches = -1;
}
static void draw_status_line(int line, int statusy)
{
uae_u8 *buf;
xlinebuffer = row_map[line];
buf = xlinebuffer;
draw_status_line_single(buf, statusy, gfxvidinfo.outwidth);
}
static void finish_drawing_frame(void)
{
int i;
lockscr();
if (gfxvidinfo.outwidth > 600)
{
if (aga_mode)
pfield_do_linetoscr = (line_draw_func)pfield_do_linetoscr_0_640_AGA;
else
pfield_do_linetoscr = (line_draw_func)pfield_do_linetoscr_0_640;
pfield_do_fill_line = (line_draw_func)pfield_do_fill_line_0_640;
}
else
{
if (aga_mode)
pfield_do_linetoscr = (line_draw_func)pfield_do_linetoscr_0_AGA;
else
pfield_do_linetoscr = (line_draw_func)pfield_do_linetoscr_0;
pfield_do_fill_line = (line_draw_func)pfield_do_fill_line_0;
}
for (i = 0; i < max_ypos_thisframe && i < gfxvidinfo.outheight; i++) {
int line = i + thisframe_y_adjust_real;
if (line >= linestate_first_undecided)
break;
pfield_draw_line(line, i);
}
if (currprefs.leds_on_screen) {
for (i = 0; i < TD_TOTAL_HEIGHT; i++) {
int line = gfxvidinfo.outheight - TD_TOTAL_HEIGHT + i;
draw_status_line(line, i);
}
}
do_flush_screen();
}
STATIC_INLINE void check_picasso(void)
{
#ifdef PICASSO96
if (picasso_requested_on == picasso_on)
return;
picasso_on = picasso_requested_on;
if (!picasso_on)
clear_inhibit_frame(IHF_PICASSO);
else
set_inhibit_frame(IHF_PICASSO);
gfx_set_picasso_state(picasso_on);
picasso_enablescreen(picasso_requested_on);
notice_new_xcolors();
count_frame();
#endif
}
void vsync_handle_check(void)
{
int changed = check_prefs_changed_gfx();
if (changed) {
reset_drawing();
notice_new_xcolors();
}
check_prefs_changed_cd();
check_prefs_changed_audio();
check_prefs_changed_custom();
check_prefs_changed_cpu();
check_picasso();
}
void vsync_handle_redraw(void)
{
if (framecnt == 0)
{
finish_drawing_frame();
}
if (quit_program < 0) {
quit_program = -quit_program;
set_inhibit_frame(IHF_QUIT_PROGRAM);
set_special(SPCFLAG_BRK | SPCFLAG_MODE_CHANGE);
return;
}
count_frame();
if (framecnt == 0)
init_drawing_frame();
}
void hsync_record_line_state(int lineno)
{
if (framecnt != 0)
return;
linestate_first_undecided = lineno + 1;
}
void reset_drawing(void)
{
lores_reset();
linestate_first_undecided = 0;
init_aspect_maps();
init_row_map();
memset(spixels, 0, sizeof spixels);
memset(&spixstate, 0, sizeof spixstate);
init_drawing_frame();
}
void drawing_init(void)
{
gen_pfield_tables();
#ifdef PICASSO96
if (!isrestore()) {
picasso_on = 0;
picasso_requested_on = 0;
gfx_set_picasso_state(0);
}
#endif
xlinebuffer = gfxvidinfo.bufmem;
inhibit_frame = 0;
reset_drawing();
}