Analog Devices ee-120 Application Notes

a
a
aa
Phone: (800) ANALOG-D, FAX: (781) 461-3010, EMAIL: dsp.support@analog.com, FTP: ftp.analog.com, WEB: www.analog.com/dsp
Engineer To Engineer Note EE-120
Technical Notes on using Analog Devices’ DSP components and development tools
Interfacing Assembly Language Programs to C
Contributed by David Levine
Writing code in a high-level language (such as C) makes it possible to quickly develop portable programs. However, there are times when you may want to use assembly language for part of an application. Assembly code may be needed for access to special instructions or system registers outside of the C domain, or may be selectively used to achieve higher performance. This note contains helpful steps a programmer can use to interface assembly code and C safely and efficiently.
How to write a subroutine in assembly language that can be called from C
3. The C compiler normally prefaces the name of external entry points with an underscore. You can simply declare the function with an underscore (as the compiler does). If you are using the function from assembler programs as well, you might want your function's name to be just as you write it. Then you will also need to tell the C compiler that it's an asm function, via 'extern "asm" {}' around the prototype. Or you can use #pragma linkage_name before the prototype to specify the actual entry point name.
2
4. A good way to determine how arguments will be passed is to write a dummy function in C and compile it with the "stop after compile" option set ("-S" command line option). Here's an example; the assignments to the global show where the arguments can be found upon entry to "asmfunc".
1. Familiarize yourself with the general features of the C runtime model. This should include the general notion of a stack, how arguments are handled, and also the various data types and their sizes. This information can be found in the C compiler manual.
2. Create a C interface definition, or "prototype", so that the C program knows the name of your function and the types of its arguments.
/* Sample file for exploring compiler interface... */
/*global variables ... assign arguments to them so that we can track which registers were used. There’s a global corresponding in type to each argument. */
int global_a; float global_b; int * global_p;
/***** the function itself *****/
The prototype will also determine how the arguments are passed.
1
C does allow a function to be used without a prototype. In that case, C will assume that all the arguments, as they appear in the call, are the proper type, without conversions; this may not be what is desired! C will also assume that the return type is integer.
Copyright 2000, Analog Devices, Inc. All rights reserved. Analog Devices assumes no responsibility for customer product design or the use or application of customers’ products or for any infringements of patents or rights of others which may result from Analog Devices assistance. All trademarks and logos are property of their respective holders. Information furnished by Analog Devices Applications and Development Tools Engineers is believed to be accurate and reliable, however no responsibility is as sumed by Analog Devices regarding the technical accuracy of the content provided in all Analog Devices’ Engineer-to-Engineer Notes.
1
int asmfunc(int a, float b, int * p) {
/* do some assignments so .s file
will show where args are */ global_a = a; global_b = b; global_p = p;
/* value gets loaded into the
return register */ return 12345;
2
This is being phased in, and may not be available in
older versions of the tools. Check the documentation.
}
For the TigerSHARC processor, this produces the following (compiled -S -O). Note that TigerSHARC passes up to four arguments in registers; check the runtime model for your processor to see how many arguments are passed in registers and in which registers. (The optimizer has gratuitously reversed the order of the assignments.)
// PROCEDURE: asmfunc
.global _asmfunc;
_asmfunc:
J8 = j31 + 12345;; [j31 + _global_p] = J6;; [j31 + _global_b] = XR5;; cjmp (NP) (ABS); [j31 + _global_a] = J4;;
If arguments are on the stack, they will be addressed via an offset from the stack pointer or frame pointer.
5. For a simple function, this may be all the information you need. The body of function then performs the computation, sets up a return value, and returns to the caller.
Some General Caveats:
The interrupt routines in particular rely on having a stack available.
IMPORTANT: The compiler assumes that
the overall machine state does not change during execution. This is processor specific. If the machine has modes, like integer/fractional, and they are changed in the assembly function, then they must be reset to their original values before exiting from the function. If the machine performs circular buffering when certain registers are non-zero, those registers must not be left altered (unless suitable reservations have been made on the related registers). This is more of a concern on some processors than others are.
6. For a more complicated function, you might find it useful to follow the general runtime model completely, and use the runtime stack for local storage. Again, a simple C program, passed through the compiler, will provide a good template to build on. Alternatively, you may find it just as convenient to use local static storage for temporaries. If you call other functions, maintaining the basic stack model will facilitate use of the debugger.
IMPORTANT: The runtime model defines certain registers as "scratch" and others as
How To Create An Assembly Language Program That Can Call A C Function
"preserved" or "dedicated". The scratch registers are available for immediate use without concern. You may also use the preserved registers, if you need more room (or if you're working with existing code), but you must save them first and then restore them before returning.
Dedicated registers should not be used for anything other than their intended purpose.
Calling "out" from assembly language to C is a bit more complicated than writing a C-callable assembly function, but it can be managed successfully. This kind of call is particularly useful for calling existing C-callable functions, most notably in the library.
1. Familiarity with the runtime model is essential.
The runtime model conventions apply not only to the compiler, but also to the libraries, interrupt routines, and the debugger.
EE-120 Page 2
Technical Notes on using Analog Devices’ DSP components and development tools
Phone: (800) ANALOG-D, FAX: (781)461-3010, EMAIL: dsp.support@analog.com, FTP: ftp.analog.com, WEB: www.analog.com/dsp
Loading...
+ 1 hidden pages