Search code examples
assemblyx86-64att

Do 32 and 64 bit values share the same register space?


If I move a value from a 32 bit register say:

    movq %rdx,%rax
    movl %edx,%eax

Will the value stored in %rax get clobbered?


Solution

  • Yes,

    Your code:

    mov rax,rdx
    mov eax,edx
    

    Will perform the following actions.

    rax <= rdx
    high 32 bits of rax <= 0, low 32 bits of rax <= edx.
    

    Assigning a 32 bit register will zero out the higher part of that register.

    Contrast this to:

    mov rax,rdx   :  rax <= rdx
    mov ax,dx     :  High 48 bits of rax is unchanged!, low 16 bits of rax <= dx
    

    The same goes for the byte registers.

    The reason 32 bit assigns zero out the upper part of a 64 bit register is that it prevents partial register updates, which causes delays in the instruction pipeline.

    Using 16 bit code in 32 or 64 bit mode causes delays in the following scenario:

    mov rax,-1       //1 cycle
    mov ax,dx        //1 cycle
                     //stall, the values of rax-upper and ax need to be combined
    mov r8,rax       //2 cycles
    

    A better option would be

    mov rax,-1       //1 cycle
    movzx eax,dx     //runs concurrent with previous instruction, 0 cycles
    mov r8,rax       //1 cycle
    
    //Total 2 cycles, twice as fast.
    

    This code is not equivalent to the sample above it, but that's the whole point. You should avoid partial register updates when possible. Also note that movzx eax,dx is equivalent to movzx rax,dx for the reasons stated above. On x64 it is one byte shorter and therefore the preferred form.

    Note that I'm not using ATT syntax, as a matter of principle