jjzjj

6502 寻址模式详解

1bite 2023-03-28 原文

6502 共有 13 种寻址模式:

  1. A:寄存器寻址。指令形式为 OPC A;目标数据位于 A 寄存器中,属于隐含寻址;使用这种寻址模式的指令都是 1 个字节长度,需要 2 个时钟周期
  2. abs:绝对地址寻址。指令形式为 OPC $LLHH;目标数据位于绝对地址 $HHLL
  3. abs,X:X 变址绝对地址寻址。指令形式为 OPC $LLHH,X;操作数是 16 位基址;有效地址由基址加上 X 寄存器中的值计算得到
  4. abs,Y:X 变址绝对地址寻址。指令形式为 OPC $LLHH,Y;操作数是 16 位基址;有效地址由基址加上 Y 寄存器中的值计算得到
  5. #:立即数寻址。指令形式为 OPC #$BB;目标数据即操作数 BB
  6. impl:隐含寻址。目标数据由指令码决定
  7. ind:间接寻址。指令形式为 OPC ($LLHH);操作数是 16 位间接地址; 有效地址是位于间接地址处的 16 位内容;读取有效地址时,不考虑跨页
  8. X,ind:X 变址间接寻址。指令形式为 OPC ($LL,X);操作数是 8 位间接地址基址; 基址与 X 寄存器的值进行无进位加法后得到间接地址,有效地址是位于间接地址处的 16 位地址;读取有效地址时,不考虑跨页
  9. ind,Y: 间接地址 Y 变址寻址。指令形式为 OPC ($LL),Y;操作数是一个 8 位间接地址;$LL 处存储的是 16 位基址,基址加上 Y 寄存器的值得到有效地址;读取 16 位基址时,不考虑跨页
  10. rel:相对寻址。指令形式为 OPC $BB;操作数 BB 为有符号数,表示偏移;目标地址由 PC 加上 BB 得到
  11. zpg:零页寻址。指令形式为 OPC $LL;操作数是一个 8 位绝对地址,目标数据位于 $00LL
  12. zpg,X:零页 X 变址寻址。指令形式为 OPC $LL,X;操作数是一个 8 位基址;有效地址由基址与 X 寄存器的值进行无进位加法后得到
  13. zpg,Y:零页 Y 变址寻址。指令形式为 OPC $LL,Y;操作数是一个 8 位基址;有效地址由基址与 Y 寄存器的值进行无进位加法后得到

6502 采用小端序,读取 16 位地址时,先读取到的是低 8 位,然后是高 8 位

我把每个寻址模式的时序整理成了表格,以便清晰地看出 CPU 内每个时钟周期的工作状态,对于理解 6502 的工作原理应该有所帮助。表格中的单元格如果为空白,表示对应的组件没有被修改。需要注意的是,以下时序只适用于正常指令(官方指令)。非法指令在解码时和执行时会激活额外的信号线,导致同样的寻址模式会多做一些事情或者同样的寻址模式却有不同的时长,往往需要单独总结。

我还做了一个网页,可以查询每个指令的时序和详细信息,点击这里体验。

zpg

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取有效地址
T2 0 DL 0 读取目标数据
OPC = RD(PC++);
EA = RD(PC++);

zpx

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取基址
T2 0 DL 0 X DL 0 加上 X 寄存器中的偏移,计算有效地址
T3 ADD 0 读取目标数据
OPC = RD(PC++);
EA = RD(PC++) + X;

zpy

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取基址
T2 0 DL 0 Y DL 0 加上 Y 寄存器中的偏移,计算有效地址
T3 ADD 0 读取目标数据
OPC = RD(PC++);
EA = RD(PC++) + Y;

abs

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取有效地址的低 8 位
T2 PCH PCL 1 0 DL 0 读取有效地址的高 8 位
T3 DL ADD 0 读取目标数据
OPC = RD(PC++);
EA = WORD(RD(PC + 1), RD(PC));
PC += 2;

abx

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取基址的低 8 位
T2 PCH PCL 1 X DL 0 读取基址的高 8 位, 加上 X 寄存器中的偏移
T3 DL ADD 0 0 DL ACR 读取目标数据,处理跨页。 若不跨页 (ACR = 0),则跳过 T4 周期
T4 ADD 0 再次读取目标数据
OPC = RD(PC++);
EA = WORD(RD(PC + 1), RD(PC)) + X;
PC += 2;

aby

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取基址的低 8 位
T2 PCH PCL 1 Y DL 0 读取基址的高 8 位, 加上 Y 寄存器中的偏移
T3 DL ADD 0 0 DL ACR 读取目标数据,处理跨页。 若不跨页 (ACR = 0),则跳过 T4 周期
T4 ADD 0 再次读取目标数据
OPC = RD(PC++);
EA = WORD(RD(PC + 1), RD(PC)) + Y;
PC += 2;

ind

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取间址的低 8 位
T2 PCH PCL 1 0 DL 0 读取间址的高 8 位
T3 DL ADD 0 0 ADD 1 读取有效地址的低 8 位,并把间址加 1(无进位)
T4 ADD 0 0 DL 0 读取有效地址的高 8 位
T5 DL ADD 0 读取目标数据
OPC = RD(PC++);
IA = WORD(RD(PC + 1), RD(PC));
EA = WORD(RD(WORD(HI(IA), LO(IA + 1))), RD(IA));
PC += 2;

inx

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取 8 位基址
T2 0 DL 0 X DL 0 空读,加上 X 寄存器中的偏移计算间址
T3 ADD 0 0 ADD 1 读取有效地址的低 8 位,并把间址加 1(无进位)
T4 ADD 0 0 DL 0 读取有效地址的高 8 位
T5 DL ADD 0 读取目标数据
OPC = RD(PC++);
IA = BYTE(RD(PC++) + X);
EA = WORD(RD(BYTE(IA + 1)), RD(IA));

iny

Cycle ADH ADL IPC AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取 8 位间址
T2 0 DL 0 0 DL 1 读取基址的低 8 位,并把间址加 1
T3 ADD 0 Y DL 0 读取基址的高 8 位,加上 Y 寄存器中的偏移计算有效地址
T4 DL ADD 0 0 DL ACR 读取目标数据并处理跨页, 若不跨页 (ACR = 0),则跳过 T5 周期
T5 ADD 0 再次读取目标数据
OPC = RD(PC++);
IA = BYTE(RD(PC++) + X);
EA = WORD(RD(BYTE(IA + 1)), RD(IA)) + Y;

rel

Cycle ADH ADL IPC PCH PCL AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取偏移
T2 PCH PCL 0 DL PCL 0 空读,把 PC 加上偏移计算目标地址
T3 ADD 0 ADD PCH * ** 处理跨页。
* 若偏移量为正,则 BI 为 0;否则为 $FF
** 若跨页(ACR = 1)且偏移量为正,则 CI 为 1;否则为 0
T0 ADD 1 ADD 设置 PC 并读取操作码
OPC = RD(PC++);
OFF = RD(PC++);
PC = PC + OFF;

JMP abs

Cycle ADH ADL IPC PCH PCL AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取有效地址的低 8 位
T2 PCH PCL 0 0 DL 0 读取有效地址的高 8 位
T0 DL ADD 1 DL ADD 设置 PC 读取操作码
OPC = RD(PC++);
EA = WORD(RD(PC + 1), RD(PC));
PC = EA;

JMP ind

Cycle ADH ADL IPC PCH PCL AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取间址低 8 位
T2 PCH PCL 1 0 DL 0 读取间址高 8 位
T3 DL ADD 0 0 ADD 1 读取有效地址低 8 位,把间址加 1 (无进位)
T4 ADD 0 0 DL 0 读取有效地址高 8 位
T0 DL ADD 1 DL ADD 设置 PC 读取操作码
OPC = RD(PC++);
IA = WORD(RD(PC + 1), RD(PC));
EA = WORD(RD(WORD(HI(IA), LO(IA + 1))), RD(IA));
PC = EA;

JSR abs

Cycle ADH ADL IPC PCH PCL S AI BI CI DOR 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 读取有效地址的低 8 位
T2 $01 S 0 DL 0 S 0 将读到的有效地址低 8 位保存到 S 寄存器中,将原本 S 寄存器的内容保存到 ALU 中
T3 $01 ADD 0 $FF ADD 0 PCH 将 PCH 压栈
T4 $01 ADD 0 $FF ADD 0 PCL 将 PCL 压栈
T5 PCH PCL 1 读取有效地址的高 8 位
T0 DL S 1 DL S ADD 设置 PC,读取操作码,并从 ALU 中恢复 S 寄存器
OPC = RD(PC++);
EA = RD(PC++);
WR(S--, PCH);
WR(S--, PCL);
EA = WORD(RD(PC++), LO(EA));
PC = EA;

BRK

Cycle ADH ADL IPC PCH PCL S AI BI CI DOR 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 0 空读
T2 $01 S 0 $FF S 0 PCH 将 PCH 压栈
T3 $01 ADD 0 $FF ADD 0 PCL 将 PCL 压栈
T4 $01 ADD 0 $FF ADD 0 P 将 P 寄存器的内容压栈
T5 $FF $FE 0 ADD 读取中断向量的低 8 位
T6 $FF $FF 0 0 DL 0 读取中断向量的高 8 位
T0 DL ADD 1 DL ADD 设置 PC 并读取操作码
OPC = RD(PC++);
WR(S--, PCH);
WR(S--, PCL);
WR(S--, P);
PC = WORD(RD(0xFFFF), RD(0xFFFE));

RTI

Cycle ADH ADL IPC PCH PCL S P AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 空读
T2 $01 S 0 0 S 1 空读,栈指针加 1
T3 $01 ADD 0 0 ADD 1 从栈顶读 P 寄存器的内容,栈指针加 1
T4 $01 ADD 0 DL 0 ADD 1 从栈顶读 PCL 的内容,栈指针加 1
T5 $01 ADD 0 ADD 0 DL 0 从栈顶读 PCH 的内容
T0 DL ADD 1 DL ADD 设置 PC 并读取操作码
OPC = RD(PC++);
P = RD(++S);
PCL = RD(++S);
PCH = RD(++S);

RTS

Cycle ADH ADL IPC PCH PCL S AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 1 空读
T2 $01 S 0 0 S 1 空读,栈指针加 1
T3 $01 ADD 0 0 ADD 1 从栈顶读 PCL,栈指针加 1
T4 $01 ADD 0 ADD 0 DL 0 从栈顶读 PCH
T5 DL ADD 1 DL ADD 空读,设置 PC 指针
T0 PCH PCL 1 读取操作码
OPC = RD(PC++);
PCL = RD(++S);
PCH = RD(++S);
PC++;

PHA, PHP

Cycle ADH ADL IPC S AI BI CI DOR 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 0 空读
T2 $01 S 0 $FF S 0 A/P 将 A 或 P 寄存器的内容压栈
T0 PCH PCL 1 ADD 读取操作码
OPC = RD(PC++);
WR(S--, A);

PLA, PLP

Cycle ADH ADL IPC A/P S AI BI CI 说明
T0 PCH PCL 1 读取操作码
T1 PCH PCL 0 空读
T2 $01 S 0 0 S 1 空读,栈指针加 1
T3 $01 ADD 0 ADD 从栈顶读 A 或 P 的内容
T0 PCH PCL 1 DL 读取操作码,设置 A 或 P 寄存器
OPC = RD(PC++);
A = RD(++S);

参考

有关6502 寻址模式详解的更多相关文章

  1. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  4. ruby - 是否有用于序列化和反序列化各种格式的对象层次结构的模式? - 2

    给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最

  5. ruby-on-rails - environment.rb 中设置的常量在开发模式中消失 - 2

    了解Rails缓存如何工作的人可以真正帮助我。这是嵌套在Rails::Initializer.runblock中的代码:config.after_initializedoSomeClass.const_set'SOME_CONST','SOME_VAL'end现在,如果我运行script/server并发出请求,一切都很好。然而,在我的Rails应用程序的第二个请求中,一切都因单元化常量错误而变得糟糕。在生产模式下,我可以成功发出第二个请求,这意味着常量仍然存在。我已通过将以上内容更改为以下内容来解决问题:config.after_initializedorequire'some_cl

  6. Ruby:标准递归模式 - 2

    我经常迷上ruby​​的一件事是递归模式。例如,假设我有一个数组,它可能包含无限深度的数组作为元素。所以,例如:my_array=[1,[2,3,[4,5,[6,7]]]]我想创建一个方法,可以将数组展平为[1,2,3,4,5,6,7]。我知道.flatten可以完成这项工作,但这个问题是作为我经常遇到的递归问题的一个例子-因此我试图找到一个更可重用的解决方案。简而言之-我猜这种事情有一个标准模式,但我想不出任何特别优雅的东西。任何想法表示赞赏 最佳答案 递归是一种方法,它不依赖于语言。您在编写算法时要考虑两种情况:再次调用函数的情

  7. ruby - 在 Ruby 中查找多个正则表达式匹配的模式和位置 - 2

    这应该是一个简单的问题,但我找不到任何相关信息。给定一个Ruby中的正则表达式,对于每个匹配项,我需要检索匹配的模式$1、$2,但我还需要匹配位置。我知道=~运算符为我提供了第一个匹配项的位置,而string.scan(/regex/)为我提供了所有匹配模式。如果可能,我需要在同一步骤中获得两个结果。 最佳答案 MatchDatastring.scan(regex)do$1#Patternatfirstposition$2#Patternatsecondposition$~.offset(1)#Startingandendingpo

  8. ruby - sinatra 框架的 MVC 模式 - 2

    我想开始使用“Sinatra”框架进行编码,但我找不到该框架的“MVC”模式。是“MVC-Sinatra”模式或框架吗? 最佳答案 您可能想查看Padrino这是一个围绕Sinatra构建的框架,可为您的项目提供更“类似Rails”的感觉,但没有那么多隐藏的魔法。这是使用Sinatra可以做什么的一个很好的例子。虽然如果您需要开始使用这很好,但我个人建议您将它用作学习工具,以对您来说最有意义的方式使用Sinatra构建您自己的应用程序。写一些测试/期望,写一些代码,通过测试-重复:)至于ORM,你还应该结帐Sequel其中(imho

  9. ruby-on-rails - Rails 如何创建数据模式种子数据 - 2

    有没有一种方法可以自动生成种子数据文件并创建种子数据,就像您在下面链接中的Laravel中看到的那样?LaravelDatabaseMigrations&Seed我在另一个应用程序上看到在Rails的db文件夹下创建了一些带有时间戳的文件,其中包含种子数据。创建它的好方法是什么? 最佳答案 我建议你使用Fabrication的组合gem和Faker.Fabrication允许您编写一个模式来构建您的对象,而Faker为您提供虚假数据,如姓名、电子邮件、电话号码等。这是制造商的样子:Fabricator(:user)dousernam

  10. ruby-on-rails - Ruby on Rails 应用程序的只读模式 - 2

    我有一个交互式RubyonRails应用程序,我想在特定时间将其置于“只读模式”。这将允许用户读取他们需要的数据,但阻止他们执行写入数据库的操作。执行此操作的一种方法是在数据库中放置一个true/false变量,该变量在进行任何写入之前进行检查。我的问题。有没有更优雅的解决方案来解决这个问题? 最佳答案 如果你真的想阻止任何数据库写入,我能想到的最简单的方法是覆盖readonly?始终返回true的模型方法,无论是在选定模型中还是对于所有ActiveRecord模型。如果模型设置为只读(通常通过调用#readonly!来完成),任何

随机推荐