mirror of https://github.com/bsnes-emu/bsnes.git
156 lines
4.0 KiB
C++
156 lines
4.0 KiB
C++
#ifdef PPU_CPP
|
|
|
|
unsigned PPU::Screen::get_palette(unsigned color) {
|
|
#if defined(ARCH_LSB)
|
|
return ((uint16*)ppu.cgram)[color];
|
|
#else
|
|
color <<= 1;
|
|
return (ppu.cgram[color + 0] << 0) + (ppu.cgram[color + 1] << 8);
|
|
#endif
|
|
}
|
|
|
|
unsigned PPU::Screen::get_direct_color(unsigned p, unsigned t) {
|
|
return ((t & 7) << 2) | ((p & 1) << 1) |
|
|
(((t >> 3) & 7) << 7) | (((p >> 1) & 1) << 6) |
|
|
((t >> 6) << 13) | ((p >> 2) << 12);
|
|
}
|
|
|
|
uint16 PPU::Screen::addsub(unsigned x, unsigned y, bool halve) {
|
|
if(!regs.color_mode) {
|
|
if(!halve) {
|
|
unsigned sum = x + y;
|
|
unsigned carry = (sum - ((x ^ y) & 0x0421)) & 0x8420;
|
|
return (sum - carry) | (carry - (carry >> 5));
|
|
} else {
|
|
return (x + y - ((x ^ y) & 0x0421)) >> 1;
|
|
}
|
|
} else {
|
|
unsigned diff = x - y + 0x8420;
|
|
unsigned borrow = (diff - ((x ^ y) & 0x8420)) & 0x8420;
|
|
if(!halve) {
|
|
return (diff - borrow) & (borrow - (borrow >> 5));
|
|
} else {
|
|
return (((diff - borrow) & (borrow - (borrow >> 5))) & 0x7bde) >> 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
void PPU::Screen::scanline() {
|
|
unsigned main_color = get_palette(0);
|
|
unsigned sub_color = (self.regs.pseudo_hires == false && self.regs.bgmode != 5 && self.regs.bgmode != 6)
|
|
? regs.color : main_color;
|
|
|
|
for(unsigned x = 0; x < 256; x++) {
|
|
output.main[x].color = main_color;
|
|
output.main[x].priority = 0;
|
|
output.main[x].source = 6;
|
|
|
|
output.sub[x].color = sub_color;
|
|
output.sub[x].priority = 0;
|
|
output.sub[x].source = 6;
|
|
}
|
|
|
|
window.render(0);
|
|
window.render(1);
|
|
}
|
|
|
|
void PPU::Screen::render_black() {
|
|
uint32* data = self.output + self.vcounter() * 1024;
|
|
if(self.interlace() && self.field()) data += 512;
|
|
memset(data, 0, self.display.width << 2);
|
|
}
|
|
|
|
uint16 PPU::Screen::get_pixel_main(unsigned x) {
|
|
auto main = output.main[x];
|
|
auto sub = output.sub[x];
|
|
|
|
if(!regs.addsub_mode) {
|
|
sub.source = 6;
|
|
sub.color = regs.color;
|
|
}
|
|
|
|
if(!window.main[x]) {
|
|
if(!window.sub[x]) {
|
|
return 0x0000;
|
|
}
|
|
main.color = 0x0000;
|
|
}
|
|
|
|
if(main.source != 5 && regs.color_enable[main.source] && window.sub[x]) {
|
|
bool halve = false;
|
|
if(regs.color_halve && window.main[x]) {
|
|
if(!regs.addsub_mode || sub.source != 6) halve = true;
|
|
}
|
|
return addsub(main.color, sub.color, halve);
|
|
}
|
|
|
|
return main.color;
|
|
}
|
|
|
|
uint16 PPU::Screen::get_pixel_sub(unsigned x) {
|
|
auto main = output.sub[x];
|
|
auto sub = output.main[x];
|
|
|
|
if(!regs.addsub_mode) {
|
|
sub.source = 6;
|
|
sub.color = regs.color;
|
|
}
|
|
|
|
if(!window.main[x]) {
|
|
if(!window.sub[x]) {
|
|
return 0x0000;
|
|
}
|
|
main.color = 0x0000;
|
|
}
|
|
|
|
if(main.source != 5 && regs.color_enable[main.source] && window.sub[x]) {
|
|
bool halve = false;
|
|
if(regs.color_halve && window.main[x]) {
|
|
if(!regs.addsub_mode || sub.source != 6) halve = true;
|
|
}
|
|
return addsub(main.color, sub.color, halve);
|
|
}
|
|
|
|
return main.color;
|
|
}
|
|
|
|
void PPU::Screen::render() {
|
|
uint32* data = self.output + self.vcounter() * 1024;
|
|
if(self.interlace() && self.field()) data += 512;
|
|
|
|
if(!self.regs.pseudo_hires && self.regs.bgmode != 5 && self.regs.bgmode != 6) {
|
|
for(unsigned i = 0; i < 256; i++) {
|
|
data[i] = video.palette[self.regs.display_brightness << 15 | get_pixel_main(i)];
|
|
}
|
|
} else {
|
|
for(unsigned i = 0; i < 256; i++) {
|
|
*data++ = video.palette[self.regs.display_brightness << 15 | get_pixel_sub(i)];
|
|
*data++ = video.palette[self.regs.display_brightness << 15 | get_pixel_main(i)];
|
|
}
|
|
}
|
|
}
|
|
|
|
PPU::Screen::Screen(PPU& self) : self(self) {
|
|
}
|
|
|
|
PPU::Screen::~Screen() {
|
|
}
|
|
|
|
void PPU::Screen::Output::plot_main(unsigned x, unsigned color, unsigned priority, unsigned source) {
|
|
if(priority > main[x].priority) {
|
|
main[x].color = color;
|
|
main[x].priority = priority;
|
|
main[x].source = source;
|
|
}
|
|
}
|
|
|
|
void PPU::Screen::Output::plot_sub(unsigned x, unsigned color, unsigned priority, unsigned source) {
|
|
if(priority > sub[x].priority) {
|
|
sub[x].color = color;
|
|
sub[x].priority = priority;
|
|
sub[x].source = source;
|
|
}
|
|
}
|
|
|
|
#endif
|