至少对于x86,使用段寄存器执行TLS.默认段寄存器%ds隐含在寻址存储器的指令中.当访问TLS时,线程使用另一个段寄存器 – %gs用于i386,%fs用于x86-64 – 在调度线程时保存/恢复,就像其他寄存器在上下文切换中一样.
因此,可以通过以下方式访问流程范围的变量:
mov (ADDR) -> REG ; load memory `myVar` to REG.
这隐含着:
mov %DS:(ADDR) -> REG
对于TLS,编译器生成:
mov %FS:(ADDR) -> REG ; load thread-local address `myVar` to REG.
实际上,即使变量的地址在不同的线程中看起来是相同的,例如,
fprintf(stdout, "%p\n", & myVar); /* in separate threads... */
事实上,每个线程对段寄存器使用不同的值,这意味着它们映射到物理存储器的不同区域.如果要将地址从一个线程传递到另一个线程,则无法访问它在第一个线程中表示的内存 – 另一个段寄存器值在第二个线程中生效.
Windows使用相同的方案(它可以交换%fs和%gs的角色 – 不确定)和OS X.对于其他架构,ELF ABI有一个深入的technical guide到TLS.它缺少对ARM体系结构的讨论,并且有关于IA-64和Alpha的详细信息,因此它显示了它的年龄.