appleramdac: HW cursor fixes.

- Add mask so that hardware cursor cannot be drawn beyond the right edge of the frame buffer.
- Add invert pixels. Invert pixels are used in the I-beam cursor and the Watch cursor.
This commit is contained in:
joevt 2024-02-18 17:14:42 -08:00 committed by Maxim Poliakovski
parent b0dc893a05
commit 57d919e424

View File

@ -180,34 +180,44 @@ void AppleRamdac::measure_hw_cursor(uint8_t *fb_ptr) {
} }
void AppleRamdac::draw_hw_cursor(uint8_t *src_buf, uint8_t *dst_buf, int dst_pitch) { void AppleRamdac::draw_hw_cursor(uint8_t *src_buf, uint8_t *dst_buf, int dst_pitch) {
uint8_t *dst_row = dst_buf; int num_pixels = this->video_width - this->cursor_xpos;
if (num_pixels <= 0)
return;
if (num_pixels > 32)
num_pixels = 32;
this->measure_hw_cursor(src_buf); this->measure_hw_cursor(src_buf);
if (this->cursor_xpos >= this->video_width) int num_words = unsigned(num_pixels + 15) / 16;
return;
src_buf += this->fb_pitch * this->cursor_ypos; uint64_t mask0 = (~0LL) << ((num_pixels >= 16) ? 0 : ((16 - num_pixels) * 4));
dst_row += this->cursor_ypos * dst_pitch + this->cursor_xpos * sizeof(uint32_t); uint64_t mask1 = (num_pixels <= 16) ? 0LL : ((~0LL) << ((32 - num_pixels) * 4));
dst_pitch -= 32 * sizeof(uint32_t);
for (int h = 0; h < this->cursor_height; h++) { uint8_t *src_row = src_buf + this->fb_pitch * this->cursor_ypos;
for (int x = 0; x < 2; x++) { // two sets of 16 pixels uint8_t *dst_row = dst_buf + this->cursor_ypos * dst_pitch +
uint64_t pix_data = READ_QWORD_BE_A(src_buf + x * 8); this->cursor_xpos * sizeof(uint32_t);
if (!pix_data) { // skip processing of 16 transparent pixels
dst_row += 16 * sizeof(uint32_t); uint32_t *color = &this->cursor_clut[0];
break;
} for (int h = this->cursor_height; h > 0; h--) {
for (int p = 0; p < 16; p++) { uint8_t* dst_16 = dst_row;
for (int x = 0; x < num_words; x++) {
uint8_t* dst_1 = dst_16;
uint64_t pix_data = READ_QWORD_BE_A(src_row + x * sizeof(uint64_t)) & (x ? mask1 : mask0);
while (pix_data) {
uint8_t pix = pix_data >> 60; // each pixel is 4 bits wide uint8_t pix = pix_data >> 60; // each pixel is 4 bits wide
if (pix & 8) { // check control bit: 0 - transparent, 1 - opaque if (pix & 8) { // check control bit: 0 - transparent, 1 - opaque
WRITE_DWORD_LE_A(dst_row, this->cursor_clut[pix & 7]); WRITE_DWORD_LE_A(dst_1, color[pix & 7]);
} else if (pix & 1) {
uint32_t c = (((READ_DWORD_LE_A(dst_1) >> 7) & 0x010101) * 0xFFU) ^ 0xFFFFFFU;
WRITE_DWORD_LE_A(dst_1, c);
} }
pix_data <<= 4; pix_data <<= 4;
dst_row += sizeof(uint32_t); dst_1 += sizeof(uint32_t);
} }
dst_16 += 16 * sizeof(uint32_t);
} }
src_buf += this->fb_pitch; src_row += this->fb_pitch;
dst_row += dst_pitch; dst_row += dst_pitch;
} }
} }