

Copyright 1990 AT&T  
All Rights Reserved  
Printed In USA

Published by Prentice-Hall, Inc.  
A Division of Simon & Schuster  
Englewood Cliffs, New Jersey 07632

No part of this publication may be reproduced or transmitted in any form or by any means—graphic, electronic, electrical, mechanical, or chemical, including photocopying, recording in any medium, taping, by any computer or information storage and retrieval systems, etc., without prior permissions in writing from AT&T.

#### IMPORTANT NOTE TO USERS

While every effort has been made to ensure the accuracy of all information in this document, AT&T assumes no liability to any party for any loss or damage caused by errors or omissions or by statements of any kind in this document, its updates, supplements, or special editions, whether such errors are omissions or statements resulting from negligence, accident, or any other cause. AT&T further assumes no liability arising out of the application or use of any product or system described herein; nor any liability for incidental or consequential damages arising from the use of this document. AT&T disclaims all warranties regarding the information contained herein, whether expressed, implied or statutory, including implied warranties of merchantability or fitness for a particular purpose. AT&T makes no representation that the interconnection of products in the manner described herein will not infringe on existing or future patent rights, nor do the descriptions contained herein imply the granting or license to make, use or sell equipment constructed in accordance with this description.

AT&T reserves the right to make changes without further notice to any products herein to improve reliability, function, or design.

#### TRADEMARKS

UNIX is a registered trademark of AT&T.

The publisher offers discounts on this book when ordered in bulk quantities.  
For more information, write:

Special Sales  
Prentice-Hall, Inc.  
College Technical and Reference Division  
Englewood Cliffs, New Jersey 07632

or

call 201-592-2498

For single copies, call 201-767-5937

10 9 8 7 6 5 4 3 2 1

ISBN 0-13-877663-6



1070.350

**UNIX**  
**PRESS**

A Prentice Hall Title

# SystemV Application Binary Interface Motorola 68000 Supplement

## Contents

|          |                                                     |      |
|----------|-----------------------------------------------------|------|
| <b>1</b> | <b>INTRODUCTION</b>                                 |      |
|          | Motorola 68000 Family and the System V ABI          | 1-1  |
|          | How to Use the Motorola 68000 Family ABI Supplement | 1-2  |
| <b>2</b> | <b>SOFTWARE INSTALLATION</b>                        |      |
|          | Software Distribution Formats                       | 2-1  |
|          | <i>nicht kopiert</i>                                |      |
| <b>3</b> | <b>LOW-LEVEL SYSTEM INFORMATION</b>                 |      |
|          | Machine Interface                                   | 3-1  |
|          | Function Calling Sequence                           | 3-10 |
|          | Operating System Interface                          | 3-19 |
|          | Coding Examples                                     | 3-31 |
| <b>4</b> | <b>OBJECT FILES</b>                                 |      |
|          | ELF Header                                          | 4-1  |
|          | Sections                                            | 4-2  |
|          | Symbol Table                                        | 4-3  |
|          | Relocation                                          | 4-4  |
| <b>5</b> | <b>PROGRAM LOADING AND DYNAMIC LINKING</b>          |      |
|          | Program Loading                                     | 5-1  |
|          | Dynamic Linking                                     | 5-5  |

Table of Contents

**6****LIBRARIES**

|                        |     |
|------------------------|-----|
| System Library         | 6-1 |
| C Library              | 6-3 |
| System Data Interfaces | 6-4 |

**Figures and Tables**

|                                                                 |      |
|-----------------------------------------------------------------|------|
| <b>Figure 3-1:</b> Scalar Types                                 | 3-2  |
| <b>Figure 3-2:</b> Structure Smaller Than a Long Word           | 3-3  |
| <b>Figure 3-3:</b> No Padding                                   | 3-4  |
| <b>Figure 3-4:</b> Internal Padding                             | 3-4  |
| <b>Figure 3-5:</b> Internal and Tail Padding                    | 3-5  |
| <b>Figure 3-6:</b> union Allocation                             | 3-5  |
| <b>Figure 3-7:</b> Bit-Field Ranges                             | 3-6  |
| <b>Figure 3-8:</b> Bit Numbering                                | 3-7  |
| <b>Figure 3-9:</b> Left-to-Right Allocation                     | 3-7  |
| <b>Figure 3-10:</b> Boundary Alignment                          | 3-7  |
| <b>Figure 3-11:</b> Storage Unit Sharing                        | 3-8  |
| <b>Figure 3-12:</b> union Allocation                            | 3-8  |
| <b>Figure 3-13:</b> Unnamed Bit-Fields                          | 3-8  |
| <b>Figure 3-14:</b> Processor Registers                         | 3-11 |
| <b>Figure 3-15:</b> Standard Stack Frame                        | 3-12 |
| <b>Figure 3-16:</b> Function Prologue                           | 3-13 |
| <b>Figure 3-17:</b> Integral and Pointer Arguments              | 3-15 |
| <b>Figure 3-18:</b> Floating-Point Arguments                    | 3-15 |
| <b>Figure 3-19:</b> Structure and Union Arguments               | 3-16 |
| <b>Figure 3-20:</b> Function Epilogue                           | 3-16 |
| <b>Figure 3-21:</b> Function Epilogue                           | 3-17 |
| <b>Figure 3-22:</b> Virtual Address Configuration               | 3-20 |
| <b>Figure 3-23:</b> Exceptions and Signals                      | 3-23 |
| <b>Figure 3-24:</b> Declaration for main                        | 3-24 |
| <b>Figure 3-25:</b> Condition Code Register Fields              | 3-25 |
| <b>Figure 3-26:</b> Initial Process Stack                       | 3-26 |
| <b>Figure 3-27:</b> Auxiliary Vector                            | 3-27 |
| <b>Figure 3-28:</b> Auxiliary Vector Types, a_type              | 3-27 |
| <b>Figure 3-29:</b> Example Process Stack                       | 3-30 |
| <b>Figure 3-30:</b> Position-Independent Function Prologue      | 3-33 |
| <b>Figure 3-31:</b> Absolute Load and Store                     | 3-34 |
| <b>Figure 3-32:</b> Position-Independent Load and Store         | 3-35 |
| <b>Figure 3-33:</b> Absolute Direct Function Call               | 3-35 |
| <b>Figure 3-34:</b> Position-Independent Direct Function Call   | 3-36 |
| <b>Figure 3-35:</b> Absolute Indirect Function Call             | 3-36 |
| <b>Figure 3-36:</b> Position-Independent Indirect Function Call | 3-37 |
| <b>Figure 3-37:</b> Branch Instruction, Both Models             | 3-37 |

# 1 INTRODUCTION

---

**Motorola 68000 Family and the System V  
ABI**

1-1

---

**How to Use the Motorola 68000 Family ABI  
Supplement**

Evolution of the ABI Specification

1-2

1-2

Table of Contents

---

## **Motorola 68000 Family and the System V ABI**

The **System V Application Binary Interface**, or **ABI**, defines a system interface for compiled application programs. Its purpose is to establish a standard binary interface for application programs on systems that implement UNIX System V Release 4.0 or some other operating system that complies with the **System V Interface Definition, Issue 3**.

This document is a supplement to the generic **System V ABI**, and it contains information specific to System V implementations built on the Motorola 68000 processor architecture family. The generic term 'Motorola 68000 family' is used in this specification to mean the Motorola MC68020, MC68030, and MC68040 processor architectures; it does not refer to the MC68000, MC68008, or MC68010.

Together, these two specifications, the generic **System V ABI** and the **Motorola 68000 Family System V ABI Supplement**, constitute a complete *System V Application Binary Interface* specification for systems that implement the architecture of the Motorola 68000 family.

## How to Use the Motorola 68000 Family ABI Supplement

This document is a supplement to the generic **System V ABI** and contains information referenced in the generic specification that may differ when System V is implemented on different processors. Therefore, the generic **ABI** is the prime reference document, and this supplement is provided to fill gaps in that specification.

As with the **System V ABI**, this specification references other publicly-available reference documents, including the

- MC68020 32-Bit Microprocessor User's Manual, MC68020UM/AD
- MC68030 Enhanced 32-Bit Microprocessor User's Manual, MC68030UM/AD
- MC68040 32-Bit Microprocessor User's Manual, MC68040UM/AD
- M68000 Programmer's Reference Manual, M68000PM/AD

available from Motorola.

All the information referenced by this supplement should be considered part of this specification, and just as binding as the requirements and data explicitly included here.

### Evolution of the ABI Specification

The **System V Application Binary Interface** will evolve over time to address new technology and market requirements, and will be reissued at intervals of approximately three years. Each new edition of the specification is likely to contain extensions and additions that will increase the potential capabilities of applications that are written to conform to the **ABI**.

As with the **System V Interface Definition**, the **ABI** will implement **Level 1** and **Level 2** support for its constituent parts. **Level 1** support indicates that a portion of the specification will continue to be supported indefinitely, while **Level 2** support means that a portion of the specification may be withdrawn or altered after the next edition of the **ABI** is made available. That is, a portion of the specification moved to **Level 2** support in an edition of the **ABI** specification will remain in effect at least until the following edition of the specification is published.

These Level 1 and Level 2 classifications and qualifications apply to this Supplement, as well as to the generic specification. All components of the **ABI** and of this supplement have Level 1 support unless they are explicitly labeled as Level 2.

## **3 LOW-LEVEL SYSTEM INFORMATION**

---

### **Machine Interface**

|                         |     |
|-------------------------|-----|
| Processor Architecture  | 3-1 |
| Data Representation     | 3-1 |
| ■ Fundamental Types     | 3-1 |
| ■ Aggregates and Unions | 3-3 |

---

### **Function Calling Sequence**

|                                          |      |
|------------------------------------------|------|
| Registers and the Stack Frame            | 3-10 |
| Integral and Pointer Arguments           | 3-14 |
| Floating-Point Arguments                 | 3-15 |
| Structure and Union Arguments            | 3-16 |
| Functions Returning Scalars or No Value  | 3-16 |
| Functions Returning Structures or Unions | 3-17 |

---

### **Operating System Interface**

|                               |      |
|-------------------------------|------|
| Virtual Address Space         | 3-19 |
| ■ Page Size                   | 3-19 |
| ■ Virtual Address Assignments | 3-19 |
| ■ Managing the Process Stack  | 3-21 |
| ■ Coding Guidelines           | 3-21 |
| Processor Execution Modes     | 3-22 |
| Exception Interface           | 3-23 |
| Process Initialization        | 3-24 |
| ■ Registers                   | 3-24 |
| ■ Process Stack               | 3-25 |

|                                        |      |
|----------------------------------------|------|
| <b>Coding Examples</b>                 | 3-31 |
| Code Model Overview                    | 3-32 |
| Position-Independent Function Prologue | 3-33 |
| Data Objects                           | 3-34 |
| Function Calls                         | 3-35 |
| Branching                              | 3-37 |
| C Stack Frame                          | 3-39 |
| Variable Argument List                 | 3-40 |
| Allocating Stack Space Dynamically     | 3-41 |

## Machine Interface

### Processor Architecture

The *MC68020 32-Bit Microprocessor User's Manual*, the *MC68030 Enhanced 32-Bit Microprocessor User's Manual*, and the *MC68040 32-Bit Microprocessor User's Manual* define the 68000 family processor architecture. The *M68000 Programmer's Reference Manual* defines the programming model. An MC68851 Paged Memory Management Unit (PMMU) may be present with the MC68020. An MC68881 or MC68882 Floating Point Coprocessor (FPCP) is assumed to be present with the MC68020 or MC68030. Programs intended to execute directly on the processor use the instruction set, instruction encodings, and instruction semantics of this architecture. A program may use only the instructions defined for the MC68040. Refer to the *M68000 Programmer's Reference Manual* for information regarding proper instruction usage for the MC68020, MC68030, and MC68040 processors.

To be ABI-conforming, the processor must implement the architecture's instructions, perform the specified operations, and produce the specified results. The ABI neither places performance constraints on systems nor specifies what instructions must be implemented in hardware. A software emulation of the architecture could conform to the ABI.

Some processors might support the 68000 and MC68881/2 FPCP architectures as subsets, providing additional instructions or capabilities. Programs that use those capabilities explicitly do not conform to the 68000 ABI. Executing those programs on machines without the additional capabilities gives undefined behavior.

### Data Representation

#### Fundamental Types

Figure 3-1 shows the correspondence between ANSI C's scalar types and the processor's.

Figure 3-1: Scalar Types

| Type           | C               | sizeof | Alignment<br>(bytes) | 68000                         |
|----------------|-----------------|--------|----------------------|-------------------------------|
| Integral       | signed char     | 1      | 1                    | signed byte                   |
|                | char            |        |                      |                               |
|                | unsigned char   | 1      | 1                    | unsigned byte                 |
|                | short           | 2      | 2                    | signed word (2 bytes)         |
|                | signed short    |        |                      |                               |
|                | unsigned short  | 2      | 2                    | unsigned word                 |
|                | int             |        |                      |                               |
|                | signed int      | 4      | 4                    | signed long word<br>(4 bytes) |
|                | long            |        |                      |                               |
|                | signed long     |        |                      |                               |
| Pointer        | enum            |        |                      |                               |
|                | unsigned int    | 4      | 4                    | unsigned long word            |
| Floating-point | unsigned long   |        |                      |                               |
|                | any-type *      | 4      | 4                    | unsigned long word            |
|                | any-type (*) () |        |                      |                               |
| Floating-point | float           | 4      | 4                    | single-precision              |
|                | double          | 8      | 8                    | double-precision              |
|                | long double     | 16     | 8                    | extended-precision            |

A null pointer (for all types) has the value zero.

**NOTE**

If a double or long double appears on the stack, not as a part of an aggregate or union, then it is aligned to 4 bytes.

## Aggregates and Unions

An array assumes the alignment of its elements' type. The size of any object, including arrays, structures, and unions, always is a multiple of the object's alignment. Structure and union objects may, therefore, require padding to meet size and alignment constraints.

- The alignment of a structure or a union is the maximum of the alignment of its elements.
- Each member is assigned to the lowest available offset with the appropriate alignment. This may require *internal padding*, depending on the previous member.
- A structure's size is increased, if necessary, to make it a multiple of the structure's alignment. This may require *tail padding*, depending on the last member.

**NOTE**

Aggregates or unions residing on the stack only require 4-byte alignment.

In the following examples, members' byte offsets appear in the upper left corners.

Figure 3-2: Structure Smaller Than a Long Word

```
struct {
    char c;
};
```

Byte aligned, sizeof is 1  
0 c

Figure 3-3: No Padding

```
struct {
    char c;
    char d;
    short s;
    long n;
};
```

Long word aligned, sizeof is 8

|   |   |   |   |   |   |   |
|---|---|---|---|---|---|---|
| 0 | c | 1 | d | 2 | s |   |
| 4 |   |   |   |   |   | n |

Figure 3-4: Internal Padding

```
struct {
    char c;
    short s;
};
```

Word aligned, sizeof is 4

|   |   |   |     |   |   |  |
|---|---|---|-----|---|---|--|
| 0 | c | 1 | pad | 2 | s |  |
|---|---|---|-----|---|---|--|

Figure 3-5: Internal and Tail Padding

```
struct {
    char c;
    double d;
    short s;
};
```

8-byte aligned (4-byte on stack), sizeof is 24

|    |     |    |     |  |  |   |
|----|-----|----|-----|--|--|---|
| 0  | c   | 1  | pad |  |  |   |
| 4  | pad |    |     |  |  | d |
| 8  | d   |    |     |  |  |   |
| 12 | d   |    |     |  |  |   |
| 16 | s   | 18 | pad |  |  |   |
| 20 | pad |    |     |  |  |   |

Figure 3-6: union Allocation

```
union {
    char c;
    short s;
    int j;
};
```

Long word aligned, sizeof is 4

|   |     |   |     |  |  |   |
|---|-----|---|-----|--|--|---|
| 0 | c   | 1 | pad |  |  |   |
| 0 | pad |   |     |  |  | s |
| 0 | j   |   |     |  |  |   |

**Bit-Fields**

C struct and union definitions may have *bit-fields*, defining integral objects with a specified number of bits.

Figure 3-7: Bit-Field Ranges

| Bit-field Type | Width $w$ | Range                     |
|----------------|-----------|---------------------------|
| signed char    |           | $-2^{w-1}$ to $2^{w-1}-1$ |
| char           | 1 to 8    | 0 to $2^w-1$              |
| unsigned char  |           | 0 to $2^w-1$              |
| signed short   |           | $-2^{w-1}$ to $2^{w-1}-1$ |
| short          | 1 to 16   | 0 to $2^w-1$              |
| unsigned short |           | 0 to $2^w-1$              |
| signed int     |           | $-2^{w-1}$ to $2^{w-1}-1$ |
| int            | 1 to 32   | 0 to $2^w-1$              |
| enum           |           | 0 to $2^w-1$              |
| unsigned int   |           | 0 to $2^w-1$              |
| signed long    |           | $-2^{w-1}$ to $2^{w-1}-1$ |
| long           | 1 to 32   | 0 to $2^w-1$              |
| unsigned long  |           | 0 to $2^w-1$              |

"Plain" bit-fields always have non-negative values. Although they may have type `char`, `short`, `int`, or `long` (which can have negative values), these bit-fields are extracted into a long word with zero fill. Bit-fields obey the same size and alignment rules as other structure and union members, with the following additions.

- Bit-fields are allocated from left to right (most to least significant).
- A bit-field must entirely reside in a storage unit appropriate for its declared type. Thus a bit-field never crosses its unit boundary.
- Bit-fields may share a storage unit with other `struct/union` members, including members that are not bit-fields. Of course, `struct` members occupy different parts of the storage unit.
- Unnamed bit-fields' types do not affect the alignment of a structure or union, although individual bit-fields member offsets obey the alignment constraints.

The following examples show `struct` and `union` members' byte offsets in the upper left corners; bit numbers appear in the lower corners.

Figure 3-8: Bit Numbering



Figure 3-9: Left-to-Right Allocation



Figure 3-10: Boundary Alignment



**Figure 3-11: Storage Unit Sharing**

```
struct {          Word aligned, sizeof is 2
    char c;
    short s:8;
};
```



As the examples show, int bit-fields (including signed and unsigned) pack more densely than smaller base types. One can use char and short bit-fields to force particular alignments, but int generally works better.

**Figure 3-12: union Allocation**

```
union {          Word aligned, sizeof is 2
    char c;
    short s:8;
};
```

**Figure 3-13: Unnamed Bit-Fields**

```
struct {          Byte aligned, sizeof is 9
    char c;
    int :0;
    char d;
    short :9;
    char e;
    char :0;
};
```



## Function Calling Sequence

This section discusses the standard function calling sequence, including stack frame layout, register usage, parameter passing, etc. The operating system interface and C programs use this calling sequence. See "Coding Examples" later in this chapter for more information on C.

**NOTE**

All of the coding examples in this section are simply illustrations of a sample implementation.

## Registers and the Stack Frame

The 68000 provides 8 data and 8 address registers, which are global to a running program. Additionally, the MC68040, or the MC68881/2 Floating Point Coprocessor, provides 8 global floating-point registers. Brief register descriptions appear in Figure 3-14; more complete information appears later.

**Figure 3-14: Processor Registers**

| Type                  | Names |         | Usage                                       |
|-----------------------|-------|---------|---------------------------------------------|
| 68000                 | %d0   |         |                                             |
|                       | %d1   |         |                                             |
|                       | %a0   |         |                                             |
|                       | %a1   |         | Scratch registers                           |
|                       | %d2   |         |                                             |
|                       | ...   |         |                                             |
|                       | %d7   |         |                                             |
|                       | %a2   |         |                                             |
|                       | ...   |         |                                             |
|                       | %a5   |         |                                             |
| MC68040,<br>MC68881/2 | %a6   | %fp     | Frame pointer (if implemented)              |
|                       | %a7   | %sp     | Stack pointer                               |
|                       |       | %pc     | Program counter                             |
|                       |       | %ccr    | Condition code register                     |
| MC68040,<br>MC68881/2 | %fp0  |         | Scratch registers                           |
|                       | %fp1  |         |                                             |
|                       | %fp2  |         |                                             |
|                       | ...   |         |                                             |
|                       | %fp7  |         |                                             |
|                       |       | %fpcr   | Floating Point Control Register             |
|                       |       | %fpsr   | Floating Point Status Register              |
|                       |       | %fp iar | Floating Point Instruction Address Register |

In addition to the registers, each function has a frame on the run-time stack. This stack grows downward from high addresses. Figure 3-15 shows the stack frame organization right after function prologue processing has allocated the frame and saved the registers (see below).

Figure 3-15: Standard Stack Frame

| Base                   | Offset            | Contents                                            | Purpose            |                |
|------------------------|-------------------|-----------------------------------------------------|--------------------|----------------|
|                        | +4+4*n            | argument long word n<br>...<br>argument long word 0 | incoming arguments | High Addresses |
|                        | +4                | return address                                      |                    |                |
| %fp (optional)         |                   | previous %fp (optional)<br>unspecified<br>...       |                    |                |
| SP (SP after prologue) | variable size     | local storage and register save area                |                    |                |
| SP                     | stack top, unused |                                                     |                    | Low addresses  |

Several key points about the stack frame deserve mention.

- The stack frame is long word aligned.
- Functions may save registers %d2 through %d7, %a2 through %a6, and %fp2 through %fp7 as necessary, without saving unused registers.
- All incoming arguments reside on the stack. "Coding Examples" below explains how variable argument lists may be implemented.
- A frame pointer may be implemented.
- Other areas depend on the compiler and the code being compiled. The standard calling sequence does not define a maximum stack frame size, nor does it restrict how a language system uses the "unspecified" areas of the standard stack frame.

Registers %d2 through %d7, %a2 through %a6, and %fp2 through %fp7, which are visible to both a calling and a called function, "belong" to the calling function. In other words, a called function must save these registers' values before it changes them, restoring their values before it returns. Remaining registers "belong" to the called function. If a calling function wants to preserve such a register value

across a function call, it must save the value in its local stack frame.

Across function boundaries, the standard function prologue saves registers %d2 through %d7, %a2 through %a6, and %fp2 through %fp7 (as needed) and allocates stack space, including the required areas of Figure 3-15 and any private space it needs. The example below illustrates this, saving registers %fp, %d7, %a5, and %fp2 and allocating 80 bytes for local storage.

Figure 3-16: Function Prologue

```
fcn:
    link.l    %fp,-80
    movm.l    %d7/%a5,-(%sp)
    fmovm.x   %fp2,-(%sp)
```

The `movm` instruction manipulates registers as part of the normal function prologue and epilogue. As explained later, the function epilogue executes a `movm` instruction to unwind the stack and restore the saved registers to their original condition.

**NOTE** Strictly speaking, a function does not need the `movm` instructions if it preserves the registers as described above. Although some functions can be optimized to eliminate the `movm` instructions, the general case uses the standard prologue and epilogue.

Some registers have assigned roles.

%fp or %a6

The *frame pointer*, if used by an individual function, holds the address of the local storage within a stack frame, referenced as negative offsets from %fp.

%sp or %a7

The *stack pointer* holds the limit of the current stack frame, which is the address of the stack's topmost valid long word. The long word to which %sp points is "in" the valid stack.

|      |                                                                                                                                                                                                                                                                                          |
|------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| %d0  | <i>Integral return values appear in %d0.</i>                                                                                                                                                                                                                                             |
| %a0  | <i>Pointer return values appear in %a0. When calling a function that returns a structure or union, the caller allocates space for the return value and sets %a0 to its address. A function that returns a structure or union value places the same address in %a0 before it returns.</i> |
| %fp0 | <i>Floating-point return values appear in this register.</i>                                                                                                                                                                                                                             |

Except as specified here, %d0, %d1, %a0, %a1, %fp0, and %fp1 are scratch registers. Functions do not need to preserve their values for the caller.

*Local registers %d2 through %d7, %a2 through %a5, and %fp2 through %fp7 have no specified role in the standard calling sequence.*

Signals can interrupt processes [see `signal(BA_OS)`]. Functions called during signal handling have no unusual restrictions on their use of registers. Moreover, if a signal handling function returns, the process resumes its original execution path with registers restored to their original values. Thus programs and compilers may freely use all registers without the danger of signal handlers changing their values.

## Integral and Pointer Arguments

As mentioned, a function receives all integral and pointer argument long words on the stack. Functions pass all integer-valued arguments as long words, expanding signed or unsigned bytes and words as needed.

**NOTE**

The following examples use the frame pointer. Functions not using %fp would find arguments at different locations.

Figure 3-17: Integral and Pointer Arguments

| Call                                    | Argument                 | Callee                                  |
|-----------------------------------------|--------------------------|-----------------------------------------|
| <code>g(1, 2, 3,<br/>(void *)0);</code> | 1<br>2<br>3<br>(void *)0 | 8(%fp)<br>12(%fp)<br>16(%fp)<br>20(%fp) |

## Floating-Point Arguments

The stack also holds floating-point arguments: single-precision values use one long word, double-precision use two, and extended-precision use four. The example below uses only double-precision arguments.

Figure 3-18: Floating-Point Arguments

| Call                                    | Argument                                                                                        | Callee                                             |
|-----------------------------------------|-------------------------------------------------------------------------------------------------|----------------------------------------------------|
| <code>h(1.414, 1,<br/>2.998e10);</code> | long word 0, 1.414<br>long word 1, 1.414<br>1<br>long word 0, 2.998e10<br>long word 1, 2.998e10 | 8(%fp)<br>12(%fp)<br>16(%fp)<br>20(%fp)<br>24(%fp) |