我向你介紹:GB Interceptor。它是一個(gè)適配器,連接在未修改的Game Boy和盒式磁帶之間,并通過USB提供游戲視頻流。
B站鏈接:https://www.bilibili.com/video/BV1h8411K7c4/
YouTube鏈接:https://www.youtube.com/watch?v=6mOJtrFnawk
上面的視頻應(yīng)該能讓你很好地了解它的功能、工作原理以及它的局限性。本文將詳細(xì)介紹其工作原理的技術(shù)細(xì)節(jié)。如果您對(duì)如何訂購(gòu)和構(gòu)建自己的GB Interceptor感興趣,請(qǐng)查看github(https://github.com/Staacks/gbinterceptor)和訪問訂購(gòu)和構(gòu)建視頻(https://www.youtube.com/watch?v=Lg92tVkEE98)。
我們?yōu)槭裁葱枰@個(gè)?
解釋我為什么開發(fā)和建造GB Interceptor的最好方法是解釋我試圖用它解決的問題。幾個(gè)月前,一位俄羅斯方塊愛好者就這個(gè)問題與我取得了聯(lián)系:一場(chǎng)在線俄羅斯方塊錦標(biāo)賽,參賽者在比賽中展示了他們的游戲。
今天,Game Boy的視頻流并沒有什么異常。模擬器可以很容易地做到這一點(diǎn),而現(xiàn)代Game Boy變體,如Analogue Pocket,提供可以捕獲的HDMI輸出。還有一些MOD可以將HDMI添加到Game Boy的原始硬件中,因此從Game Boys獲取視頻流是一個(gè)長(zhǎng)期以來需要被解決的挑戰(zhàn)。
在俄羅斯方塊比賽中做這件事的一個(gè)不同尋常的細(xì)節(jié)是,玩家必須依靠他們?cè)趥€(gè)人Game Boys上訓(xùn)練的肌肉記憶。將他們換成不熟悉的現(xiàn)代設(shè)備或模擬器將嚴(yán)重阻礙他們的競(jìng)爭(zhēng)能力。此外,你可以想象,一場(chǎng)比賽要求每個(gè)參賽者先把他們心愛的Game Boys通過mod做視頻流支持改造,這樣的比賽不會(huì)受到歡迎。
因此,我們需要一種方法,在不修改正在播放的游戲的情況下,從未經(jīng)修改的Game Boys獲取視頻。理想的形式是任何人都可以使用,而無需復(fù)雜的軟件或額外的硬件,如HDMI抓取器。
工作原理的基本概念
最后,在Game Boy上沒有mod的情況下,唯一可以訪問帶有游戲數(shù)據(jù)的連接器是cartridge插槽。畢竟,整個(gè)游戲數(shù)據(jù)都要經(jīng)過那里。因此,我們的想法是創(chuàng)建一個(gè)適配器,將cartridge直接連接到Game Boy,并且只添加攔截傳輸數(shù)據(jù)副本的功能。
GB Interceptor連接到筆記本電腦,該筆記本電腦以VLC顯示其視頻流。
然而,這意味著我們無法隨機(jī)訪問感興趣的數(shù)據(jù),也無法在RAM中看到Game Boy的CPU從cartridge的原始指令中合成的數(shù)據(jù)。特別是,我們看不到視頻RAM,但這非常不錯(cuò),因?yàn)樗鼘谄聊簧侠L制圖像所需的所有內(nèi)容。我們需要?jiǎng)?chuàng)建自己的VRAM副本。
為此,我必須編寫一個(gè)仿真器,從cartridge總線向其提供數(shù)據(jù)。為此,我使用了rp2040(樹莓派Pi Pico的微控制器),并將其內(nèi)核劃分為Game Boy的兩個(gè)主要處理部分。一個(gè)內(nèi)核模擬CPU以重新創(chuàng)建VRAM副本,另一個(gè)內(nèi)核則模擬Game Boy的圖形單元PPU4。
CPU仿真實(shí)際上是這里最棘手的部分,因?yàn)樗仨毟弦源蠹s1MHz的速率推出事件的內(nèi)存總線。如果PPU仿真落后,它會(huì)導(dǎo)致像flicker一樣的短暫glitch,但如果CPU仿真落后,最終會(huì)錯(cuò)過內(nèi)存總線上的事件。不僅RAM的模擬副本可能永遠(yuǎn)不同步,而且仿真器甚至無法解釋后續(xù)指令??偩€上的事件并不總是下一條指令,因?yàn)镚ame Boy的CPU可能需要幾個(gè)周期來執(zhí)行某些指令,而其他指令則在一個(gè)周期內(nèi)完成。因此,仿真器必須跟蹤在某個(gè)特定指令之后,在某個(gè)事件再次被視為指令之前,需要忽略多少個(gè)周期。如果我們只錯(cuò)過了其中的一個(gè),就幾乎不可能再次得到正確的答案。
這加上在32位CPU上模擬8位CPU的開銷,使rp2040有必要從其默認(rèn)的125 MHz超頻到225 MHz。rp2040通??梢暂p松處理這個(gè)問題,但我還是很想看看是否可以提高我的代碼的效率。
由于PPU仿真并不是那么關(guān)鍵,實(shí)際上在Game Boy的vblank期間,當(dāng)沒有繪制圖像時(shí),它會(huì)定期獲得一些空閑時(shí)間,因此它還可以處理USB通信。
硬件
實(shí)現(xiàn)這一功能的實(shí)際硬件是一個(gè)樹莓派Pico,帶有一些總線收發(fā)器,將其GPIO端口連接到cartridge總線。從該總線的32個(gè)引腳中,兩個(gè)用于+5V和接地,一個(gè)用于模擬音頻,一個(gè)用來控制Game Boy的復(fù)位狀態(tài)。其他28個(gè)引腳連接到rp2040,因此rp2040可以訪問16個(gè)地址引腳、8個(gè)數(shù)據(jù)引腳和4個(gè)總線控制引腳時(shí)鐘、讀取、寫入和芯片選擇。由于這些都使用5V邏輯,我使用的WiFi Game Boy cartridge(https://there.oughta.be/a/wifi-game-boy-cartridge)已經(jīng)支持將信號(hào)轉(zhuǎn)換為rp2040的3.3V。
還有兩個(gè)GPIO未使用。一個(gè)觀察用于+5V線上的電壓,以檢查Game Boy是否打開,另一個(gè)控制狀態(tài)LED并讀取模式按鈕。
GB Interceptor的PCB。
其余的cartridge基于樹莓派rp2040的最小硬件設(shè)計(jì)示例(https://datasheets.raspberrypi.com/rp2040/hardware-design-with-rp2040.pdf)。這包括一個(gè)振蕩器、閃存、一個(gè)電壓轉(zhuǎn)換器和一個(gè)USB端口,我用Type C替換了它。
一個(gè)樹莓派 Pico連接到Game Boy cartridge的內(nèi)存總線的轉(zhuǎn)換器就差不多了。原理圖和PCB設(shè)計(jì)可以在項(xiàng)目的github存儲(chǔ)庫(kù)(https://github.com/Staacks/gbinterceptor/tree/main/pcb)中找到。
實(shí)現(xiàn)
真正讓GB Interceptor工作的是它的軟件,當(dāng)然也可以在github(https://github.com/Staacks/gbinterceptor/tree/main/firmware)上找到。在下文中,我將介紹下它的一些細(xì)節(jié)。
USB video class
GB Interceptor使用TinyUSB的USB video class實(shí)現(xiàn)流式傳輸生成的圖像,因此理論上不需要驅(qū)動(dòng)程序,它應(yīng)該只顯示為網(wǎng)絡(luò)攝像頭。從理論上講。不幸的是,這只能在Linux上按預(yù)期工作,我可以在VLC、OBS、Zoom或ffmpeg中直接使用GB Interceptor。在Windows和Android上,許多應(yīng)用程序似乎在視頻流的格式上有問題。例如,在Windows上,VLC(盡管在Linux上工作)提示沒有找到合適的格式,而OBS在不需要任何設(shè)置或驅(qū)動(dòng)程序的情況下工作得很好。在Windows上,這是一個(gè)好消息,因?yàn)槟梢允褂肙BS作為虛擬網(wǎng)絡(luò)攝像頭,將GB Interceptor流轉(zhuǎn)發(fā)給任何對(duì)格式挑剔的軟件。在github上可以找到一個(gè)測(cè)試過的主機(jī)軟件列表(https://github.com/Staacks/gbinterceptor/wiki/Host-software-compatibility)。
不幸的是,在撰寫本文時(shí),我無法在MacOS上獲得任何視頻,我還不知道為什么。由于某些原因,它甚至不會(huì)觸發(fā)TinyUSB來啟用視頻流,因此我并不完全相信它是這種格式。記住,我還沒有在MacOS上做過很多測(cè)試,TinyUSB中的 video class實(shí)現(xiàn)是最近才開始的,也是實(shí)驗(yàn)性的,我希望將來能解決這個(gè)問題。即使我不能讓 video class在這里工作,也應(yīng)該可以通過USB總線上的UART傳送圖像,并使用簡(jiǎn)單的Python腳本將其轉(zhuǎn)換為系統(tǒng)上的視頻流。您可以在github上查看此Issue的當(dāng)前狀態(tài)(https://github.com/Staacks/gbinterceptor/issues/1)。
那么,這種不尋常的格式是什么?很明顯,這是從Game Boy 160x144像素的分辨率開始的,我可以想象,這可能會(huì)讓一些期待現(xiàn)代1080p流的軟件感到驚訝。但是,當(dāng)我們研究rp2040的全速USB端口及其對(duì)TinyUSB實(shí)現(xiàn)的同步傳輸?shù)挠绊懰a(chǎn)生的限制時(shí),它會(huì)變得更加復(fù)雜。這種組合意味著此端點(diǎn)的最大緩沖區(qū)大小為1023字節(jié),由于同步傳輸每1ms發(fā)生一次,因此每秒可獲得1023000字節(jié)。
如果我們只看Game Boy的原始圖像,這就綽綽有余了。Game Boy的“顏色深度”為2位,因此一個(gè)圖像幀為5760字節(jié)。大約每秒60幀,我們只需要345600字節(jié),這就是為什么如果其他所有的都失敗了,我認(rèn)為自定義UART協(xié)議是MacOS上一個(gè)有趣的替代方案。
然而,我們不希望需要驅(qū)動(dòng)程序或其他軟件。我們想要的是一種能正常工作的格式,但遺憾的是,目前還沒有被廣泛接受的2位彩色格式。相反,有很多壓縮格式,我們沒有足夠的計(jì)算能力,還有一些被認(rèn)為是廣泛支持的未壓縮彩色格式,其中大多數(shù)使用16位像素。相反,我們使用了一種據(jù)稱也被廣泛支持的稍微更高效的格式:每像素12位的NV12。12位由亮度(灰度亮度)的每像素8位和顏色信息的四個(gè)像素共享的16位(因此每像素多4位)組成。
好消息是,整個(gè)幀的顏色數(shù)據(jù)存儲(chǔ)在最后,因此我們可以將其設(shè)置為灰色或綠色,并可以忽略它。事實(shí)上,我們可以將之前的數(shù)據(jù)視為具有8位灰度數(shù)據(jù)的簡(jiǎn)單160x144像素緩沖區(qū),這對(duì)于我們的目的來說或多或少是理想的。
當(dāng)然,壞消息是,它所占用的數(shù)據(jù)仍然是原始2位圖像所需數(shù)據(jù)的6倍。我們的每秒1023000字節(jié)現(xiàn)在限制在29fps。
因此,總的來說,我們有一個(gè)分辨率為160x144的29fps NV12流。并非所有這些視頻會(huì)議工具都支持。
順便說一句,雖然GB Interceptor因此只推出了29fps,但它仍然以60fps的速度在內(nèi)部工作,并混合這些幀以模擬舊LCD的延遲。它只是在USB總線調(diào)用時(shí)推出最新的混合幀。
可編程IO
現(xiàn)在,在我解釋了如何從GB Interceptor中獲取結(jié)果之后,讓我們來談?wù)劻硪欢耍喝绾螌artridge總線上的通信連接到rp2040。
當(dāng)我試著在我的WiFi Game Boy Cartridge用ESP8266監(jiān)聽一個(gè)事件時(shí),非常痛苦。中斷太慢,而讓CPU 輪詢的方式觀察時(shí)鐘線也不可行。rp2040有一個(gè)訣竅:可編程IO。這些是簡(jiǎn)單的狀態(tài)機(jī),可以直接訪問GPIO引腳以及CPU的FIFO緩沖區(qū)。
我們需要做的就是等待時(shí)鐘線變低,然后同時(shí)讀取連接到Game Boy內(nèi)存總線的剩余27個(gè)GPIO引腳,并將結(jié)果寫入FIFO。為此,我們只需要一個(gè)PIO,它只執(zhí)行四條指令:
wait1pin28;WaitforCLKtogohigh
wait 0 pin 28 ;Wait for falling flank of CLK
mov isr pins ;Read all GPIO pins to the input shift register
push;PushtheISRtotheFIFO
這樣,CPU很方便就可以從FIFO中提取這些事件中的一個(gè),并將其打包成一個(gè)32位整數(shù)。
仿真器部分
現(xiàn)在我們看看如何處理這些事件。我希望你對(duì)Game Boy的工作原理有一個(gè)基本的了解。對(duì)于那些不熟悉Game Boy開發(fā)的人,我總是推薦Michael Steil的《Ultimate Game Boy Talk》(https://www.youtube.com/watch?v=HyzD8pNlpwI)。
如上所述,基本思想是rp2040的一個(gè)內(nèi)核解釋傳入的總線事件,使其遵循與Game Boy的CPU相同的指令。也就是說,它模擬Game Boy CPU,以便重新創(chuàng)建VRAM(和OAM)的精確副本。然后,第二個(gè)內(nèi)核充當(dāng)PPU,并從VRAM副本中渲染圖像。這主要是一個(gè)基本Game Boy模擬器的實(shí)現(xiàn),但我想談?wù)劊ɑ驅(qū)懸幌拢┮恍┎煌帯?/p>
條件跳轉(zhuǎn)和IO
首先,在這個(gè)場(chǎng)景中,有幾個(gè)事情變得簡(jiǎn)單得多。想想程序計(jì)數(shù)器和條件跳轉(zhuǎn)。我們不必執(zhí)行這些。真正的Game Boy無論如何都會(huì)獲取下一條指令。無論它是遞增PC的下一條指令,還是跳到完全不同的地址,都無關(guān)緊要。真正的Game Boy將獲取下一條指令,我們不必?fù)?dān)心指令來自何處。
這解決了一個(gè)看似最大的問題:我們看不到任何硬件I/O寄存器。特別是,我們看不到來自游戲板的輸入!如果我們看不到玩家的輸入,我們應(yīng)該如何模擬游戲?好吧,幾乎所有存在的代碼都會(huì)比較游戲板輸入以檢查按下了哪個(gè)按鈕,并對(duì)按鈕觸發(fā)的代碼進(jìn)行條件跳轉(zhuǎn)。我們的模擬器將簡(jiǎn)單地遵循這些相同的指令,而不必關(guān)心它是否由按下按鈕觸發(fā)。
你可以說GB Interceptor是一個(gè)模擬器。
只有當(dāng)來自I/O寄存器的數(shù)據(jù)最終到達(dá)VRAM時(shí),這才成為問題。想象一下,將gamepad的值添加到一個(gè)基地址,以計(jì)算顯示D-Pad當(dāng)前狀態(tài)的圖像的tile
索引。CPU將獲得獲取游戲板寄存器值的指令,并向其添加一個(gè)數(shù)字,我們的仿真器將不知道該操作的正確結(jié)果。然后將此結(jié)果寫入VRAM,我們不知道該位置中的內(nèi)容。
然而,這些應(yīng)僅適用于較小的視覺差異。我不知道有哪個(gè)例子是用gamepad I/O完成的,但我有一個(gè)DIV寄存器的例子。在俄羅斯方塊中,它被用作隨機(jī)數(shù)的來源,大多數(shù)時(shí)候它通過條件跳轉(zhuǎn)來分支代碼,以選擇下一個(gè)不同的塊,或者在游戲模式B中生成初始的垃圾塊堆。這也決定了模式B中垃圾堆的一個(gè)塊是空的還是滿的,所以我們也得到了相同的垃圾堆布局。但這些垃圾塊也有一種隨機(jī)的視覺樣式,它不是基于分支代碼,而是添加到基本tile索引中的隨機(jī)數(shù)。
結(jié)果是,我們?cè)贕B Interceptor上看到了相同的垃圾堆棧布局,但各個(gè)塊的外觀不同。這是無害的,只有當(dāng)你將圖像與Game Boys屏幕進(jìn)行比較時(shí),你才會(huì)注意到。
左:原始Game Boy屏幕在俄羅斯方塊模式B下的圖像。右:與GB Interceptor渲染的場(chǎng)景相同。方塊的布局是相同的,但各個(gè)塊有不同的設(shè)計(jì)。
只有當(dāng)整個(gè)準(zhǔn)備好的數(shù)據(jù)流從一個(gè)I/O寄存器寫入VRAM時(shí),我們才會(huì)遇到真正的麻煩。我所知道的(而且我能想到的)唯一的例子是連接電纜。在這里,我們可以看到B模式方塊的相同示例,但在俄羅斯方塊的雙人模式中。問題是,兩個(gè)玩家應(yīng)該擁有相同的方塊樣式。因此,首先啟動(dòng)游戲的Game Boy將生成方塊并通過鏈接電纜將其發(fā)送給第二個(gè)堆棧。第二個(gè)將數(shù)據(jù)直接寫入VRAM,沒有任何檢查或條件跳轉(zhuǎn),我們看不到任何內(nèi)容。
俄羅斯方塊為雙人模式。左:首先啟動(dòng)帶有GB Interceptor的Game Boy Color,方塊呈現(xiàn)為1人模式。右:另一個(gè)Game Boy首先啟動(dòng),我們無法看到方塊,因?yàn)樗峭ㄟ^鏈接電纜接收的。
因此,在雙人俄羅斯方塊中,如果是在Game Boy中首先開始游戲,GB Interceptor可以正常工作(除了各個(gè)區(qū)塊的不同視覺風(fēng)格),但如果是在第二個(gè)Game Boy中,它會(huì)產(chǎn)生無法使用的輸出。
時(shí)鐘、DIV寄存器和暫停指令
說到DIV寄存器,這實(shí)際上是一個(gè)我們可以模擬的I/O寄存器。由于我們從Game Boy獲得了準(zhǔn)確的時(shí)鐘,我們可以計(jì)算模擬寄存器與真實(shí)寄存器同步,而不會(huì)有任何偏離的危險(xiǎn)。只有兩個(gè)問題:
1.初始值是未知的-至少對(duì)我來說是未知的。當(dāng)執(zhí)行盒帶中的代碼時(shí),DIV寄存器的狀態(tài)取決于Game Boy模型,在某些情況下,它還取決于該模型引導(dǎo)序列期間的用戶交互。例如,如果在引導(dǎo)序列期間更改Game Boy color的顏色模式,DIV寄存器將在開始時(shí)具有不同的值。我不確定Interceptor在引導(dǎo)序列期間是否在總線上看到足夠的動(dòng)作來彌補(bǔ)這一點(diǎn),但我也不完全排除這種情況。
2.當(dāng)Game Boy進(jìn)入暫停狀態(tài)時(shí),我們失去了參考時(shí)鐘,對(duì)于大多數(shù)游戲,每幀至少發(fā)生一次。在這里,rp2040的時(shí)鐘必須精確接管,如果我們有更多的計(jì)算空間,這應(yīng)該是可能的。(比如,如果有人可以優(yōu)化我的代碼)
問題是,我們實(shí)際上測(cè)量了在實(shí)際游戲開始之前的引導(dǎo)序列中,每個(gè)Game Boy時(shí)鐘周期出現(xiàn)了多少rp2040時(shí)鐘周期。在這里,我們可以觀察數(shù)千個(gè)周期,并且應(yīng)該能夠從rp2040中獲得非常精確的替代時(shí)鐘。不幸的是,出于性能原因,我只使用兩個(gè)時(shí)鐘的整數(shù)比,通常為每個(gè)Game Boy時(shí)鐘對(duì)應(yīng)225 rp2040時(shí)鐘。這意味著在暫停狀態(tài)期間,舍入誤差將導(dǎo)致每100個(gè)周期大約一個(gè)周期的誤差。
所以,也許我們可以進(jìn)行分?jǐn)?shù)時(shí)鐘計(jì)數(shù),但目前,因?yàn)樗挥绊慸iv寄存器,無論如何我都無法正確初始化,所以這沒有實(shí)現(xiàn)。
同步CPU和PPU
當(dāng)我們正在將模擬器與真實(shí)Game Boy同步時(shí)……我們當(dāng)然也需要將PPU與真實(shí)GameBoy同步。否則,任何需要更改VRAM中間幀的效果都會(huì)導(dǎo)致故障,至少在VRAM中隨機(jī)更新數(shù)據(jù)時(shí),我們會(huì)看到一些撕裂效果。
問題是,在內(nèi)存總線上找不到PPU的蹤跡。我們必須通過游戲的行為來推斷PPU的狀態(tài),這也必須與PPU同步——至少要知道它何時(shí)可以寫入VRAM。這里的大問題是,游戲可以使用許多不同的方式來做到這一點(diǎn)。
最常見的方法是vsync中斷。大多數(shù)游戲只是讓Game Boy在達(dá)到vsync時(shí)觸發(fā)一個(gè)中斷,我們可以看到這個(gè)中斷的代碼何時(shí)被執(zhí)行,因此我們可以簡(jiǎn)單地調(diào)整我們自己的仿真PPU的定時(shí),以便在同一時(shí)刻進(jìn)入vsync。
不幸的是,有很多其他選擇可以做到這一點(diǎn)。對(duì)于需要擠出更多VRAM訪問權(quán)限的游戲(例如在Donkey Kong Land中實(shí)現(xiàn)),另一個(gè)常見的方法是以緊密循環(huán)的方式讀取LY寄存器,并定期將其與特定的行號(hào)進(jìn)行比較。條件跳轉(zhuǎn)返回到LY讀取,直到到達(dá)正確的行,并且代碼僅超出條件跳轉(zhuǎn)。幸運(yùn)的是,開發(fā)人員可以在未到達(dá)時(shí)通過jump指令來節(jié)省幾個(gè)周期,所以許多游戲都是這樣做的,這允許在Interceptor中簡(jiǎn)單檢測(cè)這些緊密的循環(huán)。
然而,會(huì)有不同方法的游戲(比如我的Wifi cartridge),在檢測(cè)到這些其他方法之前,GB Interceptor的輸出會(huì)出現(xiàn)故障。
檢測(cè)中斷
哦,雖然中斷是同步PPU的福音,但它們最初并不容易檢測(cè)。我們需要跟蹤每一條指令,以及Game Boy為每條指令需要多少周期,以確定內(nèi)存總線上的哪個(gè)事件將是下一條指令。Game Boy在執(zhí)行過程中跳到另一個(gè)點(diǎn),并花費(fèi)幾個(gè)額外的周期來執(zhí)行,這在這里并沒有什么幫助。
看看《The Legend of Zelda - Link’s Awakening》在原版Game Boy上的第一次vsync中斷:
AddressDataInstruction
01a2 fb EI
01a3 c3 JP a16
01a4 bd
01a5 03
81a5 71
03bd 3e IRQ
82bd 01
82bd 01
dffe 81
dffd a5
0040 c3 JP a16
0041 25
0042 05
804224
當(dāng)忽略中斷時(shí),我們會(huì)錯(cuò)誤地將第7行中的0x3e解釋為操作碼。要或多或少地確定我們正在看到中斷,唯一的方法是實(shí)現(xiàn)GB Interceptor,使其在當(dāng)前事件被誤解為指令之前讀取幾個(gè)周期以識(shí)別中斷,而實(shí)際上它只是內(nèi)存總線上的垃圾,而CPU需要一點(diǎn)時(shí)間進(jìn)入中斷。
幸運(yùn)的是,Game Boy在中斷時(shí)跳轉(zhuǎn)到幾個(gè)固定地址,所以我們要注意這些地址。但由于這些地址理論上也可以從常規(guī)代碼中調(diào)用,因此我們混合了更多的指示符,特別是堆棧指針的行為。在中斷調(diào)用期間,當(dāng)前PC被推到堆棧上,因此SP寄存器被遞減兩次,Game Boy寫入兩個(gè)遞減的地址。通常這些地址并不指向?qū)儆诤袔У牡刂?,但這些地址在內(nèi)存總線上仍然可見,因此這增加了我們檢測(cè)中斷的信心。
唯一的問題是,Game Boy實(shí)際上不需要一直這樣做,也不需要在內(nèi)存總線上顯示SP地址,因?yàn)闆]有任何游戲cartridge關(guān)心這些操作。因此,我們可以在這里看到不同設(shè)備之間的一些差異并不奇怪。以下僅為原始GameBoy(DMG)、GameBoy Color和Analogue Pocket的中斷調(diào)用:
DMG GBC Pocket
Address Data Address Data Address Data
03bd 3e 03bd 3e 03bd 3e
82bd 01 83be 00 dfff 00
82bd 01 dfff 00 dffe 01
dffe 81 dffe 80 dffd 9b
dffd a5 dffd 00 0040 c3
0040 c3 0040 c3 0040 c3 < Next instruction
如果我們仔細(xì)觀察,我們會(huì)發(fā)現(xiàn)一些細(xì)微的差異:DMG在遞減SP地址之前也會(huì)顯示SP地址,GBC只顯示它實(shí)際寫入的兩個(gè)遞減地址,Pocket則會(huì)提前一個(gè)周期執(zhí)行此操作。考慮到所有這些情況,當(dāng)然會(huì)降低中斷檢測(cè)的可靠性,并且目前它無法與Pocket的變體一起正常工作。
意圖和構(gòu)建說明
我認(rèn)為這些是實(shí)現(xiàn)過程中最有趣的部分。如果你已經(jīng)讀到了這里,那你是一個(gè)真正的8bit極客!
如果你想了解更多細(xì)節(jié),你現(xiàn)在必須深入github上的代碼(https://github.com/Staacks/gbinterceptor),在那里你還可以找到硬件設(shè)計(jì)文件和案例材料。我希望在代碼和硬件設(shè)計(jì)方面都會(huì)有一些社區(qū)貢獻(xiàn),所以如果這篇文章發(fā)表后幾個(gè)月過去了,這也將主要發(fā)生在github上。
如果你想建立自己的GB Interceptor,你也應(yīng)該觀看訂購(gòu)和構(gòu)建視頻。
B站鏈接:https://www.bilibili.com/video/BV17G4y1y7Yg/
Youtube鏈接:https://youtu.be/Lg92tVkEE98
我希望你喜歡這個(gè)項(xiàng)目!
致謝
如果沒有許多人在我之前研究、測(cè)試和推動(dòng)Game Boy,并且(最重要的是,也是我自己寫這些文章的原因)記錄了他們的工作,這個(gè)項(xiàng)目就不可能存在。以下是我最重要的一些資源:
1,gbdev.io(https://gbdev.io/),尤其是它的Pan Docs是我了解Game Boy工作原理的主要來源。
2,在Joonas Javanainen的網(wǎng)頁(yè)上(https://gekkio.fi/)可以找到許多硬件細(xì)節(jié)和一些復(fù)雜的細(xì)節(jié)。
3,雖然有很多網(wǎng)站都有Game Boy的opcode表,但我發(fā)現(xiàn)Megan Sullivan的網(wǎng)站是最方便的(https://meganesulli.com/blog/game-boy-opcodes/),我一直都在訪問它。
審核編輯 :李倩
-
GB
+關(guān)注
關(guān)注
0文章
102瀏覽量
45397 -
適配器
+關(guān)注
關(guān)注
8文章
1957瀏覽量
68075 -
模擬器
+關(guān)注
關(guān)注
2文章
877瀏覽量
43260
原文標(biāo)題:GB Interceptor:值得擁有的Game Boy 捕捉器
文章出處:【微信號(hào):Arm軟件開發(fā)者,微信公眾號(hào):Arm軟件開發(fā)者】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論