• 首发公号:Rand_cs,求关注支持

Mapper

mapper,这个概念来源于 memory mapping,又叫做 Memory Management Circuit,它是解决地址映射的一种电路,简单来说就是决定物理内存如何映射到 CPU 或者 PPU 的地址空间

mapper 可以用来支持增加卡带的 RAM 甚至支持额外的音频通道,但更一般的目的就是控制物理内存到地址空间的映射,突破游戏 40KB 的限制

为什么说是 40KB 的限制,因为早期一般的游戏最大就是

16

×

2

=

32

K

B

16 \times 2 = 32KB

16×2=32KB 的 PRG,以及

8

K

B

8KB

8KB 的 CHR,加起来就是 40KB,而更复杂的 mapper 硬件可以使得游戏突破这个限制。

mapper 的种类太多太多,不同 NES 版本的 mapper 也有所不同,各种杂七杂八的加起来有好几百个,不过这里我们挑几个常见的游戏使用的 mapper 来说明。

NROM

mapper 000,排在第零个,最简单的一种 mapper,像超级马里奥就使用的是 NROM。NROM 也分种类,NROM-128 的存放 PRG 的 ROM(后面简称 PRG-ROM) 只有 16KB,而 NROM-256 有 256KB。

  • PRG 的第一个 bank(16KB) 映射到 0x8000-0xBFFF,最后一个 bank 映射到 0xC000-0xFFFF。
  • PPU 的 VRAM 开头 8KB 映射到 CHR,也就是 PatternTable

这就是最简单的 NROM,与我们前面讲述 CPU 和 PPU 时一致,当时不是说不讨论复杂情况吗,其实就是在用 NROM 来举例子。

MMC1

mapper 001,使用 MMC1 的游戏有双截龙,恶魔城等,来看其 bank 和映射关系:

  • CPU 0x6000-0x7FFF 映射到 8KB 的 PRG RAM bank,这是可选的
  • CPU 0x8000-0xBFFF 映射到 PRG ROM 的一个 bank,这个 bank 要么是可切换的要么固定为第一个
  • CPU 0xC000-0xFFFF 映射到 PRG ROM 的一个 bank,这个 bank 要么是可切换的要么固定为最后一个
  • PPU 0x0000-0x0FFF 映射到一个 4KB 可切换的 CHR bank
  • PPU 0x1000-0x1FFF 映射到一个 4KB 可切换的 CHR bank

MMC1 这个 mapper 就高级多了,它有一系列的端口寄存器,我们可以操作这些端口来配置 MMC1。来简单看看有哪些端口:

Control

0x8000-0x9FFF,向这部分地址空间任意一地址写入数据都会写入寄存器 0,有朋友可能会有疑问,这部分地址空间不是映射到 PRG 吗,怎么又与一个寄存器相连了?虽然我不清楚具体电路,但是不难推断这是没问题的,PRG 程序代码是只读的,但这里是写入。很多地方端口相同但读写不同的情况下,映射可能也有所不同,比如前面我讲述串口时其中很多的端口就是这样子的,所以这里是不冲突没有问题的,接着来看 0x8000-0x9FFF 这部分地址空间表示的端口:

  • bit0:0 表示设置为 horizontal 镜像,1 表示设置为 vertical 镜像
  • bit1:0 表示设置为 single 镜像
  • bit2:0 表示 bank switching 发生在 0xC000-0xFFFF,1 表示 bank switching 发生在 0x8000-0xBFFF
  • bit3:0 表示 bank 大小为 32KB,也就是一次性交换 32KB(0x8000-0xFFFF),1 表示按照 bit2 的设置来交换
  • bit4:0 表示一次性交换 8KB 的 CHR,1 表示交换 2 个分开的 4KB CHR banks
  • bit7:置 1 表示清空这个寄存器

这里说明一下这个 bank 没有特定的大小,都是根据硬件设置而来,而且 PRG CHR 等都用 bank 来做交换的基本单位,可能有些混淆这里说明一下。

CHR bank 0

0xA000-0xBFFF,这部分地址空间连接到了另一个端口,向这个端口写入 CHR bank number 可以选择 CHR 的那一块 bank 映射到 PPU 的 0x0000-0x0FFF,如果 CHR bank 的大小在 Control 寄存器设置为 8KB,那么就会选择一个 bank 映射到 0x0000-0x1FFF。

CHR bank 1

0xC000-0xDFFF,道理同上,向这部分地址空间任意一地址写入 CHR bank number 可以将选择的 bank 映射到 PPU 的 0x1000-0x1FFF

PRG bank

0xE000-0xFFFF,同理,写入 PRG bank number 来选取一个 PRG bank,至于这个 bank 多大,映射到 CPU 地址空间中的 0x8000 处还是 0xC000 处要视 Control 寄存器的设置决定。

UNROM

下面接着来看 mapper 002,使用这个 mapper 的著名游戏有魂斗罗,洛克人等等。UNROM 比较厉害,最高可支持 4M 的 PRG,要知道 M 这个单位在那时对游戏来说是个很大的单位了。虽然 UNROM 可以支持 4M 的大容量 PRG,但其实 UNROM 还没 MMC1 复杂,来看其 banks 的规划(映射关系)

  • CPU 0x8000-0xBFFF,16KB 可切换的 PRG ROM bank
  • CPU 0xCFFF-0xFFFF,16KB 固定的 PRG ROM bank,这部分地址空间固定映射到最后一块 PRG bank
  • CHR 容量 8KB,就映射到 PPU 开头的 2KB

就只有这些,还是挺简单的,UNROM 只支持 2 种镜像垂直和水平,而且设置方式是制作卡带时就焊接好的(solder pad,如果我没理解错的话),它有一个 bank 选择寄存器,向 0x8000-0xFFFF 这部分地址空间的任一地址写入 PRG bank number 即可选择一 PRG bank 映射到 0x8000

不知道细心看的朋友有没有一个疑惑,像魂斗罗,洛克人这类的游戏都是打游戏,特别是魂斗罗,我之前讲述的文章里面包括了很多魂斗罗的例子,CHR 容量 8K 够用吗?显然所有的图案表加起来肯定超过 2 个也就 8KB。

像魂斗罗这类的游戏有些特殊啊,它们没有 CHR ROM,有的是 CHR RAM(可以将 PPU 的开头 8KB 视作 CHR RAM),通过 FCEUX 可以知道魂斗罗的 PRG ROM 为 128KB,没有 CHR ROM,它的 PatternTable 就在 PRG 里面,是游戏运行期间 CPU 控制通过 PPU 端口将 PatternTable 从 PRG 复制到 CHR RAM

CNROM

mapper003 CNROM,这个 mapper 也比较简单,较为出名的游戏有勇者斗恶龙,高桥名人的冒险岛等等,前面关于 mapper 都写得差不多,应该有这概念了,后面我就简述了

banks

  • PRG ROM 只有 16KB 或者 32KB,跟 mapper 000 一样,不可 bank switching
  • CHR ROM 较大,up to 2M,CHR bank size 为 8KB,CHR banks 之间切换,映射到 PPU 的 0x0000-0x1FFF

register

register 只有一个,类似 UNROM,向 0x8000-0xFFFF 这部分地址空间的任一地址写入 CHR bank number 即可选择一 CHR bank 映射到 0x0000-0x1FFF

MMC3

mapper004 MMC3,使用 MMC3 的游戏很多很多,较为出名的我比较喜欢的有热血系列比如说热血格斗传说,热血物语等等。

Banks

MMC3 对于 banks 的规划就很精细,很多,意思也大同小异,我就不重复写了,直接看 wiki 上的资料:

  • CPU 0x6000-0x7FFF: 8 KB PRG RAM bank (optional)
  • CPU 0x8000-0x9FFF (or 0xC000-0xDFFF): 8 KB switchable PRG ROM bank
  • CPU 0xA000-0xBFFF: 8 KB switchable PRG ROM bank
  • CPU 0xC000-0xDFFF (or 0x8000-0x9FFF): 8 KB PRG ROM bank, fixed to the second-last bank
  • CPU 0xE000-0xFFFF: 8 KB PRG ROM bank, fixed to the last bank
  • PPU 0x0000-0x07FF (or 0x1000-0x17FF): 2 KB switchable CHR bank
  • PPU 0x0800-0x0FFF (or 0x1800-0x1FFF): 2 KB switchable CHR bank
  • PPU 0x1000-0x13FF (or 0x0000-0x03FF): 1 KB switchable CHR bank
  • PPU 0x1400-0x17FF (or 0x0400-0x07FF): 1 KB switchable CHR bank
  • PPU 0x1800-0x1BFF (or 0x0800-0x0BFF): 1 KB switchable CHR bank
  • PPU 0x1C00-0x1FFF (or 0x0C00-0x0FFF): 1 KB switchable CHR bank

Registers

I Bank Select

0x8000-0x9FFE,这之间的偶数地址连接到 Bank Select 寄存器,顾名思义,写入这个寄存器可以控制映射方式,具体如下,我还是直接贴 wiki 的资料:

只用关注前面几位就行,不同的 RRR 映射方式不同,上图写得很清楚了,我就不再多做解释。

II Bank Data

0x8001-0x9FFF,这之间的奇数地址连接到 Bank Data 寄存器,就是向这个寄存器写入 bank number 来选取一个 bank 然后然后按照 Bank Select 寄存器中的方式映射。

III Mirroring

0xA000-0xBFFE,这之间的偶数地址连接到 Mirroring 寄存器,这个寄存器的 bit0 为 0 的话表示 垂直镜像,为 1 的话表示 水平镜像

IV PRG RAM

0xA001-0xBFFF 奇数部分地址连接到此寄存器,这个寄存器的 bit7 可以使能 PRG RAM(0x6000-0x7FFF),简而言之可以允许你存档了。

MMC3 是允许产生 IRQ 中断的,前面说了 CPU 有三种中断,RESET,NMI,IRQ,前两种的来源都说过,而 IRQ 的中断源就来自这的 mapper,与之相关的有 4 个寄存器:

V IRQ Latch

0xC000-0xDFFE 偶数部分,这个寄存器里面存放着 IRQ 计数器开始计数的数字

VI IRQ reload

0xC000-0xDFFF 奇数部分,向这个寄存器写入任何数据都会使得 IRQ Latch 中存放的数重新加载到内部的计数器

VII IRQ disable

0xE000-0xFFFE 偶数部分,向这个寄存器写入任何数据都会禁止 IRQ 产生

VIII IRQ enable

0xE001-0xFFFF 奇数部分,向这个寄存器写入任何数据都会使能 IRQ 产生

至于产生中断后干什么,那就看 IRQ 的中断处理程序,mapper 产生的 IRQ 可以用来屏幕分割,之前不是说了可以利用 sprite 0 hit 来判断 scanline 的渲染情况,渲染到那儿了,这里也可以用 IRQ,这也是 IRQ 最主要的作用。

好了,上述就是 mapper 的一些内容,更多详情有兴趣的还是请看 wiki 上的资料。到此,结合之前讲述的所有内容,应该对 NES 的 PRG,CHR,mapper,CPU,PPU 之间是如何配合使得游戏运行起来有一个大致的认识了。

本文就到这里了,有什么问题还请批评指正,也欢迎大家来同我交流学习。

  • 首发公号:Rand_cs,求关注

童年神机小霸王(七) Mapper的更多相关文章

  1. [BJOI2019]勘破神机(斯特林数,数论)

    [BJOI2019]勘破神机(斯特林数,数论) 题面 洛谷 题解 先考虑\(m=2\)的情况. 显然方案数就是\(f_i=f_{i-1}+f_{i-2}\),即斐波那契数,虽然这里求出来是斐波那契的第 ...

  2. [BJOI2019]勘破神机

    [BJOI2019]勘破神机 推式子好题 m=2,斐波那契数列,$f_{n+1}$项 不妨$++l,++r$,直接求$f_n$ 求$\sum C(f_n,k)$,下降幂转化成阶乘幂,这样都是多项式了, ...

  3. 红米5/红米5 Plus逼出最强魅蓝Note6?降价后已成性价比神机

    从品牌到产品命名,小米旗下的红米与魅族旗下的魅蓝似乎是一对天生的对手,如今小米即将发布千元全面屏的红米5/红米5 Plus,暂时没有全面屏手机推出的魅蓝也拿出了自己的应对策略,魅蓝的办法简单粗暴:直接 ...

  4. 【LOJ】#3090. 「BJOI2019」勘破神机

    LOJ#3090. 「BJOI2019」勘破神机 为了这题我去学习了一下BM算法.. 很容易发现这2的地方是\(F_{1} = 1,F_{2} = 2\)的斐波那契数列 3的地方是\(G_{1} = ...

  5. 神机iPhone6停产,苹果产业链应该感谢它还是痛恨它?

    据国内媒体报道,一些苹果上游供应商已经接到通知,iPhone6系列将会在5月底彻底停产,一时间,竟在网络上引发汹涌的怀念之情.iPhone6的特别之处在于它是苹果第一款大屏幕的智能手机,标志着库克彻底 ...

  6. 深入浅出Mybatis系列七-mapper映射文件配置之insert、update、delete

    注:本文转载自南轲梦 注:博主 Chloneda:个人博客 | 博客园 | Github | Gitee | 知乎 上篇文章<深入浅出Mybatis系列(六)---objectFactory.p ...

  7. luogu P5320 [BJOI2019]勘破神机

    传送门 首先我们要知道要求什么.显然每次放方块要放一大段不能从中间分开的部分.设\(m=2\)方案为\(f\),\(m=3\)方案为\(g\),\(m=2\)可以放一个竖的,或者两个横的,所以\(f_ ...

  8. [BJOI2019]勘破神机(斯特林数+二项式定理+数学)

    题意:f[i],g[i]分别表示用1*2的骨牌铺2*n和3*n网格的方案数,求ΣC(f(i),k)和ΣC(g(i),k),对998244353取模,其中l<=i<=r,1<=l< ...

  9. loj 3090 「BJOI2019」勘破神机 - 数学

    题目传送门 传送门 题目大意 设$F_{n}$表示用$1\times 2$的骨牌填$2\times n$的网格的方案数,设$G_{n}$$表示用$1\times 2$的骨牌填$3\times n$的网 ...

  10. #loj3090 [BJOI2019] 勘破神机

    简单线性代数练习题 首先翻开具体数学生成函数一章,可以发现\(F(n),G(n)\)满足以下递推式 \[F(n)=F(n-1)+F(n-2),F(0)=1,F(1)=1\] \[G(n)=4G(n-2 ...

随机推荐

  1. Linux_aarch64_head.S到main.c的环境建立

    PS:要转载请注明出处,本人版权所有. PS: 这个只是基于<我自己>的理解, 如果和你的原则及想法相冲突,请谅解,勿喷. 环境说明   无 前言   最开始,我仅仅是对linux比较感兴 ...

  2. 力扣977(Java)-有序数组的平方(简单)

    题目: 给你一个按 非递减顺序 排序的整数数组 nums,返回 每个数字的平方 组成的新数组,要求也按 非递减顺序 排序. 示例 1: 输入:nums = [-4,-1,0,3,10]输出:[0,1, ...

  3. 力扣599(java&python)- 两个列表的最小索引总和(简单)

    题目: 假设 Andy 和 Doris 想在晚餐时选择一家餐厅,并且他们都有一个表示最喜爱餐厅的列表,每个餐厅的名字用字符串表示. 你需要帮助他们用最少的索引和找出他们共同喜爱的餐厅. 如果答案不止一 ...

  4. 力扣205(java)-同构字符串(简单)

    题目: 给定两个字符串 s 和 t ,判断它们是否是同构的. 如果 s 中的字符可以按某种映射关系替换得到 t ,那么这两个字符串是同构的. 每个出现的字符都应当映射到另一个字符,同时不改变字符的顺序 ...

  5. 深入理解C++中的RVO

    前言 考虑存在这样一个类如HeavyObject,其拷贝赋值操作比较耗时,通常你在使用函数返回这个类的一个对象时会习惯使用哪一种方式?或者会根据具体场景选择某一种方式? // style 1 Heav ...

  6. 实时计算 Flink 版总体介绍

    简介: 实时计算 Flink 版(Alibaba Cloud Realtime Compute for Apache Flink,Powered by Ververica)是阿里云基于 Apache ...

  7. 如何保证 Serverless 业务部署更新的一致性?

    简介: 代码在其他场景被更新,需要我们在当前得到感知,这个事情其实是非常重要的,和代码的安全发布密不可少.而此时,通过 Serverless Devs 是可以做到的. 作者|Anycodes​ 从我做 ...

  8. dotnet 修复多框架 TargetFrameworks 包含不受支持平台导致构建失败

    本文将告诉大家如何修复 dotnet 项目里的多框架 TargetFrameworks 如果包含了当前系统无法支持的平台时,如何进行跳过.解决在 Linux 平台构建时提示 Mac Catalyst ...

  9. dotnetCampus.UITest.WPF 一个支持中文用例的界面单元测试框架

    本文来安利大家一个支持使用中文做用例名的 WPF 界面 UI 单元测试框架 卖点 有没有觉得命名太难?有没有觉得单元测试的命名更难?没错,这是一个业界的大问题.很多团队都会因为单元测试的用例函数命名太 ...

  10. dotnet 6 引用 NAudio 的旧版本构建不通过

    本文告诉大家在使用 NAudio 的旧版本导致构建不通过问题,解决方法是升级到 1.10 或以上版本 在更新 dotnet 6 项目时,使用了 NAudio 的旧版本,构建失败,提示 MC1000 如 ...