
100
Evaluating and Programming the 29K RISC Family
2.2.2 crt0 Preparations before Calling main()
Application code normally begins execution at address
start
in the
crt0.s
mod-
ule. The previous section described the environment prepared by a HIF conforming
operating system before the code in
crt0
.s is executed. The
crt0.s
code makes final
preparations before the application
main()
procedure is called.
The code in
crt0.s
first copies the Am29027 shadow
mode
register value, passed
in
gr96
and
gr97
, to memory location __29027Mode. If a system does not have an
Am29027 floating–point coprocessor then there is no useful data passed in these reg-
isters. However, application code linked with floating–point libraries which make
use of the Am29027 coprocessor, will access the shadow memory location to deter-
mine the coprocessor operating
mode
value.
The
setvec
system call is then used to supply the operating system with the ad-
dresses of the user’s SPILL and FILL handler code which is located in
crt0.s
. Be-
cause this code normally runs in User mode address space, and the user has the option
to tailor the operation of this code, an operating system can not know in advance
(pre–
crt0.s
) the required SPILL and FILL handler code operation.
When procedure
main()
is called, it is passed two parameters; the
argc
parame-
ter indicates the number of elements in
argv
; the second parameter,
argv,
is a pointer
to an array of the character strings:
main(argc, argv)
argc;
argv[];
int
char*
The
getargs
HIF service is used to get the address of the
argv
array. In many
real–time applications there are no parameters passed to
main()
. However, to support
porting of benchmark application programs, many systems arrange for
main()
pa-
rameters to be loaded into a user’s data space. The
crt0.s
code walks through the
array looking for a NULL terminating string; in so doing, it determines the
argc
val-
ue. The register stack pointer was lowered by the
start()
procedure’s prologue code
to create a procedure activation record for passing parameters to
main()
.
To aid run–time libraries a memory variable, __LibInit, is defined in uninitial-
ized data memory space (BSS) by the library code. If any library code needs initial-
ization before use, then the __LibInit variable will be assigned to point to a library
routine which will perform all necessary initialization. This is accomplished by the
linker matching–up the BSS __LibInit variable with an initialized __LibInit variable
defined in the library code. The
crt0.s
code checks to see if the __LibInit variable
contains a non zero address, if so, the procedure is called.
The application
main()
procedure is ready to be called by
start()
. It is not ex-
pected that
main()
will return. Real–time programs typically never exit. However,
benchmark programs do, and this is accomplished by calling the HIF
exit
service. If a
main()
routine does not explicitly call
exit
then it will return to
start()
, where
exit
is
called should
main()
return.