131 Week 16 - Inline Assembler Flashcards
Inline assembly
A way of embedding assembly code into C/C++ using the compiler.
Reason for wanting inline assembly
Allows you to outsmart gcc logic.
Allows for specific hardware optimisation
Allows for close to hardware programming (drivers)
Can improve performance
Inline assembly syntax
Use keyword asm or __asm__ to insert assembly. E.g., asm(“mov r0, r1”);
Passing parameters to inline assembly
there are optional output operands, input operands and clobbered register list that can be included for the inline assembly.
Inline assembly parameters syntax
asm(“Assembly code” : output operands : input operands : list of clobbered registers);
E.g., int a = 100, b;
asm (“movl r0, %[a];” “movl %0, r0;” : “=r” ( b ) : [a] “r” ( a ) : ”r0”);
Input/output constraints
I - immediate value
J - an immediate value in the range 0-4095
M - constant in the range 0-32
m - any valid memory address
r - general register (r0-r15)
X - any operand
Adding an = before the constraint specifies it is for output.
No = before the constraint specifies it is for input
Using inputs/outputs
Input/output for the inline assembly for inline assembly is defined by:
[assemblyName] “constraint” (codeName)
You can access the input/output using %assemblyName
E.g., [B] “=r” (b) for using a C variable b as an input register which can be used in the inline assembly through %B
Clobbered registers
Clobbered registers are registers that are modified during the inline assembly but not stated as an input or output register.
They need to be declared as otherwise the compiler does not know the register is being used and could overwrite it or optimise incorrectly.
Status registers may also need to be clobbered such as “cc” (conditional code flags) and “memory” (flags if instruction accesses unknown memory).
Register allocation
A compiler must assign variables to a processor register but 2 variables cannot be assigned to the same register at any point.
Can use spilling to store variable values if needed.
Volatile
Some assembly code must be executed exactly where it is put in the code and exactly how it is written e.g., when working with hardware or memory-mapped IO.
The volatile keyword tells the compiler not to move the code or not to remove it as it may do these while trying to optimise code.
Syntax:
volatile asm(“assembly”);
__volatile__ __asm__(“assembly”);