You may find it more convenient to write functions in either C or assembly and then mix them later. This can also be done. In fact, it’s downright easy. Functions can be written in assembly and then called from either C or C++, and vice versa; assembly routines can be called from C or C++ source code. Here, we’ll examine mixing C and assembly routines, but refer to the ARM documentation (ARM 2007d) for infor- mation on working with C++. When using mixed language programming, you want to ensure that your assembly routines follow the AAPCS standard and your C code uses C calling conventions.
EXAMPLE 18.3
You may have a function defined in C that you want to use in an assembly routine.
The code below shows a simple function that is called in the assembly routine with a BL instruction.
C source code appears as
int g(int a, int b, int c, int d, int e) {
return a + b + c + d + e;
}
Assembly source code appears as
;int f(int i) {return g(i, 2*i, 3*i, 4*i, 5*i);}
PRESERVE8 EXPORT f
AREA f, CODE, READONLY
IMPORT g ; i is in r0
STR lr, [sp, #4] ; preserve lr
ADD r1, r0, r0 ; compute 2*i (2nd param) ADD r2, r1, r0 ; compute 3*i (3rd param) ADD r3, r1, r2 ; compute 5*i
STR r3, [sp, #−4]! ; 5th param on stack
ADD r3, r1, r1 ; compute 4*i (4th param)
BL g ; branch to C function
ADD sp, sp, #4 ; remove 5th param LDR pc, [sp], #4 ; return
END
EXAMPLE 18.4
The code below shows an example of calling an assembly language function from C code. The program copies one string over the top of another string, and the copying routine is written entirely in assembly.
C source code appears as
#include <stdio.h >
extern void strcpy(char *d, const char *s);
extern void init_serial(void);
int main() {
const char *srcstr = “First string - source”;
char dststr[] = “Second string - destination”;
/* dststr is an array since we’re */
/* going to change it */
init_serial();
printf(“Before copying:\n”);
printf(“%s\n %s\n”,srcstr, dststr);
strcopy(dststr, srcstr);
printf(“After copying:\n”);
printf(“%s\n %s\n”,srcstr, dststr);
return(0);
}
Assembly source code appears as PRESERVE8
AREA SCopy, CODE, READONLY EXPORT strcopy
strcopy
; r0 points to destination string
; r1 points to source string LDRB r2, [r1], #1 ; load byte and update address STRB r2, [r0], #1 ; store byte and update address CMP r2, #0 ; check for zero terminator BNE strcopy ; keep going if not
BX lr ; return
END
In some cases, features of the processor are not readily available in C and C++.
For example, the conversion instructions in the Cortex-M4 for fixed-point and float- ing-point values we considered in Chapter 9 are not accessible in C and C++. The example below shows how to use the embedded assembly features to create a set of conversion routines for specific formats that can easily be reused.
EXAMPLE 18.5
The code below contains two routines for conversion between signed S16 format values and single-precision floating-point values. Recall that the S16 format speci- fies a short signed integer of 16 bits. In this example, we are simulating sensor data in the form of a signed fixed-point 16-bit format with 8 fraction bits. The range of input data is {−128, 127 + 255/256}, with a numeric separation of 1/256.
The conversion routine utilizing the VCVT.S16,F32 instruction is shown below.
Recall that this instruction operates on two FPU registers, so a move from the input source to an FPU register is required.
AREA FixedFloatCvtRoutines, CODE, READONLY THUMB
EXPORT CvtShorts8x8ToFloat CvtShorts8x8ToFloat
; Use the VCVT instruction to convert a short in
; signed 8x8 format to a floating-point single-
; precision value and return the float value.
; The input short is in register r0.
; First move it to a float register - no
; format conversion will take place
VMOV.F32 s0, r0 ; transfer the short to a
; floating-point register
VCVT.F32.S16 s0, s0, #8 ; perform the conversion
BX lr ; return
END
A sample C program to use this conversion routine is shown below. The input data is in short integer format representing the signed 8x8 format (check for yourself that these values are correct).
//Input data in S16 format with 8 fraction bits.
#include <stdio.h >
extern void EnableFPU(void);
extern float CvtShorts8x8ToFloat(short i);
int main(void) {
short Input[10] = {
1408, // 5.5 (0x0580) 384, // 1.5 (0x180)
−672, // −2.625 (0xFD60)
−256, // −1.0 (0xFF00)
641, // 2.50390625 (2.5 + 1/256)(0x0281) 192, // .75 (0x00C0)
−32768, // neg max, −128.0 (0x8000)
32767, // pos max, 127 + 255/256 (0x7FFF) −32, // −0.125 (0xFFE0)
0
};
int i;
short InVal;
float OutVal;
for (i = 0; i < 11; i ++) {
OutVal = CvtShorts8x8ToFloat(Input[i]);
//Operate on the float value }
}
The conversion routine is stored in a separate file. Multiple routines may be placed in this file and called as needed by the C program. In this way, a library of routines utilizing functions not readily available from the high-level languages may be created to make use of features in the processor.
For further reading, you should consult the ARM documentation about calling C++ functions from assembly and calling assembly from C++. Examples can be found in the RealView Compilation Tools Developer Guide (ARM 2007a).