Calling between C and Assembly

Một phần của tài liệu ARM assembly language (Trang 413 - 416)

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).

Một phần của tài liệu ARM assembly language (Trang 413 - 416)

Tải bản đầy đủ (PDF)

(448 trang)