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