我决定自学汇编语言.
我已经意识到,如果我尝试更改任何段寄存器的值,我的程序将无法编译.
我发现的每篇文章都说我确实可以改变至少4个段寄存器的值,那么是什么给出了?
我真的只对为什么在这一点上感兴趣,我没有任何改变这些地址的真正目的.
你说你对原因感兴趣,所以:
在实模式中,段是物理存储器的64K"窗口",这些窗口间隔16个字节.在受保护模式下,段是物理或虚拟内存的窗口,其大小和位置由操作系统确定,并且具有许多其他属性,包括进程必须具有的访问权限级别.
从这里开始,我所说的一切都是指保护模式.
内存中有一个称为全局描述符表(GDT)的表,它保存有关这些窗口大小和位置以及其他属性的信息.每个进程可能还有本地描述符表,它们的工作方式类似,所以我只关注GDT.
加载到段寄存器中的值称为段选择器.它是GDT或LDT的索引,带有一些额外的安全信息.当然,如果程序试图加载超出GDT范围的描述符,则会发生异常.此外,如果进程没有足够的权限来访问该段,或者其他内容无效,则会发生异常.
发生异常时,内核会处理它.这种异常可能被归类为分段错误.操作系统会杀死你的程序.
最后一点需要注意:在x86指令集中,您无法将立即值加载到段寄存器中.必须将中间寄存器或内存操作数或POP用于段寄存器.
MOV DS, 160 ;INVALID - won't assemble MOV AX, 160 ;VALID - assembles, but will probably result in an MOV DS, AX ;exception, and thus the death of your program
我认为应该指出,该架构允许大量的细分市场.但AFAIK,在主流x86操作系统方面,段寄存器仅用于几个目的:
除了用于TLS的每个线程的段之外,实际上只使用少数段(处理器的数量),并且仅由OS执行.应用程序可以完全忽略段寄存器.
这是由于OS设计,而不是任何技术限制.可能有嵌入式操作系统需要用户空间程序来处理段寄存器,尽管我不知道.