forked from markfloryan/pdr
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathx86-64bit.tex
546 lines (478 loc) · 27.6 KB
/
x86-64bit.tex
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
\begin{quotation}
\ldots
\end{quotation}
\noindent {\em This chapter was derived from a document written by Adam Ferrari and later updated by Alan Batson, Mike Lack, Anita
Jones, and Aaron Bloomfield}
\section{Introduction}
This small guide, in combination with the material covered in the
class lectures on assembly language programming, should provide enough
information to do the assembly language labs for this class. In this
guide, we describe the basics of 64-bit x86 assembly language
programming, covering a small but useful subset of the available
instructions and assembler directives. However, real x86 programming
is a large and extremely complex universe, much of which is beyond the
useful scope of this class. For example, there exists real (albeit
older) x86 code running in the world was written using the 16-bit
subset of the x86 instruction set. Using the 16-bit programming model
can be quite complex~-- it has a segmented memory model, more
restrictions on register usage, and so on. This was expanded into a
32-bit programming model in 1985, but that had the limit of only 4 Gb
of memory. In this guide we'll restrict our attention to the more
modern aspects of 64-bit x86 programming, and delve into the
instruction set only in enough detail to get a basic feel for
programming x86 compatible chips at the hardware level.
\section{Registers}
Modern 64-bit x86 processors have sixteen 64-bit general purpose
registers, as depicted in Figure~\ref{x86-register-diagram.fig}. The
register names for the first eight registers are mostly historical in
nature; the last eight registers were given sequential numbers. For
example, RAX used to be EAX (in the 32-bit machine), which used to be
called the ``accumulator'' since it was used by a number of arithmetic
operations, and RCX (32-bit version: ECX) was known as the ``counter''
since it was used to hold a loop index. Whereas most of the registers
have lost their special purposes in the modern instruction set, by
convention, two are sometimes reserved for special purposes~-- the
stack pointer (RSP) and the base pointer (RBP).
\begin{figure}[h]
\begin{center}
\includegraphics[width=4in]{x86-64bit/x86-register-diagram.pdf}
\end{center}
\caption{The x86 register set}
\label{x86-register-diagram.fig}
\end{figure}
In all cases, subsections of the registers may be used. For example,
the least significant 2 bytes of RAX can be treated as a 16-bit
register called AX. The least significant byte of AX can be used as a
single 8-bit register called AL, while the most significant byte of AX
can be used as a single 8-bit register called AH. It is important to
realize that these names refer to the same physical register. When a
two-byte quantity is placed into DX, the update affects the value of
RDX (in particular, the least significant 16 bits of RDX). These
``sub-registers'' are mainly hold-overs from older, 16-bit versions of
the instruction set. However, they are sometimes convenient when
dealing with data that are smaller than 64-bits (e.g., 1-byte ASCII
characters). Note that four of the registers (EAX, EBX, ECX, and EDX)
have an addition ``sub-register'' spot: the second to last byte of the
register.
When referring to registers in assembly language, the names are not
case-sensitive. For example, the names RAX and rax refer to the same
register.
\section{Memory and Addressing Modes}
\subsection{Declaring Static Data Regions}
You can declare static data regions (analogous to global variables) in
x86 assembly using special assembler directives for this purpose.
Data declarations should be preceded by the .DATA directive. Following
this directive, the directives DB, DW, and DD can be used to declare
one, two, and four byte data locations, respectively. Declared
locations can be labeled with names for later reference - this is
similar to declaring variables by name, but abides by some lower level
rules. For example, locations declared in sequence will be located in
memory next to one another. Some example declarations are depicted in
Listing~\ref{x86-declaring-memory-regions.lst}.
\begin{figure}[h]
\lstinputlisting[caption=Declaring x86 memory regions,label={x86-declaring-memory-regions.lst},backgroundcolor=\color{white},frame=trBL,linewidth=7in,xleftmargin=0in,language={[x86masm]Assembler}]{x86-64bit/code/memory-regions.s}
\end{figure}
The last example in Listing~\ref{x86-declaring-memory-regions.lst}
illustrates the declaration of an array. Unlike in high level
languages where arrays can have many dimensions and are accessed by
indices, arrays in assembly language are simply a number of cells
located contiguously in memory. Two other common methods used for
declaring arrays of data are the TIMES directive and the use of string
literals. The TIMES directive tells the assembler to duplicate an
expression a given number of times. For example, the statement ``TIMES
4 DB 2'' is equivalent to ``2, 2, 2, 2''. Some examples of declaring
arrays are depicted in Listing~\ref{x86-declaring-arrays.lst}.
\begin{figure}[h]
\lstinputlisting[caption=Declaring x86 arrays in memory,label={x86-declaring-arrays.lst},backgroundcolor=\color{white},frame=trBL,linewidth=7in,xleftmargin=0in,language={[x86masm]Assembler}]{x86-64bit/code/declaring-arrays.s}
\end{figure}
\subsection{Addressing Memory}
\label{addressing-memory.sec}
Modern x86-compatible processors are capable of addressing up to
$2^{64}$ bytes of memory; that is, memory addresses are 64-bits wide.
For example, in Listings~\ref{x86-declaring-memory-regions.lst} and
\ref{x86-declaring-arrays.lst}, where we used labels to refer to memory
regions, these labels are actually replaced by the assembler with
64-bit quantities that specify addresses in memory. In addition to
supporting referring to memory regions by labels (i.e. constant
values), the x86 provides a flexible scheme for computing and
referring to memory addresses:
{\bf x86 Addressing Mode Rule}~-- Up to two of the 64-bit registers
and a 64-bit signed constant can be added together to compute a memory
address. One of the registers can be optionally pre-multiplied by 2,
4, or 8.
To see this memory addressing rule in action, we'll look at some
example mov instructions. As we'll see later in
Section~\ref{data-movement-instructions.sec}, the {\tt mov} instruction
moves data between registers and memory. This instruction has two
operands~-- the first is the destination (where we're moving data {\em
to}) and the second specifies the source (where we're getting the
data {\em from}). Some examples of mov instructions using address
computations that obey the above rule are shown in
Listing~\ref{x86-valid-addressing-modes.lst}.
\begin{figure}[h]
\lstinputlisting[caption={Valid x86 addressing modes},label={x86-valid-addressing-modes.lst},backgroundcolor=\color{white},frame=trBL,linewidth=7in,xleftmargin=0in,language={[x86masm]Assembler}]{x86-64bit/code/valid-addressing-modes.s}
\end{figure}
Some examples of incorrect address calculations are shown in Listing~\ref{x86-invalid-addressing-modes.lst}.
\begin{figure}[h]
\lstinputlisting[caption={Invalid x86 addressing modes},label={x86-invalid-addressing-modes.lst},backgroundcolor=\color{white},frame=trBL,linewidth=7in,xleftmargin=0in,language={[x86masm]Assembler}]{x86-64bit/code/invalid-addressing-modes.s}
\end{figure}
\subsection{Size Directives}
In general, the intended size of the of the data item at a given
memory address can be inferred from the assembly code instruction in
which it is referenced. For example, in all of the above instructions,
the size of the memory regions could be inferred from the size of the
register operand~-- when we were loading a 64-bit register, the
assembler could infer that the region of memory we were referring to
was 8 bytes wide. When we were storing the value of a one byte
register to memory, the assembler could infer that we wanted the
address to refer to a single byte in memory. However, in some cases
the size of a referred-to memory region is ambiguous. Consider the
instruction {\tt mov [ebx], 2}.
Should this instruction move the value 2 into the single byte at
address EBX? Perhaps it should move the 64-bit integer representation
of 2 into the 4-bytes starting at address EBX. Since either is a valid
possible interpretation, the assembler must be explicitly directed as
to which is correct. The size directives {\tt BYTE PTR}, {\tt WORD
PTR}, {\tt DWORD PTR}, and {\tt QWORD PTR} serve this purpose. For
examples, see Listing~\ref{x86-size-directives.lst}.
\begin{figure}[h]
\lstinputlisting[caption={x86 size directive usage},label={x86-size-directives.lst},backgroundcolor=\color{white},frame=trBL,linewidth=7in,xleftmargin=0in,language={[x86masm]Assembler}]{x86-64bit/code/size-directives.s}
\end{figure}
\section{Instructions}
Machine instructions generally fall into three categories: data movement, arithmetic/logic,
and control-flow. In this section, we will look at important examples of x86 instructions from
each category. This section should not be considered an exhaustive list of x86 instructions, but
rather a useful subset.
In this section, we will use the following notation:
\begin{itemlist}
\item $<$reg64$>$ - means any 64-bit register described in Section 2, for example, ESI.
\item $<$reg16$>$ - means any 16-bit register described in Section 2, for example, BX.
\item $<$reg32$>$ - means any 32-bit register described in Section 2, for example, BX.
\item $<$reg8$>$ - means any 8-bit register described in Section 2, for example AL.
\item $<$reg$>$ - means any of the above.
\item $<$mem$>$ - will refer to a memory address, as described in Section~\ref{addressing-memory.sec}, for example {\tt [EAX]}, or
{\tt [var+4]}, or {\tt DWORD PTR [RAX+RBX]}.
\item $<$con64$>$ - means any 64-bit constant.
\item $<$con32$>$ - means any 32-bit constant.
\item $<$con16$>$ - means any 16-bit constant.
\item $<$con8$>$ - means any 8-bit constant.
\item $<$con$>$ - means any of the above sized constants.
\end{itemlist}
\subsection{Data Movement Instructions}
\label{data-movement-instructions.sec}
\asminstructionsummary{mov}
{mov $<$reg$>$,$<$reg$>$\newline mov $<$reg$>$,$<$mem$>$\newline mov $<$mem$>$,$<$reg$>$\newline mov $<$reg$>$,$<$const$>$\newline mov $<$mem$>$,$<$const$>$}
{The mov instruction moves the data item referred to by its second
operand (i.e. register contents, memory contents, or a constant
value) into the location referred to by its first operand (i.e. a
register or memory). While register-to-register moves are possible,
direct memory-to-memory moves are not. In cases where memory
transfers are desired, the source memory contents must first be
loaded into a register, then can be stored to the destination memory
address.}
{mov rax, rbx \linebreak mov BYTE PTR [var], 5 }
{; transfer ebx to eax\linebreak; store the value 5 into the byte at\linebreak; memory location ``var''}
{2.2in}{3.5in}
\asminstructionsummary{push}
{push $<$reg$>$\newline push $<$mem$>$\newline push $<$con64$>$}
{The push instruction places its operand onto the top of the hardware supported
stack in memory. Specifically, push first decrements RSP by 8, then places its
operand into the contents of the 64-bit location at address [RSP]. RSP (the stack
pointer) is decremented by push since the x86 stack grows down~-- i.e. the stack
grows from high addresses to lower addresses.}
{push rax\newline push [var]}
{; push the contents of rax onto the stack\newline
; push the 8 bytes at address ``var'' onto stack}
{1in}{4.7in}
\asminstructionsummary{pop}
{pop $<$reg$>$\newline pop $<$mem$>$}
{The pop instruction removes the 8-byte data element from the top of
the hardware supported stack into the specified operand (i.e.
register or memory location). Specifically, pop first moves the 8
bytes located at memory location [RSP] into the specified register or
memory location, and then increments SP by 8.}
{pop rdi \newline pop [rbx]}
{; pop the top element of the stack into RDI.\newline
; pop the top element of the stack into memory at\newline
; the four bytes starting at location RBX.}
{1in}{4.7in}
\asminstructionsummary{lea}
{lea $<$reg64$>$,$<$mem$>$}
{The lea instruction places the address specified by its second
operand into the register specified by its first operand. Note, the
contents of the memory location are not loaded~-- only the effective
address is computed and placed into the register. This is useful
for obtaining a ``pointer'' into a memory region.}
{lea rax, [var]\newline lea rdi, [rbx+4*rsi]}
{; the address of ``var'' is placed in RAX \newline
; the value RBX+4*RSI is placed in RDI}
{1.85in}{3.85in}
\subsection{Arithmetic and Logic Instructions}
\asminstructionsummary{add, sub}
{add $<$reg$>$,$<$reg$>$ \hspace{0.5in} sub $<$reg$>$,$<$reg$>$ \newline
add $<$reg$>$,$<$mem$>$ \hspace{0.5in} sub $<$reg$>$,$<$mem$>$ \newline
add $<$mem$>$,$<$reg$>$ \hspace{0.5in} sub $<$mem$>$,$<$reg$>$ \newline
add $<$reg$>$,$<$con$>$ \hspace{0.5in} sub $<$reg$>$,$<$con$>$ \newline
add $<$mem$>$,$<$con$>$ \hspace{0.5in} sub $<$mem$>$,$<$con$>$}
{The add instruction adds together its two operands, storing the
result in its first operand. Similarly, the sub instruction
subtracts its second operand from its first. Note, whereas both
operands may be registers, at most one operand may be a memory
location.}
{add rax, 10\newline
sub [var], rsi\newline\newline\newline
add BYTE PTR [var], 10}
{; add 10 to the contents of RAX. \newline
; subtract the contents of RSI from \newline
; the 64-bit integer stored at \newline
; memory location ``var''.\newline
; add 10 to the single byte stored\newline
; at memory address ``var''.\newline}
{2.2in}{3.5in}
\asminstructionsummary{inc, dec}
{inc $<$reg$>$ \hspace{0.5in} dec $<$reg$>$\newline
inc $<$mem$>$ \hspace{0.5in} dec $<$mem$>$}
{The inc instruction increments the contents of its operand by one,
and similarly dec decrements the contents of its operand by one.}
{dec rax \newline inc DWORD PTR [var]}
{; subtract one from the contents of RAX. \newline
; add one to the 64-bit integer stored at \newline
; memory location ``var''.}
{1.8in}{3.9in}
\asminstructionsummary{imul}
{imul $<$reg64$>$,$<$reg64$>$\newline
imul $<$reg64$>$,$<$mem$>$\newline
imul $<$reg64$>$,$<$reg64$>$,$<$con$>$\newline
imul $<$reg64$>$,$<$mem$>$,$<$con$>$}
{The imul instruction has two basic formats: two-operand (first two
syntax listings above) and three-operand (last two syntax listings
above). The two-operand form multiplies its two operands together and
stores the result in the first operand. The result (i.e., first)
operand must be a register. The three operand form multiplies its
second and third operands together and stores the result in its first
operand. Again, the result operand must be a register. Furthermore,
the third operand is restricted to being a constant value.}
{imul rax, [var]\newline\newline\newline
imul rsi, rdi, 25}
{; multiply the contents of RAX by the \newline
; 64-bit contents of the memory location \newline
; ``var''. Store the result in RAX.\newline
; multiply the contents of RDI by 25. \newline
; Store the result in RSI.}
{1.8in}{3.9in}
\asminstructionsummary{idiv}
{idiv $<$reg64$>$\newline
idiv $<$mem$>$}
{The idiv instruction is used to divide the contents of the 64 bit
integer RDX:RAX (constructed by viewing RDX as the most significant
eight bytes and RAX as the least significant eight bytes) by the
specified operand value. The quotient result of the division is
stored into RAX, while the remainder is placed in RDX. This
instruction must be used with care. Before executing the
instruction, the appropriate value to be divided must be placed into
RDX and RAX. Clearly, this value is overwritten when the idiv
instruction is executed.}
{idiv rbx \newline\newline\newline idiv DWORD PTR [var]}
{; divide the contents of RDX:RAX by the \newline
; contents of RBX. Place the quotient \newline
; in RAX and the remainder in RDX.\newline
; same as above, but divide by the \newline
; 64-bit value stored at memory \newline
; location ``var''.}
{1.9in}{3.8in}
\asminstructionsummary{and, or, xor}
{and $<$reg$>$,$<$reg$>$ \hspace{0.25in} or $<$reg$>$,$<$reg$>$ \hspace{0.25in} xor $<$reg$>$,$<$reg$>$ \newline
and $<$reg$>$,$<$mem$>$ \hspace{0.25in} or $<$reg$>$,$<$mem$>$ \hspace{0.25in} xor $<$reg$>$,$<$mem$>$ \newline
and $<$mem$>$,$<$reg$>$ \hspace{0.25in} or $<$mem$>$,$<$reg$>$ \hspace{0.25in} xor $<$mem$>$,$<$reg$>$ \newline
and $<$reg$>$,$<$con$>$ \hspace{0.25in} or $<$reg$>$,$<$con$>$ \hspace{0.25in} xor $<$reg$>$,$<$con$>$ \newline
and $<$mem$>$,$<$con$>$ \hspace{0.25in} or $<$mem$>$,$<$con$>$ \hspace{0.25in} xor $<$mem$>$,$<$con$>$}
{These instructions perform the specified logical operation (logical
bitwise and, or, and exclusive or, respectively) on their operands,
placing the result in the first operand location.}
{and rax, 0fH \newline xor rdx, rdx}
{; clear all but the last 4 bits of RAX. \newline ; set the contents of RDX to zero.}
{1.7in}{4in}
\asminstructionsummary{not}
{not $<$reg$>$ \newline not $<$mem$>$}
{Performs the logical negation of the operand contents (i.e., flips all bit values).}
{not BYTE PTR [var]}
{; negate all bits in the byte at the \newline ; memory location ``var''.}
{1.7in}{4in}
\asminstructionsummary{neg}
{neg $<$reg$>$ \newline neg $<$mem$>$}
{Performs the arithmetic (i.e., two's complement) negation of the operand contents.}
{neg rax \newline neg [var]}
{; negate the contents of RAX. \newline ; negate the contests of ``var''}
{1.4in}{4.3in}
\asminstructionsummary{shl, shr}
{shl $<$reg$>$,$<$con8$>$ \hspace{0.5in} shr $<$reg$>$,$<$con8$>$\newline
shl $<$mem$>$,$<$con8$>$ \hspace{0.5in} shr $<$mem$>$,$<$con8$>$\newline
shl $<$reg$>$,cl \hspace{0.92in} shr $<$reg$>$,cl\newline
shl $<$mem$>$,cl \hspace{0.92in} shr $<$mem$>$,cl}
{These instructions shift the bits in their first operand's contents
left and right ({\tt shl} and {\tt shr}, respectively), padding the
resulting empty bit positions with zeros. The shifted operand can be
shifted up to 31 places. The number of bits to shift is specified
by the second operand, which can be either an 8-bit constant or the
register CL. In either case, shifts counts of greater then 31 are
performed modulo 64.}
{shl rax 5 \newline\newline shr [var] 3}{; shift the contents of rax left by 5 bit \newline ; positions \newline ; shift the contents of ``var'' right by 3 \newline ; bit positions}{1.4in}{4.3in}
\subsection{Control Flow Instructions}
In this section, we will refer to labeled locations in the program
text as $<$label$>$. Labels can be inserted anywhere in x86 assembly
code text by entering a label name followed by a colon. For example,
consider the code fragment in Listing~\ref{x86-labeled-code-location.lst}.
The second instruction in this code fragment is labeled ``begin''.
Elsewhere in the code, we can refer to the memory location that this
instruction is located at in memory using the more convenient symbolic
name ``begin'' instead of having to refer to the memory address as an
integer.
\begin{figure}[h]
\lstinputlisting[caption={x86 labeled code location},label={x86-labeled-code-location.lst},backgroundcolor=\color{white},frame=trBL,linewidth=4.75in,xleftmargin=2.25in,language={[x86masm]Assembler}]{x86-64bit/code/labeled-code-location.s}
\end{figure}
\asminstructionsummary{jmp}
{jmp $<$label$>$}
{Transfers program control flow to the instruction at the memory
location indicated by the operand.}
{jmp begin}{; jumps to the ``begin'' label}
{1.5in}{3.7in}
\asminstructionsummary{jCC}
{je $<$label$>$ ; Jump when equal\newline
jne $<$label$>$ ; Jump when not equal\newline
jz $<$label$>$ ; Jump when last result was zero\newline
jg $<$label$>$ ; Jump when greater than\newline
jge $<$label$>$ ; Jump when greater than or equal to\newline
jl $<$label$>$ ; Jump when less than\newline
jle $<$label$>$ ; Jump when less than or equal to}
{These instructions are conditional jumps that are based on the status
of a set of {\bf condition codes} that are stored in a special
register called the {\em machine status word}. The contents of the
machine status word include information about the last arithmetic
operation performed. For example, one bit of this word indicates if
the last result was zero. Another indicates if the last result was
negative. Based on these condition codes, a number of conditional
jumps can be performed. For example, the jz instruction performs a
jump to the specified operand label if the result of the last
arithmetic operation (e.g. add, sub, etc.) was zero. Otherwise,
control proceeds to the next instruction in sequence after the jz.
These conditional jumps are the underlying support needed to
implement high-level language features such as ``if'' statements and
loops (e.g. ``while'' and ``for''). \vspace{6pt}\newline
A number of the conditional branches are given names that are
intuitively based on the last operation performed being a special
compare instruction, cmp (see below). For example, conditional
branches such as jle and jne are based on first performing a cmp
operation on the desired operands.}
{cmp rax, rbx \newline jle done}
{; if the contents of rax are less than or \newline
; equal to the contents of ebx, jump to the \newline
; code location labeled ``done''.}
{1.4in}{4.3in}
\asminstructionsummary{cmp}
{cmp $<$reg$>$,$<$reg$>$\newline
cmp $<$reg$>$,$<$mem$>$\newline
cmp $<$mem$>$,$<$reg$>$\newline
cmp $<$reg$>$,$<$con$>$\newline
cmp $<$mem$>$,$<$con$>$}
{Compares the two specified operands, setting the condition codes in
the machine status word appropriately. In fact, this instruction is
equivalent to the sub instruction, except the result of the
subtraction is discarded.}
{cmp DWORD PTR [var], 10\newline jeq loop}
{; if the 4 bytes stored at memory\newline
; location ``var'' equal the 4-byte\newline
; integer value 10, then jump to the\newline
; code location labeled loop}
{2.2in}{3.5in}
\asminstructionsummary{call}
{call $<$label$>$}
{This instruction implements a subroutine call that operates in
cooperation with the subroutine return instruction, ret, described
below. This instruction first pushes the current code location onto
the hardware supported stack in memory (see the push instruction for
details), and then performs an unconditional jump to the code location
indicated by the label operand. The added value of this instruction
(as compared to the simple jmp instruction) is that it saves the
location to return to when the subroutine completes.}
{call my\_subroutine}
{; jumps to the ``my\_subroutine'' label, \newline
; pushing the return address onto the \newline
; stack}
{1.8in}{3.9in}
\asminstructionsummary{ret}
{ret}
{In cooperation with the call instruction, the ret instruction
implements a subroutine return mechanism. This instruction first
pops a code location off the hardware supported in-memory stack (see
the pop instruction for details). It then performs an unconditional
jump to the retrieved code location.}
{ret}{; returns to the address on the top of the stack}
{1in}{4.7in}
\section{Basic Program Structure}
\label{x86-basic-program-structure.sec}
\begin{wrapfigure}{r}{0.37\textwidth}
\lstinputlisting[backgroundcolor=\color{white},frame=trBL,linewidth=2.5in,xleftmargin=0.25in,label={x86-return2.s.lst},language={[x86masm]Assembler},caption={x86 code to return 2}]{x86-64bit/code/return2.s}
\vspace{-0.25in}
\end{wrapfigure}
Given the above repertoire of instructions, you are in a position to
examine the basic skeletal structure of an assembly language
subroutine suitable for linking into C++ code. Unlike C++, which is
often used for the development of complete software systems, assembly
language is most often used in cooperation with other languages such
as Fortran, C, and C++. Commonly, most of a project is implemented in
the more convenient high-level language, and assembly language is used
sparingly to implement extremely low-level hardware interfaces or
performance-critical ``inner loops.'' Thus, in addition to
understanding how to program in assembly language, it is equally
important to understand how to link assembly language code into
high-level language programs.
Before examining the linkage conventions, we must first examine the
basic structure of an assembly language file. To do this, we can
compare a very simple assembly language file to an equivalent C++
file. In Listings~\ref{x86-return2.s.lst} and \ref{x86-return2.cpp.lst} we see
two files, one in C++, the other in x86 assembly. Each file includes a
function (albeit an ugly one) to return the integer value 2.
Note that the fucntion shown in assembly in
Listing~\ref{x86-return2.s.lst} returns a 32-bit value, since it is
put into register eax. This corresponds to retnring an {\tt int} in C
or C++. If we wanted to return a 64-bit value, which corresponds to
returning a {\tt long}, then we would put the return value into rax.
The top of the assembly file contains two directives that indicate the
instruction set and memory model we will use for all work in this
class (note, there are other possibilities~-- one might use only the
older 80286 instruction set for wider compatibility, for example).
Next, where in the C++ file we find the declaration of the global
variable ``var'', in the assembly file we find the use of the .DATA
and DD directives (described in Section 3.1) to reserve and initialize
a 4-byte (i.e., integer-sized) memory region labeled ``var''.
\begin{wrapfigure}{r}{0.37\textwidth}
\lstinputlisting[backgroundcolor=\color{white},frame=trBL,linewidth=2.5in,xleftmargin=0.25in,label={x86-return2.cpp.lst},language=C++,caption={C++ code to return 2}]{x86-64bit/code/return2.cpp}
\vspace{-0.25in}\end{wrapfigure}
Next in each file, we find the declaration of the function named
returnTwo. In the C++ file we have declared the function to be extern
``C''. This declaration indicates that the C++ compiler should use C
naming conventions when labeling the function returnTwo in the
resulting object file that it produces. In fact, this naming
convention means that the function returnTwo should map to the label
\_returnTwo in the object code. In the assembly code, we have labeled
the beginning of the subroutine \_returnTwo using the PROC directive,
and have declared the label \_returnTwo to be public. Again, the
result of these actions will be that the subroutine will map to the
symbol \_returnTwo in the object code that the assembler generates.
The function bodies are straight-forward. As we will see in more
detail in Section 6, return values for functions are placed into EAX
by convention, hence the instruction to move the contents of ``var''
into EAX in the assembly code.
\begin{wrapfigure}{r}{0.48\textwidth}
\lstinputlisting[backgroundcolor=\color{white},frame=trBL,linewidth=4in,xleftmargin=0.25in,label={x86-calling-return2.cpp.lst},language=C++,caption={Calling returnTwo() from C++}]{x86-64bit/code/calling-return2.cpp}
\vspace{-0.25in}
\end{wrapfigure}
Given these equivalent function definitions, use of either version of
the function is the same. A sample call to the function returnTwo is
depicted in Listing~\ref{x86-calling-return2.cpp.lst}. This C++ code
could be linked to either definition of the function and would produce
the same results (note, we could not link to both definitions, or the
linker would produce a ``multiply defined symbol'' error. The
mechanics of program linking will be discussed in an associated
document that relates to the specific programming environment that you
will use to assemble and run programs.