bsnes/higan/sfc/ppu/screen/screen.cpp

181 lines
6.4 KiB
C++
Raw Normal View History

PPU::Screen::Screen(PPU& self) : self(self) {
}
auto PPU::Screen::scanline() -> void {
Update to v096r07 release. byuu says: Changelog: - configuration files are now stored in localpath() instead of configpath() - Video gamma/saturation/luminance sliders are gone now, sorry - added Video Filter->Blur Emulation [1] - added Video Filter->Scanline Emulation [2] - improvements to GBA audio emulation (fixes Minish Cap) [Jonas Quinn] [1] For the Famicom, this does nothing. For the Super Famicom, this performs horizontal blending for proper pseudo-hires translucency. For the Game Boy, Game Boy Color, and Game Boy Advance, this performs interframe blending (each frame is the average of the current and previous frame), which is important for things like the GBVideoPlayer. [2] Right now, this only applies to the Super Famicom, but it'll come to the Famicom in the future. For the Super Famicom, this option doesn't just add scanlines, it simulates the phosphor decay that's visible in interlace mode. If you observe an interlaced game like RPM Racing on a real SNES, you'll notice that even on perfectly still screens, the image appears to shake. This option emulates that effect. Note 1: the buffering right now is a little sub-optimal, so there will be a slight speed hit with this new support. Since the core is now generating native ARGB8888 colors, it might as well call out to the interface to lock/unlock/refresh the video, that way it can render directly to the screen. Although ... that might not be such a hot idea, since the GBx interframe blending reads from the target buffer, and that tends to be a catastrophic option for performance. Note 2: the balanced and performance profiles for the SNES are completely busted again. This WIP took 6 1/2 hours, and I'm exhausted. Very much not looking forward to working on those, since those two have all kinds of fucked up speedup tricks for non-interlaced and/or non-hires video modes. Note 3: if you're on Windows and you saved your system folders somewhere else, now'd be a good time to move them to %localappdata%/higan
2016-01-15 10:06:51 +00:00
line = self.output + self.vcounter() * 1024;
if(self.display.interlace && self.field()) line += 512;
//the first hires pixel of each scanline is transparent
//note: exact value initializations are not confirmed on hardware
math.main.color = get_color(0);
math.sub.color = math.main.color;
math.main.color_enable = !(self.window.regs.col_main_mask & 1);
math.sub.color_enable = !(self.window.regs.col_sub_mask & 1) && regs.back_color_enable;
math.transparent = true;
math.addsub_mode = false;
math.color_halve = regs.color_halve && !regs.addsub_mode && math.main.color_enable;
}
auto PPU::Screen::run() -> void {
Update to v096r07 release. byuu says: Changelog: - configuration files are now stored in localpath() instead of configpath() - Video gamma/saturation/luminance sliders are gone now, sorry - added Video Filter->Blur Emulation [1] - added Video Filter->Scanline Emulation [2] - improvements to GBA audio emulation (fixes Minish Cap) [Jonas Quinn] [1] For the Famicom, this does nothing. For the Super Famicom, this performs horizontal blending for proper pseudo-hires translucency. For the Game Boy, Game Boy Color, and Game Boy Advance, this performs interframe blending (each frame is the average of the current and previous frame), which is important for things like the GBVideoPlayer. [2] Right now, this only applies to the Super Famicom, but it'll come to the Famicom in the future. For the Super Famicom, this option doesn't just add scanlines, it simulates the phosphor decay that's visible in interlace mode. If you observe an interlaced game like RPM Racing on a real SNES, you'll notice that even on perfectly still screens, the image appears to shake. This option emulates that effect. Note 1: the buffering right now is a little sub-optimal, so there will be a slight speed hit with this new support. Since the core is now generating native ARGB8888 colors, it might as well call out to the interface to lock/unlock/refresh the video, that way it can render directly to the screen. Although ... that might not be such a hot idea, since the GBx interframe blending reads from the target buffer, and that tends to be a catastrophic option for performance. Note 2: the balanced and performance profiles for the SNES are completely busted again. This WIP took 6 1/2 hours, and I'm exhausted. Very much not looking forward to working on those, since those two have all kinds of fucked up speedup tricks for non-interlaced and/or non-hires video modes. Note 3: if you're on Windows and you saved your system folders somewhere else, now'd be a good time to move them to %localappdata%/higan
2016-01-15 10:06:51 +00:00
if(self.vcounter() == 0) return;
Update to v078 release. byuu says: Finally, a new release. I have been very busy finishing up SNES box, cartridge and PCB scanning plus cataloguing the data, however this release still has some significant improvements. Most notably would be randomization on startup. This will help match the behavior of real hardware and uninitialized memory + registers. It should help catch homebrew software that forgets to initialize things properly. Of course, I was not able to test the complete library, so it is possible that if I've randomized anything that should be constant, that this could cause a regression. You can disable this randomization for netplay or to work around any incompatibilities by editing bsnes.cfg and setting snes.random to false. The GUI also received some updates. Widget sizes are now computed based on font sizes, giving it a perfectly native look (because it is native.) I've also added a hotkey remapping screen to the input settings. Not only can you remap inputs to controllers now, but those who did not know the hotkey bindings can now quickly see which ones exist and what they are mapped to. Changelog (since v077): - memory and most registers are now randomly initialized on power-up - fixed auto joypad polling issue in Super Star Wars - fixed .nec and .rtc file extensions (they were missing the dot) [krom] - PPU/accuracy now clears overscan region on any frame when it is disabled - PPU/compatibility no longer auto-blends hires pixels (use NTSC filter for this) - added hotkey remapping dialog to input settings window - added a few new hotkeys, including quick-reset - phoenix API now auto-sizes widgets based on font sizes - file dialog once again remembers previously selected file when possible
2011-04-30 13:12:15 +00:00
bool hires = self.regs.pseudo_hires || self.regs.bgmode == 5 || self.regs.bgmode == 6;
auto sscolor = get_pixel_sub(hires);
auto mscolor = get_pixel_main();
Update to v096r07 release. byuu says: Changelog: - configuration files are now stored in localpath() instead of configpath() - Video gamma/saturation/luminance sliders are gone now, sorry - added Video Filter->Blur Emulation [1] - added Video Filter->Scanline Emulation [2] - improvements to GBA audio emulation (fixes Minish Cap) [Jonas Quinn] [1] For the Famicom, this does nothing. For the Super Famicom, this performs horizontal blending for proper pseudo-hires translucency. For the Game Boy, Game Boy Color, and Game Boy Advance, this performs interframe blending (each frame is the average of the current and previous frame), which is important for things like the GBVideoPlayer. [2] Right now, this only applies to the Super Famicom, but it'll come to the Famicom in the future. For the Super Famicom, this option doesn't just add scanlines, it simulates the phosphor decay that's visible in interlace mode. If you observe an interlaced game like RPM Racing on a real SNES, you'll notice that even on perfectly still screens, the image appears to shake. This option emulates that effect. Note 1: the buffering right now is a little sub-optimal, so there will be a slight speed hit with this new support. Since the core is now generating native ARGB8888 colors, it might as well call out to the interface to lock/unlock/refresh the video, that way it can render directly to the screen. Although ... that might not be such a hot idea, since the GBx interframe blending reads from the target buffer, and that tends to be a catastrophic option for performance. Note 2: the balanced and performance profiles for the SNES are completely busted again. This WIP took 6 1/2 hours, and I'm exhausted. Very much not looking forward to working on those, since those two have all kinds of fucked up speedup tricks for non-interlaced and/or non-hires video modes. Note 3: if you're on Windows and you saved your system folders somewhere else, now'd be a good time to move them to %localappdata%/higan
2016-01-15 10:06:51 +00:00
*line++ = (self.regs.display_brightness << 15) | (hires ? sscolor : mscolor);
*line++ = (self.regs.display_brightness << 15) | (mscolor);
}
auto PPU::Screen::get_pixel_sub(bool hires) -> uint16 {
if(self.regs.display_disable || (!self.regs.overscan && self.vcounter() >= 225)) return 0;
uint priority = 0;
if(self.bg1.output.sub.priority) {
priority = self.bg1.output.sub.priority;
if(regs.direct_color && (self.regs.bgmode == 3 || self.regs.bgmode == 4 || self.regs.bgmode == 7)) {
math.sub.color = get_direct_color(self.bg1.output.sub.palette, self.bg1.output.sub.tile);
} else {
math.sub.color = get_color(self.bg1.output.sub.palette);
}
}
if(self.bg2.output.sub.priority > priority) {
priority = self.bg2.output.sub.priority;
math.sub.color = get_color(self.bg2.output.sub.palette);
}
if(self.bg3.output.sub.priority > priority) {
priority = self.bg3.output.sub.priority;
math.sub.color = get_color(self.bg3.output.sub.palette);
}
if(self.bg4.output.sub.priority > priority) {
priority = self.bg4.output.sub.priority;
math.sub.color = get_color(self.bg4.output.sub.palette);
}
if(self.sprite.output.sub.priority > priority) {
priority = self.sprite.output.sub.priority;
math.sub.color = get_color(self.sprite.output.sub.palette);
}
if(math.transparent = (priority == 0)) math.sub.color = get_color(0);
if(!hires) return 0;
if(!math.sub.color_enable) return math.main.color_enable ? math.sub.color : (uint16)0;
return addsub(
math.main.color_enable ? math.sub.color : (uint16)0,
math.addsub_mode ? math.main.color : fixed_color()
);
}
auto PPU::Screen::get_pixel_main() -> uint16 {
if(self.regs.display_disable || (!self.regs.overscan && self.vcounter() >= 225)) return 0;
uint priority = 0;
if(self.bg1.output.main.priority) {
priority = self.bg1.output.main.priority;
if(regs.direct_color && (self.regs.bgmode == 3 || self.regs.bgmode == 4 || self.regs.bgmode == 7)) {
math.main.color = get_direct_color(self.bg1.output.main.palette, self.bg1.output.main.tile);
} else {
math.main.color = get_color(self.bg1.output.main.palette);
}
math.sub.color_enable = regs.bg1_color_enable;
}
if(self.bg2.output.main.priority > priority) {
priority = self.bg2.output.main.priority;
math.main.color = get_color(self.bg2.output.main.palette);
math.sub.color_enable = regs.bg2_color_enable;
}
if(self.bg3.output.main.priority > priority) {
priority = self.bg3.output.main.priority;
math.main.color = get_color(self.bg3.output.main.palette);
math.sub.color_enable = regs.bg3_color_enable;
}
if(self.bg4.output.main.priority > priority) {
priority = self.bg4.output.main.priority;
math.main.color = get_color(self.bg4.output.main.palette);
math.sub.color_enable = regs.bg4_color_enable;
}
if(self.sprite.output.main.priority > priority) {
priority = self.sprite.output.main.priority;
math.main.color = get_color(self.sprite.output.main.palette);
math.sub.color_enable = regs.oam_color_enable && self.sprite.output.main.palette >= 192;
}
if(priority == 0) {
math.main.color = get_color(0);
math.sub.color_enable = regs.back_color_enable;
}
if(!self.window.output.sub.color_enable) math.sub.color_enable = false;
math.main.color_enable = self.window.output.main.color_enable;
if(!math.sub.color_enable) return math.main.color_enable ? math.main.color : (uint16)0;
if(regs.addsub_mode && math.transparent) {
math.addsub_mode = false;
math.color_halve = false;
} else {
math.addsub_mode = regs.addsub_mode;
math.color_halve = regs.color_halve && math.main.color_enable;
}
return addsub(
math.main.color_enable ? math.main.color : (uint16)0,
math.addsub_mode ? math.sub.color : fixed_color()
);
}
auto PPU::Screen::addsub(uint x, uint y) -> uint16 {
if(!regs.color_mode) {
if(!math.color_halve) {
uint sum = x + y;
uint carry = (sum - ((x ^ y) & 0x0421)) & 0x8420;
return (sum - carry) | (carry - (carry >> 5));
} else {
return (x + y - ((x ^ y) & 0x0421)) >> 1;
}
} else {
uint diff = x - y + 0x8420;
uint borrow = (diff - ((x ^ y) & 0x8420)) & 0x8420;
if(!math.color_halve) {
return (diff - borrow) & (borrow - (borrow >> 5));
} else {
return (((diff - borrow) & (borrow - (borrow >> 5))) & 0x7bde) >> 1;
}
}
}
auto PPU::Screen::get_color(uint palette) -> uint16 {
palette <<= 1;
self.regs.cgram_iaddr = palette;
return ppu.cgram[palette + 0] + (ppu.cgram[palette + 1] << 8);
}
auto PPU::Screen::get_direct_color(uint palette, uint tile) -> uint16 {
//palette = -------- BBGGGRRR
//tile = ---bgr-- --------
//output = 0BBb00GG Gg0RRRr0
return ((palette << 7) & 0x6000) + ((tile >> 0) & 0x1000)
+ ((palette << 4) & 0x0380) + ((tile >> 5) & 0x0040)
+ ((palette << 2) & 0x001c) + ((tile >> 9) & 0x0002);
}
auto PPU::Screen::fixed_color() const -> uint16 {
return (regs.color_b << 10) | (regs.color_g << 5) | (regs.color_r << 0);
}
auto PPU::Screen::reset() -> void {
regs.addsub_mode = random(false);
regs.direct_color = random(false);
regs.color_mode = random(false);
regs.color_halve = random(false);
regs.bg1_color_enable = random(false);
regs.bg2_color_enable = random(false);
regs.bg3_color_enable = random(false);
regs.bg4_color_enable = random(false);
regs.oam_color_enable = random(false);
regs.back_color_enable = random(false);
regs.color_r = random(0);
regs.color_g = random(0);
regs.color_b = random(0);
}