From 941c05a7e198b1b74076bd91b73e8a2dd011e38e Mon Sep 17 00:00:00 2001 From: alloncm Date: Wed, 23 Nov 2022 01:06:46 +0200 Subject: [PATCH] Document BG to OBJ priority in CGB mode (#454) Co-authored-by: Eldred Habert Closes https://github.com/gbdev/pandocs/issues/443 --- src/LCDC.md | 2 ++ src/OAM.md | 2 ++ src/Tile_Maps.md | 42 +++++++++++++++++++++++++++++++++++++----- 3 files changed, 41 insertions(+), 5 deletions(-) diff --git a/src/LCDC.md b/src/LCDC.md index db565928..ea87546f 100644 --- a/src/LCDC.md +++ b/src/LCDC.md @@ -110,6 +110,8 @@ When Bit 0 is cleared, the background and window lose their priority - the sprites will be always displayed on top of background and window, independently of the priority flags in OAM and BG Map attributes. +When Bit 0 is set, pixel priority is resolved [as described here](<#BG-to-OBJ Priority in CGB Mode>). + ## Using LCDC LCDC is a powerful tool: each bit controls a lot of behavior, and can be diff --git a/src/OAM.md b/src/OAM.md index c2c9c64c..017870ec 100644 --- a/src/OAM.md +++ b/src/OAM.md @@ -122,4 +122,6 @@ This can be exploited to only hide parts of an object behind the background ([video demonstration](https://youtu.be/B8sJGgCVvnk)). A similar behaviour [can be seen on the NES](https://forums.nesdev.org/viewtopic.php?f=10&t=16861). +**In CGB Mode**, BG vs. OBJ priority is declared in more than one register, [please see this page](<#BG-to-OBJ Priority in CGB Mode>) for more details. + ::: diff --git a/src/Tile_Maps.md b/src/Tile_Maps.md index c071ebbb..cfb7bfa7 100644 --- a/src/Tile_Maps.md +++ b/src/Tile_Maps.md @@ -33,15 +33,47 @@ Bit 3 Tile VRAM Bank number (0=Bank 0, 1=Bank 1) Bit 2-0 Background Palette number (BGP0-7) ``` -When Bit 7 is set, the corresponding BG tile will have priority above -all OBJs (regardless of the priority bits in OAM memory). There's also -a Master Priority flag in LCDC register Bit 0 which overrides all other -priority bits when cleared. - Note that, if the map entry at `0:9800` is tile \$2A, the attribute at `1:9800` doesn't define properties for ALL tiles \$2A on-screen, but only the one at `0:9800`! +### BG-to-OBJ Priority in CGB Mode + +In CGB Mode, the priority between the BG (and window) layer and the OBJ layer is declared in three different places: +- [BG Map Attribute bit 7](<#BG Map Attributes (CGB Mode only)>) +- [LCDC bit 0](<#LCDC.0 — BG and Window enable/priority>) +- [OAM Attributes bit 7](<#Byte 3 — Attributes/Flags>) + +We can infer the following rules from the table below: +* If the BG color index is 0, the OBJ will always have priority; +* Otherwise, if LCDC bit 0 is clear, the OBJ will always have priority; +* Otherwise, if both the BG Attributes and the OAM Attributes have bit 7 clear, the OBJ will have priority; +* Otherwise, BG will have priority. + +The following table shows the relations between the 3 flags: + +LCDC bit 0 | OAM attr bit 7 | BG attr bit 7 | Priority +:---------:|:--------------:|:-------------:|--------- +0 | 0 | 0 | OBJ +0 | 0 | 1 | OBJ +0 | 1 | 0 | OBJ +0 | 1 | 1 | OBJ +1 | 0 | 0 | OBJ +1 | 0 | 1 | BG color 1–3, otherwise OBJ +1 | 1 | 0 | BG color 1–3, otherwise OBJ +1 | 1 | 1 | BG color 1–3, otherwise OBJ + +[This test ROM](https://github.com/alloncm/MagenTests) can be used to observe the above. + +::: warning + +Keep in mind that: +* OAM Attributes bit 7 will grant OBJ priority when **clear**, not when **set**. +* Priority between all OBJs is resolved **before** priority with the BG layer is considered. + Please refer [to this page](<#Drawing priority>) for more details. + +::: + ## Background (BG) The [SCY and SCX](<#FF42–FF43 — SCY, SCX: Viewport Y position, X position>)