The main difference between the direct and the indirect call, is that:
- the direct call uses an instruction
call
with a fixed address as argument. After the linker has done its job, this address will be included in the opcode.
- the indirect call uses an instruction
call
with a register as argument (here rax
). The register is previously loaded either directly with the fixed address of the subroutine that is to be called, or with a value fetched from somewhere else, such as another register or a place in memory where the subroutine’s address was previously stored.
As a consequence, the direct call will always call the same subroutine, whereas the indirect call could call different subroutines, depending of what was loaded in the register before the call is made. Depending on the cpu, the indirect call might be a little slower since the indirection requires an extra effort.
A typical use case for the indirect call in assembly would be to implement what would be a call to a function pointer in C or a virtual member function in C++. In your example it’s the use of the function pointer f_sub
in the C source code. The key take-away here is that the same code (use of function pointer in C or indirect call in assembler) could call any C function or assembly subroutine that has the same interface and the choice is made at runtime.
The other differences between the two files are cosmetic, except for the load of the subroutine’s address into rax
.