Contents

Part I: Intel® Itanium™ Instruction Set Descriptions

1 About this Manual ............................................................................................................. 3:1

1.1 Overview of Volume 1: Application Architecture.......................................................... 3:1

1.1.1 Part 1: Application Architecture Guide ........................................................... 3:1

1.1.2 Part 2: Optimization Guide for the Intel® Itanium™ Architecture ........................................ 3:2

1.2 Overview of Volume 2: System Architecture ............................................................... 3:2

1.2.1 Part 1: System Architecture Guide ................................................................. 3:2

1.2.2 Part 2: System Programmer’s Guide...................................................................... 3:3

1.2.3 Appendices..................................................................................................... 3:4

1.3 Overview of Volume 3: Instruction Set Reference ....................................................... 3:4

1.3.1 Part 1: Intel® Itanium™ Instruction Set Descriptions...................................... 3:4

1.3.2 Part 2: IA-32 Instruction Set Descriptions....................................................... 3:4

1.4 Terminology................................................................................................................. 3:5

1.5 Related Documents ..................................................................................................... 3:5

1.6 Revision History .......................................................................................................... 3:6

2 Instruction Reference ......................................................................................................... 3:9

2.1 Instruction Page Conventions ..................................................................................... 3:9

2.2 Instruction Descriptions ............................................................................................. 3:11

3 Pseudo-Code Functions .................................................................................................... 3:249

4 Instruction Formats........................................................................................................... 3:257

4.1 Format Summary ..................................................................................................... 3:258

4.2 A-Unit Instruction Encodings ................................................................................... 3:264

4.2.1 Integer ALU ................................................................................................ 3:264

4.2.2 Integer Compare ........................................................................................ 3:267

4.2.3 Multimedia .................................................................................................. 3:270

4.3 I-Unit Instruction Encodings .................................................................................... 3:274

4.3.1 Multimedia and Variable Shifts........................................................................ 3:274

4.3.2 Integer Shifts .............................................................................................. 3:279

4.3.3 Test Bit ....................................................................................................... 3:281

4.3.4 Miscellaneous I-Unit Instructions................................................................ 3:282

4.3.5 GR/BR Moves ............................................................................................ 3:284

4.3.6 GR/Predicate/IP Moves .............................................................................. 3:285

4.3.7 GR/AR Moves (I-Unit)................................................................................. 3:285

4.3.8 Sign/Zero Extend/Compute Zero Index ...................................................... 3:286

4.4 M-Unit Instruction Encodings .................................................................................. 3:287

4.4.1 Loads and Stores ....................................................................................... 3:287

4.4.2 Line Prefetch .............................................................................................. 3:301

4.4.3 Semaphores ............................................................................................... 3:303

4.4.4 Set/Get FR ................................................................................................. 3:304
4.4.5 Speculation and Advanced Load Checks ................................................. 3:305
4.4.6 Cache/Synchronization/RSE/ALAT ...................................................... 3:306
4.4.7 GR/AR Moves (M-Unit) ................................................................. 3:307
4.4.8 GR/CR Moves ............................................................................... 3:308
4.4.9 Miscellaneous M-Unit Instructions .................................................... 3:309
4.4.10 System/Memory Management .......................................................... 3:310
4.5 B-Unit Instruction Encodings ............................................................... 3:315
4.5.1 Branches ....................................................................................... 3:315
4.5.2 Branch Predict and Nop ................................................................. 3:319
4.5.3 Miscellaneous B-Unit Instructions ..................................................... 3:321
4.6 F-Unit Instruction Encodings ............................................................... 3:322
4.6.1 Arithmetic ..................................................................................... 3:324
4.6.2 Parallel Floating-point Select ......................................................... 3:325
4.6.3 Compare and Classify ..................................................................... 3:325
4.6.4 Approximation .............................................................................. 3:326
4.6.5 Minimum/Maximum and Parallel Compare ...................................... 3:327
4.6.6 Merge and Logical ....................................................................... 3:328
4.6.7 Conversion .................................................................................... 3:328
4.6.8 Status Field Manipulation .............................................................. 3:329
4.6.9 Miscellaneous F-Unit Instructions ................................................. 3:330
4.7 X-Unit Instruction Encodings ............................................................... 3:330
4.7.1 Miscellaneous X-Unit Instructions .................................................. 3:330
4.7.2 Move Long Immediate 64 ............................................................... 3:332
4.8 Immediate Formation .......................................................................... 3:333

5 Resource and Dependency Semantics .......................................................... 3:335
5.1 Reading and Writing Resources .............................................................. 3:335
5.2 Dependencies and Serialization ............................................................ 3:335
5.3 Resource and Dependency Table Format Notes .................................... 3:336
5.3.1 Special Case Instruction Rules ...................................................... 3:338
5.3.2 RAW Dependency Table ............................................................... 3:338
5.3.3 WAW Dependency Table .............................................................. 3:346
5.3.4 WAR Dependency Table .............................................................. 3:350
5.3.5 Listing of Rules Referenced in Dependency Tables ......................... 3:350
5.4 Support Tables .................................................................................. 3:351

Part II: IA-32 Instruction Set Descriptions

1 Base IA-32 Instruction Reference .......................................................... 3:359
1.1 Additional Intel® Itanium™ Faults ....................................................... 3:359
1.2 Interpreting the IA-32 Instruction Reference Pages ......................... 3:360
1.2.1 IA-32 Instruction Format .............................................................. 3:360
1.2.2 Operation .................................................................................... 3:363
1.2.3 Flags Affected ........................................................................... 3:366
1.2.4 FPU Flags Affected ................................................................. 3:366
1.2.5 Protected Mode Exceptions ....................................................... 3:367
1.2.6 Real-address Mode Exceptions .................................................. 3:367
1.2.7 Virtual-8086 Mode Exceptions ............................................................... 3:368
1.2.8 Floating-point Exceptions ........................................................................ 3:368
1.3 IA-32 Base Instruction Reference ............................................................... 3:368

2 IA-32 Intel® MMX™ Technology Instruction Reference ................................. 3:747

3 IA-32 Streaming SIMD Extension Instruction Reference .................................. 3:811
  3.1 IA-32 Streaming SIMD Extension Instructions ............................................. 3:811
  3.2 About the Intel® Architecture Streaming SIMD Extensions ........................ 3:811
  3.3 Single Instruction Multiple Data ................................................................. 3:812
  3.4 New Data Types ....................................................................................... 3:812
  3.5 Streaming SIMD Extension Registers ....................................................... 3:813
  3.6 Extended Instruction Set .......................................................................... 3:813
      3.6.1 Instruction Group Review .................................................................. 3:814
  3.7 IEEE Compliance ..................................................................................... 3:821
      3.7.1 Real Number System ....................................................................... 3:822
      3.7.2 Operating on NaNs ......................................................................... 3:827
  3.8 Data Formats ......................................................................................... 3:828
      3.8.1 Memory Data Formats ..................................................................... 3:828
      3.8.2 Streaming SIMD Extension Register Data Formats .......................... 3:828
  3.9 Instruction Formats .................................................................................. 3:830
  3.10 Instruction Prefixes ............................................................................... 3:830
  3.11 Reserved Behavior and Software Compatibility ....................................... 3:831
  3.12 Notations ............................................................................................. 3:831
  3.13 SIMD Integer Instruction Set Extensions ............................................... 3:908
  3.14 Cacheability Control Instructions ............................................................ 3:922
## Figures

**Part I: Intel® Itanium™ Instruction Set Descriptions**

<table>
<thead>
<tr>
<th>Figure</th>
<th>Description</th>
<th>Page</th>
</tr>
</thead>
<tbody>
<tr>
<td>2-1</td>
<td>Add Pointer</td>
<td>3:13</td>
</tr>
<tr>
<td>2-2</td>
<td>Stack Frame</td>
<td>3:14</td>
</tr>
<tr>
<td>2-3</td>
<td>Operation of br.ctop and br.cexit</td>
<td>3:21</td>
</tr>
<tr>
<td>2-4</td>
<td>Operation of br.wtop and br.wexit</td>
<td>3:21</td>
</tr>
<tr>
<td>2-5</td>
<td>Deposit Example (merge_form)</td>
<td>3:46</td>
</tr>
<tr>
<td>2-6</td>
<td>Deposit Example (zero_form)</td>
<td>3:46</td>
</tr>
<tr>
<td>2-7</td>
<td>Extract Example</td>
<td>3:49</td>
</tr>
<tr>
<td>2-8</td>
<td>Floating-point Merge Negative Sign Operation</td>
<td>3:73</td>
</tr>
<tr>
<td>2-9</td>
<td>Floating-point Merge Sign Operation</td>
<td>3:73</td>
</tr>
<tr>
<td>2-10</td>
<td>Floating-point Merge Sign and Exponent Operation</td>
<td>3:73</td>
</tr>
<tr>
<td>2-11</td>
<td>Floating-point Mix Left</td>
<td>3:76</td>
</tr>
<tr>
<td>2-12</td>
<td>Floating-point Mix Right</td>
<td>3:76</td>
</tr>
<tr>
<td>2-13</td>
<td>Floating-point Mix Left-Right</td>
<td>3:76</td>
</tr>
<tr>
<td>2-14</td>
<td>Floating-point Pack</td>
<td>3:88</td>
</tr>
<tr>
<td>2-15</td>
<td>Floating-point Parallel Merge Negative Sign Operation</td>
<td>3:100</td>
</tr>
<tr>
<td>2-16</td>
<td>Floating-point Parallel Merge Sign Operation</td>
<td>3:100</td>
</tr>
<tr>
<td>2-17</td>
<td>Floating-point Parallel Merge Sign and Exponent Operation</td>
<td>3:100</td>
</tr>
<tr>
<td>2-18</td>
<td>Floating-point Swap</td>
<td>3:123</td>
</tr>
<tr>
<td>2-19</td>
<td>Floating-point Swap Negate Left</td>
<td>3:123</td>
</tr>
<tr>
<td>2-20</td>
<td>Floating-point Swap Negate Right</td>
<td>3:123</td>
</tr>
<tr>
<td>2-21</td>
<td>Floating-point Sign Extend Left</td>
<td>3:125</td>
</tr>
<tr>
<td>2-22</td>
<td>Floating-point Sign Extend Right</td>
<td>3:125</td>
</tr>
<tr>
<td>2-23</td>
<td>Function of getf.exp</td>
<td>3:128</td>
</tr>
<tr>
<td>2-24</td>
<td>Function of getf.sig</td>
<td>3:128</td>
</tr>
<tr>
<td>2-25</td>
<td>Mix Example</td>
<td>3:151</td>
</tr>
<tr>
<td>2-26</td>
<td>Mux1 Operation (8-bit elements)</td>
<td>3:168</td>
</tr>
<tr>
<td>2-27</td>
<td>Mux2 Examples (16-bit elements)</td>
<td>3:169</td>
</tr>
<tr>
<td>2-28</td>
<td>Pack Operation</td>
<td>3:173</td>
</tr>
<tr>
<td>2-29</td>
<td>Parallel Add Examples</td>
<td>3:175</td>
</tr>
<tr>
<td>2-30</td>
<td>Parallel Average Example</td>
<td>3:178</td>
</tr>
<tr>
<td>2-31</td>
<td>Parallel Average with Round Away from Zero Example</td>
<td>3:179</td>
</tr>
<tr>
<td>2-32</td>
<td>Parallel Average Subtract Example</td>
<td>3:181</td>
</tr>
<tr>
<td>2-33</td>
<td>Parallel Compare Example</td>
<td>3:183</td>
</tr>
<tr>
<td>2-34</td>
<td>Parallel Maximum Example</td>
<td>3:185</td>
</tr>
<tr>
<td>2-35</td>
<td>Parallel Minimum Example</td>
<td>3:186</td>
</tr>
<tr>
<td>2-36</td>
<td>Parallel Multiply Operation</td>
<td>3:187</td>
</tr>
<tr>
<td>2-37</td>
<td>Parallel Multiply and Shift Right Operation</td>
<td>3:188</td>
</tr>
<tr>
<td>2-38</td>
<td>Parallel Sum of Absolute Difference Example</td>
<td>3:193</td>
</tr>
<tr>
<td>2-39</td>
<td>Parallel Shift Left Example</td>
<td>3:194</td>
</tr>
<tr>
<td>2-40</td>
<td>Parallel Subtract Example</td>
<td>3:199</td>
</tr>
<tr>
<td>2-41</td>
<td>Function of setf.exp</td>
<td>3:214</td>
</tr>
<tr>
<td>2-42</td>
<td>Function of setf.sig</td>
<td>3:214</td>
</tr>
<tr>
<td>2-43</td>
<td>Shift Left and Add Pointer</td>
<td>3:218</td>
</tr>
<tr>
<td>2-44</td>
<td>Shift Right Pair</td>
<td>3:220</td>
</tr>
<tr>
<td>2-45</td>
<td>Unpack Operation</td>
<td>3:240</td>
</tr>
<tr>
<td>4-1</td>
<td>Bundle Format</td>
<td>3:257</td>
</tr>
</tbody>
</table>

**Part II: IA-32 Instruction Set Descriptions**
<table>
<thead>
<tr>
<th>Section</th>
<th>Page</th>
</tr>
</thead>
<tbody>
<tr>
<td>1-1 Bit Offset for BIT[EAX,21]</td>
<td>3:366</td>
</tr>
<tr>
<td>1-2 Memory Bit Indexing</td>
<td>3:366</td>
</tr>
<tr>
<td>1-3 Version Information in Registers EAX</td>
<td>3:428</td>
</tr>
<tr>
<td>2-1 Operation of the MOVQ Instruction</td>
<td>3:749</td>
</tr>
<tr>
<td>2-2 Operation of the MOVD Instruction</td>
<td>3:751</td>
</tr>
<tr>
<td>2-3 Operation of the PACKSSDW Instruction</td>
<td>3:753</td>
</tr>
<tr>
<td>2-4 Operation of the PACKUSWB Instruction</td>
<td>3:756</td>
</tr>
<tr>
<td>2-5 Operation of the PADDW Instruction</td>
<td>3:758</td>
</tr>
<tr>
<td>2-6 Operation of the PADDSDW Instruction</td>
<td>3:761</td>
</tr>
<tr>
<td>2-7 Operation of the PADDUSB Instruction</td>
<td>3:764</td>
</tr>
<tr>
<td>2-8 Operation of the PAND Instruction</td>
<td>3:767</td>
</tr>
<tr>
<td>2-9 Operation of the PANDN Instruction</td>
<td>3:769</td>
</tr>
<tr>
<td>2-10 Operation of the PCMPEQW Instruction</td>
<td>3:771</td>
</tr>
<tr>
<td>2-11 Operation of the PCMPGTW Instruction</td>
<td>3:774</td>
</tr>
<tr>
<td>2-12 Operation of the PMADDWD Instruction</td>
<td>3:777</td>
</tr>
<tr>
<td>2-13 Operation of the PMULHW Instruction</td>
<td>3:779</td>
</tr>
<tr>
<td>2-14 Operation of the PMULLW Instruction</td>
<td>3:781</td>
</tr>
<tr>
<td>2-15 Operation of the POR Instruction</td>
<td>3:783</td>
</tr>
<tr>
<td>2-16 Operation of the PSLLW Instruction</td>
<td>3:785</td>
</tr>
<tr>
<td>2-17 Operation of the PSRAW Instruction</td>
<td>3:788</td>
</tr>
<tr>
<td>2-18 Operation of the PSRLW Instruction</td>
<td>3:791</td>
</tr>
<tr>
<td>2-19 Operation of the PSUBW Instruction</td>
<td>3:794</td>
</tr>
<tr>
<td>2-20 Operation of the PSUBSW Instruction</td>
<td>3:797</td>
</tr>
<tr>
<td>2-21 Operation of the PSUBUSB Instruction</td>
<td>3:800</td>
</tr>
<tr>
<td>2-22 High-order Unpacking and Interleaving of Bytes with the PUNPCKHBW Instruction</td>
<td>3:803</td>
</tr>
<tr>
<td>2-23 Low-order Unpacking and Interleaving of Bytes with the PUNPCKLBW Instruction</td>
<td>3:806</td>
</tr>
<tr>
<td>2-24 Operation of the PXOR Instruction</td>
<td>3:809</td>
</tr>
<tr>
<td>3-1 Packed Single-FP Data Type</td>
<td>3:812</td>
</tr>
<tr>
<td>3-2 Streaming SIMD Extension Register Set</td>
<td>3:813</td>
</tr>
<tr>
<td>3-3 Packed Operation</td>
<td>3:814</td>
</tr>
<tr>
<td>3-4 Scalar Operation</td>
<td>3:814</td>
</tr>
<tr>
<td>3-5 Packed Shuffle Operation</td>
<td>3:816</td>
</tr>
<tr>
<td>3-6 Unpack High Operation</td>
<td>3:817</td>
</tr>
<tr>
<td>3-7 Unpack Low Operation</td>
<td>3:817</td>
</tr>
<tr>
<td>3-8 Binary Real Number System</td>
<td>3:822</td>
</tr>
<tr>
<td>3-9 Binary Floating-point Format</td>
<td>3:823</td>
</tr>
<tr>
<td>3-10 Real Numbers and NaNs</td>
<td>3:825</td>
</tr>
<tr>
<td>3-11 Four Packed FP Data in Memory (at address 1000H)</td>
<td>3:828</td>
</tr>
</tbody>
</table>
# Tables

**Part I: Intel® Itanium™ Instruction Set Descriptions**

<table>
<thead>
<tr>
<th>Page</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>2-1</td>
<td>Instruction Page Description .........................................................</td>
</tr>
<tr>
<td>2-2</td>
<td>Instruction Page Font Conventions ......................................................</td>
</tr>
<tr>
<td>2-3</td>
<td>Register File Notation ........................................................................</td>
</tr>
<tr>
<td>2-4</td>
<td>C Syntax Differences ............................................................................</td>
</tr>
<tr>
<td>2-5</td>
<td>Pervasive Conditions Not Included in Instruction Description Code ...........</td>
</tr>
<tr>
<td>2-6</td>
<td>Branch Types .......................................................................................</td>
</tr>
<tr>
<td>2-7</td>
<td>Branch Whether Hint ............................................................................</td>
</tr>
<tr>
<td>2-8</td>
<td>Sequential Prefetch Hint .......................................................................</td>
</tr>
<tr>
<td>2-9</td>
<td>Branch Cache Deallocation Hint ..........................................................</td>
</tr>
<tr>
<td>2-10</td>
<td>Long Branch Types ...............................................................................</td>
</tr>
<tr>
<td>2-11</td>
<td>IP-relative Branch Predict Whether Hint ...............................................</td>
</tr>
<tr>
<td>2-12</td>
<td>Indirect Branch Predict Whether Hint ..................................................</td>
</tr>
<tr>
<td>2-13</td>
<td>Importance Hint ....................................................................................</td>
</tr>
<tr>
<td>2-14</td>
<td>ALAT Clear Completer ...........................................................................</td>
</tr>
<tr>
<td>2-15</td>
<td>Comparison Types ................................................................................</td>
</tr>
<tr>
<td>2-16</td>
<td>64-bit Comparison Relations for Normal and unc Compares .......................</td>
</tr>
<tr>
<td>2-17</td>
<td>64-bit Comparison Relations for Parallel Compares ..................................</td>
</tr>
<tr>
<td>2-18</td>
<td>Immediate Range for 32-bit Compares ..................................................</td>
</tr>
<tr>
<td>2-19</td>
<td>Memory Compare and Exchange Size .......................................................</td>
</tr>
<tr>
<td>2-20</td>
<td>Compare and Exchange Semaphore Types ...............................................</td>
</tr>
<tr>
<td>2-21</td>
<td>Result Ranges for czx ..........................................................................</td>
</tr>
<tr>
<td>2-22</td>
<td>Specified pc Mnemonic Values ............................................................</td>
</tr>
<tr>
<td>2-23</td>
<td>sf Mnemonic Values ..............................................................................</td>
</tr>
<tr>
<td>2-24</td>
<td>Floating-point Class Relations .............................................................</td>
</tr>
<tr>
<td>2-25</td>
<td>Floating-point Classes .........................................................................</td>
</tr>
<tr>
<td>2-26</td>
<td>Floating-point Comparison Types ..........................................................</td>
</tr>
<tr>
<td>2-27</td>
<td>Floating-point Comparison Relations ......................................................</td>
</tr>
<tr>
<td>2-28</td>
<td>Fetch and Add Semaphore Types ............................................................</td>
</tr>
<tr>
<td>2-29</td>
<td>Floating-point Parallel Comparison Results ............................................</td>
</tr>
<tr>
<td>2-30</td>
<td>Floating-point Parallel Comparison Relations .........................................</td>
</tr>
<tr>
<td>2-31</td>
<td>sz Completers ........................................................................................</td>
</tr>
<tr>
<td>2-32</td>
<td>Load Types ...........................................................................................</td>
</tr>
<tr>
<td>2-33</td>
<td>Load Hints ............................................................................................</td>
</tr>
<tr>
<td>2-34</td>
<td>fsz Completers .....................................................................................</td>
</tr>
<tr>
<td>2-35</td>
<td>FP Load Types .....................................................................................</td>
</tr>
<tr>
<td>2-36</td>
<td>lftype Mnemonic Values .........................................................................</td>
</tr>
<tr>
<td>2-37</td>
<td>lhint Mnemonic Values ..........................................................................</td>
</tr>
<tr>
<td>2-38</td>
<td>Move to BR Whether Hints ....................................................................</td>
</tr>
<tr>
<td>2-39</td>
<td>Indirect Register File Mnemonics ..........................................................</td>
</tr>
<tr>
<td>2-40</td>
<td>Mux Permutations for 8-bit Elements .....................................................</td>
</tr>
<tr>
<td>2-41</td>
<td>Pack Saturation Limits ..........................................................................</td>
</tr>
<tr>
<td>2-42</td>
<td>Parallel Add Saturation Completers ......................................................</td>
</tr>
<tr>
<td>2-43</td>
<td>Parallel Add Saturation Limits ..............................................................</td>
</tr>
<tr>
<td>2-44</td>
<td>Pcmp Relations ....................................................................................</td>
</tr>
<tr>
<td>2-45</td>
<td>PMPYSHR Shift Options .........................................................................</td>
</tr>
<tr>
<td>2-46</td>
<td>Parallel Subtract Saturation Completers ...............................................</td>
</tr>
<tr>
<td>2-47</td>
<td>Parallel Subtract Saturation Limits .......................................................</td>
</tr>
<tr>
<td>2-48</td>
<td>Store Types .........................................................................................</td>
</tr>
<tr>
<td>Page</td>
<td>Section</td>
</tr>
<tr>
<td>--------</td>
<td>-------------------------------------------------------------------------------------------</td>
</tr>
<tr>
<td>4-40</td>
<td>4-40 Line Prefetch Hint Completer</td>
</tr>
<tr>
<td>4-41</td>
<td>4-41 Opcode 0 System/Memory Management 3-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-39</td>
<td>4-39 Store Hint Completer</td>
</tr>
<tr>
<td>4-38</td>
<td>4-38 Load Hint Completer</td>
</tr>
<tr>
<td>4-37</td>
<td>4-37 Floating-point Load Pair +Imm Opcode Extensions</td>
</tr>
<tr>
<td>4-36</td>
<td>4-36 Floating-point Load Pair/Set FR Opcode Extensions</td>
</tr>
<tr>
<td>4-34</td>
<td>4-34 Floating-point Load/Lfetch +Reg Opcode Extensions</td>
</tr>
<tr>
<td>4-33</td>
<td>4-33 Floating-point Load/Lfetch Opcode Extensions</td>
</tr>
<tr>
<td>4-32</td>
<td>4-32 Semaphore/Get FR Opcode Extensions</td>
</tr>
<tr>
<td>4-31</td>
<td>4-31 Integer Load/Store +Imm Opcode Extensions</td>
</tr>
<tr>
<td>4-30</td>
<td>4-30 Integer Load +Reg Opcode Extensions</td>
</tr>
<tr>
<td>4-29</td>
<td>4-29 Integer Load/Store Opcode Extensions</td>
</tr>
<tr>
<td>4-28</td>
<td>4-28 Floating-point Load/Store/Load Pair/Set FR 1-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-26</td>
<td>4-26 Move to BR Whether Hint Completer</td>
</tr>
<tr>
<td>4-25</td>
<td>4-25 Misc I-Unit 6-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-24</td>
<td>4-24 Misc I-Unit 3-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-23</td>
<td>4-23 Test Bit Opcode Extensions</td>
</tr>
<tr>
<td>4-22</td>
<td>4-22 Deposit Opcode Extensions</td>
</tr>
<tr>
<td>4-21</td>
<td>4-21 Integer Shift/Test Bit/Test NaT 2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-20</td>
<td>4-20 Variable Shift Opcode 7 2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-19</td>
<td>4-19 Multimedia Opcode 7 Size 4 2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-18</td>
<td>4-18 Multimedia Opcode 7 Size 2 2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-17</td>
<td>4-17 Multimedia Opcode 7 Size 1 2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-16</td>
<td>4-16 Multimedia and Variable Shift 1-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-15</td>
<td>4-15 Multimedia ALU Size 4 4-bit+2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-14</td>
<td>4-14 Multimedia ALU Size 2 4-bit+2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-13</td>
<td>4-13 Multimedia ALU Size 1 4-bit+2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-12</td>
<td>4-12 Multimedia ALU 2-bit+1-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-11</td>
<td>4-11 Integer Compare Immediate Opcode Extensions</td>
</tr>
<tr>
<td>4-10</td>
<td>4-10 Integer Compare Opcode Extensions</td>
</tr>
<tr>
<td>4-9</td>
<td>4-9 Integer ALU 4-bit+2-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-8</td>
<td>4-8 Integer ALU 2-bit+1-bit Opcode Extensions</td>
</tr>
<tr>
<td>4-7</td>
<td>4-7 Special Instruction Notations</td>
</tr>
<tr>
<td>4-6</td>
<td>4-6 Instruction Field Names</td>
</tr>
<tr>
<td>4-5</td>
<td>4-5 Instruction Field Color Key</td>
</tr>
<tr>
<td>4-4</td>
<td>4-4 Instruction Format Summary</td>
</tr>
<tr>
<td>4-3</td>
<td>4-3 Major Opcode Assignments</td>
</tr>
<tr>
<td>4-2</td>
<td>4-2 Template Field Encoding and Instruction Slot Mapping</td>
</tr>
<tr>
<td>4-1</td>
<td>4-1 Relationship between Instruction Type and Execution Unit Type</td>
</tr>
<tr>
<td>3-1</td>
<td>3-1 Instruction Field Names</td>
</tr>
<tr>
<td>3-0</td>
<td>3-0 Instruction Field Color Key</td>
</tr>
<tr>
<td>3-49</td>
<td>3-49 Store Hints</td>
</tr>
<tr>
<td>3-229</td>
<td>xsz Mnemonic Values</td>
</tr>
<tr>
<td>3-232</td>
<td>2-51 Test Bit Relations for Normal and unc tbits</td>
</tr>
<tr>
<td>3-235</td>
<td>2-53 Test NaT Relations for Normal and unc tnats</td>
</tr>
<tr>
<td>3-238</td>
<td>2-55 Memory Exchange Size</td>
</tr>
<tr>
<td>3-249</td>
<td>3-1 Pseudo-Code Functions</td>
</tr>
<tr>
<td>3-257</td>
<td>4-1 Relationship between Instruction Type and Execution Unit Type</td>
</tr>
</tbody>
</table>
Part II: IA-32 Instruction Set Descriptions

1-1 Register Encodings Associated with the +rb, +rw, and +rd Nomenclature ........................................3:361
1-2 Exception Mnemonics, Names, and Vector Numbers ........................................................................3:367
1-3 Floating-point Exception Mnemonics and Names ...........................................................................3:368
1-4 Information Returned by CPUID Instruction ..................................................................................3:427
1-5 Feature Flags Returned in EDX Register .........................................................................................3:428
1-6 FPATAN Zeros and NaNs .................................................................................................................3:497
1-7 FPREM Zeros and NaNs .....................................................................................................................3:499
1-8 FPREM1 Zeros and NaNs ..................................................................................................................3:502
1-9 FSUB Zeros and NaNs .......................................................................................................................3:531
1-10 FSUBR Zeros and NaNs ...................................................................................................................3:534
1-11 FYL2X Zeros and NaNs ..................................................................................................................3:547
1-12 FYL2XP1 Zeros and NaNs ................................................................................................................3:549
1-13 IDIV Operands ...............................................................................................................................3:552
1-14 INT Cases ......................................................................................................................................3:566
1-15 LAR Descriptor Validity ..................................................................................................................3:601
1-16 LEA Address and Operand Sizes .................................................................................................3:606
1-17 Repeat Conditions .........................................................................................................................3:686

4-44 Opcode 1 System/Memory Management 6-bit Opcode Extensions .................................................3:312
4-45 IP-Relative Branch Types .............................................................................................................3:315
4-46 Indirect/Miscellaneous Branch Opcode Extensions ......................................................................3:316
4-47 Indirect Branch Types ..................................................................................................................3:316
4-48 Indirect Return Branch Types .....................................................................................................3:317
4-49 Sequential Prefetch Hint Completer ............................................................................................3:317
4-50 Branch Whether Hint Completer ...............................................................................................3:317
4-51 Indirect Call Whether Hint Completer .........................................................................................3:317
4-52 Branch Cache Deallocation Hint Completer ..............................................................................3:318
4-53 Indirect Predict/Nop Opcode Extensions .....................................................................................3:319
4-54 Branch Importance Hint Completer ..........................................................................................3:320
4-55 IP-Relative Predict Whether Hint Completer ...............................................................................3:320
4-56 Indirect Predict Whether Hint Completer ...................................................................................3:320
4-57 Miscellaneous Floating-point 1-bit Opcode Extensions ................................................................3:322
4-58 Opcode 0 Miscellaneous Floating-point 6-bit Opcode Extensions ..............................................3:322
4-59 Opcode 1 Miscellaneous Floating-point 6-bit Opcode Extensions ..............................................3:323
4-60 Reciprocal Approximation 1-bit Opcode Extensions ....................................................................3:323
4-61 Floating-point Status Field Completer ........................................................................................3:323
4-62 Floating-point Arithmetic 1-bit Opcode Extensions ....................................................................3:324
4-63 Fixed-point Multiply Add and Select Opcode Extensions ............................................................3:324
4-64 Floating-point Compare Opcode Extensions ..............................................................................3:325
4-65 Floating-point Class 1-bit Opcode Extensions .......................................................................... 3:325
4-66 Misc X-Unit 3-bit Opcode Extensions ..........................................................................................3:330
4-67 Misc X-Unit 6-bit Opcode Extensions ..........................................................................................3:331
4-68 Move Long 1-bit Opcode Extensions ...........................................................................................3:332
4-69 Long Branch Types .....................................................................................................................3:332
4-70 Immediate Formation ..................................................................................................................3:333
5-1 Semantics of Dependency Codes ..................................................................................................3:337
5-2 RAW Dependencies Organized by Resource ..............................................................................3:339
5-3 WAW Dependencies Organized by Resource .............................................................................3:346
5-4 WAR Dependencies Organized by Resource ...............................................................................3:350
5-5 Instruction Classes ........................................................................................................................3:351
<table>
<thead>
<tr>
<th>Section</th>
<th>Title</th>
<th>Page</th>
</tr>
</thead>
<tbody>
<tr>
<td>3-1</td>
<td>Real Number Notation</td>
<td>3:823</td>
</tr>
<tr>
<td>3-2</td>
<td>Denormalization Process</td>
<td>3:826</td>
</tr>
<tr>
<td>3-3</td>
<td>Results of Operations with NAN Operands</td>
<td>3:828</td>
</tr>
<tr>
<td>3-4</td>
<td>Precision and Range of Streaming SIMD Extension Datatype</td>
<td>3:829</td>
</tr>
<tr>
<td>3-5</td>
<td>Real Number and NaN Encodings</td>
<td>3:829</td>
</tr>
<tr>
<td>3-6</td>
<td>Streaming SIMD Extension Instruction Behavior with Prefixes</td>
<td>3:830</td>
</tr>
<tr>
<td>3-7</td>
<td>SIMD Integer Instructions – Behavior with Prefixes</td>
<td>3:830</td>
</tr>
<tr>
<td>3-8</td>
<td>Cacheability Control Instruction Behavior with Prefixes</td>
<td>3:831</td>
</tr>
<tr>
<td>3-9</td>
<td>Key to Streaming SIMD Extension Naming Convention</td>
<td>3:832</td>
</tr>
</tbody>
</table>
Part I: Intel® Itanium™ Instruction Set Descriptions
The Intel® Itanium™ architecture is a unique combination of innovative features such as explicit parallelism, predication, speculation and more. The architecture is designed to be highly scalable to fill the ever increasing performance requirements of various server and workstation market segments. The Itanium architecture features a revolutionary 64-bit instruction set architecture (ISA) which applies a new processor architecture technology called EPIC, or Explicitly Parallel Instruction Computing. A key feature of the Itanium architecture is IA-32 instruction set compatibility.

The Intel® Itanium Architecture Software Developer’s Manual provides a comprehensive description of the programming environment, resources, and instruction set visible to both the application and system programmer. In addition, it also describes how programmers can take advantage of the features of the Itanium architecture to help them optimize code.

1.1 Overview of Volume 1: Application Architecture

This volume defines the Itanium application architecture, including application level resources, programming environment, and the IA-32 application interface. This volume also describes optimization techniques used to generate high performance software.

1.1.1 Part 1: Application Architecture Guide


Chapter 2, “Introduction to the Intel® Itanium™ Architecture” provides an overview of the architecture.

Chapter 3, “Execution Environment” describes the Itanium register set used by applications and the memory organization models.

Chapter 4, “Application Programming Model” gives an overview of the behavior of Itanium application instructions (grouped into related functions).

Chapter 5, “Floating-point Programming Model” describes the Itanium floating-point architecture (including integer multiply).

Chapter 6, “IA-32 Application Execution Model in an Intel® Itanium™ System Environment” describes the operation of IA-32 instructions within the Itanium System Environment from the perspective of an application programmer.
1.1.2 **Part 2: Optimization Guide for the Intel® Itanium™ Architecture**

Chapter 1, “About the Optimization Guide” gives an overview of the optimization guide.

Chapter 2, “Introduction to Programming for the Intel® Itanium™ Architecture” provides an overview of the application programming environment for the Itanium architecture.

Chapter 3, “Memory Reference” discusses features and optimizations related to control and data speculation.

Chapter 4, “Predication, Control Flow, and Instruction Stream” describes optimization features related to predication, control flow, and branch hints.

Chapter 5, “Software Pipelining and Loop Support” provides a detailed discussion on optimizing loops through use of software pipelining.

Chapter 6, “Floating-point Applications” discusses current performance limitations in floating-point applications and features that address these limitations.

1.2 **Overview of Volume 2: System Architecture**

This volume defines the Itanium system architecture, including system level resources and programming state, interrupt model, and processor firmware interface. This volume also provides a useful system programmer’s guide for writing high performance system software.

1.2.1 **Part 1: System Architecture Guide**

Chapter 1, “About this Manual” provides an overview of all volumes in the *Intel® Itanium Architecture Software Developer’s Manual*.

Chapter 2, “Intel® Itanium™ System Environment” introduces the environment designed to support execution of Itanium-based operating systems running IA-32 or Itanium-based applications.

Chapter 3, “System State and Programming Model” describes the Itanium architectural state which is visible only to an operating system.

Chapter 4, “Addressing and Protection” defines the resources available to the operating system for virtual to physical address translation, virtual aliasing, physical addressing, and memory ordering.

Chapter 5, “Interruptions” describes all interruptions that can be generated by a processor based on the Itanium architecture.

Chapter 6, “Register Stack Engine” describes the architectural mechanism which automatically saves and restores the stacked subset (GR32 – GR 127) of the general register file.

Chapter 7, “Debugging and Performance Monitoring” is an overview of the performance monitoring and debugging resources that are available in the Itanium architecture.
Chapter 8, “ Interruption Vector Descriptions” lists all interruption vectors.

Chapter 9, “ IA-32 Interruption Vector Descriptions” lists IA-32 exceptions, interrupts and intercepts that can occur during IA-32 instruction set execution in the Itanium System Environment.

Chapter 10, “ Itanium™-based Operating System Interaction Model with IA-32 Applications” defines the operation of IA-32 instructions within the Itanium System Environment from the perspective of an Itanium-based operating system.

Chapter 11, “ Processor Abstraction Layer” describes the firmware layer which abstracts processor implementation-dependent features.

1.2.2 Part 2: System Programmer’s Guide

Chapter 1, “ About the System Programmer’s Guide” gives an introduction to the second section of the system architecture guide.

Chapter 2, “ MP Coherence and Synchronization” describes multi-processing synchronization primitives and the Itanium memory ordering model.

Chapter 3, “ Interruptions and Serialization” describes how the processor serializes execution around interruptions and what state is preserved and made available to low-level system code when interruptions are taken.

Chapter 4, “ Context Management” describes how operating systems need to preserve Itanium register contents and state. This chapter also describes system architecture mechanisms that allow an operating system to reduce the number of registers that need to be spilled/filled on interruptions, system calls, and context switches.

Chapter 5, “ Memory Management” introduces various memory management strategies.

Chapter 6, “ Runtime Support for Control and Data Speculation” describes the operating system support that is required for control and data speculation.

Chapter 7, “ Instruction Emulation and Other Fault Handlers” describes a variety of instruction emulation handlers that Itanium-based operating system are expected to support.

Chapter 8, “ Floating-point System Software” discusses how processors based on the Itanium architecture handle floating-point numeric exceptions and how the software stack provides complete IEEE-754 compliance.

Chapter 9, “ IA-32 Application Support” describes the support an Itanium-based operating system needs to provide to host IA-32 applications.

Chapter 10, “ External Interrupt Architecture” describes the external interrupt architecture with a focus on how external asynchronous interrupt handling can be controlled by software.

Chapter 11, “ I/O Architecture” describes the I/O architecture with a focus on platform issues and support for the existing IA-32 I/O port space.

Chapter 12, “ Performance Monitoring Support” describes the performance monitor architecture with a focus on what kind of support is needed from Itanium-based operating systems.
Chapter 13, “Firmware Overview” introduces the firmware model, and how various firmware layers (PAL, SAL, EFI) work together to enable processor and system initialization, and operating system boot.

1.2.3 Appendices

Appendix A, “Code Examples” provides OS boot flow sample code.

1.3 Overview of Volume 3: Instruction Set Reference

This volume is a comprehensive reference to the Itanium and IA-32 instruction sets, including instruction format/encoding.

1.3.1 Part 1: Intel® Itanium™ Instruction Set Descriptions


Chapter 2, “Instruction Reference” provides a detailed description of all Itanium instructions, organized in alphabetical order by assembly language mnemonic.

Chapter 3, “Pseudo-Code Functions” provides a table of pseudo-code functions which are used to define the behavior of the Itanium instructions.

Chapter 4, “Instruction Formats” describes the encoding and instruction format instructions.

Chapter 5, “Resource and Dependency Semantics” summarizes the dependency rules that are applicable when generating code for processors based on the Itanium architecture.

1.3.2 Part 2: IA-32 Instruction Set Descriptions

Chapter 1, “Base IA-32 Instruction Reference” provides a detailed description of all base IA-32 instructions, organized in alphabetical order by assembly language mnemonic.

Chapter 2, “IA-32 Intel® MMX™ Technology Instruction Reference” provides a detailed description of all IA-32 Intel® MMX™ technology instructions designed to increase performance of multimedia intensive applications. Organized in alphabetical order by assembly language mnemonic.

Chapter 3, “IA-32 Streaming SIMD Extension Instruction Reference” provides a detailed description of all IA-32 Streaming SIMD Extension instructions designed to increase performance of multimedia intensive applications, and is organized in alphabetical order by assembly language mnemonic.
1.4 Terminology

The following definitions are for terms related to the Itanium architecture and will be used throughout this document:

**Instruction Set Architecture (ISA)** – Defines application and system level resources. These resources include instructions and registers.

**Itanium Architecture** – The new ISA with 64-bit instruction capabilities, new performance-enhancing features, and support for the IA-32 instruction set.


**Itanium System Environment** – The operating system environment that supports the execution of both IA-32 and Itanium-based code.

**IA-32 System Environment** – The operating system privileged environment and resources as defined by the *Intel Architecture Software Developer’s Manual*. Resources include virtual paging, control registers, debugging, performance monitoring, machine checks, and the set of privileged instructions.

**Itanium-based Firmware** – The Processor Abstraction Layer (PAL) and System Abstraction Layer (SAL).

**Processor Abstraction Layer (PAL)** – The firmware layer which abstracts processor features that are implementation dependent.

**System Abstraction Layer (SAL)** – The firmware layer which abstracts system features that are implementation dependent.

1.5 Related Documents

The following documents can be downloaded at the Intel’s Developer Site at http://developer.intel.com:

- **Intel® Itanium™ Processor Reference Manual for Software Development** – This document (Document number 245320) describes model-specific architectural features incorporated into the Intel® Itanium™ processor, the first processor based on the Itanium architecture. This document has been re-titled and replaces the *Intel® Itanium™ Architecture Software Developer’s Manual, Volume 4: Itanium™ Processor Programmer’s Guide*.

- **Intel® Architecture Software Developer’s Manual** – This set of manuals describes the Intel 32-bit architecture. They are readily available from the Intel Literature Department by calling 1-800-548-4725 and requesting Document Numbers 243190, 243191, and 243192.

- **Itanium™ Software Conventions and Runtime Architecture Guide** – This document (Document number 245358) defines general information necessary to compile, link, and execute a program on an Itanium-based operating system.

- **Itanium™ Processor Family System Abstraction Layer Specification** – This document (Document number 245359) specifies requirements to develop platform firmware for Itanium-based systems.
- **Extensible Firmware Interface Specification** – This document defines a new model for the interface between operating systems and platform firmware.

## 1.6 Revision History

<table>
<thead>
<tr>
<th>Date of Revision</th>
<th>Revision Number</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>December 2001</td>
<td>2.0</td>
<td>Volume 1:</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Faults in \texttt{id.c} that hits ALAT clarification (Section 4.4.5.3.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>IA-32 related changes (Section 6.2.5.4, Section 6.2.3, Section 6.2.4, Section 6.2.5.3).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Load instructions change (Section 4.4.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Volume 2:</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Class pr-writers-int clarification (Table A-5).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PAL_MC_DRAIN} clarification (Section 4.4.6.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{VHPT} walk and forward progress change (Section 4.1.1.2).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>IA-32 IBR/DBR match clarification (Section 7.1.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{ISR} figure changes (pp. 8-5, 8-26, 8-33 and 8-36).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PAL_CACHE_FLUSH} return argument change - added new status return argument (Section 11.8.3).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PAL} self-test Control and \texttt{PAL_A} procedure requirement change - added new arguments, figures, requirements (Section 11.2).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PAL_CACHE_FLUSH} clarifications (Section 11).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Non-speculative reference clarification (Section 4.4.6).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{RID} and Preferred Page Size usage clarification (Section 4.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{VHPT} read atomicity clarification (Section 4.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{IIP} and WC flush clarification (Section 4.4.5).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Revised \texttt{RSE} and \texttt{PMC} typographical errors (Section 6.4).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Revised \texttt{DV} table (Section A.4).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Memory attribute transitions - added new requirements (Section 4.4).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>MCA for WC/UC aliasing change (Section 4.4.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Bus lock deprecation - changed behavior of \texttt{DCR _tc} bit (Section 3.3.4.1, Section 3.6.8, Section 11.8.3).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PAL_PROC_GET/SET_FEATURES} changes - extend calls to allow implementation-specific feature control (Section 11.8.3).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Split \texttt{PAL_A} architecture changes (Section 11.1.6).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Simple barrier synchronization clarification (Section 13.4.2).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Limited speculation clarification - added hardware-generated speculative references (Section 4.4.6).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PAL} memory accesses and restrictions clarification (Section 11.9).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PSP} validity on \texttt{INITs} from \texttt{PAL_MC_ERROR_INFO} clarification (Section 11.8.3).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>Speculation attributes clarification (Section 4.4.6).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{PAL_A FIT} entry, \texttt{PAL_VM_TR_READ}, \texttt{PSP}, \texttt{PAL_VERSION} clarifications (Sections 11.8.3 and 11.3.2.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>TLB searching clarifications (Section 4.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>IA-32 related changes (Section 10.3, Section 10.3.2, Section 10.3.2, Section 10.3.3.1, Section 10.10.1).</td>
</tr>
<tr>
<td></td>
<td></td>
<td>\texttt{IPSR_ri} and \texttt{ISR_ei} changes (Table 3-2, Section 3.3.5.1, Section 3.3.5.2, Section 5.5, Section 8.3, and Section 2.2).</td>
</tr>
<tr>
<td>Date of Revision</td>
<td>Revision Number</td>
<td>Description</td>
</tr>
<tr>
<td>------------------</td>
<td>-----------------</td>
<td>-------------</td>
</tr>
</tbody>
</table>
| July 2000        | 1.1             | Volume 1:  
Processor Serial Number feature removed (Chapter 3).  
Clarification on exceptions to instruction dependency (Section 3.4.3).  
Volume 2:  
Clarifications regarding “reserved” fields in ITIR (Chapter 3).  
Instruction and Data translation must be enabled for executing IA-32 instructions (Chapters 3, 4 and 10).  
FCR/FDR mappings, and clarification to the value of PSR.ri after an RFI (Chapters 3 and 4).  
Clarification regarding ordering data dependency.  
Out-of-order IPI delivery is now allowed (Chapters 4 and 5).  
Content of EFLAG field changed in IIM (p. 9-24).  
PAL_CHECK and PAL_INIT calls – exit state changes (Chapter 11).  
PAL_CHECK processor state parameter changes (Chapter 11).  
PAL_BUS_GET/SET_FEATURES calls – added two new bits (Chapter 11).  
PAL_MC_ERROR_INFO call – Changes made to enhance and simplify the call to provide more information regarding machine check (Chapter 11).  
PAL_ENTER_IA_32_Env call changes – entry parameter represents the entry order; SAL needs to initialize all the IA-32 registers properly before making this call (Chapter 11).  
PAL_CACHE_FLUSH – added a new cache_type argument (Chapter 11).  
PAL_SHUTDOWN – removed from list of PAL calls (Chapter 11).  
Clarified memory ordering changes (Chapter 13).  
Clarification in dependence violation table (Appendix A).  
Volume 3:  
fmix instruction page figures corrected (Chapter 2).  
Clarification of “reserved” fields in ITIR (Chapters 2 and 3).  
Modified conditions for alloc/loadrs/flushrs instruction placement in bundle/instruction group (Chapters 2 and 4).  
IA-32 JMPE instruction page typo fix (p. 5-238).  
Processor Serial Number feature removed (Chapter 5). |
| January 2000     | 1.0             | Initial release of document. |
This chapter describes the function of each Itanium instruction. The pages of this chapter are sorted alphabetically by assembly language mnemonic.

2.1 Instruction Page Conventions

The instruction pages are divided into multiple sections as listed in Table 2-1. The first three sections are present on all instruction pages. The last three sections are present only when necessary. Table 2-2 lists the font conventions which are used by the instruction pages.

<table>
<thead>
<tr>
<th>Section Name</th>
<th>Contents</th>
</tr>
</thead>
<tbody>
<tr>
<td>Format</td>
<td>Assembly language syntax, instruction type and encoding format</td>
</tr>
<tr>
<td>Description</td>
<td>Instruction function in English</td>
</tr>
<tr>
<td>Operation</td>
<td>Instruction function in C code</td>
</tr>
<tr>
<td>FP Exceptions</td>
<td>IEEE floating-point traps</td>
</tr>
<tr>
<td>Interruptions</td>
<td>Prioritized list of interruptions that may be caused by the instruction</td>
</tr>
<tr>
<td>Serialization</td>
<td>Serializing behavior or serialization requirements</td>
</tr>
</tbody>
</table>

Table 2-2. Instruction Page Font Conventions

<table>
<thead>
<tr>
<th>Font</th>
<th>Interpretation</th>
</tr>
</thead>
<tbody>
<tr>
<td>regular</td>
<td>(Format section) Required characters in an assembly language mnemonic</td>
</tr>
<tr>
<td>italic</td>
<td>(Format section) Assembly language field name that must be filled with one of a range of legal values listed in the Description section</td>
</tr>
<tr>
<td>code</td>
<td>(Operation section) C code specifying instruction behavior</td>
</tr>
<tr>
<td>code_italic</td>
<td>(Operation section) Assembly language field name corresponding to a italic field listed in the Format section</td>
</tr>
</tbody>
</table>

In the Format section, register addresses are specified using the assembly mnemonic field names given in the third column of Table 2-3. For instructions that are predicated, the Description section assumes that the qualifying predicate is true (except for instructions that modify architectural state when their qualifying predicate is false). The test of the qualifying predicate is included in the Operation section (when applicable).

In the Operation section, registers are addressed using the notation `reg[addr].field`. The register file being accessed is specified by `reg`, and has a value chosen from the second column of Table 2-3. The `addr` field specifies a register address as an assembly language field name or a register mnemonic. For the general, floating-point, and predicate register files which undergo register renaming, `addr` is the register address prior to renaming and the renaming is not shown. The `field` option specifies a named bit field within the register. If `field` is absent, then all fields of the register are accessed. The only exception is when referencing the data field of the general registers (64-bits not including the NaT bit) where the notation `GR[addr]` is used. The syntactical differences between the code found in the Operation section and ANSI C is listed in Table 2-4.
The Operation section contains code that specifies only the execution semantics of each instruction and does not include any behavior relating to instruction fetch (e.g., interrupts and faults caused during fetch). The Interruptions section does not list any faults that may be caused by instruction fetch or by mandatory RSE loads. The code to raise certain pervasive faults and actions is not included in the code in the Operation section. These faults and actions are listed in Table 2-5. The Single step trap applies to all instructions and is not listed in the Interruptions section.

### Table 2-3. Register File Notation

<table>
<thead>
<tr>
<th>Register File</th>
<th>C Notation</th>
<th>Assembly Mnemonic</th>
<th>Indirect Access</th>
</tr>
</thead>
<tbody>
<tr>
<td>Application registers</td>
<td>AR</td>
<td>ar</td>
<td></td>
</tr>
<tr>
<td>Branch registers</td>
<td>BR</td>
<td>b</td>
<td></td>
</tr>
<tr>
<td>Control registers</td>
<td>CR</td>
<td>cr</td>
<td></td>
</tr>
<tr>
<td>CPU identification registers</td>
<td>CPUID</td>
<td>cpuid</td>
<td></td>
</tr>
<tr>
<td>Data breakpoint registers</td>
<td>DBR</td>
<td>dbbr</td>
<td>Y</td>
</tr>
<tr>
<td>Instruction breakpoint registers</td>
<td>IBR</td>
<td>lbr</td>
<td></td>
</tr>
<tr>
<td>Data TLB translation cache</td>
<td>DTC</td>
<td>n/a</td>
<td></td>
</tr>
<tr>
<td>Data TLB translation registers</td>
<td>DTR</td>
<td>dtr</td>
<td>Y</td>
</tr>
<tr>
<td>Floating-point registers</td>
<td>FR</td>
<td>f</td>
<td></td>
</tr>
<tr>
<td>General registers</td>
<td>GR</td>
<td>r</td>
<td></td>
</tr>
<tr>
<td>Instruction TLB translation cache</td>
<td>ITC</td>
<td>n/a</td>
<td></td>
</tr>
<tr>
<td>Instruction TLB translation registers</td>
<td>ITR</td>
<td>itr</td>
<td>Y</td>
</tr>
<tr>
<td>Protection key registers</td>
<td>PKR</td>
<td>pkr</td>
<td>Y</td>
</tr>
<tr>
<td>Performance monitor configuration registers</td>
<td>PMC</td>
<td>pmc</td>
<td>Y</td>
</tr>
<tr>
<td>Performance monitor data registers</td>
<td>PMD</td>
<td>pmd</td>
<td>Y</td>
</tr>
<tr>
<td>Predicate registers</td>
<td>PR</td>
<td>p</td>
<td></td>
</tr>
<tr>
<td>Region registers</td>
<td>RR</td>
<td>rr</td>
<td>Y</td>
</tr>
</tbody>
</table>

### Table 2-4. C Syntax Differences

<table>
<thead>
<tr>
<th>Syntax</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>(msb:lsb), (bit)</td>
<td>Bit field specifier. When appended to a variable, denotes a bit field extending from the most significant bit specified by “msb” to the least significant bit specified by “lsb”. Including bits “msb” and “lsb”. If “msb” and “lsb” are equal then a single bit is accessed. The second form denotes a single bit.</td>
</tr>
<tr>
<td>u&gt;, u&gt;&gt;, u&lt;, u&lt;=</td>
<td>Unsigned inequality relations. Variables on either side of the operator are treated as unsigned.</td>
</tr>
<tr>
<td>u&gt;&gt;, u&gt;&gt;=</td>
<td>Unsigned right shift. Zeroes are shifted into the most significant bit position.</td>
</tr>
<tr>
<td>u+</td>
<td>Unsigned addition. Operands are treated as unsigned, and zero-extended.</td>
</tr>
<tr>
<td>u*</td>
<td>Unsigned multiplication. Operands are treated as unsigned.</td>
</tr>
</tbody>
</table>

### Table 2-5. Pervasive Conditions Not Included in Instruction Description Code

<table>
<thead>
<tr>
<th>Condition</th>
<th>Action</th>
</tr>
</thead>
<tbody>
<tr>
<td>Read of a register outside the current frame.</td>
<td>An undefined value is returned (no fault).</td>
</tr>
<tr>
<td>Access to a banked general register (GR 16 through GR 31).</td>
<td>The GR bank specified by PSR.bn is accessed.</td>
</tr>
<tr>
<td>PSR.ss is set.</td>
<td>A Single Step trap is raised.</td>
</tr>
</tbody>
</table>
2.2 Instruction Descriptions

The remainder of this chapter provides a description of each of the Itanium instructions.
Add

Format: 
\begin{align*}
(qp) \text{ add } & r_1 = r_2, r_3, \quad \text{register_form} \quad \text{A1} \\
(qp) \text{ add } & r_1 = r_2, r_3, 1, \quad \text{plus1_form, register_form} \quad \text{A1} \\
(qp) \text{ add } & r_1 = \text{imm}, r_3, \quad \text{pseudo-op} \quad \text{A4} \\
(qp) \text{ adds } & r_1 = \text{imm14}, r_3, \quad \text{imm14_form} \quad \text{A4} \\
(qp) \text{ addl } & r_1 = \text{imm22}, r_3, \quad \text{imm22_form} \quad \text{A5}
\end{align*}

Description: The two source operands (and an optional constant 1) are added and the result placed in GR $r_1$. In the register form the first operand is GR $r_2$; in the imm14_form the first operand is taken from the sign-extended imm14 encoding field; in the imm22_form the first operand is taken from the sign-extended imm22 encoding field. In the imm22_form, GR $r_3$ can specify only GRs 0, 1, 2 and 3.

The plus1_form is available only in the register_form (although the equivalent effect in the immediate forms can be achieved by adjusting the immediate).

The immediate-form pseudo-op chooses the imm14_form or imm22_form based upon the size of the immediate operand and the value of $r_3$.

Operation:
\begin{verbatim}
if (PR[qp]) {
  check_target_register(r1);
  if (register_form) { // register form
    tmp_src = GR[r2];
  } else if (imm14_form) { // 14-bit immediate form
    tmp_src = sign_ext(imm14, 14);
  } else { // 22-bit immediate form
    tmp_src = sign_ext(imm22, 22);
  }
  tmp_nat = (register_form ? GR[r2].nat : 0);
  if (plus1_form)
    GR[r1] = tmp_src + GR[r3] + 1;
  else
    GR[r1] = tmp_src + GR[r3];
  GR[r1].nat = tmp_nat || GR[r3].nat;
}
\end{verbatim}

Interruptions: Illegal Operation fault
Add Pointer

Format:

- \((qp)\) addp4 \(r_1 = r_2, r_3\)
- \((qp)\) addp4 \(r_1 = \text{imm14}, r_3\)

register_form A1
imm14_form A4

Description: The two source operands are added. The upper 32 bits of the result are forced to zero, and then bits \(\{31:30\}\) of \(GR\ r_3\) are copied to bits \(\{62:61\}\) of the result. This result is placed in \(GR\ r_1\). In the register_form the first operand is \(GR\ r_2\); in the imm14_form the first operand is taken from the sign-extended imm14 encoding field.

Figure 2-1. Add Pointer

![Figure 2-1. Add Pointer](image)

Operation:

```c
if (PR[qp]) {
    check_target_register(r1);
    tmp_src = (register_form ? GR[r2] : sign_ext(imm14, 14));
    tmp_nat = (register_form ? GR[r2].nat : 0);
    tmp_res = tmp_src + GR[r3];
    tmp_res = zero_ext(tmp_res{31:0}, 32);
    tmp_res{62:61} = GR[r3]{31:30};
    GR[r1] = tmp_res;
    GR[r1].nat = tmp_nat || GR[r3].nat;
}
```

Interruptions: Illegal Operation fault
Allocate Stack Frame

Format: \[(q)p\] alloc \( r_f = \text{ar.pfs}, i, l, o, r \]

Description: A new stack frame is allocated on the general register stack, and the Previous Function State register (PFS) is copied to GR \( r_f \). The change of frame size is immediate. The write of GR \( r_f \) and subsequent instructions in the same instruction group use the new frame. This instruction cannot be predicated.

The four parameters, \( i \) (size of inputs), \( l \) (size of locals), \( o \) (size of outputs), and \( r \) (size of rotating) specify the sizes of the regions of the stack frame.

**Figure 2-2. Stack Frame**

![Stack Frame Diagram]

The size of the frame \( (\text{sof}) \) is determined by \( i + l + o \). Note that this instruction may grow or shrink the size of the current register stack frame. The size of the local region \( (\text{sol}) \) is given by \( i + l \). There is no real distinction between inputs and locals. They are given as separate operands in the instruction only as a hint to the assembler about how the local registers are to be used.

The rotating registers must fit within the stack frame and be a multiple of 8 in number. If this instruction attempts to change the size of CFM.sor, and the register rename base registers (CFM.rrb.gr, CFM.rrb.fr, CFM.rrb.pr) are not all zero, then the instruction will cause a Reserved Register/Field fault.

Although the assembler does not allow illegal combinations of operands for alloc, illegal combinations can be encoded in the instruction. Attempting to allocate a stack frame larger than 96 registers, or with the rotating region larger than the stack frame, or with the size of locals larger than the stack frame, or specifying a qualifying predicate other than PR 0, will cause an Illegal Operation fault.

This instruction must be the first instruction in an instruction group and must either be in instruction slot 0 or in instruction slot 1 of a template having a stop after slot 0; otherwise, the results are undefined.

If insufficient registers are available to allocate the desired frame alloc will stall the processor until enough dirty registers are written to the backing store. Such mandatory RSE stores may cause the data related faults listed below.
Operation: // tmp_sof, tmp_sol, tmp_sor are the fields encoded in the instruction
tmp_sof = i + l + o;
tmp_sol = i + l;
tmp_sor = r u>> 3;
check_target_register_sof(r, tmp_sof);
if (tmp_sof u> 96 || r u> tmp_sof || tmp_sol u> tmp_sof || qp != 0)
    illegal_operation_fault();
if (tmp_sor != CFM.sor &&
    (CFM.rrb.gr != 0 || CFM.rrb.fr != 0 || CFM.rrb.pr != 0))
    reserved_register_field_fault();
alat_frame_update(0, tmp_sof - CFM.sof);
rse_new_frame(CFM.sof, tmp_sof);// Make room for new registers; Mandatory
    // RSE stores can raise faults listed below.
CFM.sof = tmp_sof;
CFM.sol = tmp_sol;
CFM.sor = tmp_sor;
GR[r] = AR[PFS];
GR[r].nat = 0;

Interruptions:
Illegal Operation fault
Reserved Register/Field fault
Unimplemented Data Address fault
VHPT Data fault
Data Nested TLB fault
Data TLB fault
Alternate Data TLB fault
Data Page Not Present fault
Data NaT Page Consumption fault
Data Key Miss fault
Data Key Permission fault
Data Access Rights fault
Data Dirty Bit fault
Data Access Bit fault
Data Debug fault
Logical And

Format:  
\[(qp) \text{ and } r_1 = r_2, r_3 \quad \text{register form} \quad A1\]
\[(qp) \text{ and } r_1 = \text{imm}_8, r_3 \quad \text{imm}_8 \text{ form} \quad A3\]

Description: The two source operands are logically ANDed and the result placed in GR \( r_1 \). In the register_form the first operand is GR \( r_2 \); in the \text{imm}_8_form the first operand is taken from the \text{imm}_8 encoding field.

Operation:  
if (PR[qp]) {
  check_target_register(r_1);
  tmp_src = (register_form ? GR[r_2] : sign_ext(imm_8, 8));
  tmp_nat = (register_form ? GR[r_2].nat : 0);
  GR[r_1] = tmp_src & GR[r_3];
  GR[r_1].nat = tmp_nat || GR[r_3].nat;
}

Interruptions: Illegal Operation fault
And Complement

Format: \[(qp) \text{ andcm } r_1 = r_2, r_3\]  
\[(qp) \text{ andcm } r_1 = \text{imm}_8, r_3\]  
- register_form A1  
- imm8_form A3

Description: The first source operand is logically ANDed with the 1’s complement of the second source operand and the result placed in GR \(r_1\). In the register_form the first operand is GR \(r_2\); in the imm8_form the first operand is taken from the \(imm_8\) encoding field.

Operation:  
\[
\text{if (PR[qp])} \{
\quad \text{check_target_register}(r_1); \\
\quad \text{tmp}_\text{src} = (\text{register_form} \ ? \ \text{GR}[r_2] : \text{sign_ext} (\text{imm}_8, 8)); \\
\quad \text{tmp}_\text{nat} = (\text{register_form} \ ? \ \text{GR}[r_2].\text{nat} : 0); \\
\quad \text{GR}[r_2] = \text{tmp}_\text{src} \& \neg \text{GR}[r_3] ; \\
\quad \text{GR}[r_1].\text{nat} = \text{tmp}_\text{nat} \big|\big| \text{GR}[r_3].\text{nat} ; \\
\}
\]

Interruptions: Illegal Operation fault
Branch

Format:

\[(qp) \text{btype}.bwh.ph.dh \text{target}_{25} \]
\[(qp) \text{btype}.bwh.ph.dh b_1 = \text{target}_{25} \]
\[\text{br}.\text{bwh}.\text{ph}.\text{dh} \text{target}_{25} \]
\[(qp) \text{br}.\text{btype}.bwh.ph.dh b_2 \]
\[(qp) \text{br}.\text{btype}.bwh.ph.dh b_1 = b_2 \]

Description:
A branch condition is evaluated, and either a branch is taken, or execution continues with the next sequential instruction. The execution of a branch logically follows the execution of all previous non-branch instructions in the same instruction group. On a taken branch, execution begins at slot 0.

Branches can be either IP-relative, or indirect. For IP-relative branches, the \text{target}_{25} operand, in assembly, specifies a label to branch to. This is encoded in the branch instruction as a signed immediate displacement \((\text{imm}_{21})\) between the target bundle and the bundle containing this instruction \((\text{imm}_{21} = \text{target}_{25} – \text{IP} >> 4)\). For indirect branches, the target address is taken from BR \(b_2\).

Table 2-6. Branch Types

<table>
<thead>
<tr>
<th>btype</th>
<th>Function</th>
<th>Branch Condition</th>
<th>Target Address</th>
</tr>
</thead>
<tbody>
<tr>
<td>cond or none</td>
<td>Conditional branch</td>
<td>Qualifying predicate</td>
<td>IP-rel or Indirect</td>
</tr>
<tr>
<td>call</td>
<td>Conditional procedure call</td>
<td>Qualifying predicate</td>
<td>IP-rel or Indirect</td>
</tr>
<tr>
<td>ret</td>
<td>Conditional procedure return</td>
<td>Qualifying predicate</td>
<td>Indirect</td>
</tr>
<tr>
<td>ia</td>
<td>Invoke IA-32 instruction set</td>
<td>Unconditional</td>
<td>Indirect</td>
</tr>
<tr>
<td>cloop</td>
<td>Counted loop branch</td>
<td>Loop count</td>
<td>IP-rel</td>
</tr>
<tr>
<td>ctop, cexit</td>
<td>Mod-scheduled counted loop</td>
<td>Loop count and epilog count</td>
<td>IP-rel</td>
</tr>
<tr>
<td>wtop, wexit</td>
<td>Mod-scheduled while loop</td>
<td>Qualifying predicate and epilog count</td>
<td>IP-rel</td>
</tr>
</tbody>
</table>

There are two pseudo-ops for unconditional branches. These are encoded like a conditional branch \((\text{btype} = \text{cond})\), with the \(qp\) field specifying PR 0, and with the \(bwh\) hint of sptk.

The branch type determines how the branch condition is calculated and whether the branch has other effects (such as writing a link register). For the basic branch types, the branch condition is simply the value of the specified predicate register. These basic branch types are:

- **cond**: If the qualifying predicate is 1, the branch is taken. Otherwise it is not taken.
- **call**: If the qualifying predicate is 1, the branch is taken and several other actions occur:
  - The current values of the Current Frame Marker (CFM), the EC application register and the current privilege level are saved in the Previous Function State application register.
  - The caller’s stack frame is effectively saved and the callee is provided with a frame containing only the caller’s output region.
  - The rotation rename base registers in the CFM are reset to 0.
  - A return link value is placed in BR \(b_1\).
return: If the qualifying predicate is 1, the branch is taken and the following occurs:
- CFM, EC, and the current privilege level are restored from PFS. (The privilege level is restored only if this does not increase privilege.)
- The caller’s stack frame is restored.
- If the return lowers the privilege, and PSR.lp is 1, then a Lower-Privilege Transfer trap is taken.

ia: The branch is taken unconditionally, if it is not intercepted by the OS. The effect of the branch is to invoke the IA-32 instruction set (by setting PSR.is to 1) and begin processing IA-32 instructions at the virtual linear target address contained in BR $b_2{[31:0]}$. If the qualifying predicate is not PR 0, an Illegal Operation fault is raised. If instruction set transitions are disabled (PSR.di is 1), then a Disabled Instruction Set Transition fault is raised. The IA-32 target effective address is calculated relative to the current code segment, i.e. 
$$EIP{[31:0]} = BR{[31:0]} - CSD.base.$$
The IA-32 instruction set can be entered at any privilege level, provided PSR.di is 0. If PSR.dfh is 1, a Disabled FP Register fault is raised on the target IA-32 instruction. No register bank switch nor change in privilege level occurs during the instruction set transition.

Software must ensure the code segment descriptor (CSD) and selector (CS) are loaded before issuing the branch. If the target EIP value exceeds the code segment limit or has a code segment privilege violation, an IA-32_Exception(GPFault) is raised on the target IA-32 instruction. For entry into 16-bit IA-32 code, if BR $b_2$ is not within 64K-bytes of CSD.base a GPFault is raised on the target instruction. EFLAG.rf is unmodified until the successful completion of the first IA-32 instruction. PSR.da, PSR.id, PSR.ia, PSR.dd, and PSR.ed are cleared to zero after br.ia completes execution and before the first IA-32 instruction begins execution. EFLAG.rf is not cleared until the target IA-32 instruction successfully completes.

Software must issue a mf instruction before the branch if memory ordering is required between IA-32 processor consistent and Itanium unordered memory references. The processor does not ensure Itanium-instruction-set-generated writes into the instruction stream are seen by subsequent IA-32 instruction fetches. br.ia does not perform an instruction serialization operation. The processor does ensure that prior writes (even in the same instruction group) to GRs and FRs are observed by the first IA-32 instruction. Writes to ARs within the same instruction group as br.ia are not allowed, since br.ia may implicitly reads all ARs. If an illegal RAW dependency is present between an AR write and br.ia, the first IA-32 instruction fetch and execution may or may not see the updated AR value.

IA-32 instruction set execution leaves the contents of the ALAT undefined. Software can not rely on ALAT values being preserved across an instruction set transition. All registers left in the current register stack frame are undefined across an instruction set transition. On entry to IA-32 code, existing entries in the ALAT are ignored. If the register stack contains any dirty registers, an Illegal Operation fault is raised on the br.ia instruction. The current register stack frame is forced to zero. To flush the register file of dirty registers, the flushrs instruction must be issued in an instruction group preceding the br.ia instruction. To enhance the performance of the instruction set transition, software can start the register stack flush in parallel with starting the IA-32 instruction set by 1) ensuring flushrs is exactly one instruction group before the br.ia, and 2) br.ia is in the first B-slot. br.ia should always be executed in the first B-slot with a hint of “static-taken” (default), otherwise processor performance will be degraded.

If a br.ia causes any Itanium traps (e.g., Single Step trap, Taken Branch trap, or Unimplemented Instruction Address trap), IIP will contain the original 64-bit target IP. (The value will not have been zero extended from 32 bits.)
Another branch type is provided for simple counted loops. This branch type uses the Loop Count application register (LC) to determine the branch condition, and does not use a qualifying predicate:

- **cloop**: If the LC register is not equal to zero, it is decremented and the branch is taken.

In addition to these simple branch types, there are four types which are used for accelerating modulo-scheduled loops (see also “Modulo-scheduled Loop Support” on page 1:68 in *Volume 1: Application Architecture*). Two of these are for counted loops (which use the LC register), and two for while loops (which use the qualifying predicate). These loop types use register rotation to provide register renaming, and they use predication to turn off instructions that correspond to empty pipeline stages.

The Epilog Count application register (EC) is used to count epilog stages and, for some while loops, a portion of the prolog stages. In the epilog phase, EC is decremented each time around and, for most loops, when EC is one, the pipeline has been drained, and the loop is exited. For certain types of optimized, unrolled software-pipelined loops, the target of a `br.cexit` or `br.wexit` is set to the next sequential bundle. In this case, the pipeline may not be fully drained when EC is one, and continues to drain while EC is zero.

For these modulo-scheduled loop types, the calculation of whether the branch is taken or not depends on the kernel branch condition (LC for counted types, and the qualifying predicate for while types) and on the epilog condition (whether EC is greater than one or not).

These branch types are of two categories: top and exit. The top types (ctop and wtop) are used when the loop decision is located at the bottom of the loop body and therefore a taken branch will continue the loop while a fall through branch will exit the loop. The exit types (cexit and wexit) are used when the loop decision is located somewhere other than the bottom of the loop and therefore a fall through branch will continue the loop and a taken branch will exit the loop. The exit types are also used at intermediate points in an unrolled pipelined loop. (For more details, see “Modulo-scheduled Loop Support” on page 1:68 in *Volume 1: Application Architecture*).

The modulo-scheduled loop types are:

- **ctop** and **cexit**: These branch types behave identically, except in the determination of whether to branch or not. For `br.ctop`, the branch is taken if either LC is non-zero or EC is greater than one. For `br.cexit`, the opposite is true. It is not taken if either LC is non-zero or EC is greater than one and is taken otherwise.

These branch types also use LC and EC to control register rotation and predicate initialization. During the prolog and kernel phase, when LC is non-zero, LC counts down. When `br.ctop` or `br.cexit` is executed with LC equal to zero, the epilog phase is entered, and EC counts down. When `br.ctop` or `br.cexit` is executed with LC equal to zero and EC equal to one, a final decrement of EC and a final register rotation are done. If LC and EC are equal to zero, register rotation stops. These other effects are the same for the two branch types, and are described in Figure 2-3.
**Figure 2-3. Operation of br.ctop and br.cexit**

**Figure 2-4. Operation of br.wtop and br.wexit**

**wtop and wexit:** These branch types behave identically, except in the determination of whether to branch or not. For `br.wtop`, the branch is taken if either the qualifying predicate is one or EC is greater than one. For `br.wexit`, the opposite is true. It is not taken if either the qualifying predicate is one or EC is greater than one, and is taken otherwise.

These branch types also use the qualifying predicate and EC to control register rotation and predicate initialization. During the prolog phase, the qualifying predicate is either zero or one, depending upon the scheme used to program the loop. During the kernel phase, the qualifying predicate is one. During the epilog phase, the qualifying predicate is zero, and EC counts down. When `br.wtop` or `br.wexit` is executed with the qualifying predicate equal to zero and EC equal to one, a final decrement of EC and a final register rotation are done. If the qualifying predicate and EC are zero, register rotation stops. These other effects are the same for the two branch types, and are described in Figure 2-4.
The loop-type branches (br.cloop, br.ctop, br.cexit, br.wtop, and br.wexit) are only allowed in instruction slot 2 within a bundle. Executing such an instruction in either slot 0 or 1 will cause an Illegal Operation fault, whether the branch would have been taken or not.

Read after Write (RAW) and Write after Read (WAR) dependency requirements are slightly different for branch instructions. Changes to BRs, PRs, and PFS by non-branch instructions are visible to a subsequent branch instruction in the same instruction group (i.e., a limited RAW is allowed for these resources). This allows for a low-latency compare-branch sequence, for example. The normal RAW requirements apply to the LC and EC application registers, and the RRBs.

Within an instruction group, a WAR dependency on PR 63 is not allowed if both the reading and writing instructions are branches. For example, a br.wtop or br.wexit may not use PR[63] as its qualifying predicate and PR[63] cannot be the qualifying predicate for any branch preceding a br.wtop or br.wexit in the same instruction group.

For dependency purposes, the loop-type branches effectively always write their associated resources, whether they are taken or not. The cloop type effectively always writes LC. When LC is 0, a cloop branch leaves it unchanged, but hardware may implement this as a re-write of LC with the same value. Similarly, br.ctop and br.cexit effectively always write LC, EC, the RRBs, and PR[63]. br.wtop and br.wexit effectively always write EC, the RRBs, and PR[63].

Values for various branch hint completers are shown in the following tables. Whether Prediction Strategy hints are shown in Table 2-7. Sequential Prefetch hints are shown in Table 2-8. Branch Cache Deallocation hints are shown in Table 2-9. See “Branch Prediction Hints” on page 1:70 in Volume 1: Application Architecture.

**Table 2-7. Branch Whether Hint**

<table>
<thead>
<tr>
<th>bwh Completer</th>
<th>Branch Whether Hint</th>
</tr>
</thead>
<tbody>
<tr>
<td>spnt</td>
<td>Static Not-Taken</td>
</tr>
<tr>
<td>sptk</td>
<td>Static Taken</td>
</tr>
<tr>
<td>dpnt</td>
<td>Dynamic Not-Taken</td>
</tr>
<tr>
<td>dptk</td>
<td>Dynamic Taken</td>
</tr>
</tbody>
</table>

**Table 2-8. Sequential Prefetch Hint**

<table>
<thead>
<tr>
<th>ph Completer</th>
<th>Sequential Prefetch Hint</th>
</tr>
</thead>
<tbody>
<tr>
<td>few or none</td>
<td>Few lines</td>
</tr>
<tr>
<td>many</td>
<td>Many lines</td>
</tr>
</tbody>
</table>

**Table 2-9. Branch Cache Deallocation Hint**

<table>
<thead>
<tr>
<th>dh Completer</th>
<th>Branch Cache Deallocation Hint</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Don’t deallocate</td>
</tr>
<tr>
<td>clr</td>
<td>Deallocate branch information</td>
</tr>
</tbody>
</table>

**Operation:**

```plaintext
if (ip_relative_form) // determine branch target
tmp_IP = IP + sign_ext((imm_21 << 4), 25);
else // indirect_form
tmp_IP = BR[b2];

if (btype != 'ia') // for Itanium branches,
tmp_IP = tmp_IP & ~0xf; // ignore bottom 4 bits of target
lower_priv_transition = 0;
```
switch (btype) {
    case 'cond': // simple conditional branch
        tmp_taken = PR[qp];
        break;
    case 'call': // call saves a return link
        tmp_taken = PR[qp];
        if (tmp_taken) {
            BR[b1] = IP + 16;

            AR[PFS].pfm = CFM;
            AR[PFS].pec = AR[EC];
            AR[PFS].ppl = PSR.cpl;

            alat_frame_update(CFM.sol, 0);
            rse_preserve_frame(CFM.sol);
            CFM.sof -= CFM.sol; // new frame size is size of outs
            CFM.sol = 0;
            CFM.sor = 0;
            CFM.rrb.gr = 0;
            CFM.rrb.fr = 0;
            CFM.rrb.pr = 0;
        }
        break;
    case 'ret': // return restores stack frame
        tmp_taken = PR[qp];
        if (tmp_taken) {
            // tmp_growth indicates the amount to move logical TOP *up*:
            // tmp_growth = sizeof(previous out) - sizeof(current frame)
            // a negative amount indicates a shrinking stack
            tmp_growth = (AR[PFS].pfm.sol - AR[PFS].pfm.sof) - CFM.sof;
            alat_frame_update(-AR[PFS].pfm.sol, 0);
            rse_fatal = rse_restore_frame(AR[PFS].pfm.sol, tmp_growth, CFM.sof);
            if (rse_fatal) { // See Section 6.4 "RSE Operation" in Volume 2
                CFM.sof = 0;
                CFM.sol = 0;
                CFM.sor = 0;
                CFM.rrb.gr = 0;
                CFM.rrb.fr = 0;
                CFM.rrb.pr = 0;
            } else // normal branch return
                CFM = AR[PFS].pfm;

            rse_enable_current_frame_load();
            AR[EC] = AR[PFS].pec;
            if (PSR.cpl u< AR[PFS].ppl) { // ... and restores privilege
                PSR.cpl = AR[PFS].ppl;
                lower_priv_transition = 1;
            }
        }
        break;
    case 'ia': // switch to IA mode
        tmp_taken = 1;
        if (qp != 0)
            illegal_operation_fault();
        if (ARP[BPSTORE] != ARP[BSP])
            illegal_operation_fault();
        if (PSR.di)
            disabled_instruction_set_transition_fault();
PSR.is = 1; // set IA-32 Instruction Set Mode
CFM.sof = 0;  // force current stack frame
CFM.sol = 0;  // to zero
CFM.sor = 0;
CFM.rrb.gr = 0;
CFM.rrb.fr = 0;
CFM.rrb.pr = 0;
rsi_invalidate_non_current_regs();
// compute effective instruction pointer
    EIP{31:0} = tmp_IP{31:0} - AR[CSD].Base;

// Note the register stack is disabled during IA-32 instruction
// set execution
break;

case 'cloop':  // simple counted loop
    if (slot != 2)
        illegal_operation_fault();
    tmp_taken = (AR[LC] != 0);
    if (AR[LC] != 0)
        AR[LC]--;
    break;

    case 'ctop':
    case 'cexit':  // SW pipelined counted loop
        if (slot != 2)
            illegal_operation_fault();
        if (btype == 'ctop')
            tmp_taken = ((AR[LC] != 0) || (AR[EC] u> 1));
        if (btype == 'cexit')
            tmp_taken = !(AR[LC] != 0) || (AR[EC] u> 1));
        if (AR[LC] != 0)
            AR[LC]--;
        AR[EC] = AR[EC];
        PR[63] = 1;
        rotate_regs();
        } else if (AR[EC] != 0) {
            AR[LC] = AR[LC];
            AR[EC]--;
            PR[63] = 0;
            rotate_regs();
        } else {
            AR[LC] = AR[LC];
            AR[EC] = AR[EC];
            PR[63] = 0;
            CFM.rrb.gr = CFM.rrb.gr;
            CFM.rrb.fr = CFM.rrb.fr;
            CFM.rrb.pr = CFM.rrb.pr;

        }
        break;

    case 'wtop':
    case 'wexit':  // SW pipelined while loop
        if (slot != 2)
            illegal_operation_fault();
        if (btype == 'wtop')
            tmp_taken = (PR[qp] || (AR[EC] u> 1));
        if (btype == 'wexit')
            tmp_taken = !(PR[qp] || (AR[EC] u> 1));
        if (PR[qp]) {
            AR[EC] = AR[EC];
            PR[63] = 0;
            rotate_regs();
        } else if (AR[EC] != 0) {
            AR[EC]--;
            PR[63] = 0;
rotate_regs();

} else {
    AR[EC] = AR[EC];
    PR[63] = 0;
    CFM.rrb.gr = CFM.rrb.gr;
    CFM.rrb.fr = CFM.rrb.fr;
    CFM.rrb.pr = CFM.rrb.pr;
}
break;

if (tmp_taken) {
    taken_branch = 1;
    IP = tmp_IP; // set the new value for IP
    if ((PSR.it && unimplemented_virtual_address(tmp_IP))
        || (PSR.it && unimplemented_physical_address(tmp_IP)))
        unimplemented_instruction_address_trap(lower_priv_transition, tmp_IP);
    if (lower_priv_transition && PSR.lp)
        lower_privilege_transfer_trap();
    if (PSR.tb)
        taken_branch_trap();
}

**Interruptions:**
- Illegal Operation fault
- Disabled Instruction Set Transition fault
- Unimplemented Instruction Address trap
- Lower-Privilege Transfer trap
- Taken Branch trap

Additional Faults on IA-32 target instructions:
- IA-32_Exception(GPFault)
- Disabled FP Reg Fault if PSR.dfh is 1
**Break**

**Format:**

- `(qp) break imm21`  
  `- pseudo-op`  
- `(qp) break.i imm21`  
  `- i_unit_form I19`  
- `(qp) break.b imm21`  
  `- b_unit_form B9`  
- `(qp) break.m imm21`  
  `- m_unit_form M37`  
- `(qp) break.f imm21`  
  `- f_unit_form F15`  
- `(qp) break.x imm62`  
  `- x_unit_form X1`

**Description:**

A Break Instruction fault is taken. For the i_unit_form, f_unit_form and m_unit_form, the value specified by `imm21` is zero-extended and placed in the Interruption Immediate control register (IIM).

For the b_unit_form, `imm21` is ignored and the value zero is placed in the Interruption Immediate control register (IIM).

For the x_unit_form, the lower 21 bits of the value specified by `imm62` is zero-extended and placed in the Interruption Immediate control register (IIM). The L slot of the bundle contains the upper 41 bits of `imm62`.

A `break.i` instruction may be encoded in an MLI-template bundle, in which case the L slot of the bundle is ignored.

This instruction has five forms, each of which can be executed only on a particular execution unit type. The pseudo-op can be used if the unit type to execute on is unimportant.

**Operation:**

```c
if (PR[qp]) {
    if (b_unit_form)
        immediate = 0;
    else if (x_unit_form)
        immediate = zero_ext(imm62, 21);
    else // i_unit_form || m_unit_form || f_unit_form
        immediate = zero_ext(imm21, 21);
    break_instruction_fault(immediate);
}
```

**Interruptions:** Break Instruction fault
Branch Long

**Format:**

\[(qp) \text{brl.btype.bwh.ph.dh } target_{64} \]
\[(qp) \text{brl.btype.bwh.ph.dh } b_1 = target_{64} \]
\[b_1,ph \text{.dh } target_{64} \]

**Description:**

A branch condition is evaluated, and either a branch is taken, or execution continues with the next sequential instruction. The execution of a branch logically follows the execution of all previous non-branch instructions in the same instruction group. On a taken branch, execution begins at slot 0.

Long branches are always IP-relative. The \(target_{64}\) operand, in assembly, specifies a label to branch to. This is encoded in the long branch instruction as an immediate displacement (\(imm_{60}\)) between the target bundle and the bundle containing this instruction (\(imm_{60} = target_{64} - \text{IP} \gg 4\)). The L slot of the bundle contains 39 bits of \(imm_{60}\).

### Table 2-10. Long Branch Types

<table>
<thead>
<tr>
<th>(btype)</th>
<th>Function</th>
<th>Branch Condition</th>
<th>Target Address</th>
</tr>
</thead>
<tbody>
<tr>
<td>cond or none</td>
<td>Conditional branch</td>
<td>Qualifying predicate</td>
<td>IP-relative</td>
</tr>
<tr>
<td>call</td>
<td>Conditional procedure call</td>
<td>Qualifying predicate</td>
<td>IP-relative</td>
</tr>
</tbody>
</table>

There is a pseudo-op for long unconditional branches, encoded like a conditional branch (\(btype = \text{cond}\)), with the \(qp\) field specifying PR 0, and with the \(bwh\) hint of sptk.

The branch type determines how the branch condition is calculated and whether the branch has other effects (such as writing a link register). For all long branch types, the branch condition is simply the value of the specified predicate register:

- **cond**: If the qualifying predicate is 1, the branch is taken. Otherwise it is not taken.
- **call**: If the qualifying predicate is 1, the branch is taken and several other actions occur:
  - The current values of the Current Frame Marker (CFM), the EC application register and the current privilege level are saved in the Previous Function State application register.
  - The caller’s stack frame is effectively saved and the callee is provided with a frame containing only the caller’s output region.
  - The rotation rename base registers in the CFM are reset to 0.
  - A return link value is placed in BR \(b_1\).

Read after Write (RAW) and Write after Read (WAR) dependency requirements for long branch instructions are slightly different than for other instructions but are the same as for branch instructions. See page 3:22 for details.

This instruction must be immediately followed by a stop; otherwise its behavior is undefined.

Values for various branch hint completers are the same as for branch instructions. Whether Prediction Strategy hints are shown in Table 2-7 on page 3:22, Sequential Prefetch hints are shown in Table 2-8 on page 3:22, and Branch Cache Deallocation hints are shown in Table 2-9 on page 3:22. See “Branch Prediction Hints” on page 1:70 in Volume 1: Application Architecture.

This instruction is not implemented on the Itanium processor, which takes an Illegal Operation fault whenever a long branch instruction is encountered, regardless of whether the branch is taken or not. To support the Itanium processor, the operating system is required to provide an Illegal Operation fault handler which emulates taken and not-taken long branches. Presence of this instruction is indicated by a 1 in the \(lb\) bit of CPUID register 4. See Section 3.1.11 “Processor Identification Registers” in Volume 1: Application Architecture.
tmp_IP = IP + (imm_{60} << 4); // determine branch target
if (!followed_by_stop())
  undefined_behavior();
if (!long_branch_implemented())
  illegal_operation_fault();

switch (btype) {
  case 'cond': // simple conditional branch
    tmp_taken = PR[qp];
    break;
  case 'call': // call saves a return link
    tmp_taken = PR[qp];
    if (tmp_taken) {
      BR[b1] = IP + 16;
      AR[PFS].pfm = CFM; // ... and saves the stack frame
      AR[PFS].pec = AR[EC];
      AR[PFS].ppl = PSR.cpl;
      alat_frame_update(CFM.sol, 0);
      rse_preserve_frame(CFM.sol);
      CFM.sof -= CFM.sol; // new frame size is size of outs
      CFM.sol = 0;
      CFM.sor = 0;
      CFM.rrb.gr = 0;
      CFM.rrb.fr = 0;
      CFM.rrb.pr = 0;
    }
    break;
}
if (tmp_taken) {
  taken_branch = 1;
  IP = tmp_IP; // set the new value for IP
  if ((PSR.it && unimplemented_virtual_address(tmp_IP))
    || (!PSR.it && unimplemented_physical_address(tmp_IP)))
    unimplemented_instruction_address_trap(0,tmp_IP);
  if (PSR.tb)
    taken_branch_trap();
}

Interruptions: Illegal Operation fault
              Taken Branch trap
              Unimplemented Instruction Address trap
Branch Predict

Format:

- `brp.ipwh.ih target_{25}, tag_{13}`  
  - ip_relative_form B6
- `brp.indwh.ih b_{2}, tag_{13}`  
  - indirect_form B7
- `brp.ret.indwh.ih b_{2}, tag_{13}`  
  - return_form, indirect_form B7

Description: This instruction can be used to provide to hardware early information about a future branch. It has no effect on architectural machine state, and operates as a `nop` instruction except for its performance effects.

The `tag_{13}` operand, in assembly, specifies the address of the branch instruction to which this prediction information applies. This is encoded in the branch predict instruction as a signed immediate displacement (`imm_{9}`) between the bundle containing the presaged branch and the bundle containing this instruction (`imm_{9} = tag_{13} – IP >> 4`).

The `target_{25}` operand, in assembly, specifies the label that the presaged branch will have as its target. This is encoded in the branch predict instruction exactly as in branch instructions, with a signed immediate displacement (`imm_{21}`) between the target bundle and the bundle containing this instruction (`imm_{21} = target_{25} – IP >> 4`). The indirect_form can be used to presage an indirect branch. In the indirect_form, the target of the presaged branch is given by BR $b_{2}$.

The return_form is used to indicate that the presaged branch will be a return.

Other hints can be given about the presaged branch. Values for various hint completers are shown in the following tables. For more details, refer to “Branch Prediction Hints” on page 1:70 in Volume 1: Application Architecture.

The `ipwh` and `indwh` completers provide information about how best the branch condition should be predicted, when the branch is reached.

**Table 2-11. IP-relative Branch Predict Whether Hint**

<table>
<thead>
<tr>
<th><code>ipwh</code> Completer</th>
<th>IP-relative Branch Predict Whether Hint</th>
</tr>
</thead>
<tbody>
<tr>
<td>sptk</td>
<td>Presaged branch should be predicted Static Taken</td>
</tr>
<tr>
<td>loop</td>
<td>Presaged branch will be br.cloop, br.ctop, or br.wtop</td>
</tr>
<tr>
<td>exit</td>
<td>Presaged branch will be br.cexit or br.wexit</td>
</tr>
<tr>
<td>dptk</td>
<td>Presaged branch should be predicted Dynamically</td>
</tr>
</tbody>
</table>

**Table 2-12. Indirect Branch Predict Whether Hint**

<table>
<thead>
<tr>
<th><code>indwh</code> Completer</th>
<th>Indirect Branch Predict Whether Hint</th>
</tr>
</thead>
<tbody>
<tr>
<td>sptk</td>
<td>Presaged branch should be predicted Static Taken</td>
</tr>
<tr>
<td>dptk</td>
<td>Presaged branch should be predicted Dynamically</td>
</tr>
</tbody>
</table>

The `ih` completer can be used to mark a small number of very important branches (e.g., an inner loop branch). This can signal to hardware to use faster, smaller prediction structures for this information.

**Table 2-13. Importance Hint**

<table>
<thead>
<tr>
<th><code>ih</code> Completer</th>
<th>Branch Predict Importance Hint</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Less important</td>
</tr>
<tr>
<td>imp</td>
<td>More important</td>
</tr>
</tbody>
</table>
Operation:

```c
    tmp_tag = IP + sign_ext((timm << 4), 13);
    if (ip_relative_form) {
        tmp_target = IP + sign_ext((imm2 << 4), 25);
        tmp_wh = ipwh;
    } else { // indirect_form
        tmp_target = BR[b2];
        tmp_wh = indwh;
    }
    branch_predict(tmp_wh, ih, return_form, tmp_target, tmp_tag);
```

Interruptions: None
**Bank Switch**

**Format:**
- `bsw.0` zero_form B8
- `bsw.1` one_form B8

**Description:**
This instruction switches to the specified register bank. The `zero_form` specifies Bank 0 for GR16 to GR31. The `one_form` specifies Bank 1 for GR16 to GR31. After the bank switch the previous register bank is no longer accessible but does retain its current state. If the new and old register banks are the same, `bsw` is effectively a `nop`, although there may be a performance degradation.

A `bsw` instruction must be the last instruction in an instruction group. Otherwise, an Illegal Operation fault is taken. Instructions in the same instruction group that access GR16 to GR31 reference the previous register bank. Subsequent instruction groups reference the new register bank.

This instruction is privileged.

This instruction cannot be predicated.

**Operation:**
```c
if (!followed_by_stop())
    illegal_operation_fault();

if (PSR.cpl != 0)
    privileged_operation_fault(0);

if (zero_form)
    PSR.bn = 0;
else // one_form
    PSR.bn = 1;
```

**Interruptions:**
- Illegal Operation fault
- Privileged Operation fault

**Serialization:**
This instruction does not require any additional instruction or data serialization operation. The bank switch occurs synchronously with its execution.
Speculation Check

Format:

\[
\begin{align*}
(qp) & \text{ chk.s } r_2, \text{target}_25 & \text{ pseudo-op} \\
(qp) & \text{ chk.s.i } r_2, \text{target}_25 & \text{ control_form, i_unit_form, gr_form} \quad \text{I20} \\
(qp) & \text{ chk.s.m } r_2, \text{target}_25 & \text{ control_form, m_unit_form, gr_form} \quad \text{M20} \\
(qp) & \text{ chk.s.f2 \text{ } fr} & \text{ control_form, fr_form} \quad \text{M21} \\
(qp) & \text{ chk.a.aclr } r_1, \text{target}_25 & \text{ data_form, gr_form} \quad \text{M22} \\
(qp) & \text{ chk.a.aclr } f_1, \text{target}_25 & \text{ data_form, fr_form} \quad \text{M23} \\
\end{align*}
\]

Description: The result of a control- or data-speculative calculation is checked for success or failure. If the check fails, a branch to \text{target}_25 is taken.

In the control_form, success is determined by a NaT indication for the source register. If the NaT bit corresponding to GR \text{r}_2 is 1 (in the gr_form), or FR \text{f}_2 contains a NaTVal (in the fr_form), the check fails.

In the data_form, success is determined by the ALAT. The ALAT is queried using the general register specifier \text{r}_1 (in the gr_form), or the floating-point register specifier \text{f}_1 (in the fr_form). If no ALAT entry matches, the check fails. An implementation may optionally cause the check to fail independent of whether an ALAT entry matches. A \text{chk.a} with general register specifier \text{r}_0 or floating-point register specifiers \text{f}_0 or \text{f}_1 always fails.

The \text{target}_25 operand, in assembly, specifies a label to branch to. This is encoded in the instruction as a signed immediate displacement \((imm_{21})\) between the target bundle and the bundle containing this instruction \((imm_{21} = \text{target}_25 - \text{IP} >> 4)\).

The branching behavior of this instruction can be optionally unimplemented. If the instruction would have branched, and the branching behavior is not implemented, then a Speculative Operation fault is taken and the value specified by \text{imm}_{21} is zero-extended and placed in the Interruption Immediate control register (IIM). The fault handler emulates the branch by sign-extending the IIM value, adding it to IIP and returning.

The control_form of this instruction for checking general registers can be encoded on either an I-unit or an M-unit. The pseudo-op can be used if the unit type to execute on is unimportant.

For the data_form, if an ALAT entry matches, the matching ALAT entry can be optionally invalidated, based on the value of the aclr completer (See Table 2-14).

<table>
<thead>
<tr>
<th>aclr Completer</th>
<th>Effect on ALAT</th>
</tr>
</thead>
<tbody>
<tr>
<td>clr</td>
<td>Invalidate matching ALAT entry</td>
</tr>
<tr>
<td>nc</td>
<td>Don’t invalidate</td>
</tr>
</tbody>
</table>

Table 2-14. ALAT Clear Completer

Note that if the clr value of the aclr completer is used and the check succeeds, the matching ALAT entry is invalidated. However, if the check fails (which may happen even if there is a matching ALAT entry), any matching ALAT entry may optionally be invalidated, but this is not required. Recovery code for data speculation, therefore, cannot rely on the absence of a matching ALAT entry.
Operation:

```c
if (PR[qp]) {  
    if (control_form) {  
        if (fr_form && (tmp_isrcode = fp_reg_disabled(f2, 0, 0, 0)))  
            disabled_fp_register_fault(tmp_isrcode, 0);  
        check_type = gr_form ? CHKS_GENERAL : CHKS_FLOAT;  
        fail = (gr_form && GR[r2].nat) || (fr_form && FR[f2] == NATVAL);  
    } else { // data_form  
        if (gr_form) {  
            reg_type = GENERAL;  
            check_type = CHKA_GENERAL;  
            alat_index = r1;  
            always_fail = (alat_index == 0);  
        } else { // fr_form  
            reg_type = FLOAT;  
            check_type = CHKA_FLOAT;  
            alat_index = f1;  
            always_fail = ((alat_index == 0) || (alat_index == 1));  
        }
    fail = (always_fail || (!alat_cmp(reg_type, alat_index)));  
}  
if (fail) {  
    if (check_branch_implemented(check_type)) {  
        taken_branch = 1;  
        IP = IP + sign_ext((imm22 << 4), 25);  
        if ((PSR.it && unimplemented_virtual_address(IP))  
            || (PSR.it && unimplemented_physical_address(IP)))  
            unimplemented_instruction_address_trap(0, IP);  
        if (PSR.tb)  
            taken_branch_trap();  
    } else  
        speculation_fault(check_type, zero_ext(imm22, 21));  
} else if (data_form && (aclr == 'clr'))  
    alat_inval_single_entry(reg_type, alat_index);  
}
```

Interruptions:

- Disabled Floating-point Register fault
- Unimplemented Instruction Address trap
- Speculative Operation fault
- Taken Branch trap
Clear RRB

Format:  

```
clrrrb                     all_form   B8
clrrrb.pr                pred_form   B8
```

Description: In the all_form, the register rename base registers (CFM.rrb.gr, CFM.rrb.fr, and CFM.rrb.pr) are cleared. In the pred_form, the single register rename base register for the predicates (CFM.rrb.pr) is cleared.

This instruction must be the last instruction in an instruction group, or an Illegal Operation fault is taken.

This instruction cannot be predicated.

Operation:  

```
if (!followed_by_stop())
    illegal_operation_fault();

if (all_form) {
    CFM.rrb.gr = 0;
    CFM.rrb.fr = 0;
    CFM.rrb.pr = 0;
} else { // pred_form
    CFM.rrb.pr = 0;
}
```

Interruptions: Illegal Operation fault
**Compare**

**Format:**

- \((qp)\) cmp.crel.cype \(p_1, p_2 = r_2, r_3\)  
- \((qp)\) cmp.crel.cype \(p_1, p_2 = \text{imm}_8, r_3\)  
- \((qp)\) cmp.crel.cype \(p_1, p_2 = r_0, r_3\)  
- \((qp)\) cmp.crel.cype \(p_1, p_2 = r_3, r_0\)

**register_form**  
- \(A6\)

**imm8_form**  
- \(A8\)

**parallel_inequality_form**  
- \(A7\)

**pseudo-op**  
- \(A6\)

**Description:**

The two source operands are compared for one of ten relations specified by \(crel\). This produces a boolean result which is 1 if the comparison condition is true, and 0 otherwise. This result is written to the two predicate register destinations, \(p_1\) and \(p_2\). The way the result is written to the destinations is determined by the compare type specified by \(ctype\).

The compare types describe how the predicate targets are updated based on the result of the comparison. The normal type simply writes the compare result to one target, and the complement to the other. The parallel types update the targets only for a particular comparison result. This allows multiple simultaneous OR-type or multiple simultaneous AND-type compares to target the same predicate register.

The unc type is special in that it first initializes both predicate targets to 0, independent of the qualifying predicate. It then operates the same as the normal type. The behavior of the compare types is described in Table 2-15. A blank entry indicates the predicate target is left unchanged.

**Table 2-15. Comparison Types**

<table>
<thead>
<tr>
<th>(ctype)</th>
<th>pseudo-op of</th>
<th>(PR[qp] == 0)</th>
<th>(PR[qp] == 1)</th>
</tr>
</thead>
<tbody>
<tr>
<td>(none)</td>
<td></td>
<td>(PR[p_1]) (PR[p_2])</td>
<td>(PR[p_1]) (PR[p_2])</td>
</tr>
<tr>
<td>(unc)</td>
<td></td>
<td>0 0 (PR[p_1]) (PR[p_2])</td>
<td>0 0 (PR[p_1]) (PR[p_2])</td>
</tr>
<tr>
<td>(or)</td>
<td>or and.cm</td>
<td>0 0 (PR[p_1]) (PR[p_2])</td>
<td>0 1 (PR[p_1]) (PR[p_2])</td>
</tr>
<tr>
<td>(and)</td>
<td>or and.cm</td>
<td>0 1 (PR[p_1]) (PR[p_2])</td>
<td>0 0 (PR[p_1]) (PR[p_2])</td>
</tr>
<tr>
<td>(orcm)</td>
<td></td>
<td>1 1 (PR[p_1]) (PR[p_2])</td>
<td>1 1 (PR[p_1]) (PR[p_2])</td>
</tr>
<tr>
<td>(andcm)</td>
<td></td>
<td>0 0 (PR[p_1]) (PR[p_2])</td>
<td>0 0 (PR[p_1]) (PR[p_2])</td>
</tr>
</tbody>
</table>

In the register_form the first operand is GR \(r_2\); in the imm8_form the first operand is taken from the sign-extended \(\text{imm}_8\) encoding field; and in the parallel_inequality_form the first operand must be GR 0. The parallel_inequality_form is only used when the compare type is one of the parallel types, and the relation is an inequality (\(>\), \(>=\), \(<\), \(<=\)). See below.

If the two predicate register destinations are the same (\(p_1\) and \(p_2\) specify the same predicate register), the instruction will take an Illegal Operation fault, if the qualifying predicate is 1, or if the compare type is unc.

Of the ten relations, not all are directly implemented in hardware. Some are actually pseudo-ops. For these, the assembler simply switches the source operand specifiers and/or switches the predicate target specifiers and uses an implemented relation. For some of the pseudo-op compares in the imm8_form, the assembler subtracts 1 from the immediate value, making the allowed immediate range slightly different. Of the six parallel compare types, three of the types are actually pseudo-ops. The assembler simply uses the negative relation with an implemented type.
implemented relations and how the pseudo-ops map onto them are shown in Table 2-16 (for normal and unc type compares), and Table 2-17 (for parallel type compares).

### Table 2-16. 64-bit Comparison Relations for Normal and unc Compares

<table>
<thead>
<tr>
<th>crel</th>
<th>Compare Relation (a rel b)</th>
<th>Register Form is a pseudo-op of</th>
<th>Immediate Form is a pseudo-op of</th>
<th>Immediate Range</th>
</tr>
</thead>
<tbody>
<tr>
<td>eq</td>
<td>a == b</td>
<td>eq</td>
<td>eq</td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>ne</td>
<td>a (!=) b</td>
<td></td>
<td></td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>lt</td>
<td>a (&lt;) b, unsigned</td>
<td>lt a (&lt;) b p₁ (&lt;) p₂</td>
<td>lt a⁻¹ p₁ (&lt;) p₂</td>
<td>-127 .. 128</td>
</tr>
<tr>
<td>le</td>
<td>a (\le) b</td>
<td>lt a (\le) b p₁ (&lt;) p₂</td>
<td>lt a⁻¹ p₁ (&lt;) p₂</td>
<td>-127 .. 128</td>
</tr>
<tr>
<td>gt</td>
<td>a (&gt;) b</td>
<td>lt a (&gt;) b p₁ (&lt;) p₂</td>
<td>lt a⁻¹ p₁ (&lt;) p₂</td>
<td>-127 .. 128</td>
</tr>
<tr>
<td>ge</td>
<td>a (\ge) b</td>
<td>lt a (\ge) b p₁ (&lt;) p₂</td>
<td>lt a⁻¹ p₁ (&lt;) p₂</td>
<td>-127 .. 128</td>
</tr>
<tr>
<td>ltu</td>
<td>a (&lt;) b, unsigned</td>
<td>ltu a (&lt;) b p₁ (&lt;) p₂</td>
<td>ltu a⁻¹ p₁ (&lt;) p₂</td>
<td>0 .. 127, 2⁶₄⋅128 .. 2⁶₄⋅1</td>
</tr>
<tr>
<td>leu</td>
<td>a (\le) b</td>
<td>ltu a (\le) b p₁ (&lt;) p₂</td>
<td>ltu a⁻¹ p₁ (&lt;) p₂</td>
<td>0 .. 127, 2⁶₄⋅128 .. 2⁶₄</td>
</tr>
<tr>
<td>gtu</td>
<td>a (&gt;) b</td>
<td>ltu a (&gt;) b p₁ (&lt;) p₂</td>
<td>ltu a⁻¹ p₁ (&lt;) p₂</td>
<td>0 .. 127, 2⁶₄⋅128 .. 2⁶₄</td>
</tr>
<tr>
<td>geu</td>
<td>a (\ge) b</td>
<td>ltu a (\ge) b p₁ (&lt;) p₂</td>
<td>ltu a⁻¹ p₁ (&lt;) p₂</td>
<td>0 .. 127, 2⁶₄⋅128 .. 2⁶₄</td>
</tr>
</tbody>
</table>

The parallel compare types can be used only with a restricted set of relations and operands. They can be used with equal and not-equal comparisons between two registers or between a register and an immediate, or they can be used with inequality comparisons between a register and GR 0.

Unsigned relations are not provided, since they are not of much use when one of the operands is zero. For the parallel inequality comparisons, hardware only directly implements the ones where the first operand (GR r₁) is GR 0. Comparisons where the second operand is GR 0 are pseudo-ops for which the assembler switches the register specifiers and uses the opposite relation.

### Table 2-17. 64-bit Comparison Relations for Parallel Compares

<table>
<thead>
<tr>
<th>crel</th>
<th>Compare Relation (a rel b)</th>
<th>Register Form is a pseudo-op of</th>
<th>Immediate Range</th>
</tr>
</thead>
<tbody>
<tr>
<td>eq</td>
<td>a == b</td>
<td></td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>ne</td>
<td>a (!=) b</td>
<td></td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>lt</td>
<td>0 (&lt;) b, signed</td>
<td>gt a (&lt;) b</td>
<td>no immediate forms</td>
</tr>
<tr>
<td>le</td>
<td>0 (\le) b</td>
<td>ge a (\le) b</td>
<td></td>
</tr>
<tr>
<td>gt</td>
<td>0 (&gt;) b</td>
<td>lt a (&gt;) b</td>
<td></td>
</tr>
<tr>
<td>ge</td>
<td>0 (\ge) b</td>
<td>le a (\ge) b</td>
<td></td>
</tr>
</tbody>
</table>
Operation: if (PR[qp]) {
  if (p1 == p2)
    illegal_operation_fault();
  tmp_nat = (register_form ? GR[r2].nat : 0) || GR[r3].nat;
  if (register_form)
    tmp_src = GR[r3];
  else if (imm8_form)
    tmp_src = sign_ext(imm8, 8);
  else // parallel_inequality_form
    tmp_src = 0;
  if (crel == 'eq')
    tmp_rel = tmp_src == GR[r3];
  else if (crel == 'ne')
    tmp_rel = tmp_src != GR[r3];
  else if (crel == 'lt')
    tmp_rel = lesser_signed(tmp_src, GR[r3]);
  else if (crel == 'le')
    tmp_rel = lesser_equal_signed(tmp_src, GR[r3]);
  else if (crel == 'gt')
    tmp_rel = greater_signed(tmp_src, GR[r3]);
  else if (crel == 'ge')
    tmp_rel = greater_equal_signed(tmp_src, GR[r3]);
  else if (crel == 'ltu')
    tmp_rel = lesser(tmp_src, GR[r3]);
  else if (crel == 'leu')
    tmp_rel = lesser_equal(tmp_src, GR[r3]);
  else if (crel == 'gtu')
    tmp_rel = greater(tmp_src, GR[r3]);
  else // parallel_inequality_form
    tmp_rel = greater_equal(tmp_src, GR[r3]);
  switch (ctype) {
    case 'and': // and-type compare
      if (tmp_nat || !tmp_rel) {
        PR[p1] = 0;
        PR[p2] = 0;
      }
      break;
    case 'or': // or-type compare
      if (!tmp_nat && tmp_rel) {
        PR[p1] = 1;
        PR[p2] = 1;
      }
      break;
    case 'or.andcm': // or.andcm-type compare
      if (!tmp_nat && tmp_rel) {
        PR[p1] = 1;
        PR[p2] = 0;
      }
      break;
    case 'unc': // unc-type compare
    default: // normal compare
      if (tmp_nat) {
        PR[p1] = 0;
        PR[p2] = 0;
      } else {
        PR[p1] = tmp_rel;
        PR[p2] = !tmp_rel;
      }
      break;
  }
} else {
  if (ctype == 'unc') {
    if (p1 == p2)
      illegal_operation_fault();
    PR[p1] = 0;
    PR[p2] = 0;
  }
}

Interruptions: Illegal Operation fault
**Compare 4 Bytes**

**Format:**

\[
\begin{align*}
&q p \text{ cmp4.crel.ctype } p_1, p_2 = r_2, r_3 \\
&q p \text{ cmp4.crel.ctype } p_1, p_2 = \text{imm}_8, r_3 \\
&q p \text{ cmp4.crel.ctype } p_1, p_2 = r_0, r_3 \\
&q p \text{ cmp4.crel.ctype } p_1, p_2 = r_3, r_0
\end{align*}
\]

<table>
<thead>
<tr>
<th>Register Form</th>
<th>Immediate Form</th>
</tr>
</thead>
<tbody>
<tr>
<td>A6</td>
<td>A6</td>
</tr>
<tr>
<td>A8</td>
<td>A8</td>
</tr>
<tr>
<td>A7</td>
<td>A7</td>
</tr>
<tr>
<td>A8</td>
<td>A7</td>
</tr>
</tbody>
</table>

**Description:**
The least significant 32 bits from each of two source operands are compared for one of ten relations specified by `crel`. This produces a boolean result which is 1 if the comparison condition is true, and 0 otherwise. This result is written to the two predicate register destinations, \( p_1 \) and \( p_2 \). The way the result is written to the destinations is determined by the compare type specified by `ctype`. See the Compare instruction and Table 2-15 on page 3:35.

In the register_form the first operand is GR \( r_2 \); in the imm8_form the first operand is taken from the sign-extended imm8 encoding field; and in the parallel_inequality_form the first operand must be GR 0. The parallel_inequality_form is only used when the compare type is one of the parallel types, and the relation is an inequality (>, >=, <, <=). See the Compare instruction and Table 2-17 on page 3:36.

If the two predicate register destinations are the same (\( p_1 \) and \( p_2 \) specify the same predicate register), the instruction will take an Illegal Operation fault, if the qualifying predicate is 1, or if the compare type is unc.

Of the ten relations, not all are directly implemented in hardware. Some are actually pseudo-ops. See the Compare instruction and Table 2-16 and Table 2-17 on page 3:36. The range for immediates is given below.

**Table 2-18. Immediate Range for 32-bit Compares**

<table>
<thead>
<tr>
<th>crel</th>
<th>Compare Relation ((a \ rel \ b))</th>
<th>Immediate Range</th>
</tr>
</thead>
<tbody>
<tr>
<td>eq</td>
<td>( a == b )</td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>ne</td>
<td>( a \neq b )</td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>lt</td>
<td>( a &lt; b ) \hspace{1cm} signed</td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>le</td>
<td>( a \leq b )</td>
<td>-127 .. 128</td>
</tr>
<tr>
<td>gt</td>
<td>( a &gt; b )</td>
<td>-127 .. 128</td>
</tr>
<tr>
<td>ge</td>
<td>( a \geq b )</td>
<td>-128 .. 127</td>
</tr>
<tr>
<td>ltu</td>
<td>( a &lt; b ) \hspace{1cm} unsigned</td>
<td>0 .. 127, (2^{32} - 1)</td>
</tr>
<tr>
<td>leu</td>
<td>( a \leq b )</td>
<td>1 .. 128, (2^{32} - 1)</td>
</tr>
<tr>
<td>gtu</td>
<td>( a &gt; b )</td>
<td>1 .. 128, (2^{32} - 1)</td>
</tr>
<tr>
<td>geu</td>
<td>( a \geq b )</td>
<td>0 .. (2^{32} - 1)</td>
</tr>
</tbody>
</table>
Operation: if (PR[qp]) {
    if (p1 == p2)
        illegal_operation_fault();

    tmp_nat = (register_form ? GR[r2].nat : 0) || GR[r3].nat;

    if (register_form)
        tmp_src = GR[r2];
    else if (imm8_form)
        tmp_src = sign_ext(imm8, 8);
    else // parallel_inequality_form
        tmp_src = 0;

    if (crel == 'eq')
        tmp_rel = tmp_src[31:0] == GR[r3][31:0];
    else if (crel == 'ne')
        tmp_rel = tmp_src[31:0] != GR[r3][31:0];
    else if (crel == 'lt')
        tmp_rel = lesser_signed(sign_ext(tmp_src, 32),
                                sign_ext(GR[r3], 32));
    else if (crel == 'le')
        tmp_rel = lesser_equal_signed(sign_ext(tmp_src, 32),
                                       sign_ext(GR[r3], 32));
    else if (crel == 'gt')
        tmp_rel = greater_signed(sign_ext(tmp_src, 32),
                                 sign_ext(GR[r3], 32));
    else if (crel == 'ge')
        tmp_rel = greater_equal_signed(sign_ext(tmp_src, 32),
                                        sign_ext(GR[r3], 32));
    else if (crel == 'ltu')
        tmp_rel = lesser(zero_ext(tmp_src, 32),
                         zero_ext(GR[r3], 32));
    else if (crel == 'leu')
        tmp_rel = lesser_equal(zero_ext(tmp_src, 32),
                               zero_ext(GR[r3], 32));
    else if (crel == 'gtu')
        tmp_rel = greater(zero_ext(tmp_src, 32),
                          zero_ext(GR[r3], 32));
    else // 'geu'
        tmp_rel = greater_equal(zero_ext(tmp_src, 32),
                                zero_ext(GR[r3], 32));

    switch (ctype) { // and-type compare
        case 'and':
            if (tmp_nat || !tmp_rel) {
                PR[p1] = 0;
                PR[p2] = 0;
            }
            break;
        case 'or': // or-type compare
            if (!tmp_nat && tmp_rel) {
                PR[p1] = 1;
                PR[p2] = 1;
            }
            break;
        case 'or.andcm': // or.andcm-type compare
            if (!tmp_nat && tmp_rel) {
                PR[p1] = 1;
                PR[p2] = 0;
            }
            break;
        case 'unc': // unc-type compare
            default: // normal compare
            if (tmp_nat) {
```c
PR[p1] = 0;
PR[p2] = 0;
} else {
    PR[p1] = tmp_rel;
    PR[p2] = !tmp_rel;
    break;
}
#else {
    if (ctype == 'unc') {
        if (p1 == p2)
            illegal_operation_fault();
        PR[p1] = 0;
        PR[p2] = 0;
    }
}

**Interruptions:** Illegal Operation fault
Compare and Exchange

Format: \((qp)\ cmpxchg_{sz,.sem,.ldhint} \ r_j = [r_3], \ r_2, \ ar.ccv\)

Description: A value consisting of \(sz\) bytes is read from memory starting at the address specified by the value in \(GR\ r_3\). The value is zero extended and compared with the contents of the \(cmpxchg\) Compare Value application register (AR\[CCV\]). If the two are equal, then the least significant \(sz\) bytes of the value in \(GR\ r_2\) are written to memory starting at the address specified by the value in \(GR\ r_3\). The zero-extended value read from memory is placed in \(GR\ r_1\) and the NaT bit corresponding to \(GR\ r_1\) is cleared.

The values of the \(sz\) completer are given in Table 2-19. The \(sem\) completer specifies the type of semaphore operation. These operations are described in Table 2-20. See “Sequentiality Attribute and Ordering” on page 1:69 in Volume 2: System Architecture for details on memory ordering.

### Table 2-19. Memory Compare and Exchange Size

<table>
<thead>
<tr>
<th>sz Completer</th>
<th>Bytes Accesssed</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>2</td>
<td>2</td>
</tr>
<tr>
<td>4</td>
<td>4</td>
</tr>
<tr>
<td>8</td>
<td>8</td>
</tr>
</tbody>
</table>

### Table 2-20. Compare and Exchange Semaphore Types

<table>
<thead>
<tr>
<th>sem Completer</th>
<th>Ordering Semantics</th>
<th>Semaphore Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>acq</td>
<td>Acquire</td>
<td>The memory read/write is made visible prior to all subsequent data memory accesses.</td>
</tr>
<tr>
<td>rel</td>
<td>Release</td>
<td>The memory read/write is made visible after all previous data memory accesses.</td>
</tr>
</tbody>
</table>

If the address specified by the value in \(GR\ r_3\) is not naturally aligned to the size of the value being accessed in memory, an Unaligned Data Reference fault is taken independent of the state of the User Mask alignment checking bit, UM.ac (PSR.ac in the Processor Status Register).

The memory read and write are guaranteed to be atomic.

Both read and write access privileges for the referenced page are required. The write access privilege check is performed whether or not the memory write is performed.

This instruction is only supported to cacheable pages with write-back write policy. Accesses to NaTPages cause a Data NaT Page Consumption fault. Accesses to pages with other memory attributes cause an Unsupported Data Reference fault.

The value of the \(ldhint\) completer specifies the locality of the memory access. The values of the \(ldhint\) completer are given in Table 2-33 on page 3:136. Locality hints do not affect program functionality and may be ignored by the implementation. See “Memory Hierarchy Control and Consistency” on page 1:62 for details.
cmpxchg

**Operation:**

```c
if (PR[gp]) {
    check_target_register(r1);

    if (GR[r3].nat || GR[r2].nat)
        register_nat_consumption_fault(SEMAPHORE);

    paddr = tlb_translate(GR[r3], sz, SEMAPHORE, PSR.cpl, &mattr,
                          &tmp_unused);

    if (!ma_supports_semaphores(mattr))
        unsupported_data_reference_fault(SEMAPHORE, GR[r3]);

    if (sem == 'acq')
        val = mem_xchg_cond(AR[CCV], GR[r2], paddr, sz, UM.be, mattr,
                             ACQUIRE, lhint);
    else // 'rel'
        val = mem_xchg_cond(AR[CCV], GR[r2], paddr, sz, UM.be, mattr,
                             RELEASE, lhint);
    val = zero_ext(val, sz * 8);

    if (AR[CCV] == val)
        alat_inval_multiple_entries(paddr, sz);

    GR[r1] = val;
    GR[r1].nat = 0;
}
```

**Interruptions:**

- Illegal Operation fault
- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Dirty Bit fault
- Data Access Bit fault
- Data Debug fault
- Unaligned Data Reference fault
- Unsupported Data Reference fault
Cover Stack Frame

Format:  cover  B8

Description:  A new stack frame of zero size is allocated which does not include any registers from the previous frame (as though all output registers in the previous frame had been locals). The register rename base registers are reset. If interruption collection is disabled (PSR.ic is zero), then the old value of the Current Frame Marker (CFM) is copied to the Interruption Function State register (IFS), and IFS.v is set to one.

A cover instruction must be the last instruction in an instruction group. Otherwise, an Illegal Operation fault is taken.

This instruction cannot be predicated.

Operation:

```c
if (!followed_by_stop())
    illegal_operation_fault();

alat_frame_update(CFM.sof, 0);
rse_preserve_frame(CFM.sof);
if (PSR.ic == 0) {
    CR[IFS].ifm = CFM;
    CR[IFS].v = 1;
}

CFM.sof = 0;
CFM.sol = 0;
CFM.sor = 0;
CFM.rrb.gr = 0;
CFM.rrb.fr = 0;
CFM.rrb.pr = 0;
```

Interruptions:  Illegal Operation fault
Compute Zero Index

Format:

\[(qp) \text{czx1.l } r_1 = r_3\] one_byte_form, left_form
\[(qp) \text{czx1.r } r_1 = r_3\] one_byte_form, right_form
\[(qp) \text{czx2.l } r_1 = r_3\] two_byte_form, left_form
\[(qp) \text{czx2.r } r_1 = r_3\] two_byte_form, right_form

Description:
GR \(r_3\) is scanned for a zero element. The element is either an 8-bit aligned byte (one_byte_form) or a 16-bit aligned pair of bytes (two_byte_form). The index of the first zero element is placed in GR \(r_1\). If there are no zero elements in GR \(r_3\), a default value is placed in GR \(r_1\). Table 2-21 gives the possible result values. In the left_form, the source is scanned from most significant element to least significant element, and in the right_form it is scanned from least significant element to most significant element.

Table 2-21. Result Ranges for \text{czx}

<table>
<thead>
<tr>
<th>Size</th>
<th>Element Width</th>
<th>Range of Result if Zero Element Found</th>
<th>Default Result if No Zero Element Found</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>8 bit</td>
<td>0-7</td>
<td>8</td>
</tr>
<tr>
<td>2</td>
<td>16 bit</td>
<td>0-3</td>
<td>4</td>
</tr>
</tbody>
</table>

Operation:

\[
\text{if (PR[qp])}
\text{check_target_register(r_1);}\
\text{if (one_byte_form)} \{
\text{if (left_form)} \{
\text{if ((GR[r_3] \& 0xff00000000000000) == 0) GR[r_1] = 0;}
\text{else if ((GR[r_3] \& 0x00ff000000000000) == 0) GR[r_1] = 1;}
\text{else if ((GR[r_3] \& 0x0000ff0000000000) == 0) GR[r_1] = 2;}
\text{else if ((GR[r_3] \& 0x000000ff00000000) == 0) GR[r_1] = 3;}
\text{else if ((GR[r_3] \& 0x00000000ff000000) == 0) GR[r_1] = 4;}
\text{else if ((GR[r_3] \& 0x0000000000ff0000) == 0) GR[r_1] = 5;}
\text{else if ((GR[r_3] \& 0x000000000000ff00) == 0) GR[r_1] = 6;}
\text{else if ((GR[r_3] \& 0x00000000000000ff) == 0) GR[r_1] = 7;}
\text{else GR[r_1] = 8;}
\text{else { // right_form scan from least significant up}
\text{if ((GR[r_3] \& 0x00000000000000ff) == 0) GR[r_1] = 0;}
\text{else if ((GR[r_3] \& 0x0000000000000ff0) == 0) GR[r_1] = 1;}
\text{else if ((GR[r_3] \& 0x00000000000ff00) == 0) GR[r_1] = 2;}
\text{else if ((GR[r_3] \& 0x0000000000ff000) == 0) GR[r_1] = 3;}
\text{else if ((GR[r_3] \& 0x000000000ff0000) == 0) GR[r_1] = 4;}
\text{else if ((GR[r_3] \& 0x00000000ff00000) == 0) GR[r_1] = 5;}
\text{else if ((GR[r_3] \& 0x0000000ff000000) == 0) GR[r_1] = 6;}
\text{else if ((GR[r_3] \& 0x000000ff0000000) == 0) GR[r_1] = 7;}
\text{else if ((GR[r_3] \& 0x00000ff00000000) == 0) GR[r_1] = 8;}
\text{else { // two_byte_form
\text{if (left_form)} \{
\text{if ((GR[r_3] \& 0xffff000000000000) == 0) GR[r_1] = 0;}
\text{else if ((GR[r_3] \& 0x0000ffff00000000) == 0) GR[r_1] = 1;}
\text{else if ((GR[r_3] \& 0x000000ff00000000) == 0) GR[r_1] = 2;}
\text{else if ((GR[r_3] \& 0x00000000ff000000) == 0) GR[r_1] = 3;}
\text{else if ((GR[r_3] \& 0x0000000000ff0000) == 0) GR[r_1] = 4;}
\text{else if ((GR[r_3] \& 0x000000000000ff00) == 0) GR[r_1] = 5;}
\text{else if ((GR[r_3] \& 0x00000000000000ff) == 0) GR[r_1] = 6;}
\text{else if ((GR[r_3] \& 0x0000000000000ff0) == 0) GR[r_1] = 7;}
\text{else if ((GR[r_3] \& 0x000000000000ff00) == 0) GR[r_1] = 8;}
\text{else { // right_form scan from least significant up}
\text{if ((GR[r_3] \& 0x00000000000000ff0) == 0) GR[r_1] = 0;}
\text{else if ((GR[r_3] \& 0x000000000000ff00) == 0) GR[r_1] = 1;}
\text{else if ((GR[r_3] \& 0x00000000000ff000) == 0) GR[r_1] = 2;}
\text{else if ((GR[r_3] \& 0x0000000000ff0000) == 0) GR[r_1] = 3;}
\text{else if ((GR[r_3] \& 0x000000000ff00000) == 0) GR[r_1] = 4;}
\text{else if ((GR[r_3] \& 0x00000000ff000000) == 0) GR[r_1] = 5;}
\text{else if ((GR[r_3] \& 0x0000000ff0000000) == 0) GR[r_1] = 6;}
\text{else if ((GR[r_3] \& 0x000000ff00000000) == 0) GR[r_1] = 7;}
\text{else if ((GR[r_3] \& 0x00000ff000000000) == 0) GR[r_1] = 8;}
\text{else GR[r_1] = 4;}}}}}}
\]
GR[r_1].nat = GR[r_3].nat;

**Interruptions:** Illegal Operation fault
Deposit

Format:

\[(qp) \text{dep } r_1 = r_2, r_3, pos_6, len_4\] merge_form, register_form

\[(qp) \text{dep } r_1 = \text{imm}_1, r_3, pos_6, len_6\] merge_form, imm_form

\[(qp) \text{dep.z } r_1 = r_2, pos_6, len_6\] zero_form, register_form

\[(qp) \text{dep.z } r_1 = \text{imm}_8, pos_6, len_6\] zero_form, imm_form

Description: In the merge_form, a right justified bit field taken from the first source operand is deposited into the value in GR \(r_3\) at an arbitrary bit position and the result is placed in GR \(r_1\). In the register_form the first source operand is GR \(r_2\); and in the imm_form it is the sign-extended value specified by \(\text{imm}_1\) (either all ones or all zeroes). The deposited bit field begins at the bit position specified by the \(pos_6\) immediate and extends to the left (towards the most significant bit) a number of bits specified by the \(len\) immediate. Note that \(len\) has a range of 1-16 in the register_form and 1-64 in the imm_form. The \(pos_6\) immediate has a range of 0 to 63.

In the zero_form, a right justified bit field taken from either the value in GR \(r_2\) (in the register_form) or the sign-extended value in \(\text{imm}_8\) (in the imm_form) is deposited into GR \(r_1\) and all other bits in GR \(r_1\) are cleared to zero. The deposited bit field begins at the bit position specified by the \(pos_6\) immediate and extends to the left (towards the most significant bit) a number of bits specified by the \(len\) immediate. The \(len\) immediate has a range of 1-64 and the \(pos_6\) immediate has a range of 0 to 63.

In the event that the deposited bit field extends beyond bit 63 of the target, i.e., \(len + pos_6 > 64\), the most significant \(len + pos_6 - 64\) bits of the deposited bit field are truncated. The \(len\) immediate is encoded as \(len\) minus 1 in the instruction.

The operation of \(\text{dep } r_1 = r_2, r_3, 36, 16\) is illustrated in Figure 2-5.

**Figure 2-5. Deposit Example (merge_form)**

![Figure 2-5. Deposit Example (merge_form)](image)

The operation of \(\text{dep.z } r_1 = r_2, 36, 16\) is illustrated in Figure 2-6.

**Figure 2-6. Deposit Example (zero_form)**

![Figure 2-6. Deposit Example (zero_form)](image)
Operation:  
if (PR[qp]) {
    check_target_register(r1);

    if (imm_form) {
        tmp_src = (merge_form ? sign_ext(imm1, 1) : sign_ext(imm8, 8));
        tmp_nat = merge_form ? GR[r3].nat : 0;
        tmp_len = len6;
    } else { // register_form
        tmp_src = GR[r2];
        tmp_nat = (merge_form ? GR[r3].nat : 0) || GR[r2].nat;
        tmp_len = merge_form ? len4 : len6;
    }
    if (pos6 + tmp_len u> 64)
        tmp_len = 64 - pos6;

    if (merge_form)
        GR[r1] = GR[r3];
    else // zero_form
        GR[r1] = 0;

    GR[r1]((pos6 + tmp_len - 1):pos6) = tmp_src((tmp_len - 1):0);
    GR[r1].nat = tmp_nat;
}

Interruptions:  Illegal Operation fault
Enter Privileged Code

**Format:**  
epc

**Description:**  
This instruction increases the privilege level. The new privilege level is given by the TLB entry for the page containing this instruction. This instruction can be used to implement calls to higher-privileged routines without the overhead of an interruption.

Before increasing the privilege level, a check is performed. The PFS.ppl (previous privilege level) is checked to ensure that it is not more privileged than the current privilege level. If this check fails, the instruction takes an Illegal Operation fault.

If the check succeeds, then the privilege is increased as follows:

- If instruction address translation is enabled and the page containing the epc instruction has execute-only page access rights and the privilege level assigned to the page is higher than (numerically less than) the current privilege level, then the current privilege level is set to the privilege level field in the translation for the page containing the epc instruction. This instruction can promote but cannot demote, and the new privilege comes from the TLB entry.

If instruction address translation is disabled, then the current privilege level is set to 0 (most privileged).

Instructions after the epc in the same instruction group may be executed at the old privilege level or the new, higher privilege level. Instructions in subsequent instruction groups will be executed at the new, higher privilege level.

- If the page containing the epc instruction has any other access rights besides execute-only, or if the privilege level assigned to the page is lower or equal to (numerically greater than or equal to) the current privilege level, then no action is taken (the current privilege level is unchanged).

Note that the ITLB is actually only read once, at instruction fetch. Information from the access rights and privilege level fields from the translation is then used in executing this instruction.

This instruction cannot be predicated.

**Operation:**  
```c
if (AR[PFS].ppl u< PSR.cpl)
    illegal_operation_fault();
if (PSR.it)
    PSR.cpl = tlb_enter_privileged_code();
else
    PSR.cpl = 0;
```

**Interruptions:**  
Illegal Operation fault
Extract

Format: \[(qp) \text{ extr } r_j = r_3, pos_6, len_6\]  
\[(qp) \text{ extr.u } r_j = r_3, pos_6, len_6\]

signed_form \quad \text{I11} 
unsigned_form \quad \text{I11}

Description: A field is extracted from GR\(r_3\), either zero extended or sign extended, and placed right-justified in GR\(r_j\). The field begins at the bit position given by the second operand and extends \(len_6\) bits to the left. The bit position where the field begins is specified by the \(pos_6\) immediate. The extracted field is sign extended in the signed_form or zero extended in the unsigned_form. The sign is taken from the most significant bit of the extracted field. If the specified field extends beyond the most significant bit of GR\(r_3\), the sign is taken from the most significant bit of GR\(r_3\). The immediate value \(len_6\) can be any number in the range 1 to 64, and is encoded as \(len_6-1\) in the instruction. The immediate value \(pos_6\) can be any value in the range 0 to 63.

The operation of \text{ extr } r_j = r_3, 7, 50 \quad \text{is illustrated in Figure 2-7.}

\begin{figure}
\centering
\includegraphics[width=0.8\textwidth]{extract_example.png}
\caption{Extract Example}
\end{figure}

Operation: \[
\text{if (PR[qp])} \{
\text{check_target_register}(r_j); \\
\quad \text{tmp_len = len}_6; \\
\quad \text{if (pos}_6 + \text{tmp_len u> 64)} \\
\quad \quad \text{tmp_len = 64 - pos}_6; \\
\quad \text{if (unsigned_form)} \\
\quad \quad \text{GR}[r_j] = \text{zero_ext}(\text{shift_right}\_\text{unsigned}(\text{GR}[r_3], \text{pos}_6), \text{tmp_len}); \\
\quad \text{else // signed_form} \\
\quad \quad \text{GR}[r_j] = \text{sign_ext}(\text{shift_right}\_\text{unsigned}(\text{GR}[r_3], \text{pos}_6), \text{tmp_len}); \\
\quad \text{GR}[r_j].\text{nat} = \text{GR}[r_3].\text{nat} \};
\]

Interruptions: Illegal Operation fault
Floating-point Absolute Value

**Format:**

\[(qp) \text{ fabs } f_1 = f_3\]  

pseudo-op of: \[(qp) \text{ fmerge.s } f_1 = f_0, f_3\]

**Description:**
The absolute value of the value in FR \(f_3\) is computed and placed in FR \(f_1\).
If \(f_3\) is a NaTVal, \(f_1\) is set to NaTVal instead of the computed result.

**Operation:**
See “Floating-point Merge” on page 3:73.
Floating-point Add

Format: \( (qp) \text{ fadd.pc.sf } f_1 = f_3, f_2 \)

pseudo-op of: \( (qp) \text{ fma.pc.sf } f_1 = f_3, f_1, f_2 \)

Description: FR \( f_3 \) and FR \( f_2 \) are added (computed to infinite precision), rounded to the precision indicated by \( pc \) (and possibly FPSR.sf.pc and FPSR.sf.wre) using the rounding mode specified by FPSR.sf.rc, and placed in FR \( f_1 \). If either FR \( f_3 \) or FR \( f_2 \) is a NaTVal, FR \( f_1 \) is set to NaTVal instead of the computed result.

The mnemonic values for the opcode’s \( pc \) are given in Table 2-22. The mnemonic values for \( sf \) are given in Table 2-23. For the encodings and interpretation of the status field’s \( pc, \text{ wre}, \) and \( rc \), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Table 2-22. Specified \( pc \) Mnemonic Values

<table>
<thead>
<tr>
<th>( pc ) Mnemonic</th>
<th>Precision Specified</th>
</tr>
</thead>
<tbody>
<tr>
<td>.s</td>
<td>single</td>
</tr>
<tr>
<td>.d</td>
<td>double</td>
</tr>
<tr>
<td>none</td>
<td>dynamic</td>
</tr>
<tr>
<td></td>
<td>(i.e. use pc value in status field)</td>
</tr>
</tbody>
</table>

Table 2-23. \( sf \) Mnemonic Values

<table>
<thead>
<tr>
<th>( sf/\text{Mnemonic} )</th>
<th>Status Field Accessed</th>
</tr>
</thead>
<tbody>
<tr>
<td>.s0 or none</td>
<td>sf0</td>
</tr>
<tr>
<td>.s1</td>
<td>sf1</td>
</tr>
<tr>
<td>.s2</td>
<td>sf2</td>
</tr>
<tr>
<td>.s3</td>
<td>sf3</td>
</tr>
</tbody>
</table>

Floating-point Absolute Maximum

Format:  \((qp)\) famax.sf \(f_1 = f_2, f_3\)  

Description: The operand with the larger absolute value is placed in \(f_1\). If the magnitude of \(f_2\) equals the magnitude of \(f_3\), \(f_1\) gets \(f_3\). 

If either \(f_2\) or \(f_3\) is a NaN, \(f_1\) gets \(f_3\). 

If either \(f_2\) or \(f_3\) is a NaTVal, \(f_1\) is set to NaTVal instead of the computed result. 

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as the fcmp.lt operation. 

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation: 

\[
\text{if} \ (\text{PR}[qp]) \ {\}
\text{fp\_check\_target\_register}(f_1); \\
\text{if} \ (\text{tmp\_isrcode} = \text{fp\_reg\_disabled}(f_1, f_2, f_3, 0)) \\
\text{disabled\_fp\_register\_fault}(\text{tmp\_isrcode}, 0); \\
\text{if} \ (\text{fp\_is\_natval}(\text{FR}[f_2]) \ | \ | \ \text{fp\_is\_natval}(\text{FR}[f_3])) \ {\}
\text{FR}[f_1] = \text{NATVAL}; \\
\text{else} \ {\}
\text{fminmax\_exception\_fault\_check}(f_2, f_3, sf, \&tmp\_fp\_env); \\
\text{if} \ (\text{fp\_raise\_fault}(tmp\_fp\_env)) \\
\text{fp\_exception\_fault}(\text{fp\_decode\_fault}(tmp\_fp\_env)); \\
\text{tmp\_right} = \text{fp\_reg\_read}(\text{FR}[f_2]); \\
\text{tmp\_left} = \text{fp\_reg\_read}(\text{FR}[f_3]); \\
\text{tmp\_right\_sign} = \text{FP\_SIGN\_POSITIVE}; \\
\text{tmp\_left\_sign} = \text{FP\_SIGN\_POSITIVE}; \\
\text{tmp\_bool\_res} = \text{fp\_less\_than}(\text{tmp\_left}, \text{tmp\_right}); \\
\text{FR}[f_1] = \text{tmp\_bool\_res} ? \text{FR}[f_2] : \text{FR}[f_3]; \\
\text{fp\_update\_fpsr}(sf, \text{tmp\_fp\_env}); \\
\}
\text{fp\_update\_psr}(f_1); \\
\}

FP Exceptions: Invalid Operation (V) 
Denormal/Unnormal Operand (D) 
Software Assist (SWA) fault

Interruptions: Illegal Operation fault 
Floating-point Exception fault 
Disabled Floating-point Register fault
Floating-point Absolute Minimum

**Format:** \( (qp) \ famin.sf \ f_1 = f_2, f_3 \)

**Description:** The operand with the smaller absolute value is placed in \( f_1 \). If the magnitude of \( f_2 \) equals the magnitude of \( f_3 \), \( f_1 \) gets \( f_3 \).

If either \( f_2 \) or \( f_3 \) is a NaN, \( f_1 \) gets \( f_2 \).

If either \( f_2 \) or \( f_3 \) is a NaTVal, \( f_1 \) is set to NaTVal instead of the computed result.

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as the \( fcmp.lt \) operation.

The mnemonic values for \( sf \) are given in Table 2-23 on page 3:51.

**Operation:**

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3])) {
        FR[f_1] = NATVAL;
    } else {
        fminmax_exception_fault_check(f_2, f_3, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        tmp_left = fp_reg_read(FR[f_2]);
        tmp_right = fp_reg_read(FR[f_3]);
        tmp_left.sign = FP_SIGN_POSITIVE;
        tmp_right.sign = FP_SIGN_POSITIVE;
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        fp_update_fpsr(sf, tmp_fp_env);
    }
    fp_update_psr(f_1);
}
```

**FP Exceptions:** Invalid Operation (V)

Denormal/Unnormal Operand (D)

Software Assist (SWA) fault

**Interruptions:**

Illegal Operation fault

Floating-point Exception fault

Disabled Floating-point Register fault
Floating-point Logical And

Format: \((qp) \ f_{\text{and}} = f_2 \& f_3\)

Description: The bit-wise logical AND of the significand fields of FR\(f_2\) and FR\(f_3\) is computed. The resulting value is stored in the significand field of FR\(f_1\). The exponent field of FR\(f_1\) is set to the biased exponent for 2.0\(^{63}\) (0x1003E) and the sign field of FR\(f_1\) is set to positive (0).

If either FR\(f_2\) or FR\(f_3\) is a NaTVal, FR\(f_1\) is set to NaTVal instead of the computed result.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3])) {
        FR[f_1] = NATVAL;
    } else {
        FR[f_1].significand = FR[f_2].significand & FR[f_3].significand;
        FR[f_1].exponent = FP_INTEGER_EXP;
        FR[f_1].sign = FP_SIGN_POSITIVE;
    }
    fp_update_psr(f_1);
}
```

FP Exceptions: None

Interruptions: Illegal Operation fault Disabled Floating-point Register fault
Floating-point And Complement

Format: \((qp)\ fandcm\ f_1 = f_2.f_3\)

Description: The bit-wise logical AND of the significand field of FR \(f_2\) with the bit-wise complemented significand field of FR \(f_3\) is computed. The resulting value is stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

If either FR \(f_2\) or FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3])) {
        FR[f_1] = NATVAL;
    } else {
        FR[f_1].significand = FR[f_2].significand & ~FR[f_3].significand;
        FR[f_1].exponent = FP_INTEGER_EXP;
        FR[f_1].sign = FP_SIGN_POSITIVE;
    }
    fp_update_psr(f_1);
}
```

FP Exceptions: None

Interruptions: Illegal Operation fault Disabled Floating-point Register fault
Flush Cache

**Format:** \((qp) \text{ fc } r_3\)

**Description:** The cache line associated with the address specified by the value of GR \(r_3\) is invalidated from all levels of the processor cache hierarchy. The invalidation is broadcast throughout the coherence domain. If, at any level of the cache hierarchy, the line is inconsistent with memory it is written to memory before invalidation.

The line size affected is at least 32-bytes (aligned on a 32-byte boundary). An implementation may flush a larger region.

When executed at privilege level 0, \text{ fc} performs no access rights or protection key checks. At other privilege levels, \text{ fc} performs access rights checks as if it were a 1-byte read, but no protection key checks (regardless of PSR.pk).

The memory attribute of the page containing the affected line has no effect on the behavior of this instruction. This instruction can be used to remove a range of addresses from the cache by first changing the memory attribute to non-cacheable and then flushing the range.

This instruction follows data dependency rules; it is ordered with respect to preceding and following memory references to the same line. \text{ fc} has data dependencies in the sense that any prior stores by this processor will be included in the data written back to memory. \text{ fc} is an unordered operation, and is not affected by a memory fence (mf) instruction. It is ordered with respect to the sync.i instruction.

**Operation:**

```c
if (PR[qp]) {
  itype = NON_ACCESS|FC|READ;
  if (GR[r_3].nat)
    register_nat_consumption_fault(itype);
  tmp_paddr = tlb_translate_nonaccess(GR[r_3], itype);
  mem_flush(tmp_paddr);
}
```

**Interruptions:**

- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data TLB fault
- Data Page Not Present fault
- Data Nested TLB fault
- Data NaT Page Consumption fault
- Alternate Data TLB fault
- Data Access Rights fault
- VHPT Data fault
Floating-point Check Flags

Format: \((qp)\ fchkf.sf \ target_{25}\)

Description:
The flags in FPSR.sf.flags are compared with FPSR.s0.flags and FPSR.traps. If any flags set in FPSR.sf.flags correspond to FPSR.traps which are enabled, or if any flags set in FPSR.sf.flags are not set in FPSR.s0.flags, then a branch to \(target_{25}\) is taken.

The \(target_{25}\) operand specifies a label to branch to. This is encoded in the instruction as a signed immediate displacement (\(imm_{21}\)) between the target bundle and the bundle containing this instruction (\(imm_{21} = target_{25} – IP >> 4\)).

The branching behavior of this instruction can be optionally unimplemented. If the instruction would have branched, and the branching behavior is not implemented, then a Speculative Operation fault is taken and the value specified by \(imm_{21}\) is zero-extended and placed in the Interruption Immediate control register (IIM). The fault handler emulates the branch by sign-extending the IIM value, adding it to IIP and returning.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

```c
if (PR[qp]) {
    switch (sf) {
        case 's0':
            tmp_flags = AR[FPSR].sf0.flags;
            break;
        case 's1':
            tmp_flags = AR[FPSR].sf1.flags;
            break;
        case 's2':
            tmp_flags = AR[FPSR].sf2.flags;
            break;
        case 's3':
            tmp_flags = AR[FPSR].sf3.flags;
            break;
    }
    if ((tmp_flags & ~AR[FPSR].traps) || (tmp_flags & ~AR[FPSR].sf0.flags)) {
        if (check_branchImplemented(FCHKF)) {
            taken_branch = 1;
            IP = IP + signExt((imm_{21} << 4), 25);
            if ((PSR.it && unimplementedVirtualAddress(IP))
                || (!PSR.it && unimplementedPhysicalAddress(IP)))
                unimplementedInstructionAddressTrap(0, IP);
            if (PSR.tb)
                taken_branch_trap();
        } else
            speculation Fault(FCHKF, zeroExt(imm_{21}, 21));
    }
}
```

FP Exceptions: None

Interruptions: Speculative Operation fault \hspace{1cm} Taken Branch trap
Unimplemented Instruction Address trap
Floating-point Class

Format: \[(qp) \text{fclass,fcrel,fctype} \ p_1, p_2 = f_2, \text{fclass}_9\]

Description: The contents of FR \(f_2\) are classified according to the fclass\(_9\) completer as shown in Table 2-25. This produces a boolean result based on whether the contents of FR \(f_2\) agrees with the floating-point number format specified by fclass\(_9\), as specified by the fcrel completer. This result is written to the two predicate register destinations, \(p_1\) and \(p_2\). The result written to the destinations is determined by the compare type specified by fctype.

The allowed types are Normal (or none) and unc. See Table 2-26 on page 3:61. The assembly syntax allows the specification of membership or non-membership and the assembler swaps the target predicates to achieve the desired effect.

Table 2-24. Floating-point Class Relations

<table>
<thead>
<tr>
<th>fcrel</th>
<th>Test Relation</th>
</tr>
</thead>
<tbody>
<tr>
<td>m</td>
<td>FR (f_2) agrees with the pattern specified by fclass(_9) (is a member)</td>
</tr>
<tr>
<td>nm</td>
<td>FR (f_2) does not agree with the pattern specified by fclass(_9) (is not a member)</td>
</tr>
</tbody>
</table>

A number agrees with the pattern specified by fclass\(_9\) if:

- the number is NaTVal and fclass\(_9\) \{8\} is 1, or
- the number is a quiet NaN and fclass\(_9\) \{7\} is 1, or
- the number is a signaling NaN and fclass\(_9\) \{6\} is 1, or
- the sign of the number agrees with the sign specified by one of the two low-order bits of fclass\(_9\), and the type of the number (disregarding the sign) agrees with the number-type specified by the next 4 bits of fclass\(_9\), as shown in Table 2-25.

Note: An fclass\(_9\) of 0x1FF is equivalent to testing for any supported operand. The class names used in Table 2-25 are defined in Table 5-2 on page 3:78 in Volume 1: Application Architecture.

Table 2-25. Floating-point Classes

<table>
<thead>
<tr>
<th>fclass(_9)</th>
<th>Class</th>
<th>Mnemonic</th>
</tr>
</thead>
<tbody>
<tr>
<td>Either these cases can be tested for</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0x0100</td>
<td>NaTVal</td>
<td>@nat</td>
</tr>
<tr>
<td>0x080</td>
<td>Quiet NaN</td>
<td>@qnan</td>
</tr>
<tr>
<td>0x040</td>
<td>Signaling NaN</td>
<td>@snn</td>
</tr>
<tr>
<td>or the OR of the following two cases</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0x001</td>
<td>Positive</td>
<td>@pos</td>
</tr>
<tr>
<td>0x002</td>
<td>Negative</td>
<td>@neg</td>
</tr>
<tr>
<td>ANDed with OR of the following 4 cases</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0x004</td>
<td>Zero</td>
<td>@zero</td>
</tr>
<tr>
<td>0x008</td>
<td>Unnormalized</td>
<td>@unorm</td>
</tr>
<tr>
<td>0x010</td>
<td>Normalized</td>
<td>@norm</td>
</tr>
<tr>
<td>0x020</td>
<td>Infinity</td>
<td>@inf</td>
</tr>
</tbody>
</table>
Operation:

```c
if (PR[qp]) {
    if (p1 == p2)
        illegal_operation_fault();

    if (tmp_isrcode = fp_reg_disabled(f2, 0, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

                   && ((fclass_9[2] && fp_is_zero(FR[f2])) ||
                       (fclass_9[3] && fp_is_unorm(FR[f2]))) ||
                   (fclass_9[4] && fp_is_normal(FR[f2]))) ||
                   (fclass_9[5] && fp_is_inf(FR[f2]))
             )
            || (fclass_9[6] && fp_is_snana(FR[f2]))
            || (fclass_9[7] && fp_is_qnana(FR[f2]))
            || (fclass_9[8] && fp_is_natval(FR[f2]));

    tmp_nat = fp_is_natval(FR[f2]) && (!fclass_9[8]);

    if (tmp_nat) {
        PR[p1] = 0;
        PR[p2] = 0;
    } else {
        PR[p1] = tmp_rel;
        PR[p2] = !tmp_rel;
    }
} else {
    if (fctype == 'unc') {
        if (p1 == p2)
            illegal_operation_fault();
        PR[p1] = 0;
        PR[p2] = 0;
    }
}
```

**FP Exceptions:** None

**Interruptions:**

- Illegal Operation fault
- Disabled Floating-point Register fault
Floating-point Clear Flags

Format:  \((qp)\ fclrf.sf\)

Description: The status field’s 6-bit flags field is reset to zero.
The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:
\[
\text{if } (PR[qp]) \{
\text{fp_set_sf_flags}(sf, 0);
\}
\]

FP Exceptions: None

 Interruptions: None
Floating-point Compare

Format: \((qp) \ fcmp.\frel.\fctype.\sf \ p_1, p_2 = f_2, f_3\)  

Description: The two source operands are compared for one of twelve relations specified by \(\frel\). This produces a boolean result which is 1 if the comparison condition is true, and 0 otherwise. This result is written to the two predicate register destinations, \(p_1\) and \(p_2\). The way the result is written to the destinations is determined by the compare type specified by \(\fctype\). The allowed types are Normal (or \(\text{none}\)) and unc.

Table 2-26. Floating-point Comparison Types

<table>
<thead>
<tr>
<th>(\fctype)</th>
<th>(\text{PR}(qp)==0)</th>
<th>(\text{PR}(qp)==1)</th>
<th>(\text{PR}(qp)==0)</th>
<th>(\text{PR}(qp)==1)</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>0</td>
<td>0</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>unc</td>
<td>0</td>
<td>0</td>
<td>1</td>
<td>1</td>
</tr>
</tbody>
</table>

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

The relations are defined for each of the comparison types in Table 2-27. Of the twelve relations, not all are directly implemented in hardware. Some are actually pseudo-ops. For these, the assembler simply switches the source operand specifiers and/or switches the predicate target specifiers and uses an implemented relation.

Table 2-27. Floating-point Comparison Relations

<table>
<thead>
<tr>
<th>(\frel)</th>
<th>(\text{frel Completer Unabbreviated})</th>
<th>(\text{Relation})</th>
<th>(\text{Pseudo-op of Quiet NaN as Operand Signals Invalid})</th>
</tr>
</thead>
<tbody>
<tr>
<td>eq</td>
<td>equal</td>
<td>(f_2 == f_3)</td>
<td>No</td>
</tr>
<tr>
<td>lt</td>
<td>less than</td>
<td>(f_2 &lt; f_3)</td>
<td>Yes</td>
</tr>
<tr>
<td>le</td>
<td>less than or equal</td>
<td>(f_2 &lt;= f_3)</td>
<td>Yes</td>
</tr>
<tr>
<td>gt</td>
<td>greater than</td>
<td>(f_2 &gt; f_3)</td>
<td>Yes</td>
</tr>
<tr>
<td>ge</td>
<td>greater than or equal</td>
<td>(f_2 &gt;= f_3)</td>
<td>Yes</td>
</tr>
<tr>
<td>unord</td>
<td>unordered</td>
<td>(f_2 ? f_3)</td>
<td>No</td>
</tr>
<tr>
<td>neq</td>
<td>not equal</td>
<td>(f_2 != f_3)</td>
<td>(p_1 \leftrightarrow p_2)</td>
</tr>
<tr>
<td>nlt</td>
<td>not less than</td>
<td>(f_2 &lt; f_3)</td>
<td>(p_1 \leftrightarrow p_2)</td>
</tr>
<tr>
<td>nle</td>
<td>not less than or equal</td>
<td>(f_2 &lt;= f_3)</td>
<td>(p_1 \leftrightarrow p_2)</td>
</tr>
<tr>
<td>ngt</td>
<td>not greater than</td>
<td>(f_2 &gt; f_3)</td>
<td>(p_1 \leftrightarrow p_2)</td>
</tr>
<tr>
<td>nge</td>
<td>not greater than or equal</td>
<td>(f_2 &gt;= f_3)</td>
<td>(p_1 \leftrightarrow p_2)</td>
</tr>
<tr>
<td>ord</td>
<td>ordered</td>
<td>(f_2 ? f_3)</td>
<td>(p_1 \leftrightarrow p_2)</td>
</tr>
</tbody>
</table>
Operation:

```c
if (PR[qp]) {
    if (p1 == p2)
        illegal_operation_fault();

    if (tmp_isrcode = fp_reg_disabled(f2, f3, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        PR[p1] = 0;
        PR[p2] = 0;
    } else {
        fcmp.exception_fault_check(f2, f3, frel, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));

        tmp_fr2 = fp_reg_read(FR[f2]);
        tmp_fr3 = fp_reg_read(FR[f3]);

        if (frel == 'eq')
            tmp_rel = fp_equal(tmp_fr2, tmp_fr3);
        else if (frel == 'lt')
            tmp_rel = fp_less_than(tmp_fr2, tmp_fr3);
        else if (frel == 'le')
            tmp_rel = fp_lesser_or_equal(tmp_fr2, tmp_fr3);
        else if (frel == 'gt')
            tmp_rel = fp_less_than(tmp_fr3, tmp_fr2);
        else if (frel == 'ge')
            tmp_rel = fp_lesser_or_equal(tmp_fr3, tmp_fr2);
        else if (frel == 'unord')
            tmp_rel = fp_unordered(tmp_fr2, tmp_fr3);
        else if (frel == 'neq')
            tmp_rel = !fp_equal(tmp_fr2, tmp_fr3);
        else if (frel == 'nlt')
            tmp_rel = !fp_less_than(tmp_fr2, tmp_fr3);
        else if (frel == 'nle')
            tmp_rel = !fp_lesser_or_equal(tmp_fr2, tmp_fr3);
        else if (frel == 'ngt')
            tmp_rel = !fp_less_than(tmp_fr3, tmp_fr2);
        else if (frel == 'nge')
            tmp_rel = !fp_lesser_or_equal(tmp_fr3, tmp_fr2);
        else
            tmp_rel = !fp_unordered(tmp_fr2, tmp_fr3); // 'ord'

        PR[p1] = tmp_rel;
        PR[p2] = !tmp_rel;

        fp_update_fpsr(sf, tmp_fp_env);
    }
} else {
    if (fctype == 'unc') {
        if (p1 == p2)
            illegal_operation_fault();
        PR[p1] = 0;
        PR[p2] = 0;
    }
}
```

**FP Exceptions:**
- Invalid Operation (V)
- Denormal/Unnormal Operand (D)
- Software Assist (SWA) fault
**Interruptions:** Illegal Operation fault
Disabled Floating-point Register fault

Floating-point Exception fault
Convert Floating-point to Integer

Format:

\[(qp)\] \text{fcvt.fx.sf} f_1 = f_2 \quad \text{signed_form} \quad F10
\[(qp)\] \text{fcvt.fx.trunc.sf} f_1 = f_2 \quad \text{signed_form, trunc_form} \quad F10
\[(qp)\] \text{fcvt.fxu.sf} f_1 = f_2 \quad \text{unsigned_form} \quad F10
\[(qp)\] \text{fcvt.fxu.trunc.sf} f_1 = f_2 \quad \text{unsigned_form, trunc_form} \quad F10

Description:
FR \(f_2\) is treated as a register format floating-point value and converted to a signed (signed_form) or unsigned integer (unsigned_form) using either the rounding mode specified in the FPSR.sf,rc, or using Round-to-Zero if the trunc_form of the instruction is used. The result is placed in the 64-bit significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for 2.0^{63} (0x1003E) and the sign field of FR \(f_1\) is set to positive (0). If the result of the conversion cannot be represented as a 64-bit integer, the 64-bit integer indefinite value 0x8000000000000000 is used as the result, if the IEEE Invalid Operation Floating-point Exception fault is disabled.

If FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f_2])) {
        FR[f_1] = NATVAL;
        fp_update_psr(f_1);
    } else {
        tmp_default_result = fcvt_exception_fault_check(f_2, signed_form, trunc_form, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        if (fp_is_nan(tmp_default_result)) {
            FR[f_1].significand = INTEGER_INDEFINITE;
            FR[f_1].exponent = FP_INTEGER_EXP;
            FR[f_1].sign = FP_SIGN_POSITIVE;
        } else {
            tmp_res = fp_ieee_rnd_to_int(fp_reg_read(FR[f_2]), &tmp_fp_env);
            if (tmp_res.exponent)
                tmp_res.significand = fp_U64_rsh(tmp_res.significand, (FP_INTEGER_EXP - tmp_res.exponent));
            if (signed_form && tmp_res.sign)
                tmp_res.significand = (~tmp_res.significand) + 1;
            FR[f_1].significand = tmp_res.significand;
            FR[f_1].exponent = FP_INTEGER_EXP;
            FR[f_1].sign = FP_SIGN_POSITIVE;
        }
    }
    fp_update_fpsr(sf, tmp_fp_env);
    fp_update_psr(f_1);
    if (fp_raise_traps(tmp_fp_env))
        fp_exception_trap(fp_decode_trap(tmp_fp_env));
}
```
<table>
<thead>
<tr>
<th>FP Exceptions:</th>
<th>Inexact (I)</th>
</tr>
</thead>
<tbody>
<tr>
<td>Invalid Operation (V)</td>
<td>Software Assist (SWA) fault</td>
</tr>
<tr>
<td>Denormal/Unnormal Operand (D)</td>
<td></td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Interruptions:</th>
<th>Floating-point Exception fault</th>
</tr>
</thead>
<tbody>
<tr>
<td>Illegal Operation fault</td>
<td>Floating-point Exception trap</td>
</tr>
<tr>
<td>Disabled Floating-point Register fault</td>
<td></td>
</tr>
</tbody>
</table>
Convert Signed Integer to Floating-point

Format: \((qp)\) fcvt.xf \(f_1 = f_2\)

Description: The 64-bit significand of FR \(f_2\) is treated as a signed integer and its register file precision floating-point representation is placed in FR \(f_1\).

If FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

This operation is always exact and is unaffected by the rounding mode.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2])) {
        FR[f1] = NATVAL;
    } else {
        tmp_res = FR[f2];
        if (tmp_res.significand{63}) {
            tmp_res.significand = (~tmp_res.significand) + 1;
            tmp_res.sign = 1;
        } else
            tmp_res.sign = 0;

        tmp_res.exponent = FP_INTEGER_EXP;
        tmp_res = fp_normalize(tmp_res);

        FR[f1].significand = tmp_res.significand;
        FR[f1].exponent = tmp_res.exponent;
        FR[f1].sign = tmp_res.sign;
    }
    fp_update_psr(f1);
}
```

FP Exceptions: None

Interruptions: Illegal Operation fault Disabled Floating-point Register fault
Convert Unsigned Integer to Floating-point

Format: \((qp)\ fcvt.xuf\_pc\.sf \_f1 = f3\)

pseudo-op of: \((qp)\ fma\_pc\.sf \_f1 = f3, f1, f0\)

Description: FR \(f3\) is multiplied with FR 1, rounded to the precision indicated by \(pc\) (and possibly FPSR.\(sf\_pc\) and FPSR.\(sf\_wre\)) using the rounding mode specified by FPSR.\(sf\_rc\), and placed in FR \(f1\).

Note: Multiplying FR \(f3\) with FR 1 (a 1.0) normalizes the canonical representation of an integer in the floating-point register file producing a normal floating-point value.

If FR \(f3\) is a NaTVal, FR \(f1\) is set to NaTVal instead of the computed result.

The mnemonic values for the opcode’s \(pc\) are given in Table 2-22 on page 3:51. The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \(pc\), \(wre\), and \(rc\), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

**Fetch and Add Immediate**

**Format:**

\[(qp) \text{fetchadd}_4 . \text{sem} \text{ldhint } r_1 = [r_3], \text{inc}_3 \quad \text{four_byte_form} \quad \text{M17}

\[(qp) \text{fetchadd}_8 . \text{sem} \text{ldhint } r_1 = [r_3], \text{inc}_3 \quad \text{eight_byte_form} \quad \text{M17}

**Description:**

A value consisting of four or eight bytes is read from memory starting at the address specified by the value in GR \( r_3 \). The value is zero extended and added to the sign-extended immediate value specified by \( \text{inc}_3 \). The values that may be specified by \( \text{inc}_3 \) are: -16, -8, -4, -1, 1, 4, 8, 16. The least significant four or eight bytes of the sum are then written to memory starting at the address specified by the value in GR \( r_3 \). The zero-extended value read from memory is placed in GR \( r_1 \) and the NaT bit corresponding to GR \( r_1 \) is cleared.

The \( \text{sem} \) completer specifies the type of semaphore operation. These operations are described in Table 2-28. See “Sequentiality Attribute and Ordering” on page 1:69 in *Volume 2: System Architecture* for details on memory ordering.

**Table 2-28. Fetch and Add Semaphore Types**

<table>
<thead>
<tr>
<th>\text{sem} Completer</th>
<th>Ordering semantics</th>
<th>Semaphore operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>acq</td>
<td>Acquire</td>
<td>The memory read/write is made visible prior to all subsequent data memory accesses.</td>
</tr>
<tr>
<td>rel</td>
<td>Release</td>
<td>The memory read/write is made visible after all previous data memory accesses.</td>
</tr>
</tbody>
</table>

The memory read and write are guaranteed to be atomic for accesses to pages with cacheable, writeback memory attribute. For accesses to other memory types, atomicity is platform dependent. Details on memory attributes are described in “Memory Attributes” on page 1:63 in *Volume 2: System Architecture*.

If the address specified by the value in GR \( r_3 \) is not naturally aligned to the size of the value being accessed in memory, an Unaligned Data Reference fault is taken independent of the state of the User Mask alignment checking bit, UM.ac (PSR.ac in the Processor Status Register).

Both read and write access privileges for the referenced page are required. The write access privilege check is performed whether or not the memory write is performed.

Only accesses to UCE pages or cacheable pages with write-back write policy are permitted. Accesses to NaTPages result in a Data NaT Page Consumption fault. Accesses to pages with other memory attributes cause an Unsupported Data Reference fault.

On a processor model that supports exported \( \text{fetchadd} \), a \( \text{fetchadd} \) to a UCE page causes the fetch-and-add operation to be exported outside of the processor; if the platform does not support exported \( \text{fetchadd} \), the operation is undefined. On a processor model that does not support exported \( \text{fetchadd} \), a \( \text{fetchadd} \) to a UCE page causes an Unsupported Data Reference fault. See Section 4.4.9 “Effects of Memory Attributes on Memory Reference Instructions” in *Volume 2: System Architecture*.

The value of the \( \text{ldhint} \) completer specifies the locality of the memory access. The values of the \( \text{ldhint} \) completer are given in Table 2-33 on page 3:136. Locality hints do not affect program functionality and may be ignored by the implementation. See “Memory Hierarchy Control and Consistency” on page 1:62 in *Volume 1: Application Architecture* for details.
Operation:  
if (PR[qp]) {
    check_target_register(r1);

    if (GR[r3].nat)
        register_nat_consumption_fault(SEMAPHORE);

    size = four_byte_form ? 4 : 8;

    paddr = tlb_translate(GR[r3], size, SEMAPHORE, PSR.cpl, &mattr,
                          &tmp_unused);
    if (!ma_supports_fetchadd(mattr))
        unsupported_data_reference_fault(SEMAPHORE, GR[r3]);

    if (sem == 'acq')
        val = mem_xchg_add(inc3, paddr, size, UM.be, mattr, ACQUIRE, ldhint);
    else // 'rel'
        val = mem_xchg_add(inc3, paddr, size, UM.be, mattr, RELEASE, ldhint);

    alat_inval_multiple_entries(paddr, size);

    GR[r1] = zero_ext(val, size * 8);
    GR[r1].nat = 0;
}

Interruptions:  
Illegal Operation fault  
Register NaT Consumption fault  
Unimplemented Data Address fault  
Data Nested TLB fault  
Alternate Data TLB fault  
VHPT Data fault  
Data TLB fault  
Data Page Not Present fault  
Data NaT Page Consumption fault  
Data Key Miss fault  
Data Key Permission fault  
Data Access Rights fault  
Data Dirty Bit fault  
Data Access Bit fault  
Data Debug fault  
Unaligned Data Reference fault  
Unsupported Data Reference fault
Flush Register Stack

Format: flushrs  M25

Description: All stacked general registers in the dirty partition of the register stack are written to the backing store before execution continues. The dirty partition contains registers from previous procedure frames that have not yet been saved to the backing store. For a description of the register stack partitions, refer to Chapter 6, “Register Stack Engine” in Volume 2: System Architecture. A pending external interrupt can interrupt the RSE store loop when enabled.

After this instruction completes execution BSPSTORE is equal to BSP.

This instruction must be the first instruction in an instruction group and must either be in instruction slot 0 or in instruction slot 1 of a template having a stop after slot 0; otherwise, the results are undefined. This instruction cannot be predicated.

Operation:

```c
while (AR[BSPSTORE] != AR[BSP]) {
    rse_store(MANDATORY); // increments AR[BSPSTORE]
    deliver_unmasked_pending_external_interrupt();
}
```

Interruptions:

- Unimplemented Data Address fault
- VHPT Data fault
- Data Nested TLB fault
- Data TLB fault
- Alternate Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Dirty Bit fault
- Data Access Bit fault
- Data Debug fault
Floating-point Multiply Add

Format: \( (qp) \text{fma}.pc.sf \, f_1 = f_3, f_4, f_2 \)

Description: The product of \( f_3 \) and \( f_4 \) is computed to infinite precision and then \( f_2 \) is added to this product, again in infinite precision. The resulting value is then rounded to the precision indicated by \( pc \) (and possibly FPSR.sf.pc and FPSR.sf.wre) using the rounding mode specified by FPSR.sf.rc. The rounded result is placed in \( f_1 \).

If any of \( f_3 \), \( f_4 \), or \( f_2 \) is a NaTVal, \( f_1 \) is set to NaTVal instead of the computed result.

If \( f_2 \) is 0, an IEEE multiply operation is performed instead of a multiply and add. See “Floating-point Multiply” on page 3:78.

The mnemonic values for the opcode’s \( pc \) are given in Table 2-22 on page 3:51. The mnemonic values for \( sf \) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \( pc \), \( wre \), and \( rc \), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Operation: if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, f_4))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3]) ||
        fp_is_natval(FR[f_4])) {
        FR[f_1] = NATVAL;
        fp_update_psr(f_1);
    } else {
        tmp_default_result = fma_exception_fault_check(f_2, f_3, f_4,
            pc, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));

        if (fp_is_nan_or_inf(tmp_default_result)) {
            FR[f_1] = tmp_default_result;
        } else {
            tmp_res = fp_mul(fp_reg_read(FR[f_3]), fp_reg_read(FR[f_4]));
            if (f_2 != 0)
                tmp_res = fp_add(tmp_res, fp_reg_read(FR[f_2]), tmp_fp_env);
            FR[f_1] = fp_ieee_round(tmp_res, &tmp_fp_env);
        }

        fp_update_fpsr(sf, tmp_fp_env);
        fp_update_psr(f_1);
        if (fp_raise_traps(tmp_fp_env))
            fp_exception_trap(fp_decode_trap(tmp_fp_env));
    }
}

FP Exceptions: Invalid Operation (V) Underflow (U)
Denormal/Unnormal Operand (D) Overflow (O)
Software Assist (SWA) fault Inexact (I)
Software Assist (SWA) trap

Interruptions: Illegal Operation fault Floating-point Exception fault
Disabled Floating-point Register fault Floating-point Exception trap
Floating-point Maximum

Format: \((qp) \text{ fmax}.sf f_1 = f_2, f_3\)

Description: The operand with the larger value is placed in \(f_1\). If \(f_2\) equals \(f_3\), \(f_1\) gets \(f_3\).

If either \(f_2\) or \(f_3\) is a NaN, \(f_1\) gets \(f_3\).

If either \(f_2\) or \(f_3\) is a NaN, \(f_1\) is set to NaN instead of the computed result.

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as the \(f_{cmp}.lt\) operation.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        fminmax_exception_fault_check(f2, f3, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));

        tmp_bool_res = fp_less_than(fp_reg_read(FR[f3]),
                                    fp_reg_read(FR[f2]));

        fp_update_fpsr(sf, tmp_fp_env);
    }
    fp_update_psr(f1);
}
```

FP Exceptions: Invalid Operation (V)
Denormal/Unnormal Operand (D)
Software Assist (SWA) fault

Interruptions: Illegal Operation fault Floating-point Exception fault
Disabled Floating-point Register fault
Floating-point Merge

Format:  
\[(qp) \text{ fmerge.ns } f_1 = f_2, f_3\]  
\[(qp) \text{ fmerge.s } f_1 = f_2, f_3\]  
\[(qp) \text{ fmerge.se } f_1 = f_2, f_3\]

Description:  
Sign, exponent and significand fields are extracted from FR \(f_2\) and FR \(f_3\), combined, and the result is placed in FR \(f_1\).

For the neg_sign_form, the sign of FR \(f_2\) is negated and concatenated with the exponent and the significand of FR \(f_3\). This form can be used to negate a floating-point number by using the same register for FR \(f_2\) and FR \(f_3\).

For the sign_form, the sign of FR \(f_2\) is concatenated with the exponent and the significand of FR \(f_3\).

For the sign_exp_form, the sign and exponent of FR \(f_2\) is concatenated with the significand of FR \(f_3\).

For all forms, if either FR \(f_2\) or FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

Figure 2-8. Floating-point Merge Negative Sign Operation

Figure 2-9. Floating-point Merge Sign Operation

Figure 2-10. Floating-point Merge Sign and Exponent Operation
Operation: if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        FR[f1].significand = FR[f3].significand;
        if (neg_sign_form) {
            FR[f1].exponent = FR[f3].exponent;
            FR[f1].sign = !FR[f2].sign;
        } else if (sign_form) {
            FR[f1].exponent = FR[f3].exponent;
            FR[f1].sign = FR[f2].sign;
        } else { // sign_exp_form
            FR[f1].exponent = FR[f2].exponent;
            FR[f1].sign = FR[f2].sign;
        }
    }
}

fp_update_psr(f1);

FP Exceptions: None

Interruptions: Illegal Operation fault Disabled Floating-point Register fault
**Floating-point Minimum**

**Format:** \((qp) \text{fmin.sf } f_1 = f_2, f_3\)

**Description:** The operand with the smaller value is placed in \(f_1\). If \(f_2\) equals \(f_3\), \(f_1\) gets \(f_3\).

- If either \(f_2\) or \(f_3\) is a NaN, \(f_1\) gets \(f_3\).
- If either \(f_2\) or \(f_3\) is a NaNVal, \(f_1\) is set to NaNVal instead of the computed result.

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as the \(fcmp.lt\) operation.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

**Operation:**
```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        fminmax_exception_fault_check(f2, f3, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));

        tmp_bool_res = fp_less_than(fp_reg_read(FR[f2]),
                     fp_reg_read(FR[f3]));

        fp_update_fpsr(sf, tmp_fp_env);
    }
    fp_update_psr(f1);
}
```

**FP Exceptions:**
- Invalid Operation (V)
- Denormal/Unnormal Operand (D)
- Software Assist (SWA) fault

**Interruptions:**
- Illegal Operation fault
- Floating-point Exception fault
- Disabled Floating-point Register fault
Floating-point Mix

Format:

\[(qp) \text{ fmix.l } f_1 = f_2, f_3 \]  \hspace{1cm} \text{mix_l_form} \hspace{1cm} F9

\[(qp) \text{ fmix.r } f_1 = f_2, f_3 \]  \hspace{1cm} \text{mix_r_form} \hspace{1cm} F9

\[(qp) \text{ fmix.lr } f_1 = f_2, f_3 \]  \hspace{1cm} \text{mix_lr_form} \hspace{1cm} F9

Description:

For the mix_l_form (mix_r_form), the left (right) single precision value in FR \( f_2 \) is concatenated with the left (right) single precision value in FR \( f_3 \). For the mix_lr_form, the left single precision value in FR \( f_2 \) is concatenated with the right single precision value in FR \( f_3 \).

For all forms, the exponent field of FR \( f_j \) is set to the biased exponent for \( 2.0^{63} \) (0x1003E) and the sign field of FR \( f_1 \) is set to positive (0).

For all forms, if either FR \( f_2 \) or FR \( f_3 \) is a NaN value, FR \( f_1 \) is set to NaN instead of the computed result.

Figure 2-11. Floating-point Mix Left

<table>
<thead>
<tr>
<th>FR ( f_2 )</th>
<th>FR ( f_1 )</th>
<th>FR ( f_3 )</th>
</tr>
</thead>
<tbody>
<tr>
<td>81 80 64 63</td>
<td>32 31 0</td>
<td>81 80 64 63</td>
</tr>
<tr>
<td>FR ( f_2 )</td>
<td>FR ( f_1 )</td>
<td>FR ( f_3 )</td>
</tr>
<tr>
<td>81 80 64 63</td>
<td>32 31 0</td>
<td>81 80 64 63</td>
</tr>
</tbody>
</table>

Figure 2-12. Floating-point Mix Right

<table>
<thead>
<tr>
<th>FR ( f_2 )</th>
<th>FR ( f_1 )</th>
<th>FR ( f_3 )</th>
</tr>
</thead>
<tbody>
<tr>
<td>81 80 64 63</td>
<td>32 31 0</td>
<td>81 80 64 63</td>
</tr>
<tr>
<td>FR ( f_2 )</td>
<td>FR ( f_1 )</td>
<td>FR ( f_3 )</td>
</tr>
<tr>
<td>81 80 64 63</td>
<td>32 31 0</td>
<td>81 80 64 63</td>
</tr>
</tbody>
</table>

Figure 2-13. Floating-point Mix Left-Right

<table>
<thead>
<tr>
<th>FR ( f_2 )</th>
<th>FR ( f_1 )</th>
<th>FR ( f_3 )</th>
</tr>
</thead>
<tbody>
<tr>
<td>81 80 64 63</td>
<td>32 31 0</td>
<td>81 80 64 63</td>
</tr>
<tr>
<td>FR ( f_2 )</td>
<td>FR ( f_1 )</td>
<td>FR ( f_3 )</td>
</tr>
<tr>
<td>81 80 64 63</td>
<td>32 31 0</td>
<td>81 80 64 63</td>
</tr>
</tbody>
</table>
Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        if (mix_l_form) {
            tmp_res_hi = FR[f2].significand{63:32};
            tmp_res_lo = FR[f3].significand{63:32};
        } else if (mix_r_form) {
            tmp_res_hi = FR[f2].significand{31:0};
            tmp_res_lo = FR[f3].significand{31:0};
        } else { // mix_lr_form
            tmp_res_hi = FR[f2].significand{63:32};
            tmp_res_lo = FR[f3].significand{31:0};
        }
        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
    }

    fp_update_psr(f1);
}
```

**FP Exceptions:** None

**Interruptions:** Illegal Operation fault, Disabled Floating-point Register fault
Floating-point Multiply

Format: \((qp)\ fmpy,pc.sf\ f_1 = f_3, f_4\) pseudo-op of: \((qp)\ fma,pc.sf\ f_1 = f_3, f_4, f_0\)

Description: The product \(f_3\) and \(f_4\) is computed to infinite precision. The resulting value is then rounded to the precision indicated by \(pc\) (and possibly FPSR.\(sf,pc\) and FPSR.\(sf,wre\)) using the rounding mode specified by FPSR.\(sf,rc\). The rounded result is placed in \(f_1\).

If either \(f_3\) or \(f_4\) is a NaTVal, \(f_1\) is set to NaTVal instead of the computed result.

The mnemonic values for the opcode’s \(pc\) are given in Table 2-22 on page 3:51. The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \(pc,\ wre,\) and \(rc\), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Floating-point Multiply Subtract

Format: 
\[(qp) \text{fms} \cdot \text{pc} \cdot \text{sf} \cdot f_1 = f_3, f_4, f_2\]

Description: 
The product of FR\(_{f_3}\) and FR\(_{f_4}\) is computed to infinite precision and then FR\(_{f_2}\) is subtracted from this product, again in infinite precision. The resulting value is then rounded to the precision indicated by \(\text{pc}\) (and possibly FPSR.\(\text{sf}.\text{pc}\) and FPSR.\(\text{sf}.\text{wre}\)) using the rounding mode specified by FPSR.\(\text{sf}.\text{rc}\). The rounded result is placed in FR\(_{f_1}\).

If any of FR\(_{f_3}\), FR\(_{f_4}\), or FR\(_{f_2}\) is a NaTVal, a NaTVal is placed in FR\(_{f_1}\) instead of the computed result.

If \(f_2\) is \(f_0\), an IEEE multiply operation is performed instead of a multiply and subtract. See “Floating-point Multiply” on page 3:78.

The mnemonic values for the opcode’s \(\text{pc}\) are given in Table 2-22 on page 3:51. The mnemonic values for \(\text{sf}\) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \(\text{pc}\), \(\text{wre}\), and \(\text{rc}\), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Operation: 
\[
\text{if} (\text{PR}[qp]) \{
\text{fp_check_target_register}(f_1); \\
\text{if} (\text{tmp_isrcode} = \text{fp_reg_disabled}(f_1, f_3, f_4)) \\
\text{disabled_fp_register_fault}(\text{tmp_isrcode}, 0); \\
\text{if} (\text{fp_is_natval}(\text{FR}[f_3]) || \text{fp_is_natval}(\text{FR}[f_3]) || \\
\text{fp_is_natval}(\text{FR}[f_4])) \\
\text{FR}[f_1] = \text{NATVAL}; \\
\text{fp_update_psr}(f_1); \\
\text{else} \{ \\
\text{tmp_default_result} = \text{fmsfnma_exception_fault_check}(f_2, f_3, f_4, \\
\text{pc}, \text{sf}, \&\text{tmp_fp_env}); \\
\text{if} (\text{fp_raise_fault}(\text{tmp_fp_env})) \\
\text{fp_exception_fault}(\text{fp_decode_fault}(\text{tmp_fp_env})); \\
\text{if} (\text{fp_is_nan_or_inf}(\text{tmp_default_result})) \\
\text{FR}[f_1] = \text{tmp_default_result}; \\
\text{else} \{ \\
\text{tmp_res} = \text{fp_mul}(\text{fp_reg_read}(\text{FR}[f_3]), \text{fp_reg_read}(\text{FR}[f_4])); \\
\text{tmp_fr2} = \text{fp_reg_read}(\text{FR}[f_2]); \\
\text{tmp_fr2}.\text{sign} = !\text{tmp_fr2}.\text{sign}; \\
\text{if} (f_2 != 0) \\
\text{tmp_res} = \text{fp_add}(\text{tmp_res}, \text{tmp_fr2}, \text{tmp_fp_env}); \\
\text{FR}[f_1] = \text{fp_ieee_round}(\text{tmp_res}, \&\text{tmp_fp_env}); \\
\} \\
\text{fp_update_fpsr}(sf, \text{tmp_fp_env}); \\
\text{fp_update_psr}(f_1); \\
\text{if} (\text{fp_raise_traps}(\text{tmp_fp_env})) \\
\text{fp_exception_trap}(\text{fp_decode_trap}(\text{tmp_fp_env})); \\
\} \\
\}
\]

FP Exceptions: 
Invalid Operation (V)  Underflow (U)
Denormal/Unnormal Operand (D)  Overflow (O)
Software Assist (SWA) fault  Inexact (I)
Software Assist (SWA) trap
Interruptions:  Illegal Operation fault  Floating-point Exception fault
              Disabled Floating-point Register fault  Floating-point Exception trap
Floating-point Negate

Format: \((qp) \text{ fneg } f_1 = f_3\)  

Description: The value in FR\(f_3\) is negated and placed in FR\(f_1\).

If FR\(f_3\) is a NaTVal, FR\(f_1\) is set to NaTVal instead of the computed result.

Operation: See “Floating-point Merge” on page 3:73.
Floating-point Negate Absolute Value

**Format:**

\[(qp) \text{ fnegabs } f_1 = f_3\]  

pseudo-op of: \[(qp) \text{ fmerge.ns } f_1 = f_0, f_3\]

**Description:**

The absolute value of the value in FR \(f_3\) is computed, negated, and placed in FR \(f_1\).

If FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

**Operation:**

See “Floating-point Merge” on page 3:73.
Floating-point Negative Multiply Add

Format: \((qp) \text{fnma}.pc.sf f_1 = f_3,f_4,f_2\)

Description: The product of FR\(_{f_3}\) and FR\(_{f_4}\) is computed to infinite precision, negated, and then FR\(_{f_2}\) is added to this product, again in infinite precision. The resulting value is then rounded to the precision indicated by \(pc\) (and possibly FPSR.\(sf\.pc\) and FPSR.\(sf\.wre\)) using the rounding mode specified by FPSR.\(sf\.rc\). The rounded result is placed in FR\(_{f_1}\).

If any of FR\(_{f_3}\), FR\(_{f_4}\), or FR\(_{f_2}\) is a NaTV al, FR\(_{f_1}\) is set to NaTV al instead of the computed result.

If \(f_2\) is 0, an IEEE multiply operation is performed, followed by negation of the product. See “Floating-point Negative Multiply” on page 3:84.

The mnemonic values for the opcode’s \(pc\) are given in Table 2-22 on page 3:51. The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \(pc\), \(wre\), and \(rc\), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, f4))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3]) ||
        fp_is_natval(FR[f4]))) {
        FR[f1] = NATVAL;
        fp_update_psr(f1);
    } else {
        tmp_default_result = fms_fnma_exception_fault_check(f2, f3, f4,
                                                             pc, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        if (fp_is_nan_or_inf(tmp_default_result)) {
            FR[f1] = tmp_default_result;
        } else {
            tmp_res = fp_mul(fp_reg_read(FR[f3]), fp_reg_read(FR[f4]));
            tmp_res.sign = !tmp_res.sign;
            if (f2 != 0)
                tmp_res = fp_add(tmp_res, fp_reg_read(FR[f2]), tmp_fp_env);
            FR[f1] = fp_ieee_round(tmp_res, &tmp_fp_env);
        }
        fp_update_fpsr(sf, tmp_fp_env);
        fp_update_psr(f1);
        if (fp_raise_traps(tmp_fp_env))
            fp_exception_trap(fp_decode_trap(tmp_fp_env));
    }
}
```

FP Exceptions: Invalid Operation (V) Underflow (U)
Denormal/Unnormal Operand (D) Overflow (O)
Software Assist (SWA) fault Inexact (I)
Software Assist (SWA) trap

Interruptions: Disabled Floating-point Register fault Floating-point Exception trap Floating-point Exception fault
Floating-point Negative Multiply

Format: \((qp) \ fnmipy.pc.sf \ f_1 = f_3, f_4\)

pseudo-op of: \((qp) \ fnma.pc.sf \ f_1 = f_3, f_4, f_0\)

Description: The product \(f_3\) and \(f_4\) is computed to infinite precision and then negated. The resulting value is then rounded to the precision indicated by \(pc\) (and possibly FPSR.sf.pc and FPSR.sf.wre) using the rounding mode specified by FPSR.sf.rc. The rounded result is placed in \(F_1\).

If either \(f_3\) or \(f_4\) is a NaTVal, \(F_1\) is set to NaTVal instead of the computed result.

The mnemonic values for the opcode’s \(pc\) are given in Table 2-22 on page 3:51. The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \(pc\), \(wre\), and \(rc\), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Floating-point Normalize

Format: \[(qp) \text{fnorm}.pc.sf f_j = f_3\] pseudo-op of: \[(qp) \text{fma}.pc.sf f_j = f_3, f_1, f_0\]

Description: \(f_j\) is normalized and rounded to the precision indicated by \(pc\) (and possibly FPSR.sf.pc and FPSR.sf.wre) using the rounding mode specified by FPSR.sf.rc, and placed in \(f_1\).

If \(f_j\) is a NaTVal, \(f_1\) is set to NaTVal instead of the computed result.

The mnemonic values for the opcode’s \(pc\) are given in Table 2-22 on page 3:51. The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \(pc\), \(wre\), and \(rc\), refer to Table 5-5 and Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Floating-point Logical Or

Format: \((qp)\) for \(f_j = f_2, f_3\)

Description: The bit-wise logical OR of the significand fields of \(FR_{f_2}\) and \(FR_{f_3}\) is computed. The resulting value is stored in the significand field of \(FR_{f_j}\). The exponent field of \(FR_{f_j}\) is set to the biased exponent for 2.0\(^{63}\) (0x1003E) and the sign field of \(FR_{f_j}\) is set to positive (0).

If either \(FR_{f_2}\) or \(FR_{f_3}\) is a NaTVal, \(FR_{f_j}\) is set to NaTVal instead of the computed result.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f_j);
    if (tmp_isrcode = fp_req_disabled(f_j, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3])) {
        FR[f_j] = NATVAL;
    } else {
        FR[f_j].significand = FR[f_2].significand | FR[f_3].significand;
        FR[f_j].exponent = FP_INTEGER_EXP;
        FR[f_j].sign = FP_SIGN_POSITIVE;
    }
    fp_update_psr(f_j);
}
```

FP Exceptions: None

Interruptions: Illegal Operation fault

Disabled Floating-point Register fault
Floating-point Parallel Absolute Value

**Format:**

\[(qp) \text{ fpabs } f_j = f_3 \quad \text{pseudo-op of:} \ (qp) \text{ fpmerge.s } f_j = f_0, f_3\]

**Description:**
The absolute values of the pair of single precision values in the significand field of FR \(f_3\) are computed and stored in the significand field of FR \(f_j\). The exponent field of FR \(f_j\) is set to the biased exponent for \(2.0^63\) (0x1003E) and the sign field of FR \(f_j\) is set to positive (0).

If FR \(f_3\) is a NaTVal, FR \(f_j\) is set to NaTVal instead of the computed result.

**Operation:**

See “Floating-point Parallel Merge” on page 3:100.
Floating-point Pack

Format: \((qp) \text{ fpack } f_1 = f_2, f_3\)

Description: The register format numbers in FR \(f_2\) and FR \(f_3\) are converted to single precision memory format. These two single precision numbers are concatenated and stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

If either FR \(f_2\) or FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

Operation:

\[
\text{if (PR[qp])} \{ \\
\quad \text{fp_check_target_register}(f_1); \\
\quad \text{if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))} \\
\quad\quad \text{disabled_fp_register_fault(tmp_isrcode, 0);} \\
\quad \text{if (fp_is_natval(FR[f_2]) \| fp_is_natval(FR[f_3]))} \\
\quad\quad \text{FR[f_1] = NATVAL;} \\
\quad \text{else} \{ \\
\quad\quad \text{tmp_res_hi = fp_single(FR[f_2]);} \\
\quad\quad \text{tmp_res_lo = fp_single(FR[f_3]);} \\
\quad\quad \text{FR[f_1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);} \\
\quad\quad \text{FR[f_1].exponent = FP_INTEGER_EXP;} \\
\quad\quad \text{FR[f_1].sign = FP_SIGN_POSITIVE;} \\
\quad\quad \text{fp_update_psr(f_1);} \\
\quad \}\n\]

FP Exceptions: None

Interruptions: Illegal Operation fault Disabled Floating-point Register fault

Figure 2-14. Floating-point Pack

82-bit FR to Single Mem Format Conversions

FR \(f_1\)

81 80 64 63 0

81 80 64 63 0

81 80 64 63

82-bit FR to Single Mem Format Conversions

FR \(f_2\)

FR \(f_3\)
Floating-point Parallel Absolute Maximum

Format: \((qp)\) \(fpamax.sf \, f_1 = f_2, f_3\)

Description: The paired single precision values in the significands of \(f_2\) and \(f_3\) are compared. The operands with the larger absolute value are returned in the significand field of \(f_1\).

If the magnitude of high (low) \(f_1\) is less than the magnitude of high (low) \(f_2\), high (low) \(f_1\) gets high (low) \(f_2\). Otherwise high (low) \(f_1\) gets high (low) \(f_3\).

If high (low) \(f_2\) or high (low) \(f_3\) is a NaN, and neither \(f_2\) or \(f_3\) is a NaTVal, high (low) \(f_1\) gets high (low) \(f_3\).

The exponent field of \(f_1\) is set to the biased exponent for 2.0\(^{63}\) (0x1003E) and the sign field of \(f_1\) is set to positive (0).

If either \(f_2\) or \(f_3\) is a NaTVal, \(f_1\) is set to NaTVal instead of the computed result.

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as for the \(fp\text{cmp}.lt\) operation.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3]))
        FR[f_1] = NATVAL;
    else {
        fpminmax_exception_fault_check(f_2, f_3, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        tmp_fr2 = tmp_right = fp_reg_read_hi(f_2);
        tmp_fr3 = tmp_left = fp_reg_read_hi(f_3);
        tmp_right.sign = FP_SIGN_POSITIVE;
        tmp_left.sign = FP_SIGN_POSITIVE;
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        tmp_res_hi = fp_single(tmp_bool_res ? tmp_fr2 : tmp_fr3);
        tmp_fr2 = tmp_right = fp_reg_read_lo(f_2);
        tmp_fr3 = tmp_left = fp_reg_read_lo(f_3);
        tmp_right.sign = FP_SIGN_POSITIVE;
        tmp_left.sign = FP_SIGN_POSITIVE;
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        tmp_res_lo = fp_single(tmp_bool_res ? tmp_fr2 : tmp_fr3);
        FR[f_1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f_1].exponent = FP_INTEGER_EXP;
        FR[f_1].sign = FP_SIGN_POSITIVE;
        fp_update_fpsr(sf, tmp_fp_env);
    }
    fp_update_psr(f_1);
}
```

FP Exceptions: Invalid Operation (V)
Denormal/Unnormal Operand (D)
Software Assist (SWA) fault
Interruptions: Illegal Operation fault
Disabled Floating-point Register fault

Floating-point Exception fault
Floating-point Parallel Absolute Minimum

**Format:**

\[(qp) \ fpamin.sf \ f_1 = f_2, f_3\]

**Description:**

The paired single precision values in the significands of FR\(f_2\) or FR\(f_3\) are compared. The operands with the smaller absolute value is returned in the significand of FR\(f_1\).

If the magnitude of high (low) FR\(f_2\) is less than the magnitude of high (low) FR\(f_3\), high (low) FR\(f_1\) gets high (low) FR\(f_2\). Otherwise high (low) FR\(f_1\) gets high (low) FR\(f_3\).

If high (low) FR\(f_2\) or high (low) FR\(f_3\) is a NaN, and neither FR\(f_2\) or FR\(f_3\) is a NaTVal, high (low) FR\(f_1\) gets high (low) FR\(f_3\).

The exponent field of FR\(f_1\) is set to the biased exponent for 2.0\(^{03}\) (0x1003E) and the sign field of FR\(f_1\) is set to positive (0).

If either FR\(f_2\) or FR\(f_3\) is NaTVal, FR\(f_1\) is set to NaTVal instead of the computed result.

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as for the \textit{fp\_cmp\_lt} operation.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

**Operation:**

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        fpminmax_exception_fault_check(f2, f3, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        tmp_fr2 = tmp_left = fp_reg_read_hi(f2);
        tmp_fr3 = tmp_right = fp_reg_read_hi(f3);
        tmp_left.sign = FP_SIGN_POSITIVE;
        tmp_right.sign = FP_SIGN_POSITIVE;
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        tmp_res_hi = fp_single(tmp_bool_res ? tmp_fr2: tmp_fr3);
        tmp_fr2 = tmp_left = fp_reg_read_lo(f2);
        tmp_fr3 = tmp_right = fp_reg_read_lo(f3);
        tmp_left.sign = FP_SIGN_POSITIVE;
        tmp_right.sign = FP_SIGN_POSITIVE;
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        tmp_res_lo = fp_single(tmp_bool_res ? tmp_fr2: tmp_fr3);
        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
        fp_update_fpsr(sf, tmp_fp_env);
    }
    fp_update_psr(f1);
}
```

**FP Exceptions:**

- Invalid Operation (V)
- Denormal/Unnormal Operand (D)
- Software Assist (SWA) fault
Interruptions: Illegal Operation fault
Disabled Floating-point Register fault

Floating-point Exception fault
Floating-point Parallel Compare

Format: \[(qp)\text{ fpcmp}.frel.sf\ f_{i} = f_{2}, f_{3}\]

Description: The two pairs of single precision source operands in the significand fields of FR \(f_{2}\) and FR \(f_{3}\) are compared for one of twelve relations specified by \(frel\). This produces a boolean result which is a mask of 32 1’s if the comparison condition is true, and a mask of 32 0’s otherwise. This result is written to a pair of 32-bit integers in the significand field of FR \(f_{1}\). The exponent field of FR \(f_{1}\) is set to the biased exponent for \(2^{63}\) (0x1003E) and the sign field of FR \(f_{1}\) is set to positive (0).

Table 2-29. Floating-point Parallel Comparison Results

<table>
<thead>
<tr>
<th>(PR[qp] == 0)</th>
<th>(PR[qp] == 1)</th>
</tr>
</thead>
<tbody>
<tr>
<td><strong>result=false, No Source NaTVals</strong></td>
<td><strong>result=true, No Source NaTVals</strong></td>
</tr>
<tr>
<td>unchanged</td>
<td>0...0</td>
</tr>
<tr>
<td></td>
<td></td>
</tr>
</tbody>
</table>

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

The relations are defined for each of the comparison types in Table 2-29. Of the twelve relations, not all are directly implemented in hardware. Some are actually pseudo-ops. For these, the assembler simply switches the source operand specifiers and/or switches the predicate type specifiers and uses an implemented relation.

If either FR \(f_{2}\) or FR \(f_{3}\) is a NaTVal, FR \(f_{1}\) is set to NaTVal instead of the computed result.

Table 2-30. Floating-point Parallel Comparison Relations

<table>
<thead>
<tr>
<th>(frel)</th>
<th>(frel) Completer unabreviated</th>
<th>Relation</th>
<th>Pseudo-op of</th>
<th>Quiet NaN as Operand Signals Invalid</th>
</tr>
</thead>
<tbody>
<tr>
<td>eq</td>
<td>equal</td>
<td>(f_{2} == f_{3})</td>
<td>No</td>
<td></td>
</tr>
<tr>
<td>lt</td>
<td>less than</td>
<td>(f_{2} &lt; f_{3})</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>le</td>
<td>less than or equal</td>
<td>(f_{2} &lt;= f_{3})</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>gt</td>
<td>greater than</td>
<td>(f_{2} &gt; f_{3})</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>ge</td>
<td>greater than or equal</td>
<td>(f_{2} &gt;= f_{3})</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>unordered</td>
<td>unordered</td>
<td>(f_{2} ? f_{3})</td>
<td>No</td>
<td></td>
</tr>
<tr>
<td>neq</td>
<td>not equal</td>
<td>(!(f_{2} == f_{3}))</td>
<td>No</td>
<td></td>
</tr>
<tr>
<td>nlt</td>
<td>not less than</td>
<td>(!(f_{2} &lt; f_{3}))</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>nle</td>
<td>not less than or equal</td>
<td>(!(f_{2} &lt;= f_{3}))</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>ngt</td>
<td>not greater than</td>
<td>(!(f_{2} &gt; f_{3}))</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>nge</td>
<td>not greater than or equal</td>
<td>(!(f_{2} &gt;= f_{3}))</td>
<td>Yes</td>
<td></td>
</tr>
<tr>
<td>ord</td>
<td>ordered</td>
<td>(f_{2} ? f_{3})</td>
<td>No</td>
<td></td>
</tr>
</tbody>
</table>

Operation:

```c
if (PR[qp]) {
    fpcmp_exception_fault_check(f_2, f_3, frel, sf, &tmp_fp_env);
    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3])) {
        FR[f_1] = NaN;
    } else {
        fpcmp_exception_fault_check(f_2, f_3, frel, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        tmp_fr2 = fp_reg_read_hi(f_2);
    }
```
fpcmp

```c
    tmp_fr3 = fp_reg_read_hi(f3);
    if (frel == 'eq') tmp_rel = fp_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'lt') tmp_rel = fp_less_than(tmp_fr2, tmp_fr3);
    else if (frel == 'le') tmp_rel = fp_less_or_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'gt') tmp_rel = fp_less_than(tmp_fr3, tmp_fr2);
    else if (frel == 'ge') tmp_rel = fp_less_or_equal(tmp_fr3, tmp_fr2);
    else if (frel == 'unord') tmp_rel = fp_unordered(tmp_fr2, tmp_fr3);
    else if (frel == 'neq') tmp_rel = !fp_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'nlt') tmp_rel = !fp_less_than(tmp_fr2, tmp_fr3);
    else if (frel == 'nle') tmp_rel = !fp_less_or_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'ngt') tmp_rel = !fp_less_than(tmp_fr3, tmp_fr2);
    else if (frel == 'nge') tmp_rel = !fp_less_or_equal(tmp_fr3, tmp_fr2);
    else tmp_rel = !fp_unordered(tmp_fr2, tmp_fr3); // 'ord'

    tmp_res_hi = (tmp_rel ? 0xFFFFFFFF : 0x00000000);
    tmp_fr2 = fp_reg_read_lo(f2);
    tmp_fr3 = fp_reg_read_lo(f3);
    if (frel == 'eq') tmp_rel = fp_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'lt') tmp_rel = fp_less_than(tmp_fr2, tmp_fr3);
    else if (frel == 'le') tmp_rel = fp_less_or_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'gt') tmp_rel = fp_less_than(tmp_fr3, tmp_fr2);
    else if (frel == 'ge') tmp_rel = fp_less_or_equal(tmp_fr3, tmp_fr2);
    else if (frel == 'unord') tmp_rel = fp_unordered(tmp_fr2, tmp_fr3);
    else if (frel == 'neq') tmp_rel = !fp_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'nlt') tmp_rel = !fp_less_than(tmp_fr2, tmp_fr3);
    else if (frel == 'nle') tmp_rel = !fp_less_or_equal(tmp_fr2, tmp_fr3);
    else if (frel == 'ngt') tmp_rel = !fp_less_than(tmp_fr3, tmp_fr2);
    else if (frel == 'nge') tmp_rel = !fp_less_or_equal(tmp_fr3, tmp_fr2);
    else tmp_rel = !fp_unordered(tmp_fr2, tmp_fr3); // 'ord'

    tmp_res_lo = (tmp_rel ? 0xFFFFFFFF : 0x00000000);
    FR[fj].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
    FR[fj].exponent = FP_INTEGER_EXP;
    FR[fj].sign = FP_SIGN_POSITIVE;
    fp_update_fpsr(sf, tmp_fp_env);
    fp_update_psr(fj);
```

**FP Exceptions:**
- Invalid Operation (V)
- Denormal/Unnormal Operand (D)
- Software Assist (SWA) fault

**Interruptions:**
- Illegal Operation fault
- Floating-point Exception fault
- Disabled Floating-point Register fault
Convert Parallel Floating-point to Integer

Format:

\[(qp) \text{fpcvt.fx.sf}\ f_1 = f_2\]
\[(qp) \text{fpcvt.fx.trunc.sf}\ f_1 = f_2\]
\[(qp) \text{fpcvt.fxu.sf}\ f_1 = f_2\]
\[(qp) \text{fpcvt.fxu.trunc.sf}\ f_1 = f_2\]

\centered{\text{signed_form F10}}
\centered{\text{signed_form, trunc_form F10}}
\centered{\text{unsigned_form F10}}
\centered{\text{unsigned_form, trunc_form F10}}

Description:
The pair of single precision values in the significand field of FR \(f_2\) is converted to a pair of 32-bit signed integers (\text{signed_form}) or unsigned integers (\text{unsigned_form}) using either the rounding mode specified in the FPSR \text{sf.rc}, or using Round-to-Zero if the \text{trunc_form} of the instruction is used. The result is written as a pair of 32-bit integers into the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0). If the result of the conversion cannot be represented as a 32-bit integer, the 32-bit integer indefinite value 0x80000000 is used as the result, if the IEEE Invalid Operation Floating-point Exception fault is disabled.

If FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

The mnemonic values for \text{sf} are given in Table 2-23 on page 3:51.
Operation: if (PR[gp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_disabled(f1, f2, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2])) {
        FR[f1] = NATVAL;
        fp_update_psr(f1);
    } else {
        tmp_default_result_pair = fpcvt_exception_fault_check(f2,
            signed_form, trunc_form, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));

        if (fp_is_nan(tmp_default_result_pair.hi)) {
            tmp_res_hi = INTEGER_INDEFINITE_32_BIT;
        } else {
            tmp_res = fp_ieee_rnd_to_int_sp(fp_reg_read_hi(f2), HIGH,
                &tmp_fp_env);
            if (tmp_res.exponent)
                tmp_res.significand = fp_U64_rsh(
                    tmp_res.significand, (FP_INTEGER_EXP - tmp_res.exponent));
            if (signed_form && tmp_res.sign)
                tmp_res.significand = (~(tmp_res.significand) + 1);
            tmp_res_hi = tmp_res.significand{31:0};
        }

        if (fp_is_nan(tmp_default_result_pair.lo)) {
            tmp_res_lo = INTEGER_INDEFINITE_32_BIT;
        } else {
            tmp_res = fp_ieee_rnd_to_int_sp(fp_reg_read_lo(f2), LOW,
                &tmp_fp_env);
            if (tmp_res.exponent)
                tmp_res.significand = fp_U64_rsh(
                    tmp_res.significand, (FP_INTEGER_EXP - tmp_res.exponent));
            if (signed_form && tmp_res.sign)
                tmp_res.significand = (~(tmp_res.significand) + 1);
            tmp_res_lo = tmp_res.significand{31:0};
        }

        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;

        fp_update_fpsr(sf, tmp_fp_env);
        fp_update_psr(f1);
        if (fp_raise_traps(tmp_fp_env))
            fp_exception_trap(fp_decode_trap(tmp_fp_env));
    }
}

FP Exceptions: Invalid Operation (V) Inexact (I)
Denormal/Unnormalized Operand (D)
Software Assist (SWA) Fault

Interruptions: Illegal Operation fault Floating-point Exception fault
Disabled Floating-point Register fault Floating-point Exception trap
Floating-point Parallel Multiply Add

Format: \((qp) \text{fpma.sf} f_1 = f_3, f_4, f_2\)

Description: The pair of products of the pairs of single precision values in the significand fields of FR \(f_3\) and FR \(f_4\) are computed to infinite precision and then the pair of single precision values in the significand field of FR \(f_2\) is added to these products, again in infinite precision. The resulting values are then rounded to single precision using the rounding mode specified by FPSR.sf.rc. The pair of rounded results are stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

If any of FR \(f_3\), FR \(f_4\), or FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed results.

Note: If \(f_2\) is \(f0\) in the fpma instruction, just the IEEE multiply operation is performed. (See “Floating-point Parallel Multiply” on page 3:103.) FR \(f1\), as an operand, is not a packed pair of 1.0 values, it is just the register file format’s 1.0 value.

The mnemonic values for sf are given in Table 2-23 on page 3:51.

The encodings and interpretation for the status field’s rc are given in Table 5-6 on page 3:82.
Operation: \[
\begin{align*}
&\text{if } (PR[q]) \{
\quad \text{fp_check_target_register}(f_1); \\
&\quad \text{if} \ (\text{tmp_isrcode} = \text{fp_reg_disabled}(f_1, f_2, f_3, f_4)) \\
&\quad \quad \text{disabled_fp_register_fault(tmp_isrcode, 0);}
\end{align*}
\]
\[
\begin{align*}
&\text{if} \ (\text{fp_is_natval}(FR[f_2]) \mid \text{fp_is_natval}(FR[f_3]) \mid \\
&\text{fp_is_natval}(FR[f_4])) \{
\quad \text{FR}[f_1] = \text{NATVAL}; \\
&\quad \text{fp_update_psr}(f_1); \\
&\} \text{ else } \\
&\quad \text{tmp_default_result_pair} = \text{fpma_exception_fault_check}(f_2, f_3, f_4, sf, \&\text{tmp_fp_env});
\end{align*}
\]
\[
\begin{align*}
&\text{if} \ (\text{fp_raise_fault}(\text{tmp_fp_env})) \\
&\quad \text{fp_exception_fault(\text{fp_decode_fault(\text{tmp_fp_env}))};
\end{align*}
\]
\[
\begin{align*}
&\text{if} \ (\text{fp_is_nan_or_inf}(\text{tmp_default_result_pair}.hi)) \{
\quad \text{tmp_res_hi} = \text{fp_single}(\text{tmp_default_result_pair}.hi); \\
&\} \text{ else } \\
&\quad \text{tmp_res} = \text{fp_mul}(\text{fp_reg_read_hi}(f_3), \text{fp_reg_read_hi}(f_4)); \\
&\quad \quad \text{if} \ (f_2 \neq 0) \\
&\quad \quad \quad \text{tmp_res} = \text{fp_add}(\text{tmp_res}, \text{fp_reg_read_hi}(f_2), \text{tmp_fp_env}); \\
&\quad \quad \text{tmp_res_hi} = \text{fp_ieee_round_sp}(\text{tmp_res}, \text{HIGH}, \&\text{tmp_fp_env});
\end{align*}
\]
\[
\begin{align*}
&\text{if} \ (\text{fp_is_nan_or_inf}(\text{tmp_default_result_pair}.lo)) \{
\quad \text{tmp_res_lo} = \text{fp_single}(\text{tmp_default_result_pair}.lo); \\
&\} \text{ else } \\
&\quad \text{tmp_res} = \text{fp_mul}(\text{fp_reg_read_lo}(f_3), \text{fp_reg_read_lo}(f_4)); \\
&\quad \quad \text{if} \ (f_2 \neq 0) \\
&\quad \quad \quad \text{tmp_res} = \text{fp_add}(\text{tmp_res}, \text{fp_reg_read_lo}(f_2), \text{tmp_fp_env}); \\
&\quad \quad \text{tmp_res_lo} = \text{fp_ieee_round_sp}(\text{tmp_res}, \text{LOW}, \&\text{tmp_fp_env});
\end{align*}
\]
\[
\begin{align*}
&\text{FR}[f_1].\text{significand} = \text{fp_concatenate}(\text{tmp_res_hi}, \text{tmp_res_lo}); \\
&\text{FR}[f_1].\text{exponent} = \text{FP_INTEGER_EXP}; \\
&\text{FR}[f_1].\text{sign} = \text{FP_SIGN_POSITIVE}; \\
&\text{fp_update_fpsr}(sf, \text{tmp_fp_env}); \\
&\text{fp_update_psr}(f_1); \\
&\text{if} \ (\text{fp_raise_traps}(\text{tmp_fp_env})) \\
&\quad \text{fp_exception_trap(\text{fp_decode_trap(\text{tmp_fp_env}))};
\end{align*}
\]

**FP Exceptions:**
- Invalid Operation (V)
- Underflow (U)
- Denormal/Unnormal Operand (D)
- Overflow (O)
- Software Assist (SWA) Fault
- Inexact (I)
- Software Assist (SWA) trap
- Inexact (I)
- Software Assist (SWA) trap

**Interruptions:**
- Illegal Operation fault
- Floating-point Exception fault
- Disabled Floating-point Register fault
- Floating-point Exception trap
Floating-point Parallel Maximum

Format: \((qp) \text{fpmax.sf } f_1 = f_2, f_3\)

Description: The paired single precision values in the significands of \(f_2\) or \(f_3\) are compared. The operands with the larger value is returned in the significand of \(f_1\).

If the value of high (low) \(f_2\) is less than the value of high (low) \(f_3\), high (low) \(f_1\) gets high (low) \(f_3\). Otherwise high (low) \(f_1\) gets high (low) \(f_2\).

If high (low) \(f_2\) or high (low) \(f_3\) is a NaN, high (low) \(f_1\) gets high (low) \(f_3\).

The exponent field of \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of \(f_1\) is set to positive (0).

If either \(f_2\) or \(f_3\) is NaNVal, \(f_1\) is set to NaNVal instead of the computed result.

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as for the \(\text{fpcmp.lt}\) operation.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);  // F8
    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        fpminmax_exception_fault_check(f2, f3, sf, &tmp_fp_env);
        fp_exception_fault(tmp_fp_env);
        tmp_fr2 = tmp_right = fp_reg_read_hi(f2);
        tmp_fr3 = tmp_left = fp_reg_read_hi(f3);
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        tmp_res_hi = fp_single(tmp_bool_res ? tmp_fr2 : tmp_fr3);
        tmp_fr2 = tmp_right = fp_reg_read_lo(f2);
        tmp_fr3 = tmp_left = fp_reg_read_lo(f3);
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        tmp_res_lo = fp_single(tmp_bool_res ? tmp_fr2 : tmp_fr3);
        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
        fp_update_fpsr(sf, tmp_fp_env);
    }
    fp_update_psr(f1);
}
```

FP Exceptions: Invalid Operation (V)
Denormal/Unnormal Operand (D)
Software Assist (SWA) fault

Interruptions: Illegal Operation fault Floating-point Exception fault
Disabled Floating-point Register fault
Floating-point Parallel Merge

Format:

(qp) fpmerge.ns \( f_1 = f_2, f_3 \)  
(qp) fpmerge.s \( f_1 = f_2, f_3 \)  
(qp) fpmerge.se \( f_1 = f_2, f_3 \)

Description: For the neg_sign_form, the signs of the pair of single precision values in the significand field of FR \( f_2 \) are negated and concatenated with the exponents and the significands of the pair of single precision values in the significand field of FR \( f_3 \) and stored in the significand field of FR \( f_1 \). This form can be used to negate a pair of single precision floating-point numbers by using the same register for \( f_2 \) and \( f_3 \).

For the sign_form, the signs of the pair of single precision values in the significand field of FR \( f_2 \) are concatenated with the exponents and the significands of the pair of single precision values in the significand field of FR \( f_3 \) and stored in FR \( f_1 \).

For the sign_exp_form, the signs and exponents of the pair of single precision values in the significand field of FR \( f_2 \) are concatenated with the pair of single precision significands in the significand field of FR \( f_3 \) and stored in the significand field of FR \( f_1 \).

For all forms, the exponent field of FR \( f_1 \) is set to the biased exponent for \( 2.0^{63} \) (0x1003E) and the sign field of FR \( f_1 \) is set to positive (0).

For all forms, if either FR \( f_2 \) or FR \( f_3 \) is a NaTVal, FR \( f_1 \) is set to NaTVal instead of the computed result.

Figure 2-15. Floating-point Parallel Merge Negative Sign Operation

Figure 2-16. Floating-point Parallel Merge Sign Operation

Figure 2-17. Floating-point Parallel Merge Sign and Exponent Operation
Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3]))
        FR[f1] = NATVAL;
    else {
        if (neg_sign_form) {
            tmp_res_hi = (!FR[f2].significand{63} << 31)
                        | (FR[f3].significand{62:32});
            tmp_res_lo = (!FR[f2].significand{31} << 31)
                        | (FR[f3].significand{30:0});
        } else if (sign_form) {
            tmp_res_hi = (FR[f2].significand{63} << 31)
                        | (FR[f3].significand{62:32});
            tmp_res_lo = (FR[f2].significand{31} << 31)
                        | (FR[f3].significand{30:0});
        } else { // sign_exp_form
            tmp_res_hi = (FR[f2].significand{63:55} << 23)
                        | (FR[f3].significand{54:32});
            tmp_res_lo = (FR[f2].significand{31:23} << 23)
                        | (FR[f3].significand{22:0});
        }

        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;

    }

    fp_update_psr(f1);
}
```

**FP Exceptions:** None

**Interruptions:** Illegal Operation fault Disabled Floating-point Register fault
Floating-point Parallel Minimum

Format: \((qp)\) \(fp\text{min.sf}\ f_j = f_2, f_3\)

Description: The paired single precision values in the significands of \(FR\ f_2\) or \(FR\ f_3\) are compared. The operands with the smaller value is returned in significand of \(FR\ f_1\).

If the value of high (low) \(FR\ f_2\) is less than the value of high (low) \(FR\ f_3\), high (low) \(FR\ f_j\) gets high (low) \(FR\ f_2\). Otherwise high (low) \(FR\ f_j\) gets high (low) \(FR\ f_3\).

If high (low) \(FR\ f_2\) or high (low) \(FR\ f_3\) is a NaN, high (low) \(FR\ f_j\) gets high (low) \(FR\ f_3\).

The exponent field of \(FR\ f_j\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of \(FR\ f_j\) is set to positive (0).

If either \(FR\ f_2\) or \(FR\ f_3\) is a NaTVal, \(FR\ f_1\) is set to NaTVal instead of the computed result.

This operation does not propagate NaNs the same way as other arithmetic floating-point instructions. The Invalid Operation is signaled in the same manner as for the \(fp\text{cmp.lt}\) operation.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        fpminmax_exception_fault_check(f2, f3, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        tmp_left = fp_reg_read_hi(f2);
        tmp_right = fp_reg_read_hi(f3);
        tmp_res_hi = fp_less_than(tmp_left, tmp_right);
        tmp_res_lo = fp_single(tmp_res_hi ? tmp_left : tmp_right);
        tmp_left = fp_reg_read_lo(f2);
        tmp_right = fp_reg_read_lo(f3);
        tmp_bool_res = fp_less_than(tmp_left, tmp_right);
        tmp_res_lo = fp_single(tmp_bool_res ? tmp_left : tmp_right);
        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
        fp_update_fpsr(sf, tmp_fp_env);
    }
    fp_update_psr(f1);
}
```

FP Exceptions: Invalid Operation (V)
Denormal/Unnormal Operand (D)
Software Assist (SWA) fault

Interruptions: Illegal Operation fault
Floating-point Exception fault
Disabled Floating-point Register fault
Floating-point Parallel Multiply

Format: \[(qp) \text{fpmpy.sf} f_1 = f_3, f_4\]  
\[\text{pseudo-op of: (qp) fpma.sf} f_1 = f_3, f_4, f_0\]

Description: The pair of products of the pairs of single precision values in the significand fields of FR\(_f^3\) and FR\(_f^4\) are computed to infinite precision. The resulting values are then rounded to single precision using the rounding mode specified by FPSR.\(sf.rc\). The pair of rounded results are stored in the significand field of FR\(_f^1\). The exponent field of FR\(_f^1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR\(_f^1\) is set to positive (0).

If either FR\(_f^3\), or FR\(_f^4\) is a NaTVal, FR\(_f^1\) is set to NaTVal instead of the computed results.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

The encodings and interpretation for the status field’s \(rc\) are given in Table 5-6 on page 3:82 in Volume 1: Application Architecture.

Floating-point Parallel Multiply Subtract

Format: \((qp)\) \(fpms.sf \ f_1 = f_3, f_4, f_2\)

Description: The pair of products of the pairs of single precision values in the significand fields of FR \(f_3\) and FR \(f_4\) are computed to infinite precision and then the pair of single precision values in the significand field of FR \(f_2\) is subtracted from these products, again in infinite precision. The resulting values are then rounded to single precision using the rounding mode specified by FPSR.\(sf,rc\). The pair of rounded results are stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

Note: If any of FR \(f_3\), FR \(f_4\), or FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed results.

Mapping: If \(f_2\) is f0 in the fpms instruction, just the IEEE multiply operation is performed.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

The encodings and interpretation for the status field’s \(rc\) are given in Table 5-6 on page 3:82 in Volume 1: Application Architecture.
Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, f4))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3]) ||
        fp_is_natval(FR[f4])) {
        FR[f1] = NATVAL;
        fp_update_psr(f1);
    } else {
        tmp_default_result_pair = fpms_fpnma_exception_fault_check(f2, f3,
                                                                  f4, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));

        if (fp_is_nan_or_inf(tmp_default_result_pair.hi)) {
            tmp_res_hi = fp_single(tmp_default_result_pair.hi);
        } else {
            tmp_res = fp_mul(fp_reg_read_hi(f3), fp_reg_read_hi(f4));
            if (f2 != 0) {
                tmp_sub = fp_reg_read_hi(f2);
                tmp_sub.sign = !tmp_sub.sign;
                tmp_res = fp_add(tmp_res, tmp_sub, tmp_fp_env);
            }
            tmp_res_hi = fp_ieee_round_sp(tmp_res, HIGH, &tmp_fp_env);
        }

        if (fp_is_nan_or_inf(tmp_default_result_pair.lo)) {
            tmp_res_lo = fp_single(tmp_default_result_pair.lo);
        } else {
            tmp_res = fp_mul(fp_reg_read_lo(f3), fp_reg_read_lo(f4));
            if (f2 != 0) {
                tmp_sub = fp_reg_read_lo(f2);
                tmp_sub.sign = !tmp_sub.sign;
                tmp_res = fp_add(tmp_res, tmp_sub, tmp_fp_env);
            }
            tmp_res_lo = fp_ieee_round_sp(tmp_res, LOW, &tmp_fp_env);
        }

        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
        fp_update_fpsr(sf, tmp_fp_env);
        fp_update_psr(f1);
        if (fp_raise_traps(tmp_fp_env))
            fp_exception_trap(fp_decode_trap(tmp_fp_env));
    }
}
```

**FP Exceptions:**
- Invalid Operation (V)
- Denormal/Unnormal Operand (D)
- Overflow (O)
- Software Assist (SW A) fault
- Inexact (I)
- Software Assist (SW A) trap

**Interruptions:**
- Illegal Operation fault
- Disabled Floating-point Register fault
- Floating-point Exception fault
- Floating-point Exception trap
Floating-point Parallel Negate

Format: \[(qp) \text{fpneg } f_1 = f_3\]  
\text{pseudo-op of:} (qp) \text{fpmerge.ns } f_1 = f_3, f_3

Description: The pair of single precision values in the significand field of FR \(f_3\) are negated and stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

If FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

Operation: See “Floating-point Parallel Merge” on page 3:100.
Floating-point Parallel Negate Absolute Value

**Format:**

\[(qp)\ fpnegabs\ f_1 = f_3\]

pseudo-op of: \[(qp)\ fpmerge.ns\ f_j = f_0, f_3\]

**Description:**
The absolute values of the pair of single precision values in the significand field of FR \(f_3\) are computed, negated and stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

If FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

**Operation:**

See “Floating-point Parallel Merge” on page 3:100.
Floating-point Parallel Negative Multiply Add

**Format:**  \((qp)\) fpnma.sf \(f_1 = f_3, f_4, f_2\)

**Description:** The pair of products of the pairs of single precision values in the significand fields of FR \(f_3\) and FR \(f_4\) are computed to infinite precision, negated, and then the pair of single precision values in the significand field of FR \(f_2\) are added to these (negated) products, again in infinite precision. The resulting values are then rounded to single precision using the rounding mode specified by FPSR.sf.rc. The pair of rounded results are stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

If any of FR \(f_3\), FR \(f_4\), or FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

**Note:** If \(f_2\) is f0 in the fpnma instruction, just the IEEE multiply operation (with the product being negated before rounding) is performed.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

The encodings and interpretation for the status field’s rc are given in Table 5-6 on page 3:82 in *Volume 1: Application Architecture*. 
Operation:
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, f4))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3]) ||
        fp_is_natval(FR[f4])) {
        FR[f1] = NATVAL;
        fp_update_psr(f1);
    } else {
        tmp_default_result_pair = fpms_fpnma_exception_fault_check(f2, f3,
                                                                   f4, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));

        if (fp_is_nan_or_inf(tmp_default_result_pair.hi)) {
            tmp_res_hi = fp_single(tmp_default_result_pair.hi);
        } else {
            tmp_res = fp_mul(fp_reg_read_hi(f3), fp_reg_read_hi(f4));
            tmp_res.sign = !tmp_res.sign;
            if (f2 != 0)
                tmp_res = fp_add(tmp_res, fp_reg_read_hi(f2), tmp_fp_env);
            tmp_res_hi = fp_ieee_round_sp(tmp_res, HIGH, &tmp_fp_env);
        }

        if (fp_is_nan_or_inf(tmp_default_result_pair.lo)) {
            tmp_res_lo = fp_single(tmp_default_result_pair.lo);
        } else {
            tmp_res = fp_mul(fp_reg_read_lo(f3), fp_reg_read_lo(f4));
            tmp_res.sign = !tmp_res.sign;
            if (f2 != 0)
                tmp_res = fp_add(tmp_res, fp_reg_read_lo(f2), tmp_fp_env);
            tmp_res_lo = fp_ieee_round_sp(tmp_res, LOW, &tmp_fp_env);
        }

        FR[f1].significand = fp.concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
        fp_update_fpsr(sf, tmp_fp_env);
        fp_update_psr(f1);
        if (fp_raise_traps(tmp_fp_env))
            fp_exception_trap(fp_decode_trap(tmp_fp_env));
    }
}

FP Exceptions: Invalid Operation (V)  Underflow (U)
Denormal/Unnormal Operand (D)  Overflow (O)
Software Assist (SWA) fault  Inexact (I)
Software Assist (SWA) trap

Interruptions: Illegal Operation fault  Floating-point Exception fault
Disabled Floating-point Register fault  Floating-point Exception trap
Floating-point Parallel Negative Multiply

**Format:**

\[(qp) \text{ fpnmpy.sf } f_1 = f_3, f_4\]  
\[\text{pseudo-op of: } (qp) \text{ fpnma.sf } f_1 = f_3, f_4, f_0\]

**Description:** The pair of products of the pairs of single precision values in the significand fields of FR \(f_3\) and FR \(f_4\) are computed to infinite precision and then negated. The resulting values are then rounded to single precision using the rounding mode specified by FPSR.sf,rc. The pair of rounded results are stored in the significand field of FR \(f_1\). The exponent field of FR \(f_1\) is set to the biased exponent for \(2^{0.53} (0x1003E)\) and the sign field of FR \(f_1\) is set to positive (0).

If either FR \(f_3\) or FR \(f_4\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed results.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

The encodings and interpretation for the status field’s \(rc\) are given in Table 5-6 on page 3:82 in Volume 1: Application Architecture.

**Operation:** See “Floating-point Parallel Negative Multiply Add” on page 3:108.
Floating-point Parallel Reciprocal Approximation

**Format:**  
\[(qp) \text{fprcpa.}\text{sf} \ f_1, p_2 = f_2, f_3\]

**Description:**  
If PR \(qp\) is 0, PR \(p_2\) is cleared and FR \(f_1\) remains unchanged.

If PR \(qp\) is 1, the following will occur:

- Each half of the significand of FR \(f_1\) is either set to an approximation (with a relative error < \(2^{-8.886}\)) of the reciprocal of the corresponding half of FR \(f_3\), or set to the IEEE-754 mandated response for the quotient FR \(f_2/FR f_3\) of the corresponding half — if that half of FR \(f_2\) or of FR \(f_3\) is in the set \([-\infty, -0, +0, +\infty, \text{NaN}\].

- If either half of FR \(f_j\) is set to the IEEE-754 mandated quotient, or is set to an approximation of the reciprocal which may cause the Newton-Raphson iterations to fail to produce the correct IEEE-754 divide result, then PR \(p_2\) is set to 0, otherwise it is set to 1.

For correct IEEE divide results, when PR \(p_2\) is cleared, user software is expected to compute the quotient (FR \(f_2/FR f_3\)) for each half (using the non-parallel \(\text{frcpa}\) instruction), and merge the results into FR \(f_j\), keeping PR \(p_2\) cleared.

- The exponent field of FR \(f_1\) is set to the biased exponent for 2.0^{63} (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

- If either FR \(f_2\) or FR \(f_3\) is a NaN, FR \(f_1\) is set to NaN instead of the computed result, and PR \(p_2\) is cleared.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

**Operation:**  
```
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3])) {
        FR[f_1] = NATVAL;
        PR[p_2] = 0;
    } else {  
        tmp_default_result_pair = fprcpa_exception_fault_check(f_2, f_3, sf, 
                                                                 &tmp_fp_env, &limits_check);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        if (fp_is_nan_or_inf(tmp_default_result_pair.hi) ||
            limits_check.hi_fr3) {
            tmp_res_hi = fp_single(tmp_default_result_pair.hi);
            tmp_pred_hi = 0;
        } else {
            num = fp_normalize(fp_reg_read_hi(f_2));
            den = fp_normalize(fp_reg_read_hi(f_3));
            if (fp_is_inf(num) && fp_is_finite(den)) {
                tmp_res = FP_INFINITY;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_finite(num) && fp_is_inf(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_zero(num) && fp_is_finite(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_zero(num) && fp_is_zero(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_inf(num) && fp_is_nan(den)) {
                tmp_res = FP_INFINITY;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_zero(num) && fp_is_nan(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_nan_or_inf(num) || fp_is_nan(den)) {
                tmp_res = FP_NAN;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_inf(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_INFINITY;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_finite(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_zero(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_finite(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_INFINITY;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_zero(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_finite(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_INFINITY;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_zero(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_ZERO;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
            } else if (fp_is_finite(num) && fp_is_nan_or_inf(den)) {
                tmp_res = FP_INFINITY;
                tmp_res.sign = num.sign ^ den.sign;
                tmp_pred_hi = 0;
```
if (fp_is_nan_or_inf(tmp_default_result_pair.lo) ||
    limits_check.lo_fr3) {
  tmp_res_lo = fp_single(tmp_default_result_pair.lo);
  tmp_pred_lo = 0;
} else {
  num = fp_normalize(fp_reg_read_lo(f2));
  den = fp_normalize(fp_reg_read_lo(f3));
  if (fp_is_inf(num) && fp_is_finite(den)) {
    tmp_res = FP_INFINITY;
    tmp_res.sign = num.sign ^ den.sign;
    tmp_pred_lo = 0;
  } else if (fp_is_finite(num) && fp_is_inf(den)) {
    tmp_res = FP_ZERO;
    tmp_res.sign = num.sign ^ den.sign;
    tmp_pred_lo = 0;
  } else if (fp_is_zero(num) && fp_is_finite(den)) {
    tmp_res = FP_ZERO;
    tmp_res.sign = num.sign ^ den.sign;
    tmp_pred_lo = 0;
  } else {
    tmp_res = fp_ieee_recip(den);
    if (limits_check.lo_fr2_or_quot)
      tmp_pred_lo = 0;
    else
      tmp_pred_lo = 1;
  }
  tmp_res_lo = fp_single(tmp_res);
}

FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
FR[f1].exponent = FP_INTEGER_EXP;
FR[f1].sign = FP_SIGN_POSITIVE;
PR[p2] = tmp_pred_hi && tmp_pred_lo;
fp_update_fpsr(sf, tmp_fp_env);
}
else {
  PR[p2] = 0;
}

FP Exceptions: Invalid Operation (V)
Zero Divide (Z)
Denormal/Unnormal Operand (D)
Software Assist (SWA) fault

Interruptions: Illegal Operation fault Floating-point Exception fault
Disabled Floating-point Register fault

Volume 3: Instruction Reference
Floating-point Parallel Reciprocal Square Root Approximation

**Format:**

\[(qp) \text{fprsqrta.sf \(f_1, p_2 = f_3\)}\]

**Description:**

If PR \(qp\) is 0, PR \(p_2\) is cleared and FR \(f_1\) remains unchanged.

If PR \(qp\) is 1, the following will occur:

- Each half of the significand of FR \(f_1\) is either set to an approximation (with a relative error < \(2^{-8.831}\)) of the reciprocal square root of the corresponding half of FR \(f_3\), or set to the IEEE-754 compliant response for the reciprocal square root of the corresponding half of FR \(f_3\) — if that half of FR \(f_3\) is in the set \{-Infinity, -Finite, -0, +0, +Infinity, NaN\}.
- If either half of FR \(f_3\) is set to the IEEE-754 mandated reciprocal square root, or is set to an approximation of the reciprocal square root which may cause the Newton-Raphson iterations to fail to produce the correct IEEE-754 square root result, then PR \(p_2\) is set to 0, otherwise it is set to 1.
- For correct IEEE square root results, when PR \(p_2\) is cleared, user software is expected to compute the square root for each half (using the non-parallel \text{frsqrta} instruction), and merge the results in FR \(f_1\), keeping PR \(p_2\) cleared.
- The exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).
- If FR \(f_3\) is a NaN value, FR \(f_1\) is set to NaN instead of the computed result, and PR \(p_2\) is cleared.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

**Operation:**

```c
if (PR[qp]) {
   (fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f3, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f3]))
        FR[f1] = NATVAL;
        PR[p2] = 0;
    else {
        tmp_default_result_pair = fprsqrta_exception_fault_check(f3, sf,
            &tmp_fp_env, &limits_check);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        if (fp_is_nan(tmp_default_result_pair.hi))
            tmp_res_hi = fp_single(tmp_default_result_pair.hi);
            tmp_pred_hi = 0;
        else {
            tmp_fr3 = fp_normalize(fp_reg_read_hi(f3));
            if (fp_is_zero(tmp_fr3))
                tmp_res = FP_INFINITY;
                tmp_res.sign = tmp_fr3.sign;
                tmp_pred_hi = 0;
            else if (fp_is_pos_inf(tmp_fr3))
                tmp_res = FP_ZERO;
                tmp_pred_hi = 0;
            else {
                tmp_res = fp_ieee_recip_sqrt(tmp_fr3);
                if (limits_check.hi)
                    tmp_pred_hi = 0;
                else
                    tmp_pred_hi = 1;
            }
```
if (fp_is_nan(tmp_default_result_pair.lo)) {
    tmp_res_lo = fp_single(tmp_default_result_pair.lo);
    tmp_pred_lo = 0;
} else {
    tmp_fr3 = fp_normalize(fp_reg_read_lo(f3));
    if (fp_is_zero(tmp_fr3)) {
        tmp_res = FP_INFINITY;
        tmp_res.sign = tmp_fr3.sign;
        tmp_pred_lo = 0;
    } else if (fp_is_pos_inf(tmp_fr3)) {
        tmp_res = FP_ZERO;
        tmp_pred_lo = 0;
    } else {
        tmp_res = fp_ieee_recip_sqrt(tmp_fr3);
        if (limits_check.lo)
            tmp_pred_lo = 0;
        else
            tmp_pred_lo = 1;
    }
    tmp_res_lo = fp_single(tmp_res);
}

FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
FR[f1].exponent = FP_INTEGER_EXP;
FR[f1].sign = FP_SIGN_POSITIVE;
PR[p2] = tmp_pred_hi && tmp_pred_lo;

fp_update_fpsr(sf, tmp_fp_env);
} else {
    PR[p2] = 0;
}

**FP Exceptions:**
- Invalid Operation (V)
- Denormal/Unnormalized Operand (D)
- Software Assist (SWA) fault

**Interruptions:**
- Illegal Operation fault
- Floating-point Exception fault
- Disabled Floating-point Register fault
Floating-point Reciprocal Approximation

Format:  \((qp)\) \text{frcpa.sf} \ f_1, p_2 = f_2, f_3

Description: If PR \(qp\) is 0, PR \(p_2\) is cleared and FR \(f_1\) remains unchanged.

If PR \(qp\) is 1, the following will occur:

- FR \(f_1\) is either set to an approximation (with a relative error < \(2^{-8.886}\)) of the reciprocal of FR \(f_3\), or to the IEEE-754 mandated quotient of FR \(f_2/FR f_3\) — if either FR \(f_2\) or FR \(f_3\) is in the set \{-Infinity, -0, Pseudo-zero, +0, +Infinity, NaN, Unsupported\}.
- If FR \(f_1\) is set to the approximation of the reciprocal of FR \(f_3\), then PR \(p_2\) is set to 1; otherwise, it is set to 0.
- If FR \(f_2\) and FR \(f_3\) are such that the approximation of FR \(f_1\)'s reciprocal may cause the Newton-Raphson iterations to fail to produce the correct IEEE-754 result of FR \(f_2/FR f_3\), then a Floating-point Exception fault for Software Assist occurs. System software is expected to compute the IEEE-754 quotient (FR \(f_2/FR f_3\)), return the result in FR \(f_1\), and set PR \(p_2\) to 0.
- If either FR \(f_2\) or FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result, and PR \(p_2\) is cleared.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

\[
\begin{align*}
\text{if (PR[qp])} & \\
& \text{fp_check_target_register}(f_1); \\
& \text{if (tmp_isrcode = fp_reg_disabled}(f_1, f_2, f_3, 0)) \\
& \quad \text{disabled_fp_register_fault(tmp_isrcode, 0)}; \\
& \text{if (fp_is_natval}(FR[f_2]) \mid \text{ fp_is_natval}(FR[f_3])) \\
& \quad \text{FR}[f_1] = \text{NATVAL}; \\
& \quad \text{PR}[p_2] = 0; \\
& \text{else} \\
& \quad \text{tmp_default_result = frcpa_exception_fault_check}(f_2, f_3, sf, \text{ &tmp_fp_env}); \\
& \quad \text{if (fp_raise_fault(tmp_fp_env))} \\
& \quad \quad \text{fp_exception_fault(fp_decode_fault(tmp_fp_env))}; \\
& \text{if (fp_is_nan_or_inf}(\text{tmp_default_result})) \\
& \quad \text{FR}[f_1] = \text{tmp_default_result}; \\
& \quad \text{PR}[p_2] = 0; \\
& \text{else} \\
& \quad \quad \text{num = fp_normalize}(\text{fp_reg_read}(FR[f_2])); \\
& \quad \quad \text{den = fp_normalize}(\text{fp_reg_read}(FR[f_3])); \\
& \quad \quad \text{if (fp_is_inf}(num) \&\& \text{ fp_is_finite}(\text{den})) \\
& \quad \quad \quad \text{FR}[f_1] = \text{FP_INFINITY}; \\
& \quad \quad \quad \text{FR}[f_1].sign = \text{num.sign} ^ \text{den.sign}; \\
& \quad \quad \quad \text{PR}[p_2] = 0; \\
& \quad \text{else if (fp_is_finite}(\text{num}) \&\& \text{ fp_is_inf}(\text{den})) \\
& \quad \quad \quad \text{FR}[f_1] = \text{FP_ZERO}; \\
& \quad \quad \quad \text{FR}[f_1].sign = \text{num.sign} ^ \text{den.sign}; \\
& \quad \quad \quad \text{PR}[p_2] = 0; \\
& \quad \text{else if (fp_is_zero}(\text{num}) \&\& \text{ fp_is_finite}(\text{den})) \\
& \quad \quad \quad \text{FR}[f_1] = \text{FP_ZERO}; \\
& \quad \quad \quad \text{FR}[f_1].sign = \text{num.sign} ^ \text{den.sign}; \\
& \quad \quad \quad \text{PR}[p_2] = 0; \\
& \quad \text{else} \\
& \quad \quad \quad \text{FR}[f_1] = \text{fp_ieee_recip}(\text{den}); \\
& \quad \quad \quad \text{PR}[p_2] = 1;
\end{align*}
\]
fp_update_fpsr(sf, tmp_fp_env);
}
fp_update_psr(fj);
} else {
    PR[p2] = 0;
}

// fp_ieee_recip()

fp_ieee_recip(den) {
    RECIP_TABLE[256] = {
        0x3fc, 0x3f4, 0x3ec, 0x3e4, 0x3d6, 0x3d5, 0x3cd, 0x3c6,
        0x3be, 0x3b7, 0x3af, 0x3a8, 0x3a1, 0x399, 0x392, 0x38b,
        0x384, 0x37d, 0x376, 0x36f, 0x368, 0x361, 0x35b, 0x354,
        0x34d, 0x346, 0x33f, 0x338, 0x331, 0x32c, 0x326, 0x320,
        0x319, 0x313, 0x30d, 0x307, 0x300, 0x2fa, 0x2f4, 0x2ee,
        0x2e8, 0x2e2, 0x2dc, 0x2d7, 0x2d1, 0x2cb, 0x2c5, 0x2bf,
        0x2ba, 0x2b4, 0x2a8, 0x2a3, 0x29e, 0x299, 0x293, 0x28d,
        0x288, 0x283, 0x27e, 0x279, 0x273, 0x26e, 0x269, 0x263,
        0x25f, 0x25a, 0x255, 0x250, 0x24b, 0x246, 0x241, 0x23c,
        0x237, 0x232, 0x22e, 0x229, 0x224, 0x21f, 0x21b, 0x216,
        0x211, 0x20d, 0x208, 0x204, 0x1ff, 0x1fb, 0x1f6, 0x1f3,
        0x1f2, 0x1ed, 0x1e9, 0xe1e0, 0xe1d8, 0xe1d4,
        0x1cf, 0x1cb, 0x1c7, 0x1c3, 0x1bf, 0x1bb, 0x1b6, 0x1b2,
        0x1ae, 0x1aa, 0x1a6, 0x1a2, 0x19e, 0x19a, 0x197, 0x193,
        0x18f, 0x18b, 0x187, 0x183, 0x17f, 0x17c, 0x178, 0x174,
        0x171, 0x16d, 0x169, 0x166, 0x15e, 0x15b, 0x157, 0x154,
        0x150, 0x14d, 0x149, 0x146, 0x142, 0x13f, 0x13b, 0x138,
        0x134, 0x131, 0x12e, 0x12a, 0x127, 0x124, 0x120, 0x11d,
        0x11a, 0x117, 0x113, 0x110, 0x10d, 0x10a, 0x107, 0x103,
        0x100, 0x0f8, 0x0fa, 0xf7, 0xf4, 0xf1, 0xee, 0xe8,
        0xe6, 0xe5, 0xe2, 0xdf, 0xdc, 0xd9, 0xd6, 0xd3,
        0xda, 0x09d, 0x09c, 0x099, 0x096, 0x094, 0x091,
        0x08e, 0x08c, 0x089, 0x087, 0x084, 0x082, 0x07f,
        0x07c, 0x07a, 0x077, 0x075, 0x073, 0x070, 0x06e, 0x06b,
        0x069, 0x066, 0x064, 0x061, 0x05f, 0x05d, 0x05a, 0x058,
        0x056, 0x053, 0x051, 0x04f, 0x04c, 0x04a, 0x048, 0x045,
        0x043, 0x041, 0x03f, 0x03c, 0x03a, 0x038, 0x036, 0x033,
        0x031, 0x02f, 0x02d, 0x02b, 0x029, 0x026, 0x024, 0x022,
        0x020, 0x01e, 0x01c, 0x01a, 0x018, 0x015, 0x013, 0x011,
        0x00f, 0x00d, 0x00b, 0x009, 0x007, 0x005, 0x003, 0x001,
    };
    tmp_index = den.significand(62:55);
    tmp_res.significand = (1 << 63) | (RECIP_TABLE[tmp_index] << 53);
    tmp_res.exponent = FP_REG_EXP_ONES - 2 - den.exponent;
    tmp_res.sign = den.sign;
    return (tmp_res);
}

FP Exceptions: Invalid Operation (V)
Zero Divide (Z)
Denormal/Unnormal Operand (D)
Software Assist (SWA) fault
**Interruptions:**

- Illegal Operation fault
- Disabled Floating-point Register fault
- Floating-point Exception fault
Floating-point Reciprocal Square Root Approximation

Format: \((qp)\ frsqrta.sf \ f_1, p_2 = f_3\)

Description: If PR \(qp\) is 0, PR \(p_2\) is cleared and FR \(f_1\) remains unchanged.

If PR \(qp\) is 1, the following will occur:

- FR \(f_1\) is either set to an approximation (with a relative error < \(2^{-8.831}\)) of the reciprocal square root of FR \(f_3\), or set to the IEEE-754 mandated square root of FR \(f_3\) — if FR \(f_3\) is in the set \([-\infty, -\text{Finite}, -0, \text{Pseudo-zero}, +0, +\infty, \text{NaN}, \text{Unsupported}]\).

- If FR \(f_1\) is set to an approximation of the reciprocal square root of FR \(f_3\), then PR \(p_2\) is set to 1; otherwise, it is set to 0.

- If FR \(f_3\) is such the approximation of its reciprocal square root may cause the Newton-Raphson iterations to fail to produce the correct IEEE-754 square root result, then a Floating-point Exception fault for Software Assist occurs.

System software is expected to compute the IEEE-754 square root, return the result in FR \(f_1\), and set PR \(p_2\) to 0.

- If FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result, and PR \(p_2\) is cleared.

The mnemonic values for \(sf\) are given in Table 2-23 on page 3:51.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_3, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f_3])) {
        FR[f_1] = NATVAL;
        PR[p_2] = 0;
    } else {
        tmp_default_result = frsqrta_exception_fault_check(f_3, sf, &tmp_fp_env);
        if (fp_raise_fault(tmp_fp_env))
            fp_exception_fault(fp_decode_fault(tmp_fp_env));
        if (fp_is_nan(tmp_default_result)) {
            FR[f_1] = tmp_default_result;
            PR[p_2] = 0;
        } else {
            tmp_fr3 = fp_normalize(fp_reg_read(FR[f_3]));
            if (fp_is_zero(tmp_fr3)) {
                FR[f_1] = tmp_fr3;
                PR[p_2] = 0;
            } else if (fp_is_pos_inf(tmp_fr3)) {
                FR[f_1] = tmp_fr3;
                PR[p_2] = 0;
            } else {
                FR[f_1] = fp_ieee_recip_sqrt(tmp_fr3);
                PR[p_2] = 1;
            }
        }
    }
    fp_update_fpsr(sf, tmp_fp_env);
} else {
    PR[p_2] = 0;
}
```
frsqrta

} // fp_ieee_recip_sqrt()

fp_ieee_recip_sqrt(root) {
    RECIP_SQRT_TABLE[256] = {
        0x1a5, 0x1a0, 0x19a, 0x195, 0x18f, 0x18a, 0x185, 0x180,
        0x17a, 0x175, 0x170, 0x16b, 0x166, 0x161, 0x15d, 0x158,
        0x153, 0x14e, 0x14a, 0x145, 0x140, 0x13c, 0x138, 0x133,
        0x12f, 0x12a, 0x126, 0x122, 0x11e, 0x11a, 0x115, 0x111,
        0x10d, 0x109, 0x105, 0x101, 0x0fd, 0x0fa, 0x0f6, 0x0f2,
        0x0ee, 0x0ea, 0xe07, 0xe03, 0xd0f, 0xd0c, 0xd08, 0xd04,
        0x0d1, 0x0ce, 0xca7, 0xc03, 0xc00, 0xc0b, 0xc06, 0xc02,
        0x0b6, 0x0b3, 0xb00, 0xb04, 0xa0a, 0xa06, 0xa02, 0xa00,
        0x09d, 0x09a, 0x097, 0x094, 0x091, 0x08e, 0x08b, 0x088,
        0x085, 0x082, 0x07f, 0x07d, 0x077, 0x074, 0x071, 0x06f,
        0x06c, 0x069, 0x067, 0x064, 0x061, 0x05f, 0x05c, 0x059,
        0x056, 0x053, 0x050, 0x04f, 0x04d, 0x04a, 0x047, 0x044,
        0x041, 0x03f, 0x03e, 0x03c, 0x03a, 0x037, 0x034, 0x032,
        0x030, 0x02e, 0x02c, 0x029, 0x026, 0x024, 0x022, 0x020,
        0x01e, 0x01c, 0x01a, 0x017, 0x014, 0x011, 0x00f, 0x00d,
        0x00b, 0x009, 0x007, 0x005, 0x003, 0x001, 0x3fc, 0x3f4,
        0x3ed, 0x3e5, 0x3dd, 0x3d5, 0x3ce, 0x3c7, 0x3bf, 0x3b8,
        0x3b1, 0x3aa, 0x3a3, 0x39c, 0x395, 0x38e, 0x388, 0x381,
        0x37a, 0x374, 0x36d, 0x367, 0x361, 0x35a, 0x354, 0x34e,
        0x34a, 0x346, 0x342, 0x33c, 0x336, 0x330, 0x32b, 0x325,
        0x31f, 0x31a, 0x314, 0x30f, 0x309, 0x304, 0x2fe, 0x2f9,
        0x2f4, 0x2ee, 0x2e9, 0x2e4, 0x2df, 0x2da, 0x2d5, 0x2d0,
        0x2cb, 0x2c6, 0x2c1, 0x2bd, 0x2b8, 0x2b3, 0x2ae, 0x2aa,
        0x2a5, 0x2a1, 0x29c, 0x298, 0x295, 0x28f, 0x28a, 0x286,
        0x282, 0x27d, 0x279, 0x275, 0x271, 0x26d, 0x268, 0x264,
        0x260, 0x25d, 0x256, 0x256, 0x254, 0x24c, 0x249, 0x245,
        0x241, 0x23d, 0x239, 0x235, 0x232, 0x22e, 0x22a, 0x226,
        0x221, 0x21c, 0x21b, 0x215, 0x211, 0x208, 0x204, 0x200,
        0x1fd, 0x1f9, 0x1f6, 0x1f3, 0x1f0, 0x1ec, 0x1ed, 0x1e6,
        0x1e3, 0x1df, 0x1dc, 0x1d9, 0x1d6, 0x1d3, 0x1d0, 0x1c7,
        0x1c4, 0x1c1, 0x1be, 0x1bb, 0x1b8, 0x1b5, 0x1a4, 0x1ac,
        0x1aa,
    };
    tmp_index = (root.exponent(0) << 7) | root.significand(62:56);
    tmp_res.significand = (1 << 63) | (RECIP_SQRT_TABLE[tmp_index] << 53);
    tmp_res.exponent = FP_REG_EXP_HALF -
        ((root.exponent - FP_REG_BIAS) >> 1));
    tmp_res.sign = FP_SIGN_POSITIVE;
    return (tmp_res);
}

FP Exceptions: Invalid Operation (V)
Denormal/Unnormal Operand (D)
Software Assist (SWA) fault

Interruptions: Illegal Operation fault Floating-point Exception fault
Disabled Floating-point Register fault

Volume 3: Instruction Reference 3:119
## Floating-point Select

**Format:**  
\[(qp) \text{ fselect } f_1 = f_3, f_4, f_2\]  
\[F3\]

**Description:** The significand field of FR\textsubscript{f3} is logically AND-ed with the significand field of FR\textsubscript{f2} and the significand field of FR\textsubscript{f4} is logically AND-ed with the one’s complement of the significand field of FR\textsubscript{f2}. The two results are logically OR-ed together. The result is placed in the significand field of FR\textsubscript{f1}.

The exponent field of FR\textsubscript{f1} is set to the biased exponent for 2.0\textsuperscript{63} (0x1003E). The sign bit field of FR\textsubscript{f1} is set to positive (0).

If any of FR\textsubscript{f3}, FR\textsubscript{f4}, or FR\textsubscript{f2} is a NaTVal, FR\textsubscript{f1} is set to NaTVal instead of the computed result.

**Operation:**

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, f4))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3]) ||
        fp_is_natval(FR[f4])) {
        FR[f1] = NATVAL;
    } else {
        FR[f1].significand = (FR[f3].significand & FR[f2].significand)
                             | (FR[f4].significand & ~FR[f2].significand);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
    }
    fp_update_psr(f1);
}
```

**FP Exceptions:** None

**Interruptions:** Illegal Operation fault  
Disabled Floating-point Register fault
Floating-point Set Controls

**Format:** \( (qp) \) \( fsetc.sf \ amask_7, \ omask_7 \)

**Description:** The status field’s control bits are initialized to the value obtained by logically AND-ing the \( sf0.controls \) and \( amask_7 \) immediate field and logically OR-ing the \( omask_7 \) immediate field.

The mnemonic values for \( sf \) are given in Table 2-23 on page 3:51.

**Operation:**

```c
if (PR[qp]) {
    tmp_controls = (AR[FPSR].sf0.controls & amask_7) | omask_7;
    if (is_reserved_field(FSETC, sf, tmp_controls))
        reserved_register_field_fault();
    fp_set_sf_controls(sf, tmp_controls);
}
```

**FP Exceptions:** None

**Interruptions:** Reserved Register/Field fault
**Floating-point Subtract**

**Format:**

\[
\text{fsub}\, p_c \, s_f \, f_1 = f_3, f_2
\]

pseudo-op of: \((qp)\) fms.p_c.s_f f_1 = f_3, f_1, f_2

**Description:**

FR \(f_2\) is subtracted from FR \(f_3\) (computed to infinite precision), rounded to the precision indicated by \(p_c\) (and possibly FPSR.s_f.p_c and FPSR.s_f.wre) using the rounding mode specified by FPSR.s_f.rc, and placed in FR \(f_1\).

If either FR \(f_3\) or FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

The mnemonic values for the opcode’s \(p_c\) are given in Table 2-22 on page 3:51. The mnemonic values for \(s_f\) are given in Table 2-23 on page 3:51. For the encodings and interpretation of the status field’s \(p_c\), \(wre\), and \(rc\), refer to Table 5-5 and Table 5-6 on page 3:82 in *Volume 1: Application Architecture*.

**Operation:**

See “Floating-point Multiply Subtract” on page 3:79.
Floating-point Swap

Format:  

$$ (qp)\text{ fswap } f_1 = f_2, f_3 $$  \hspace{1cm} \text{swap\_form F9}  

$$ (qp)\text{ fswap.nl } f_1 = f_2, f_3 $$  \hspace{1cm} \text{swap\_nl\_form F9}  

$$ (qp)\text{ fswap.nr } f_1 = f_2, f_3 $$  \hspace{1cm} \text{swap\_nr\_form F9}  

Description:  

For the swap\_form, the left single precision value in FR\_\text{f2} is concatenated with the right single precision value in FR\_\text{f3}. The concatenated pair is then swapped.

For the swap\_nl\_form, the left single precision value in FR\_\text{f2} is concatenated with the right single precision value in FR\_\text{f3}. The concatenated pair is then swapped, and the left single precision value is negated.

For the swap\_nr\_form, the left single precision value in FR\_\text{f2} is concatenated with the right single precision value in FR\_\text{f3}. The concatenated pair is then swapped, and the right single precision value is negated.

For all forms, the exponent field of FR\_\text{f1} is set to the biased exponent for 2.0^{63} (0x1003E) and the sign field of FR\_\text{f1} is set to positive (0).

For all forms, if either FR\_\text{f2} or FR\_\text{f3} is a NaTVal, FR\_\text{f1} is set to NaTVal instead of the computed result.

Figure 2-18. Floating-point Swap

![Figure 2-18. Floating-point Swap](image)

Figure 2-19. Floating-point Swap Negate Left

![Figure 2-19. Floating-point Swap Negate Left](image)

Figure 2-20. Floating-point Swap Negate Right

![Figure 2-20. Floating-point Swap Negate Right](image)
**Operation:**

```c
if (PR[qp]) {
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, f3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f2]) || fp_is_natval(FR[f3])) {
        FR[f1] = NATVAL;
    } else {
        if (swap_form) {
            tmp_res_hi = FR[f3].significand{31:0};
            tmp_res_lo = FR[f2].significand{63:32};
        } else if (swap_nl_form) {
            tmp_res_hi = (!(FR[f3].significand{31} << 31)
                        | (FR[f3].significand{30:0}));
            tmp_res_lo = FR[f2].significand{63:32};
        } else { // swap_nr_form
            tmp_res_hi = FR[f3].significand{31:0};
            tmp_res_lo = (!(FR[f2].significand{63} << 31)
                        | (FR[f2].significand{62:32}));
        }

        FR[f1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f1].exponent = FP_INTEGER_EXP;
        FR[f1].sign = FP_SIGN_POSITIVE;
    }
}

fp_update_psr(f1);
```

**FP Exceptions:** None

**Interruptions:**

- Illegal Operation fault
- Disabled Floating-point Register fault
Floating-point Sign Extend

Format: 

\[(qp) \text{ fsxt}.l \ f_1 = f_2, f_3\]  sxt_l_form  F9
\[(qp) \text{ fsxt}.r \ f_1 = f_2, f_3\]  sxt_r_form  F9

Description: For the sxt_l_form (sxt_r_form), the sign of the left (right) single precision value in FR \(f_2\) is extended to 32-bits and is concatenated with the left (right) single precision value in FR \(f_3\).

For all forms, the exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

For all forms, if either FR \(f_2\) or FR \(f_3\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

Figure 2-21. Floating-point Sign Extend Left

![Figure 2-21. Floating-point Sign Extend Left](image)

Figure 2-22. Floating-point Sign Extend Right

![Figure 2-22. Floating-point Sign Extend Right](image)

Operation: 

if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);
    if (fp_is_natval(FR[f_2]) \ || \ fp_is_natval(FR[f_3]) \})
        NATVAL;
    else {
        if (sxt_l_form) {
            tmp_res_hi = (FR[f_2].significand{63} ? 0xFFFFFFFF : 0x00000000);
            tmp_res_lo = FR[f_3].significand(63:32);
        } else {
            tmp_res_hi = (FR[f_2].significand{31} ? 0xFFFFFFFF : 0x00000000);
            tmp_res_lo = FR[f_3].significand(31:0);
        }
        FR[f_1].significand = fp_concatenate(tmp_res_hi, tmp_res_lo);
        FR[f_1].exponent = FP_INTEGER_EXP;
        FR[f_1].sign = FP_SIGN_POSITIVE;
    }
    fp_update_psr(f_1);
}

FP Exceptions: None

Interruptions: Illegal Operation fault  Disabled Floating-point Register fault
Flush Write Buffers

Format: \((qp)\ fwb\)

Description: The processor is instructed to expedite flushing of any pending stores held in write or coalescing buffers. Since this operation is a hint, the processor may or may not take any action and actually flush any outstanding stores. The processor gives no indication when flushing of any prior stores is completed. An \(fwb\) instruction does not ensure ordering of stores, since later stores may be flushed before prior stores.

To ensure prior coalesced stores are made visible before later stores, software must issue a release operation between stores (see Table 4-14 on page 3:69 in Volume 2: System Architecture for a list of release operations).

This instruction can be used to help ensure stores held in write or coalescing buffers are not delayed for long periods or to expedite high priority stores out of the processors.

Operation: \(\text{if } (PR[qp]) \{\)
\(\quad \text{mem_flush_pending_stores();}\)
\(\}\)

Interruptions: None
Floating-point Exclusive Or

Format: \( (qp) \text{ fxor } f_1 = f_2 \lor f_3 \)

Description: The bit-wise logical exclusive-OR of the significand fields of FR \( f_2 \) and FR \( f_3 \) is computed. The resulting value is stored in the significand field of FR \( f_1 \). The exponent field of FR \( f_1 \) is set to the biased exponent for \( 2.0^{63} \) (0x1003E) and the sign field of FR \( f_1 \) is set to positive (0).

If either of FR \( f_2 \) or FR \( f_3 \) is a NaTVal, FR \( f_1 \) is set to NaTVal instead of the computed result.

Operation:

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, 0))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f_2]) || fp_is_natval(FR[f_3])) {
        FR[f_1] = NATVAL;
    } else {
        FR[f_1].significand = FR[f_2].significand ^ FR[f_3].significand;
        FR[f_1].exponent = FP_INTEGER_EXP;
        FR[f_1].sign = FP_SIGN_POSITIVE;
    }

    fp_update_psr(f_1);
}
```

FP Exceptions: None

Interruptions: Legal Operation fault Disabled Floating-point Register fault
Get Floating-point Value or Exponent or Significand

Format:

\[(qp) \text{ getf.s } r_1 = f_2\]
\[(qp) \text{ getf.d } r_1 = f_2\]
\[(qp) \text{ getf.exp } r_1 = f_2\]
\[(qp) \text{ getf.sig } r_1 = f_2\]

single_form, M19
double_form, M19
exponent_form, M19
significand_form, M19

Description:
In the single and double forms, the value in FR \(f_2\) is converted into a single precision (single_form) or double precision (double_form) memory representation and placed in GR \(r_1\), as shown in Figure 5-4 on page 1:84 and Figure 5-5 on page 1:85 in Volume 1: Application Architecture, respectively. In the single_form, the most-significant 32 bits of GR \(r_1\) are set to 0.

In the exponent_form, the exponent field of FR \(f_2\) is copied to bits 16:0 of GR \(r_1\) and the sign bit of the value in FR \(f_2\) is copied to bit 17 of GR \(r_1\). The most-significant 46-bits of GR \(r_1\) are set to zero.

In the significand_form, the significand field of the value in FR \(f_2\) is copied to GR \(r_1\)

For all forms, if FR \(f_2\) contains a NaTVal, then the NaT bit corresponding to GR \(r_1\) is set to 1.

Operation:

\[\text{if (PR[qp])} \{\]
  \hspace{1em} \text{check_target_register}(r_1);
  \hspace{1em} \text{if (tmp_isrcode = fp_reg_disabled(f_2, 0, 0, 0))}
     \hspace{1em} \text{disabled_fp_register_fault(tmp_isrcode, 0)};

  \text{if (single_form) } \{
    \hspace{1em} \text{GR}[r_1][31:0] = \text{fp_fr_to_mem_format(FR}[f_2], 4, 0);
    \hspace{1em} \text{GR}[r_1][63:32] = 0;
  \} \text{ else if (double_form) } \{
    \hspace{1em} \text{GR}[r_1] = \text{fp_fr_to_mem_format(FR}[f_2], 8, 0);
  \} \text{ else if (exponent_form) } \{
    \hspace{1em} \text{GR}[r_1][63:18] = 0;
    \hspace{1em} \text{GR}[r_1][16:0] = \text{FR}[f_2].\text{exponent};
    \hspace{1em} \text{GR}[r_1][17] = \text{FR}[f_2].\text{sign};
  \} \text{ else // significand_form}
     \hspace{1em} \text{GR}[r_1] = \text{FR}[f_2].\text{significand};\]

Figure 2-23. Function of getf.exp

Figure 2-24. Function of getf.sig
if (fp_is_natval(FR[fj]))
    GR[rj].nat = 1;
else
    GR[rj].nat = 0;
}

**Interruptions:**
- Illegal Operation fault
- Disabled Floating-point Register fault
Invalidate ALAT

Format:  
(qp) invala  
(qp) invala.e \(r_1\)  
(qp) invala.e \(f_1\)

complete_form  M24  
gr_form, entry_form  M26  
fr_form, entry_form  M27

Description: The selected entry or entries in the ALAT are invalidated.
In the complete_form, all ALAT entries are invalidated. In the entry_form, the ALAT is queried using the general register specifier \(r_1\) (gr_form), or the floating-point register specifier \(f_1\) (fr_form), and if any ALAT entry matches, it is invalidated.

Operation:  
if (PR[qp]) {  
    if (complete_form)  
        alat_inval();  
    else {  // entry_form  
        if (gr_form)  
            alat_inval_single_entry(GENERAL, \(r_1\));  
        else  // fr_form  
            alat_inval_single_entry(FLOAT, \(f_1\));  
    }  
}

Interruptions: None
Insert Translation Cache

Format:

(qp) itc.i \( r_2 \)  
(instruction_form) M41
(qp) itc.d \( r_2 \)  
(data_form) M41

Description:

An entry is inserted into the instruction or data translation cache. GR \( r_2 \) specifies the physical address portion of the translation. ITIR specifies the protection key, page size and additional information. The virtual address is specified by the IFA register and the region register is selected by IFA[63:61]. The processor determines which entry to replace based on an implementation-specific replacement algorithm.

The visibility of the itc instruction to externally generated purges (ptc.q, ptc.ga) must occur before subsequent memory operations. From a software perspective, this is similar to acquire semantics. Serialization is still required to observe the side-effects of a translation being present.

itc must be the last instruction in an instruction group; otherwise, its behavior (including its ordering semantics) is undefined.

The TLB is first purged of any overlapping entries as specified by Table 4-1 on page 3:43 in Volume 2: System Architecture.

This instruction can only be executed at the most privileged level, and when PSR.ic is zero. To ensure forward progress, software must ensure that PSR.ic remains 0 until rfi-ing to the instruction that requires the translation.

Operation:

```c
if (PR[qp]) {
    if (!followed_by_stop())
        undefined_behavior();
    if (PSR.ic)
        illegal_operation_fault();
    if (PSR.cpl != 0)
        privileged_operation_fault(0);
    if (GR[r2].nat)
        register_nat_consumption_fault(0);

    tmp_size = CR[ITIR].ps;
    tmp_va = CR[IFA]{60:0};
    tmp_rid = RR[CR[IFA]{63:61}].rid;
    tmp_va = align_to_size_boundary(tmp_va, tmp_size);

    if (is_reserved_field(TLB_TYPE, GR[r2], CR[ITIR]))
        reserved_register_field_fault();
    if (unimplemented_virtual_address(CR[IFA]))
        unimplemented_data_address_fault(0);

    if (instruction_form) {
        tlb_must_purge_itc_entries(tmp_rid, tmp_va, tmp_size);
        tlb_may_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);
        slot = tlb_replacement_algorithm(ITC_TYPE);
        tlb_insert_inst(slot, GR[r2], CR[ITIR], CR[IFA], tmp_rid, TC);
    } else { // data_form
        tlb_must_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);
        tlb_may_purge_itc_entries(tmp_rid, tmp_va, tmp_size);
        slot = tlb_replacement_algorithm(DTC_TYPE);
        tlb_insert_data(slot, GR[r2], CR[ITIR], CR[IFA], tmp_rid, TC);
    }
```
**Interruptions:**
- Machine Check abort
- Privileged Operation fault
- Register NaT Consumption fault
- Reserved Register/Field fault
- Unimplemented Data Address fault

**Serialization:**
For the instruction_form, software must issue an instruction serialization operation before a dependent instruction fetch access. For the data_form, software must issue a data serialization operation before issuing a data access or non-access reference dependent on the new translation.
Insert Translation Register

**Format:**

\[(qp) \quad \text{itr} \quad \text{itr}[r_3] = r_2 \quad \text{instruction_form} \quad M42\]
\[(qp) \quad \text{itr.d} \quad \text{dtr}[r_3] = r_2 \quad \text{data_form} \quad M42\]

**Description:**

A translation is inserted into the instruction or data translation register specified by the contents of \(GR[r_3]\). \(GR[r_2]\) specifies the physical address portion of the translation. ITIR specifies the protection key, page size and additional information. The virtual address is specified by the IFA register and the region register is selected by IFA\{63:61\}.

As described in Table 4-1 on page 3:43 in *Volume 2: System Architecture*, the TLB is first purged of any entries that overlap with the newly inserted translation. The translation previously contained in the TR slot specified by \(GR[r_3]\) is not purged from the processor’s TLBs. To remove previous TR translations, software must use explicit ptr instructions.

This instruction can only be executed at the most privileged level, and when PSR.ic is zero.

**Operation:**

\[
\text{if } (\text{PR}[qp]) \{ \\
\quad \text{if } (\text{PSR.ic}) \quad \text{illegal_operation_fault();} \\
\quad \text{if } (\text{PSR.cpl} \neq 0) \quad \text{privileged_operation_fault(0);} \\
\quad \text{if } (\text{GR}[r_3].\text{nat} || \text{GR}[r_2].\text{nat}) \quad \text{register_nat_consumption_fault(0);} \\
\quad \text{slot} = \text{GR}[r_3]{7:0}; \\
\quad \text{tmp_size} = \text{CR}[\text{ITIR}].\text{ps}; \\
\quad \text{tmp_va} = \text{CR}[\text{IFA}]{60:0}; \\
\quad \text{tmp_rid} = \text{RR}[\text{CR}[\text{IFA}]{63:61}].\text{rid}; \\
\quad \text{tmp_va} = \text{align_to_size_boundary}(\text{tmp_va}, \text{tmp_size}); \\
\quad \text{tmp_tr_type} = \text{instruction_form} ? \text{ITR_TYPE} : \text{DTR_TYPE}; \\
\quad \text{if } (\text{is_reserved_reg(tmp_tr_type, slot)}) \quad \text{reserved_register_field_fault();} \\
\quad \text{if } (\text{is_reserved_field(TLB_TYPE, \text{GR}[r_2], \text{CR}[\text{ITIR}])) \quad \text{reserved_register_field_fault();} \\
\quad \text{if } (\text{unimplemented_virtual_address(CR[IFA]})) \quad \text{unimplemented_data_address_fault(0);} \\
\quad \text{if } (\text{instruction_form}) \{ \\
\quad \quad \text{tlb_must_purge_itc_entries(tmp_rid, tmp_va, tmp_size);} \\
\quad \quad \text{tlb_may_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);} \\
\quad \quad \text{tlb_insert_inst(slot, \text{GR}[r_2], \text{CR}[\text{ITIR}], \text{CR}[\text{IFA}], tmp_rid, TR);} \\
\quad \} \quad \text{// data_form} \\
\quad \text{tlb_must_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);} \\
\quad \text{tlb_may_purge_itc_entries(tmp_rid, tmp_va, tmp_size);} \\
\quad \text{tlb_insert_data(slot, \text{GR}[r_2], \text{CR}[\text{ITIR}], \text{CR}[\text{IFA}], tmp_rid, TR);} \\
\} \]

**Interruptions:**

Machine Check abort

Reserved Register/Field fault

Privileged Operation fault

Unimplemented Data Address fault

Register NaT Consumption fault

**Serialization:**

For the instruction_form, software must issue an instruction serialization operation before a dependent instruction fetch access. For the data_form, software must issue a data serialization operation before issuing a data access or non-access reference dependent on the new translation.
Notes: The processor may use invalid translation registers for translation cache entries. Performance can be improved on some processor models by ensuring translation registers are allocated beginning at translation register zero and continuing contiguously upwards.
Load

Format:

\[(qp) \text{ldsz.}ldtype.\text{ldhint } r_1 = [r_3] \text{ no_base_update_form M1 }\]
\[(qp) \text{ldsz.}ldtype.\text{ldhint } r_1 = [r_3], r_2 \text{ reg_base_update_form M2 }\]
\[(qp) \text{ldsz.}ldtype.\text{ldhint } r_1 = [r_3], \text{imm}_9 \text{ imm_base_update_form M3 }\]
\[(qp) \text{ld8.fill.ldhint } r_1 = [r_2] \text{ fill_form, no_base_update_form M1 }\]
\[(qp) \text{ld8.fill.ldhint } r_1 = [r_3], r_2 \text{ fill_form, reg_base_update_form M2 }\]
\[(qp) \text{ld8.fill.ldhint } r_1 = [r_3], \text{imm}_9 \text{ fill_form, imm_base_update_form M3 }\]

Description:
A value consisting of sz bytes is read from memory starting at the address specified by the value in GR \(r_3\). The value is then zero extended and placed in GR \(r_1\). The values of the \(sz\) completer are given in Table 2-31. The NaT bit corresponding to GR \(r_1\) is cleared, except as described below for speculative loads. The \(ldtype\) completer specifies special load operations, which are described in Table 2-32.

For the fill_form, an 8-byte value is loaded, and a bit in the UNAT application register is copied into the target register NaT bit. This instruction is used for reloading a spilled register/NaT pair. See “Control Speculation” on page 1:53 in Volume 1: Application Architecture for details.

In the base update forms, the value in GR \(r_3\) is added to either a signed immediate value (\(\text{imm}_9\)) or a value from GR \(r_2\), and the result is placed back in GR \(r_3\). This base register update is done after the load, and does not affect the load address. In the reg_base_update_form, if the NaT bit corresponding to GR \(r_2\) is set, then the NaT bit corresponding to GR \(r_3\) is set and no fault is raised.

Table 2-31. \(sz\) Completers

<table>
<thead>
<tr>
<th>(sz) Completer</th>
<th>Bytes Accessed</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>1 byte</td>
</tr>
<tr>
<td>2</td>
<td>2 bytes</td>
</tr>
<tr>
<td>4</td>
<td>4 bytes</td>
</tr>
<tr>
<td>8</td>
<td>8 bytes</td>
</tr>
</tbody>
</table>

Table 2-32. Load Types

<table>
<thead>
<tr>
<th>(ldtype) Completer</th>
<th>Interpretation</th>
<th>Special Load Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Normal load</td>
<td>Certain exceptions may be deferred rather than generating a fault. Deferral causes the target register’s NaT bit to be set. The NaT bit is later used to detect deferral.</td>
</tr>
<tr>
<td>s</td>
<td>Speculative load</td>
<td>An entry is added to the ALAT. This allows later instructions to check for colliding stores. If the referenced data page has a non-speculative attribute, the target register and NaT bit is cleared, and the processor ensures that no ALAT entry exists for the target register. The absence of an ALAT entry is later used to detect deferral or collision.</td>
</tr>
<tr>
<td>a</td>
<td>Advanced load</td>
<td>An entry is added to the ALAT, and certain exceptions may be deferred. Deferral causes the target register’s NaT bit to be set, and the processor ensures that no ALAT entry exists for the target register. The absence of an ALAT entry is later used to detect deferral or collision.</td>
</tr>
<tr>
<td>sa</td>
<td>Speculative Advanced load</td>
<td>An entry is added to the ALAT, and certain exceptions may be deferred. Deferral causes the target register’s NaT bit to be set, and the processor ensures that no ALAT entry exists for the target register. The absence of an ALAT entry is later used to detect deferral or collision.</td>
</tr>
<tr>
<td>c.nc</td>
<td>Check load - no clear</td>
<td>The ALAT is searched for a matching entry. If found, no load is done and the target register is unchanged. Regardless of ALAT hit or miss, base register updates are performed, if specified. An implementation may optionally cause the ALAT lookup to fail independent of whether an ALAT entry matches. If not found, a load is performed, and an entry is added to the ALAT (unless the referenced data page has a non-speculative attribute, in which case no ALAT entry is allocated).</td>
</tr>
</tbody>
</table>

For the non-speculative load types, if NaT bit associated with GR $r_3$ is 1, a Register NaT Consumption fault is taken. For speculative and speculative advanced loads, no fault is raised, and the exception is deferred. For the base-update calculation, if the NaT bit associated with GR $r_2$ is 1, the NaT bit associated with GR $r_3$ is set to 1 and no fault is raised.

The value of the $ldhint$ completer specifies the locality of the memory access. The values of the $ldhint$ completer are given in Table 2-33. A prefetch hint is implied in the base update forms. The address specified by the value in GR $r_3$ after the base update acts as a hint to prefetch the indicated cache line. This prefetch uses the locality hints specified by $ldhint$. Prefetch and locality hints do not affect program functionality and may be ignored by the implementation. See “Memory Hierarchy Control and Consistency” on page 1:62 in Volume 1: Application Architecture for details.

<table>
<thead>
<tr>
<th>$ldtype$ Completter</th>
<th>Interpretation</th>
<th>Special Load Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>c.clr</td>
<td>Check load</td>
<td>The ALAT is searched for a matching entry. If found, the entry is removed, no load is done and the target register is unchanged. Regardless of ALAT hit or miss, base register updates are performed, if specified. An implementation may optionally cause the ALAT lookup to fail independent of whether an ALAT entry matches. If not found, a clear check load behaves like a normal load.</td>
</tr>
<tr>
<td>c.clr.acq</td>
<td>Ordered check load</td>
<td>This type behaves the same as the unordered clear form, except that the ALAT lookup (and resulting load, if no ALAT entry is found) is performed with acquire semantics.</td>
</tr>
<tr>
<td>acq</td>
<td>Ordered load</td>
<td>An ordered load is performed with acquire semantics.</td>
</tr>
<tr>
<td>bias</td>
<td>Biased load</td>
<td>A hint is provided to the implementation to acquire exclusive ownership of the accessed cache line.</td>
</tr>
</tbody>
</table>

Table 2-33. Load Hints

<table>
<thead>
<tr>
<th>$ldhint$ Completter</th>
<th>Interpretation</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Temporal locality, level 1</td>
</tr>
<tr>
<td>nt1</td>
<td>No temporal locality, level 1</td>
</tr>
<tr>
<td>nta</td>
<td>No temporal locality, all levels</td>
</tr>
</tbody>
</table>

In the no_base_update form, the value in GR $r_3$ is not modified and no prefetch hint is implied.

For the base update forms, specifying the same register address in $r_1$ and $r_3$ will cause an Illegal Operation fault.
Operation:  
if (PR[qp]) {
    size = fill_form ? 8 : sz;
    speculative = (ldtype == 's' || ldtype == 'sa');
    advanced = (ldtype == 'a' || ldtype == 'sa');
    check_clear = (ldtype == 'c.clr' || ldtype == 'c.clr.acq');
    check_no_clear = (ldtype == 'c.nc');
    check = check_clear || check_no_clear;
    acquire = (ldtype == 'acq' || ldtype == 'c.clr.acq');
    bias = (ldtype == 'bias') ? BIAS : 0;
    itype = READ;
    if (speculative) itype |= SPEC;
    if (advanced) itype |= ADVANCE;

    if ((reg_base_update_form || imm_base_update_form) && (r1 == r3))
        illegal_operation_fault();
    check_target_register(r1);
    if (reg_base_update_form || imm_base_update_form)
        check_target_register(r3);

    if (reg_base_update_form) {
        tmp_r2 = GR[r2];
        tmp_r2nat = GR[r2].nat;
    }

    if (!speculative && GR[r3].nat)  // fault on NaT address
        register_nat_consumption_fault(itype);
    defer = speculative && (GR[r3].nat || PSR.ed);  // defer exception if spec

    if (check && alat_cmp(GENERAL, r1)) {  // no load on ld.c & ALAT hit
        if (check_clear)  // remove entry on ld.c.clr or ld.c.clr.acq
            alat_inval_single_entry(GENERAL, r1);
    } else {
        if (!defer) {
            paddr = tlb_translate(GR[r3], size, itype, PSR.cpl, &mattr, &defer);
            if (!defer) {
                otype = acquire ? ACQUIRE : UNORDERED;
                val = mem_read(paddr, size, UM.be, mattr, otype, bias | ldhint);
            }
        }
        if (check_clear || advanced)  // remove any old ALAT entry
            alat_inval_single_entry(GENERAL, r1);
        if (defer) {
            if (speculative) {
                GR[r1] = natd_gr_read(paddr, size, UM.be, mattr, otype, bias | ldhint);
                GR[r1].nat = 1;
            } else {
                GR[r1] = 0;  // ld.a to sequential memory
                GR[r1].nat = 0;
            }
        } else {  // execute load normally
            if (fill_form) {  // fill NaT on ld8.fill
                bit_pos = GR[r3]{8:3};
                GR[r1] = val;
                GR[r1].nat = AR[UNAT]{bit_pos};
            } else {  // clear NaT on other types
                GR[r1] = zero_ext(val, size * 8);
                GR[r1].nat = 0;
            }
        }
    }
}

if ((check_no_clear || advanced) && ma_is_speculative(mattr))
  // add entry to ALAT
  alat_write(GENERAL, r1, paddr, size);
}

if (imm_base_update_form) { // update base register
  GR[r3] = GR[r3] + sign_ext(imm9, 9);
  GR[r3].nat = GR[r3].nat;
} else if (reg_base_update_form) {
  GR[r3] = GR[r3] + tmp_r2;
  GR[r3].nat = GR[r3].nat || tmp_r2nat;
}

if ((reg_base_update_form || imm_base_update_form) && !GR[r3].nat)
  mem_implicit_prefetch(GR[r3], ldhint | bias, itype);

**Interruptions:**

- Illegal Operation fault
- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Access Bit fault
- Data Debug fault
- Unaligned Data Reference fault
Floating-point Load

Format:

\[(qp) \text{ldf}\text{sz, fldtype, ldhint} \ f_1 = [r_3] \quad \text{no_base_update_form \ M6} \]
\[(qp) \text{ldf}\text{sz, fldtype, ldhint} \ f_1 = [r_3], r_2 \quad \text{reg_base_update_form \ M7} \]
\[(qp) \text{ldf}\text{sz, fldtype, ldhint} \ f_1 = [r_3], \text{imm}_9 \quad \text{imm_base_update_form \ M8} \]
\[(qp) \text{ldf}.f1.ldhint \ f_1 = [r_3] \quad \text{integer_form, no_base_update_form \ M6} \]
\[(qp) \text{ldf}.f1.ldhint \ f_1 = [r_3], r_2 \quad \text{integer_form, reg_base_update_form \ M7} \]
\[(qp) \text{ldf}.f1.ldhint \ f_1 = [r_3], \text{imm}_9 \quad \text{integer_form, imm_base_update_form \ M8} \]
\[(qp) \text{ldf.fill, ldhint} \ f_1 = [r_3] \quad \text{fill_form, no_base_update_form \ M6} \]
\[(qp) \text{ldf.fill, ldhint} \ f_1 = [r_3], r_2 \quad \text{fill_form, reg_base_update_form \ M7} \]
\[(qp) \text{ldf.fill, ldhint} \ f_1 = [r_3], \text{imm}_9 \quad \text{fill_form, imm_base_update_form \ M8} \]

Description:

A value consisting of \text{fsz} bytes is read from memory starting at the address specified by the value in \text{GR} r_3. The value is then converted into the floating-point register format and placed in \text{FR} f_1. See “Data Types and Formats” on page 1:77 in \text{Volume 1: Application Architecture} for details on conversion to floating-point register format. The values of the \text{fsz} completer are given in Table 2-34. The \text{fldtype} completer specifies special load operations, which are described in Table 2-35.

For the \text{integer_form}, an 8-byte value is loaded and placed in the significand field of \text{FR} f_1 without conversion. The exponent field of \text{FR} f_1 is set to the biased exponent for 2.0^{63} (0x1003E) and the sign field of \text{FR} f_1 is set to positive (0).

For the \text{fill_form}, a 16-byte value is loaded, and the appropriate fields are placed in \text{FR} f_1 without conversion. This instruction is used for reloading a spilled register. See “Control Speculation” on page 1:53 in \text{Volume 1: Application Architecture} for details.

In the base update forms, the value in \text{GR} r_3 is added to either a signed immediate value (\text{imm}_9) or a value from \text{GR} r_2, and the result is placed back in \text{GR} r_3. This base register update is done after the load, and does not affect the load address. In the \text{reg_base_update_form}, if the NaT bit corresponding to \text{GR} r_2 is set, then the NaT bit corresponding to \text{GR} r_3 is set and no fault is raised.

### Table 2-34. \text{fsz} Completers

<table>
<thead>
<tr>
<th>\text{fsz Completer}</th>
<th>\text{Bytes Accessed}</th>
<th>\text{Memory Format}</th>
</tr>
</thead>
<tbody>
<tr>
<td>s</td>
<td>4 bytes</td>
<td>Single precision</td>
</tr>
<tr>
<td>d</td>
<td>8 bytes</td>
<td>Double precision</td>
</tr>
<tr>
<td>e</td>
<td>10 bytes</td>
<td>Extended precision</td>
</tr>
</tbody>
</table>

### Table 2-35. FP Load Types

<table>
<thead>
<tr>
<th>\text{fldtype Compler}</th>
<th>\text{Interpretation}</th>
<th>\text{Special Load Operation}</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Normal load</td>
<td></td>
</tr>
<tr>
<td>s</td>
<td>Speculative load</td>
<td>Certain exceptions may be deferred rather than generating a fault. Deferral causes NaTVal to be placed in the target register. The NaTVal value is later used to detect deferral.</td>
</tr>
<tr>
<td>a</td>
<td>Advanced load</td>
<td>An entry is added to the ALAT. This allows later instructions to check for colliding stores. If the referenced data page has a non-speculative attribute, no ALAT entry is added to the ALAT and the target register is set as follows: for the \text{integer_form}, the exponent is set to 0x1003E and the sign and significand are set to zero; for all other forms, the sign, exponent and significand are set to zero. The absence of an ALAT entry is later used to detect deferral or collision.</td>
</tr>
</tbody>
</table>
For more details on speculative, advanced and check loads see “Control Speculation” on page 1:53 and “Data Speculation” on page 1:57 in Volume 1: Application Architecture. Details on memory attributes are described in “Memory Attributes” on page 1:63 in Volume 2: System Architecture.

For the non-speculative load types, if NaT bit associated with GRr3 is 1, a Register NaT Consumption fault is taken. For speculative and speculative advanced loads, no fault is raised, and the exception is deferred. For the base-update calculation, if the NaT bit associated with GRr2 is 1, the NaT bit associated with GRr3 is set to 1 and no fault is raised.

The value of the ldhint modifier specifies the locality of the memory access. The mnemonic values of ldhint are given in Table 2-33 on page 3:136. A prefetch hint is implied in the base update forms. The address specified by the value in GRr3 after the base update acts as a hint to prefetch the indicated cache line. This prefetch uses the locality hints specified by ldhint. Prefetch and locality hints do not affect program functionality and may be ignored by the implementation. See “Memory Hierarchy Control and Consistency” on page 1:62 in Volume 1: Application Architecture for details.

In the no_base_update form, the value in GRr3 is not modified and no prefetch hint is implied.

The PSR.mfl and PSR.mfh bits are updated to reflect the modification of FR f1.

Hardware support for ldfe (10-byte) instructions that reference a page that is neither a cacheable page with write-back policy nor a NaTPage is optional. On processor models that do not support such ldfe accesses, an Unsupported Data Reference fault is raised when an unsupported reference is attempted. The fault is delivered only on the normal, advanced, and check load flavors. Control-speculative flavors of ldfe always defer the Unsupported Data Reference fault.
Operation:

```c
if (PR[qp]) {
    size = (fill_form ? 16 : (integer_form ? 8 : fsz));
    speculative = (fldtype == 's' || fldtype == 'sa');
    advanced = (fldtype == 'a' || fldtype == 'sa');
    check_clear = (fldtype == 'c.clr');
    check_no_clear = (fldtype == 'c.nc');
    check = check_clear || check_no_clear;

    itype = READ;
    if (speculative) itype |= SPEC;
    if (advanced) itype |= ADVANCE;

    if (reg_base_update_form || imm_base_update_form)
        check_target_register(r3);
    fp_check_target_register(f1);
    if (tmp_isrcode = fp_reg_disabled(f1, 0, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, itype);

    if (!speculative && GR[r3].nat) // fault on NaT address
        register_nat_consumption_fault(itype);

    defer = speculative && (GR[r3].nat || PSR.ed); // defer exception if spec

    if (check && alat_cmp(FLOAT, f1)) { // no load on ldf.c & ALAT hit
        if (check_clear) // remove entry on ldf.c.clr
            alat_inval_single_entry(FLOAT, f1);
    } else {
        if (!defer) {
            paddr = tlb_translate(GR[r3], size, itype, PSR.cpl, &mattr,
                                  &defer);
            if (!defer)
                val = mem_read(paddr, size, UM.be, mattr, UNORDERED, ldhint);
        }
    }

    if (check_clear || advanced) // remove any old ALAT entry
        alat_inval_single_entry(FLOAT, f1);
    if (speculative && defer) {
        FR[f1] = NATVAL;
    } else if (advanced && !speculative && defer) {
        FR[f1] = (integer_form ? FP_INT_ZERO : FP_ZERO);
    } else {
        // execute load normally
        FR[f1] = fp_mem_to_fr_format(val, size, integer_form);
        if ((check_no_clear || advanced) && ma_is_speculative(mattr))
            // add entry to ALAT
            alat_write(FLOAT, f1, paddr, size);
    }
}

if (imm_base_update_form) { // update base register
    GR[r3] = GR[r3] + sign_ext(imm9, 9);
    GR[r3].nat = GR[r3].nat;
} else if (reg_base_update_form) {
    GR[r3] = GR[r3] + GR[r2];
    GR[r3].nat = GR[r3].nat || GR[r2].nat;
}

if (((reg_base_update_form || imm_base_update_form) && !GR[r3].nat)
    mem_implicit_prefetch(GR[r3], ldhint, itype);
fp_update_psr(f1);
}
```
Interruptions:  Illegal Operation fault
              Disabled Floating-point Register fault
              Register NaT Consumption fault
              Unimplemented Data Address fault
              Data Nested TLB fault
              Alternate Data TLB fault
              VHPT Data fault
              Data TLB fault
              Data Page Not Present fault

              Data NaT Page Consumption fault
              Data Key Miss fault
              Data Key Permission fault
              Data Access Rights fault
              Data Access Bit fault
              Data Debug fault
              Unaligned Data Reference fault
              Unsupported Data Reference fault
Floating-point Load Pair

Format:

\[(qp) \text{ldfps.fldtype.ldhint } f_1, f_2 = [r_3] \quad \text{single_form, no_base_update_form} \quad M11\]
\[(qp) \text{ldfps.fldtype.ldhint } f_1, f_2 = [r_3], 8 \quad \text{single_form, base_update_form} \quad M12\]
\[(qp) \text{ldfpd.fldtype.ldhint } f_1, f_2 = [r_3] \quad \text{double_form, no_base_update_form} \quad M11\]
\[(qp) \text{ldfpd.fldtype.ldhint } f_1, f_2 = [r_3], 16 \quad \text{double_form, base_update_form} \quad M12\]
\[(qp) \text{ldfp8.fldtype.ldhint } f_1, f_2 = [r_3] \quad \text{integer_form, no_base_update_form} \quad M11\]
\[(qp) \text{ldfp8.fldtype.ldhint } f_1, f_2 = [r_3], 16 \quad \text{integer_form, base_update_form} \quad M12\]

Description:

Eight (single_form) or sixteen (double_form/integer_form) bytes are read from memory starting at the address specified by the value in GR \[r_3\]. The value read is treated as a contiguous pair of floating-point numbers for the single_form/double_form and as integer/Parallel FP data for the integer_form. Each number is converted into the floating-point register format. The value at the lowest address is placed in FR \[f_1\], and the value at the highest address is placed in FR \[f_2\]. See “Data Types and Formats” on page 1:77 in Volume 1: Application Architecture for details on conversion to floating-point register format. The \text{fldtype} completer specifies special load operations, which are described in Table 2-35 on page 3:139.

For more details on speculative, advanced and check loads see “Control Speculation” on page 1:53 and “Data Speculation” on page 1:57 in Volume 1: Application Architecture.

For the non-speculative load types, if NaT bit associated with GR \[r_3\] is 1, a Register NaT Consumption fault is taken. For speculative and speculative advanced loads, no fault is raised, and the exception is deferred.

In the base_update_form, the value in GR \[r_3\] is added to an implied immediate value (equal to double the data size) and the result is placed back in GR \[r_3\]. This base register update is done after the load, and does not affect the load address.

The value of the \text{ldhint} modifier specifies the locality of the memory access. The mnemonic values of \text{ldhint} are given in Table 2-33 on page 3:136. A prefetch hint is implied in the base update form. The address specified by the value in GR \[r_3\] after the base update acts as a hint to prefetch the indicated cache line. This prefetch uses the locality hints specified by \text{ldhint}. Prefetch and locality hints do not affect program functionality and may be ignored by the implementation. See “Memory Hierarchy Control and Consistency” on page 1:62 in Volume 1: Application Architecture for details.

In the no_base_update form, the value in GR \[r_3\] is not modified and no prefetch hint is implied.

The PSR.mfl and PSR.mfh bits are updated to reflect the modification of FR \[f_1\] and FR \[f_2\].

There is a restriction on the choice of target registers. Register specifiers \(f_1\) and \(f_2\) must specify one odd-numbered physical FR and one even-numbered physical FR. Specifying two odd or two even registers will cause an Illegal Operation fault to be raised. The restriction is on physical register numbers after register rotation. This means that if \(f_1\) and \(f_2\) both specify static registers or both specify rotating registers, then \(f_1\) and \(f_2\) must be odd/even or even/odd. If \(f_1\) and \(f_2\) specify one static and one rotating register, the restriction depends on CFM.rrb.fr. If CFM.rrb.fr is even, the restriction is the same; \(f_1\) and \(f_2\) must be odd/even or even/odd. If CFM.rrb.fr is odd, then \(f_1\) and \(f_2\) must be even/even or odd/odd. Specifying one static and one rotating register should only be done when CFM.rrb.fr will have a predictable value (such as 0).
Operation:

if (PR[gp]) {
    size = single_form ? 8 : 16;
    speculative = (fldtype == 's' || fldtype == 'sa');
    advanced = (fldtype == 'a' || fldtype == 'sa');
    check_clear = (fldtype == 'c.clr');
    check_no_clear = (fldtype == 'c.nc');
    check = check_clear || check_no_clear;

    itype = READ;
    if (speculative) itype |= SPEC;
    if (advanced) itype |= ADVANCE;

    if (fp_reg_bank_conflict(f1, f2))
        illegal_operation_fault();

    if (base_update_form)
        check_target_register(r3);

    fp_check_target_register(f1);
    fp_check_target_register(f2);
    if (tmp_isrcode = fp_reg_disabled(f1, f2, 0, 0))
        disabled_fp_register_fault(tmp_isrcode, itype);

    if (!speculative && GR[r3].nat) // fault on NaT address
        register_nat_consumption_fault(itype);

    defer = speculative && (GR[r3].nat || PSR.ed); // defer exception if spec

    if (check && alat_cmp(FLOAT, f1)) { // no load on ldfp.c & ALAT hit
        if (check_clear) // remove entry on ldfp.c.clr
            alat_inval_single_entry(FLOAT, f1);
        else {
            if (!defer) {
                paddr = tlb_translate(GR[r3], size, itype, PSR.cpl, &mattr, &defer);
                if (!defer)
                    val = mem_read(paddr, size, UM.be, mattr, UNORDERED, ldhint);
            }

            if (check_clear || advanced) // remove any old ALAT entry
                alat_inval_single_entry(FLOAT, f1);

            if (speculative && defer) {
                FR[f1] = NATVAL;
                FR[f2] = NATVAL;
            } else if (advanced && !speculative && defer) {
                FR[f1] = (integer_form ? FP_INT_ZERO : FP_ZERO);
                FR[f2] = (integer_form ? FP_INT_ZERO : FP_ZERO);
            } else { // execute load normally
                if (UM.be) {
                    FR[f1] = fp_mem_to_fr_format(val u>> (size/2*8), size/2, integer_form);
                    FR[f2] = fp_mem_to_fr_format(val, size/2, integer_form);
                } else {
                    FR[f1] = fp_mem_to_fr_format(val, size/2, integer_form);
                    FR[f2] = fp_mem_to_fr_format(val u>> (size/2*8), size/2, integer_form);
                }

            }

            if ((check_no_clear || advanced) && ma_is_speculative(mattr))
                // add entry to ALAT
                alat_write(FLOAT, f1, paddr, size);
        }
    } else {
        if (!defer) {
            if (check_clear) // remove entry on ldfp.c.clr
                alat_inval_single_entry(FLOAT, f1);
            else {
                if (!defer) {
                    paddr = tlb_translate(GR[r3], size, itype, PSR.cpl, &mattr, &defer);
                    if (!defer)
                        val = mem_read(paddr, size, UM.be, mattr, UNORDERED, ldhint);
                }

            }

            if (check_clear || advanced) // remove any old ALAT entry
                alat_inval_single_entry(FLOAT, f1);

            if (speculative && defer) {
                FR[f1] = NATVAL;
                FR[f2] = NATVAL;
            } else if (advanced && !speculative && defer) {
                FR[f1] = (integer_form ? FP_INT_ZERO : FP_ZERO);
                FR[f2] = (integer_form ? FP_INT_ZERO : FP_ZERO);
            } else { // execute load normally
                if (UM.be) {
                    FR[f1] = fp_mem_to_fr_format(val u>> (size/2*8), size/2, integer_form);
                    FR[f2] = fp_mem_to_fr_format(val, size/2, integer_form);
                } else {
                    FR[f1] = fp_mem_to_fr_format(val, size/2, integer_form);
                    FR[f2] = fp_mem_to_fr_format(val u>> (size/2*8), size/2, integer_form);
                }

            }

            if ((check_no_clear || advanced) && ma_is_speculative(mattr))
                // add entry to ALAT
                alat_write(FLOAT, f1, paddr, size);
        }
    }
}
if (base_update_form) {
    // update base register
    GR[r3] = GR[r3] + size;
    GR[r3].nat = GR[r3].nat;
    if (!GR[r3].nat)
      mem_implicit_prefetch(GR[r3], ldhint, itype);

    fp_update_psr(f1);
    fp_update_psr(f2);
}

**Interruptions:**

- Illegal Operation fault
- Disabled Floating-point Register fault
- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Access Bit fault
- Data Debug fault
- Unaligned Data Reference fault
**Line Prefetch**

**Format:**

- \((qp) lfetch.lftype.lfhint [r_3]\)  
  no_base_update_form \(M_{13}\)
- \((qp) lfetch.lftype.lfhint [r_3], r_2\)  
  reg_base_update_form \(M_{14}\)
- \((qp) lfetch.lftype.lfhint [r_3], \text{imm}_9\)  
  imm_base_update_form \(M_{15}\)
- \((qp) lfetch.lftype.excl.lfhint [r_3]\)  
  no_base_update_form, exclusive_form \(M_{13}\)
- \((qp) lfetch.lftype.excl.lfhint [r_3], r_2\)  
  reg_base_update_form, exclusive_form \(M_{14}\)
- \((qp) lfetch.lftype.excl.lfhint [r_3], \text{imm}_9\)  
  imm_base_update_form, exclusive_form \(M_{15}\)

**Description:**

The line containing the address specified by the value in GR \(r_3\) is moved to the highest level of the data memory hierarchy. The value of the \(lfhint\) modifier specifies the locality of the memory access; see Section 4.4, “Memory Access Instructions” on page 1:51 in *Volume 2: System Architecture* for details. The mnemonic values of \(lfhint\) are given in Table 2-37.

The behavior of the memory read is also determined by the memory attribute associated with the accessed page. See Chapter 4, “Addressing and Protection” in *Volume 2: System Architecture*. Line size is implementation dependent but must be a power of two greater than or equal to 32 bytes. In the exclusive form, the cache line is allowed to be marked in an exclusive state. This qualifier is used when the program expects soon to modify a location in that line. If the memory attribute for the page containing the line is not cacheable, then no reference is made.

The completer, \(lftype\), specifies whether or not the instruction raises faults normally associated with a regular load. Table 2-36 defines these two options.

**Table 2-36. lftype Mnemonic Values**

<table>
<thead>
<tr>
<th>(lftype) Mnemonic</th>
<th>Interpretation</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Ignore faults</td>
</tr>
<tr>
<td>fault</td>
<td>Raise faults</td>
</tr>
</tbody>
</table>

In the base update forms, after being used to address memory, the value in GR \(r_3\) is incremented by either the sign-extended value in \(\text{imm}_9\) (in the \(\text{imm}_9\) base update form) or the value in GR \(r_2\) (in the reg base update form). In the reg_base_update_form, if the NaT bit corresponding to GR \(r_2\) is set, then the NaT bit corresponding to GR \(r_3\) is set – no fault is raised.

In the reg_base_update_form and the \(\text{imm}_9\) base update form, if the NaT bit corresponding to GR \(r_3\) is clear, then the address specified by the value in GR \(r_3\) after the post-increment acts as a hint to implicitly prefetch the indicated cache line. This implicit prefetch uses the locality hints specified by \(lfhint\). The implicit prefetch does not affect program functionality, does not raise any faults, and may be ignored by the implementation.

In the no_base_update_form, the value in GR \(r_3\) is not modified and no implicit prefetch hint is implied.

If the NaT bit corresponding to GR \(r_3\) is set then the state of memory is not affected. In the reg_base_update_form and \(\text{imm}_9\) base update form, the post increment of GR \(r_3\) is performed and prefetch is hinted as described above.

\(lfetch\) instructions, like hardware prefetches, are not orderable operations, i.e., they have no order with respect to prior or subsequent memory operations.
A faulting `lfetch` to an unimplemented address results in an Unimplemented Data Address fault. A non-faulting `lfetch` to an unimplemented address does not take the fault and will not issue a prefetch request, but, if specified, will perform a register post-increment.

**Operation:**

```c
if (PR[qp]) {
  itype = READ|NON_ACCESS;
  itype |= (lftype == 'fault') ? LFETCH_FAULT : LFETCH;

  if (reg_base_update_form || imm_base_update_form)
    check_target_register(r3);

  if (lftype == 'fault') { // faulting form
    if (GR[r3].nat && !PSR.ed) // fault on NaT address
      register_nat_consumption_fault(itype);
  }

  excl_hint = (exclusive_form) ? EXCLUSIVE : 0;

  if (!GR[r3].nat && !PSR.ed) { // faulting form already faulted if r3 is nat
    paddr = tlb_translate(GR[r3], 1, itype, PSR.cpl, &mattr, &defer);
    if (!defer)
      mem_promote(paddr, mattr, lfhint | excl_hint);
  }

  if (imm_base_update_form) {
    GR[r3] = GR[r3] + sign_ext(imm9, 9);
    GR[r3].nat = GR[r3].nat;
  } else if (reg_base_update_form) {
    GR[r3] = GR[r3] + GR[r2];
    GR[r3].nat = GR[r2].nat | GR[r3].nat;
  }

  if ((reg_base_update_form || imm_base_update_form) && !GR[r3].nat)
    mem_implicit_prefetch(GR[r3], lfhint | excl_hint, itype);
}
```

**Interruptions:**

- Illegal Operation fault
- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Access Bit fault
- Data Debug fault
Load Register Stack

**Format:**
loadrs  

**Description:** This instruction ensures that a specified number of bytes (registers values and/or NaT collections) below the current BSP have been loaded from the backing store into the stacked general registers. The loaded registers are placed into the dirty partition of the register stack. All other stacked general registers are marked as invalid, without being saved to the backing store.

The number of bytes to be loaded is specified in a sub-field of the RSC application register (RSC.loadrs). Backing store addresses are always 8-byte aligned, and therefore the low order 3 bits of the loadrs field (RSC.loadrs[2:0]) are ignored. This instruction can be used to invalidate all stacked registers outside the current frame, by setting RSC.loadrs to zero.

This instruction will fault with an Illegal Operation fault under any of the following conditions:

- the RSE is not in enforced lazy mode (RSC.mode is non-zero).
- CFM.sof and RSC.loadrs are both non-zero.
- an attempt is made to load up more registers than are available in the physical stacked register file.

This instruction must be the first instruction in an instruction group and must either be in instruction slot 0 or in instruction slot 1 of a template having a stop after slot 0; otherwise, the results are undefined. This instruction cannot be predicated.

**Operation:**

```c
if (AR[RSC].mode != 0)
    illegal_operation_fault();

if ((CFM.sof != 0) && (AR[RSC].loadrs != 0))
    illegal_operation_fault();

rse_ensure_regs_loaded(AR[RSC].loadrs); // can raise faults listed below
AR[RNAT] = undefined();
```

**Interruptions:**

- Illegal Operation fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Access Bit fault
- Data Debug fault
Memory Fence

Format:  
(qp) mf ordering_form M24  
(qp) mf.a acceptance_form M24

Description: This instruction forces ordering between prior and subsequent memory accesses. The ordering_form ensures all prior data memory accesses are made visible prior to any subsequent data memory accesses being made visible. It does not ensure prior data memory references have been accepted by the external platform, nor that prior data memory references are visible.

The acceptance_form prevents any subsequent data memory accesses by the processor from initiating transactions to the external platform until:

- all prior loads to sequential pages have returned data, and
- all prior stores to sequential pages have been accepted by the external platform.

The definition of “acceptance” is platform dependent. The acceptance_form is typically used to ensure the processor has “waited” until a memory-mapped I/O transaction has been “accepted”, before initiating additional external transactions. The acceptance_form does not ensure ordering, or acceptance to memory areas other than sequential pages.

Operation:  
if (PR[qp]){  
    if (acceptance_form)  
        acceptance_fence();  
    else // ordering_form  
        ordering_fence();  
}

Interruptions: None
Mix

**Format:**

- `(qp) mix1` \( r_1 = r_2, r_3 \) one_byte_form, left_form 12
- `(qp) mix2` \( r_1 = r_2, r_3 \) two_byte_form, left_form 12
- `(qp) mix4` \( r_1 = r_2, r_3 \) four_byte_form, left_form 12
- `(qp) mix1.r` \( r_1 = r_2, r_3 \) one_byte_form, right_form 12
- `(qp) mix2.r` \( r_1 = r_2, r_3 \) two_byte_form, right_form 12
- `(qp) mix4.r` \( r_1 = r_2, r_3 \) four_byte_form, right_form 12

**Description:** The data elements of GR \( r_2 \) and \( r_3 \) are mixed as shown in Figure 2-25, and the result placed in GR \( r_1 \). The data elements in the source registers are grouped in pairs, and one element from each pair is selected for the result. In the left_form, the result is formed from the leftmost elements from each of the pairs. In the right_form, the result is formed from the rightmost elements. Elements are selected alternately from the two source registers.
Figure 2-25. Mix Example
Operation:

```c
if (PR[qp]) {
    check_target_register(r);

    if (one_byte_form) {
        // one-byte elements
        x[0] = GR[r2]{7:0}; y[0] = GR[r3]{7:0};

        if (left_form)
            GR[r1] = concatenate8(x[7], y[7], x[5], y[5],
                                     x[3], y[3], x[1], y[1]);
        else // right_form
            GR[r1] = concatenate8(x[6], y[6], x[4], y[4],
                                     x[2], y[2], x[0], y[0]);
    } else if (two_byte_form) {
        // two-byte elements
        x[0] = GR[r2]{15:0}; y[0] = GR[r3]{15:0};

        if (left_form)
            GR[r1] = concatenate4(x[3], y[3], x[1], y[1]);
        else // right_form
            GR[r1] = concatenate4(x[2], y[2], x[0], y[0]);
    } else {
        // four-byte elements
        x[0] = GR[r2]{31:0}; y[0] = GR[r3]{31:0};

        if (left_form)
            GR[r1] = concatenate2(x[1], y[1]);
        else // right_form
            GR[r1] = concatenate2(x[0], y[0]);
    }

    GR[r1].nat = GR[r2].nat || GR[r3].nat;
}
```

Interruptions: Illegal Operation fault
Move Application Register

Format:

- \((qp)\) mov \(r_1 = ar_3\)  
- \((qp)\) mov \(ar_3 = r_2\)  
- \((qp)\) mov \(ar_3 = imm_8\)
- \((qp)\) mov.i \(r_1 = ar_3\)
- \((qp)\) mov.i \(ar_3 = r_2\)
- \((qp)\) mov.i \(ar_3 = imm_8\)
- \((qp)\) mov.m \(r_1 = ar_3\)
- \((qp)\) mov.m \(ar_3 = r_2\)
- \((qp)\) mov.m \(ar_3 = imm_8\)

Description: The source operand is copied to the destination register.

In the from_form, the application register specified by \(ar_3\) is copied into GR \(r_1\) and the corresponding NaT bit is cleared.

In the to_form, the value in GR \(r_2\) (in the register_form), or the sign-extended value in \(imm_8\) (in the immediate_form), is placed in AR \(ar_3\). In the register_form if the NaT bit corresponding to GR \(r_2\) is set, then a Register NaT Consumption fault is raised.

Only a subset of the application registers can be accessed by each execution unit (M or I). Table 3-3 on page 3:24 in Volume 1: Application Architecture indicates which application registers may be accessed from which execution unit type. An access to an application register from the wrong unit type causes an Illegal Operation fault.

This instruction has multiple forms with the pseudo operation eliminating the need for specifying the execution unit. Accesses of the ARs are always implicitly serialized. While implicitly serialized, read-after-write and write-after-write dependency violations must be avoided (e.g., setting CCV, followed by cmpxchg in the same instruction group, or simultaneous writes to the UNAT register by ld.fill and mov to UNAT).
mov ar

Operation:

if (PR[qp]) {
    tmp_type = (i_form ? AR_I_TYPE : AR_M_TYPE);
    if (is_reserved_reg(tmp_type, ar3))
        illegal_operation_fault();

    if (from_form) {
        check_target_register(r1);
        if (((ar3 == BSPSTORE) || (ar3 == RNAT)) && (AR[RSC].mode != 0))
            illegal_operation_fault();
        if (ar3 == ITC && PSR.si && PSR.cpl != 0)
            privileged_register_fault();
        GR[rj] = (is_ignored_reg(ar3)) ? 0 : AR[ar3];
        GR[rj].nat = 0;
    } else { // to_form
        tmp_val = (register_form) ? GR[r2] : sign_ext(imm8, 8);

        if (is_read_only_register(AR_TYPE, ar3) ||
            (((ar3 == BSPSTORE) || (ar3 == RNAT)) && (AR[RSC].mode != 0)))
            illegal_operation_fault();
        if (register_form && GR[r2].nat)
            register_nat_consumption_fault(0);
        if (is_reserved_field(AR_TYPE, ar3, tmp_val))
            reserved_register_field_fault();
        if (((is_kernel_reg(ar3) || ar3 == ITC) && (PSR.cpl != 0))
            privileged_register_fault();
        if (!is_ignored_reg(ar3)) {
            tmp_val = ignored_field_mask(AR_TYPE, ar3, tmp_val);
            // check for illegal promotion
            if (ar3 == RSC && tmp_val{3:2} u< PSR.cpl)
                tmp_val{3:2} = PSR.cpl;
            AR[ar3] = tmp_val;

            if (ar3 == BSPSTORE) {
                AR[BSP] = rse_update_internal_stack_pointers(tmp_val);
                AR[RNAT] = undefined();
            }
        }
    }
}

Interruptions:

Illegal Operation fault
Reserved Register/Field fault
Register NaT Consumption fault
Privileged Register fault
Move Branch Register

**Format:**

- \((qp)\) mov \(r_1 = b_2\)  \(\text{from\_form} \ I22\)
- \((qp)\) mov \(b_1 = r_2\)  \(\text{pseudo-op} \ I21\)
- \((qp)\) mov.mwh.ih \(b_1 = r_2, \text{tag}_{j3}\)  \(\text{to\_form} \ I21\)
- \((qp)\) mov.ret.mwh.ih \(b_1 = r_2, \text{tag}_{j3}\)  \(\text{return\_form, to\_form} \ I21\)

**Description:** The source operand is copied to the destination register.

In the from_form, the branch register specified by \(b_2\) is copied into GR \(r_1\). The NaT bit corresponding to GR \(r_1\) is cleared.

In the to_form, the value in GR \(r_2\) is copied into BR \(b_1\). If the NaT bit corresponding to GR \(r_2\) is 1, then a Register NaT Consumption fault is taken.

A set of hints can also be provided when moving to a branch register. These hints are very similar to those provided on the \(brp\) instruction, and provide prediction information about a future branch which may use the value being moved into BR \(b_1\). The return_form is used to provide the hint that this value will be used in a return-type branch.

The values for the \(mwh\) whether hint completer are given in Table 2-38. For a description of the \(ih\) hint completer see the Branch Prediction instruction and Table 2-13 on page 3:29.

<table>
<thead>
<tr>
<th>(mwh) Completer</th>
<th>Move to BR Whether Hint</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Ignore all hints</td>
</tr>
<tr>
<td>spltk</td>
<td>Static Taken</td>
</tr>
<tr>
<td>dptk</td>
<td>Dynamic</td>
</tr>
</tbody>
</table>

A pseudo-op is provided for copying a general register into a branch register when there is no hint information to be specified. This is encoded with a value of 0 for \(\text{tag}_{j3}\) and values corresponding to \textit{none} for the hint completers.

**Operation:**

```c
if (PR[qp]) {
    if (from_form) {
        check_target_register(r_1);
        GR[r_1] = BR[b_2];
        GR[r_1].nat = 0;
    } else { // to_form
        tmp_tag = IP + sign_ext((timm_9 << 4), 13);
        if (GR[r_2].nat)
            register_nat_consumption_fault(0);
        BR[b_1] = GR[r_2];
        branch_predict(mwh, ih, return_form, GR[r_2], tmp_tag);
    }
}
```

**Interruptions:**

- Illegal Operation fault
- Register NaT Consumption fault
Move Control Register

Format:  

\[(qp) \text{mov} \; r_j = cr_3 \quad \text{from form} \quad \text{M33} \]

\[(qp) \text{mov} \; cr_3 = r_2 \quad \text{to form} \quad \text{M32} \]

Description:  
The source operand is copied to the destination register.

For the from form, the control register specified by \( cr_3 \) is read and the value copied into \( GR[r_1] \).

For the to form, \( GR[r_2] \) is read and the value copied into \( CR[cr_3] \).

Control registers can only be accessed at the most privileged level. Reading or writing an interruption control register (CR16-CR25), when the PSR.ic bit is one, will result in an Illegal Operation fault.

Operation:  

\[
\text{if (PR[qp])} \{ \\
\quad \text{if (is_reserved_reg(CR_TYPE, cr_3)} \\
\quad \quad \text{to_form \&\& is_read_only_reg(CR_TYPE, cr_3)} \\
\quad \quad \text{PSR.ic \&\& is_interruption_cr(cr_3)} \\
\quad \{ \\
\quad \quad \text{illegal_operation_fault();} \\
\quad \} \\
\text{if (from_form) \} \\
\quad \text{check_target_register(r_1);} \\
\text{if (PSR.cpl != 0) \} \\
\quad \text{privileged_operation_fault(0);} \\
\text{if (from_form) \{ \\
\quad \text{if (cr_3 == IVR)} \\
\quad \quad \text{check_interrupt_request();} \\
\quad \quad \text{if (cr_3 == ITIR)} \\
\quad \quad \quad \text{GR[r_1] = impl_itir_cwi_mask(CR[ITIR]);} \\
\quad \quad \text{else} \\
\quad \quad \quad \text{GR[r_1] = CR[cr_3];} \\
\quad \quad \text{GR[r_1].nat = 0;} \\
\quad \} \text{else { // to_form} \\
\quad \quad \text{if (GR[r_2].nat) \} \\
\quad \quad \quad \text{register_nat_consumption_fault(0);} \\
\quad \quad \quad \text{if (is_reserved_field(CR_TYPE, cr_3, GR[r_2])) \} \\
\quad \quad \quad \text{reserved_register_field_fault();} \\
\quad \quad \text{if (cr_3 == EOI)} \\
\quad \quad \quad \text{end_of_interrupt();} \\
\quad \text{tmp_val = ignored_field_mask(CR_TYPE, cr_3, GR[r_2]);} \\
\quad \text{CR[cr_3] = tmp_val;} \\
\quad \text{if (cr_3 == IIPA)} \\
\quad \quad \text{last_IP = tmp_val;} \\
\} \}
\]

Interruptions:  
Illegal Operation fault  
Register NaT Consumption fault  
Privileged Operation fault  
Reserved Register/Field fault

Serialization:  
Reads of control registers reflect the results of all prior instruction groups and interruptions.

In general, writes to control registers do not immediately affect subsequent instructions. Software must issue a serialize operation before a dependent instruction uses a modified resource.

Control register writes are not implicitly synchronized with a corresponding control register read and requires data serialization.
Move Floating-point Register

Format: \((qp) \text{ mov } f_1 = f_3\) 

pseudo-op of: \((qp) \text{ merge.s } f_1 = f_3, f_3\)

Description: The value of FR \(f_3\) is copied to FR \(f_1\).

Operation: See “Floating-point Merge” on page 3:73.
Move General Register

Format: \((qp) \text{ mov } r_1 = r_3\)  

Description: The value of GR \(r_2\) is copied to GR \(r_1\).

Operation: See “Add” on page 3:12.
**Move Immediate**

**Format:**

\[(qp) \text{ mov } r_1 = \text{imm}_{22} \quad \text{pseudo-op of: (qp) addl } r_1 = \text{imm}_{22}, r0\]

**Description:**

The immediate value, \(\text{imm}_{22}\), is sign extended to 64 bits and placed in GR \(r_1\).

**Operation:**

See “Add” on page 3:12.
Move Indirect Register

Format: 
\( (qp) \text{ mov } r_1 = \text{ireg}[r_3] \) from_form M43
\( (qp) \text{ mov } \text{ireg}[r_3] = r_2 \) to_form M42

Description: The source operand is copied to the destination register.

For move from indirect register, GR \( r_3 \) is read and the value used as an index into the register file specified by \( \text{ireg} \) (see Table 2-39 below). The indexed register is read and its value is copied into GR \( r_1 \).

For move to indirect register, GR \( r_3 \) is read and the value used as an index into the register file specified by \( \text{ireg} \). GR \( r_2 \) is read and its value copied into the indexed register.

Table 2-39. Indirect Register File Mnemonics

<table>
<thead>
<tr>
<th>( \text{ireg} )</th>
<th>Register File</th>
</tr>
</thead>
<tbody>
<tr>
<td>cpuid</td>
<td>Processor Identification Register</td>
</tr>
<tr>
<td>dbbr</td>
<td>Data Breakpoint Register</td>
</tr>
<tr>
<td>ibrr</td>
<td>Instruction Breakpoint Register</td>
</tr>
<tr>
<td>pkr</td>
<td>Protection Key Register</td>
</tr>
<tr>
<td>pmcr</td>
<td>Performance Monitor Configuration Register</td>
</tr>
<tr>
<td>pmdr</td>
<td>Performance Monitor Data Register</td>
</tr>
<tr>
<td>rr</td>
<td>Region Register</td>
</tr>
</tbody>
</table>

For all register files other than the region registers, bits \( \{7:0\} \) of GR \( r_3 \) are used as the index. For region registers, bits \( \{63:61\} \) are used. The remainder of the bits are ignored.

Instruction and data breakpoint, performance monitor configuration, protection key, and region registers can only be accessed at the most privileged level. Performance monitor data registers can only be written at the most privileged level.

The CPU identification registers can only be read. There is no to_form of this instruction.

For move to protection key register, the processor ensures uniqueness of protection keys by checking new valid protection keys against all protection key registers. If any matching keys are found, duplicate protection keys are invalidated.

Apart from the PMC and PMD register files, access of a non-existent register results in a Reserved Register/Field fault. All accesses to the implementation-dependent portion of PMC and PMD register files result in implementation dependent behavior but do not fault.

Modifying a region register or a protection key register which is being used to translate:

- the executing instruction stream when PSR.it == 1, or
- the data space for an eager RSE reference when PSR.rt == 1

is an undefined operation.
Operation: if (PR[qp]) {
    if (ireg == RR_TYPE)
        tmp_index = GR[r3][63:61];
    else // all other register types
        tmp_index = GR[r3][7:0];

    if (from_form) {
        check_target_register(r1);

        if (PSR.cpl != 0 && !(ireg == PMD_TYPE || ireg == CPUID_TYPE))
            privileged_operation_fault(0);

        if (GR[r3].nat)
            register_nat_consumption_fault(0);

        if (is_reserved_reg(ireg, tmp_index))
            reserved_register_field_fault();

        if (ireg == PMD_TYPE) {
            if (((PSR.cpl != 0) && ((PSR.sp == 1) ||
                    (tmp_index > 3 &&
                    tmp_index <= IMPL_MAXGENERIC_PMCPMD &&
                    PMC[tmp_index].pm == 1)))
                GR[r1] = 0;
            else
                GR[r1] = pmd_read(tmp_index);
        } else {
            switch (ireg) {
            case CPUID_TYPE: GR[r1] = CPUID[tmp_index]; break;
            case DBR_TYPE: GR[r1] = DBR[tmp_index]; break;
            case IBR_TYPE: GR[r1] = IBR[tmp_index]; break;
            case PKR_TYPE: GR[r1] = PKR[tmp_index]; break;
            case PMC_TYPE: GR[r1] = pmc_read(tmp_index); break;
            case RR_TYPE: GR[r1] = RR[tmp_index]; break;
            }
        }
    } else { // to_form
        if (PSR.cpl != 0)
            privileged_operation_fault(0);

        if (GR[r2].nat || GR[r3].nat)
            register_nat_consumption_fault(0);

        if (is_reserved_reg(ireg, tmp_index) || is_reserved_field(ireg, tmp_index, GR[r2]))
            reserved_register_field_fault();

        if (ireg == PKR_TYPE && GR[r2]{0} == 1) { // writing valid prot key
            if ((tmp_slot = tlb_search_pkr(GR[r2]{31:8})) != NOT_FOUND)
                PKR[tmp_slot].v = 0; // clear valid bit of matching key reg
        }

        tmp_val = ignored_field_mask(ireg, tmp_index, GR[r2]);
        switch (ireg) {
        case DBR_TYPE: DBR[tmp_index] = tmp_val; break;
        case IBR_TYPE: IBR[tmp_index] = tmp_val; break;
        case PKR_TYPE: PKR[tmp_index] = tmp_val; break;
        case PMC_TYPE: pmc_write(tmp_index, tmp_val); break;
        case PMD_TYPE: pmd_write(tmp_index, tmp_val); break;
        case RR_TYPE: RR[tmp_index] = tmp_val; break;
        }
    }
}
Interrupts: Illegal Operation fault  Register NaT Consumption fault
Privileged Operation fault  Reserved Register/Field fault

Serialization: For move to data breakpoint registers, software must issue a data serialize operation before issuing a memory reference dependent on the modified register.

For move to instruction breakpoint registers, software must issue an instruction serialize operation before fetching an instruction dependent on the modified register.

For move to protection key, region, performance monitor configuration, and performance monitor data registers, software must issue an instruction or data serialize operation to ensure the changes are observed before issuing any dependent instruction.

To obtain improved accuracy, software can issue an instruction or data serialize operation before reading the performance monitors.
Move Instruction Pointer

Format: \((qp)\) \text{mov} \ r_j = \text{ip} \quad \text{I25}

Description: The Instruction Pointer (IP) for the bundle containing this instruction is copied into GR \(r_j\).

Operation: \[
\begin{align*}
\text{if (PR[qp])} & \quad \{
\quad \text{check\_target\_register} (r_j); \\
\quad \text{GR}[r_j] = \text{IP}; \\
\quad \text{GR}[r_j].\text{nat} = 0;
\quad \}
\end{align*}
\]

Interruptions: Illegal Operation fault
Move Predicates

Format:

\[
\begin{align*}
(qp) \; \text{mov} \; r_1 &= \text{pr} & \text{from\_form} & \; 125 \\
(qp) \; \text{mov} \; \text{pr} &= r_2, \text{mask}_{17} & \text{to\_form} & \; 123 \\
(qp) \; \text{mov} \; \text{pr}\_\text{rot} &= \text{imm}_{44} & \text{to\_rotate\_form} & \; 124
\end{align*}
\]

Description: The source operand is copied to the destination register.

For moving the predicates to a GR, PR \( i \) is copied to bit position \( i \) within GR \( r_1 \).

For moving to the predicates, the source can either be a general register, or an immediate value. In the to_form, the source operand is GR \( r_2 \) and only those predicates specified by the immediate value \( \text{mask}_{17} \) are written. The value \( \text{mask}_{17} \) is encoded in the instruction in an \( \text{imm}_{16} \) field such that: \( \text{imm}_{16} = \text{mask}_{17} >> 1 \). Predicate register 0 is always one. The \( \text{mask}_{17} \) value is sign extended. The most significant bit of \( \text{mask}_{17} \), therefore, is the mask bit for all of the rotating predicates. If there is a deferred exception for GR \( r_2 \) (the NaT bit is 1), a Register NaT Consumption fault is taken.

In the to_rotate_form, only the 48 rotating predicates can be written. The source operand is taken from the \( \text{imm}_{44} \) operand (which is encoded in the instruction in an \( \text{imm}_{28} \) field, such that: \( \text{imm}_{28} = \text{imm}_{44} >> 16 \)). The low 16-bits correspond to the static predicates. The immediate is sign extended to set the top 21 predicates. Bit position \( i \) in the source operand is copied to PR \( i \).

This instruction operates as if the predicate rotation base in the Current Frame Marker (CFM.rrb.pr) were zero.

Operation:

\[
\text{if } (\text{PR}[qp]) \{ \\
\text{if (from\_form)} \{ \\
\text{check\_target\_register}(r_1); \\
\text{GR}[r_1] = 1; \quad \text{// PR}[0] \text{ is always 1} \\
\text{for } (i = 1; i <= 63; i++) \{ \\
\text{GR}[r_1][i] = \text{PR}[\text{pr\_phys\_to\_virt}(i)]; \\
\} \\
\text{GR}[r_1].\text{nat} = 0; \\
\} \text{ else if (to\_form)} \{ \\
\text{if (GR}[r_2].\text{nat}) \\
\text{register\_nat\_consumption\_fault}(0); \\
\text{tmp\_src} = \text{sign\_ext}(\text{mask}_{17}, 17); \\
\text{for } (i = 1; i <= 63; i++) \{ \\
\text{if (tmp\_src}(i)) \\
\text{PR}[\text{pr\_phys\_to\_virt}(i)] = \text{GR}[r_2][i]; \\
\} \\
\} \text{ else } \{ \text{// to\_rotate\_form} \\
\text{tmp\_src} = \text{sign\_ext}(\text{imm}_{44}, 44); \\
\text{for } (i = 16; i <= 63; i++) \{ \\
\text{PR}[\text{pr\_phys\_to\_virt}(i)] = \text{tmp\_src}(i); \\
\}
\}
\]

Interruptions: Illegal Operation fault \quad Register NaT Consumption fault
Move Processor Status Register

Format:  
\[(qp) \text{ mov } r_1 = \text{psr}\text{ from_form M36}\]  
\[(qp) \text{ mov } \text{psr}.l = r_2\text{ to_form M35}\]

Description:  
The source operand is copied to the destination register. See “Processor Status Register (PSR)” on page 1:18 in Volume 2: System Architecture.

For move from processor status register, PSR bits \{36:35\} and \{31:0\} are read, and copied into GR \(r_1\). All other bits of the PSR read as zero.

For move to processor status register, GR \(r_2\) is read, bits \{31:0\} copied into PSR\{31:0\} and bits \{45:32\} are ignored. All bits of GR \(r_2\) corresponding to reserved fields of the PSR must be 0 or a Reserved Register/Field fault will result.

Moves to and from the PSR can only be performed at the most privileged level.

The contents of the interruption resources (that are overwritten when the PSR.ic bit is 1) are undefined if an interruption occurs between the enabling of the PSR.ic bit and a subsequent instruction serialize operation.

Operation:  
if (PR[qp]) {
  if (from_form)
    check_target_register(r_1);
  if (PSR.cpl != 0)
    privileged_operation_fault(0);
  if (from_form) {
    tmp_val = zero_ext(PSR{31:0}, 32); // read lower 32 bits
    tmp_val |= PSR{36:35} << 35; // read mc and it bits
    GR[r_1] = tmp_val; // other bits read as zero
    GR[r_1].nat = 0;
  } else { // to_form
    if (GR[r_2].nat)
      register_nat_consumption_fault(0);
    if (is_reserved_field(PSR_TYPE, PSR_MOVPART, GR[r_2]))
      reserved_register_field_fault();
    PSR{31:0} = GR[r_2]{31:0};
  }
}

Interruptions:  
Illegal Operation fault  
Register NaT Consumption fault  
Privileged Operation fault  
Reserved Register/Field fault

Serialization:  
Software must issue an instruction or data serialize operation before issuing instructions dependent upon the altered PSR bits. Unlike with the rsm instruction, the PSR.i bit is not treated specially when cleared.
Move User Mask

Format:  
\[(qp)\] mov \(r_j = \text{psr.um}\)  
\[(qp)\] mov \(\text{psr.um} = r_2\)  

from_form \(M36\)  
to_form \(M35\)

Description: The source operand is copied to the destination register.
For move from user mask, PSR\{5:0\} is read, zero-extend, and copied into GR \(r_1\).
For move to user mask, PSR\{5:0\} is written by bits \{5:0\} of GR \(r_2\). PSR.up can only be modified if the secure performance monitor bit (PSR.sp) is zero. Otherwise PSR.up is not modified.
Writing a non-zero value into any other parts of the PSR results in a Reserved Register/Field fault.

Operation:
\[
\begin{align*}
\text{if } \{\text{PR}(qp)\} & \{ \\
\text{if } \{\text{from\_form}\} & \\
& \text{check\_target\_register}(r_1); \\
& \text{GR}[r_1] = \text{zero\_ext}(\text{PSR}\{5:0\}, 6); \\
& \text{GR}[r_1].\text{nat} = 0; \\
\} & \text{else } \{ \text{to\_form} \\
& \text{if } \{\text{GR}[r_2].\text{nat}\} \\
& \text{register\_nat\_consumption\_fault}(0); \\
& \text{if } \{\text{is\_reserved\_field}(\text{PSR\_TYPE}, \text{PSR\_UM}, \text{GR}[r_2])\} \\
& \text{reserved\_register\_field\_fault}(); \\
& \text{PSR}\{1:0\} = \text{GR}[r_2]\{1:0\}; \\
& \text{if } \{\text{PSR.sp} == 0\} \text{ unsecured perf monitor} \\
& \text{PSR}\{2\} = \text{GR}[r_2]\{2\}; \\
& \text{PSR}\{5:3\} = \text{GR}[r_2]\{5:3\}; \\
\} & \}
\end{align*}
\]

Interruptions:  
Illegal Operation fault  
Reserved Register/Field fault  
Register NaT Consumption fault

Serialization: All user mask modifications are observed by the next instruction group.
Move Long Immediate

Format: \((qp)\) movl \(r_I = \text{imm}_{64}\)  

Description: The immediate value \(\text{imm}_{64}\) is copied to GR \(r_I\). The L slot of the bundle contains 41 bits of \(\text{imm}_{64}\).

Operation: if \(\text{PR} \[qp\] \) {
    check_target_register\(\{r_I\}\);
    \(\text{GR}\[r_I\] = \text{imm}_{64}\);
    \(\text{GR}\[r_I\].nat = 0\);
}

Interruptions: Illegal Operation fault
Mux

Format:  
\[(qp)\] mux1 \( r_1 = r_2, mbyte_{4} \)  \quad \text{one_byte_form} \quad I3  
\[(qp)\] mux2 \( r_1 = r_2, mbyte_{8} \)  \quad \text{two_byte_form} \quad I4  

Description:  A permutation is performed on the packed elements in a single source register, GR \( r_2 \), and the result is placed in GR \( r_1 \). For 8-bit elements, only some of all possible permutations can be specified. The five possible permutations are given in Table 2-40 and shown in Figure 2-26.

Table 2-40. Mux Permutations for 8-bit Elements

<table>
<thead>
<tr>
<th>mbyte_{d}</th>
<th>Function</th>
</tr>
</thead>
<tbody>
<tr>
<td>@rev</td>
<td>Reverse the order of the bytes</td>
</tr>
<tr>
<td>@mix</td>
<td>Perform a Mix operation on the two halves of GR ( r_2 )</td>
</tr>
<tr>
<td>@shuf</td>
<td>Perform a Shuffle operation on the two halves of GR ( r_2 )</td>
</tr>
<tr>
<td>@alt</td>
<td>Perform an Alternate operation on the two halves of GR ( r_2 )</td>
</tr>
<tr>
<td>@brcst</td>
<td>Perform a Broadcast operation on the least significand byte of GR ( r_2 )</td>
</tr>
</tbody>
</table>

Figure 2-26. Mux1 Operation (8-bit elements)
For 16-bit elements, all possible permutations, with and without repetitions can be specified. They are expressed with an 8-bit \textit{mhtype}_8 field, which encodes the indices of the four 16-bit data elements. The indexed 16-bit elements of GR \textit{r}_2 are copied to corresponding 16-bit positions in the target register GR \textit{r}_1. The indices are encoded in little-endian order. (The 8 bits of \textit{mhtype}_8[7:0] are grouped in pairs of bits and named \textit{mhtype}_8[3], \textit{mhtype}_8[2], \textit{mhtype}_8[1], \textit{mhtype}_8[0] in the Operation section).

\textbf{Figure 2-27. Mux2 Examples (16-bit elements)}

\begin{figure}
\centering
\includegraphics[width=\textwidth]{mux2_examples}
\end{figure}
Operation:

```c
if (PR[qp]) {
    check_target_register(r1);

    if (one_byte_form) {
        x[0] = GR[r2]{7:0};
        x[1] = GR[r2]{15:8};
        x[3] = GR[r2]{31:24};
        x[5] = GR[r2]{47:40};
        x[7] = GR[r2]{63:56};

        switch (mbtype) {
            case '@rev':
                GR[r1] = concatenate8(x[0], x[1], x[2], x[3],
                                       x[4], x[5], x[6], x[7]);
                break;
            case '@mix':
                GR[r1] = concatenate8(x[7], x[3], x[5], x[1],
                                       x[6], x[2], x[4], x[0]);
                break;
            case '@shuf':
                GR[r1] = concatenate8(x[7], x[3], x[6], x[2],
                                       x[5], x[1], x[4], x[0]);
                break;
            case '@alt':
                GR[r1] = concatenate8(x[7], x[5], x[3], x[1],
                                       x[6], x[4], x[2], x[0]);
                break;
            case '@brcst':
                GR[r1] = concatenate8(x[0], x[0], x[0], x[0],
                                       x[0], x[0], x[0], x[0]);
                break;
        }
    } else { // two_byte_form
        x[0] = GR[r2]{15:0};
        x[1] = GR[r2]{31:16};
        x[2] = GR[r2]{47:32};
        x[3] = GR[r2]{63:48};

        res[0] = x[mhtype8{1:0}];
        res[1] = x[mhtype8{3:2}];
        res[2] = x[mhtype8{5:4}];
        res[3] = x[mhtype8{7:6}];

        GR[r1] = concatenate4(res[3], res[2], res[1], res[0]);
    }
}
GR[r1].nat = GR[r2].nat;
```

Interruptions: Illegal Operation fault
No Operation

Format:  

(qp) nop  \textit{imm}_{21}  \quad \text{pseudo-op}  
(qp) nop.i  \textit{imm}_{21}  \quad \text{i\_unit\_form} \text{ I19}  
(qp) nop.b  \textit{imm}_{21}  \quad \text{b\_unit\_form} \text{ B9}  
(qp) nop.m  \textit{imm}_{21}  \quad \text{m\_unit\_form} \text{ M37}  
(qp) nop.f  \textit{imm}_{21}  \quad \text{f\_unit\_form} \text{ F15}  
(qp) nop.x  \textit{imm}_{62}  \quad \text{x\_unit\_form} \text{ X1}  

Description:  No operation is done.  

The immediate, \textit{imm}_{21} or \textit{imm}_{62}, can be used by software as a marker in program code. It is ignored by hardware.  

For the \text{x\_unit\_form}, the L slot of the bundle contains the upper 41 bits of \textit{imm}_{62}.  

A \text{nop.i} instruction may be encoded in an MLI-template bundle, in which case the L slot of the bundle is ignored.  

This instruction has five forms, each of which can be executed only on a particular execution unit type. The pseudo-op can be used if the unit type to execute on is unimportant.  

Operation:  

\begin{verbatim}
if (PR[qp]) {
    ; // no operation
}
\end{verbatim}  

Interruptions:  None
Logical Or

Format: \[(qp) \text{ or } r_j = r_2, r_3\] 
\[(qp) \text{ or } r_j = \text{imm}_8, r_3\]

Description: The two source operands are logically ORed and the result placed in GR \(r_1\). In the register form the first operand is GR \(r_2\); in the immediate form the first operand is taken from the \(\text{imm}_8\) encoding field.

Operation: 
\[
\text{if (PR[qp])} \{ \\
\quad \text{check_target_register}(r_j); \\
\quad \text{tmp}_\text{src} = (\text{register_form} ? \text{GR}[r_2] : \text{sign_ext}(\text{imm}_8, 8)); \\
\quad \text{tmp}_\text{nat} = (\text{register_form} ? \text{GR}[r_2].\text{nat} : 0); \\
\quad \text{GR}[r_1] = \text{tmp}_\text{src} \mid \text{GR}[r_3]; \\
\quad \text{GR}[r_2].\text{nat} = \text{tmp}_\text{nat} \mid \text{GR}[r_3].\text{nat}; \\
\}\]

Interruptions: Illegal Operation fault
Pack

**Format:**

- \((qp)\) pack2.sss \(r_1 = r_2, r_3\)  
  - two-byte_form, signed_saturation_form  
- \((qp)\) pack2.ass \(r_1 = r_2, r_3\)  
  - two-byte_form, unsigned_saturation_form  
- \((qp)\) pack4.sss \(r_1 = r_2, r_3\)  
  - four-byte_form, signed_saturation_form

**Description:**

32-bit or 16-bit elements from GR \(r_2\) and GR \(r_3\) are converted into 16-bit or 8-bit elements respectively, and the results are placed GR \(r_1\). The source elements are treated as signed values. If a source element cannot be represented in the result element, then saturation clipping is performed. The saturation can either be signed or unsigned. If an element is larger than the upper limit value, the result is the upper limit value. If it is smaller than the lower limit value, the result is the lower limit value. The saturation limits are given in Table 2-41.

**Table 2-41. Pack Saturation Limits**

<table>
<thead>
<tr>
<th>Size</th>
<th>Source Element Width</th>
<th>Result Element Width</th>
<th>Saturation</th>
<th>Upper Limit</th>
<th>Lower Limit</th>
</tr>
</thead>
<tbody>
<tr>
<td>2</td>
<td>16 bit</td>
<td>8 bit</td>
<td>signed</td>
<td>0x7f</td>
<td>0x80</td>
</tr>
<tr>
<td>2</td>
<td>16 bit</td>
<td>8 bit</td>
<td>unsigned</td>
<td>0xff</td>
<td>0x00</td>
</tr>
<tr>
<td>4</td>
<td>32 bit</td>
<td>16 bit</td>
<td>signed</td>
<td>0x7fff</td>
<td>0x8000</td>
</tr>
</tbody>
</table>

**Figure 2-28. Pack Operation**

[Diagram showing pack2 and pack4 operations]
Operation: if (PR[qp]) {
    check_target_register(r1);

    if (two_byte_form) {
        if (signed_saturation_form) {
            max = sign_ext(0x7f, 8);
            min = sign_ext(0x80, 8);
        } else { // unsigned_saturation_form
            max = 0xff;
            min = 0x00;
        }
        temp[0] = sign_ext(GR[r2]{15:0}, 16);
        temp[1] = sign_ext(GR[r2]{31:16}, 16);
        temp[2] = sign_ext(GR[r2]{47:32}, 16);
        temp[3] = sign_ext(GR[r2]{63:48}, 16);
        temp[4] = sign_ext(GR[r3]{15:0}, 16);
        temp[5] = sign_ext(GR[r3]{31:16}, 16);
        temp[6] = sign_ext(GR[r3]{47:32}, 16);
        temp[7] = sign_ext(GR[r3]{63:48}, 16);
        for (i = 0; i < 8; i++) {
            if (temp[i] > max)
                temp[i] = max;
            if (temp[i] < min)
                temp[i] = min;
        }
        GR[r1] = concatenate8(temp[7], temp[6], temp[5], temp[4],
                              temp[3], temp[2], temp[1], temp[0]);
    } else { // four_byte_form
        max = sign_ext(0x7fff, 16); // signed_saturation_form
        min = sign_ext(0x8000, 16);
        temp[0] = sign_ext(GR[r2]{31:0}, 32);
        temp[1] = sign_ext(GR[r2]{63:32}, 32);
        temp[2] = sign_ext(GR[r3]{31:0}, 32);
        temp[3] = sign_ext(GR[r3]{63:32}, 32);
        for (i = 0; i < 4; i++) {
            if (temp[i] > max)
                temp[i] = max;
            if (temp[i] < min)
                temp[i] = min;
        }
        GR[r1] = concatenate4(temp[3], temp[2], temp[1], temp[0]);
    }
    GR[r1].nat = GR[r2].nat || GR[r3].nat;
}

Interruptions: Illegal Operation fault
Parallel Add

Format:

\[(qp)\text{ padd1} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd1.sss} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd1.uus} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd1.uuu} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd2} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd2.sss} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd2.uus} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd2.uuu} \quad r_1 = r_2, r_3\]
\[(qp)\text{ padd4} \quad r_1 = r_2, r_3\]

Description: The sets of elements from the two source operands are added, and the results placed in GR \(r_1\).

If a sum of two elements cannot be represented in the result element and a saturation completer is specified, then saturation clipping is performed. The saturation can either be signed or unsigned, as given in Table 2-42. If the sum of two elements is larger than the upper limit value, the result is the upper limit value. If it is smaller than the lower limit value, the result is the lower limit value. The saturation limits are given in Table 2-43.

Table 2-42. Parallel Add Saturation Completers

<table>
<thead>
<tr>
<th>Completer</th>
<th>Result (r_1) treated as</th>
<th>Source (r_2) treated as</th>
<th>Source (r_3) treated as</th>
</tr>
</thead>
<tbody>
<tr>
<td>sss</td>
<td>signed</td>
<td>signed</td>
<td>signed</td>
</tr>
<tr>
<td>uus</td>
<td>unsigned</td>
<td>unsigned</td>
<td>signed</td>
</tr>
<tr>
<td>uuu</td>
<td>unsigned</td>
<td>unsigned</td>
<td>unsigned</td>
</tr>
</tbody>
</table>

Table 2-43. Parallel Add Saturation Limits

<table>
<thead>
<tr>
<th>Size</th>
<th>Element Width</th>
<th>Result (r_1) signed</th>
<th>Result (r_1) unsigned</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td>Upper Limit</td>
<td>Lower Limit</td>
</tr>
<tr>
<td>1</td>
<td>8 bit</td>
<td>0x7f</td>
<td>0x80</td>
</tr>
<tr>
<td>2</td>
<td>16 bit</td>
<td>0x7fff</td>
<td>0x8000</td>
</tr>
</tbody>
</table>

Figure 2-29. Parallel Add Examples
Operation:

```
if (PR[qp]) {
    check_target_register(r);

    if (one_byte_form) {
        // one-byte elements
        x[0] = GR[r][7:0];  y[0] = GR[r][7:0];
        x[1] = GR[r][15:8]; y[1] = GR[r][15:8];
        x[5] = GR[r][47:40]; y[5] = GR[r][47:40];
        x[7] = GR[r][63:56]; y[7] = GR[r][63:56];

        if (sss_saturation_form) {
            max = sign_ext(0x7f, 8);
            min = sign_ext(0x80, 8);
            for (i = 0; i < 8; i++) {
                temp[i] = sign_ext(x[i], 8) + sign_ext(y[i], 8);
            }
        } else if (uus_saturation_form) {
            max = 0xff;
            min = 0x00;
            for (i = 0; i < 8; i++) {
                temp[i] = zero_ext(x[i], 8) + sign_ext(y[i], 8);
            }
        } else if (uuu_saturation_form) {
            max = 0xff;
            min = 0x00;
            for (i = 0; i < 8; i++) {
                temp[i] = zero_ext(x[i], 8) + zero_ext(y[i], 8);
            }
        } else { // modulo_form
            for (i = 0; i < 8; i++) {
                temp[i] = zero_ext(x[i], 8) + zero_ext(y[i], 8);
            }
        }
    }

    GR[r] = concatenate8(temp[7], temp[6], temp[5], temp[4],
                         temp[3], temp[2], temp[1], temp[0]);

    if (two_byte_form) {
        // 2-byte elements
        x[0] = GR[r][15:0];  y[0] = GR[r][15:0];
        x[1] = GR[r][31:16]; y[1] = GR[r][31:16];

        if (sss_saturation_form) {
            max = sign_ext(0x7fff, 16);
            min = sign_ext(0x8000, 16);
        }
    }
}
```
for (i = 0; i < 4; i++) {
    temp[i] = sign_ext(x[i], 16) + sign_ext(y[i], 16);
}
} else if (uus_saturation_form) {
    max = 0xffff;
    min = 0x0000;
    for (i = 0; i < 4; i++) {
        temp[i] = zero_ext(x[i], 16) + sign_ext(y[i], 16);
    }
} else if (uuu_saturation_form) {
    max = 0xffff;
    min = 0x0000;
    for (i = 0; i < 4; i++) {
        temp[i] = zero_ext(x[i], 16) + zero_ext(y[i], 16);
    }
} else { // modulo_form
    for (i = 0; i < 4; i++) {
        temp[i] = zero_ext(x[i], 16) + zero_ext(y[i], 16);
    }
}

if (sss_saturation_form || uus_saturation_form || uuu_saturation_form) {
    for (i = 0; i < 4; i++) {
        if (temp[i] > max)
            temp[i] = max;
        if (temp[i] < min)
            temp[i] = min;
    }
} 
GR[r1] = concatenate4(temp[3], temp[2], temp[1], temp[0]);

} else { // four-byte elements
    x[0] = GR[r2][31:0]; y[0] = GR[r3][31:0];
    for (i = 0; i < 2; i++) {
        // modulo_form
        temp[i] = zero_ext(x[i], 32) + zero_ext(y[i], 32);
    }
    GR[r1] = concatenate2(temp[1], temp[0]);
} 
GR[r1].nat = GR[r2].nat || GR[r3].nat;

**Interruptions:** Illegal Operation fault
Parallel Average

Format:

- \((qp)\) pavg1 \( r_1 = r_2, r_3 \)  
  normal_form, one_byte_form \ A9
- \((qp)\) pavg1.raz \( r_1 = r_2, r_3 \)  
  raz_form, one_byte_form \ A9
- \((qp)\) pavg2 \( r_1 = r_2, r_3 \)  
  normal_form, two_byte_form \ A9
- \((qp)\) pavg2.raz \( r_1 = r_2, r_3 \)  
  raz_form, two_byte_form \ A9

Description:

The unsigned data elements of GR \( r_2 \) are added to the unsigned data elements of GR \( r_3 \). The results of the add are then each independently shifted to the right by one bit position. The high-order bits of each element are filled with the carry bits of the sums. To prevent cumulative round-off errors, an averaging is performed. The unsigned results are placed in GR \( r_1 \).

The averaging operation works as follows. In the normal_form, the low-order bit of each result is set to 1 if at least one of the two least significant bits of the corresponding sum is 1. In the raz_form, the average rounds away from zero by adding 1 to each of the sums.

Figure 2-30. Parallel Average Example
Figure 2-31. Parallel Average with Round Away from Zero Example

GR \( r_3 \):

GR \( r_2 \):

16-bit sum plus carry

1 1 1 1

+ + +

shift right 1 bit

GR \( r_1 \):

pavg2.raz

shift right 1 bit

carry bit

sum bits
Operation: if (PR[qp]) {
    check_target_register(r_j);
}

if (one_byte_form) {
    x[0] = GR[r_2][7:0];  y[0] = GR[r_3][7:0];
    x[1] = GR[r_2][15:8]; y[1] = GR[r_3][15:8];
    x[5] = GR[r_2][47:40]; y[5] = GR[r_3][47:40];
    x[7] = GR[r_2][63:56]; y[7] = GR[r_3][63:56];

    if (raz_form) {
        for (i = 0; i < 8; i++) {
            temp[i] = zero_ext(x[i], 8) + zero_ext(y[i], 8) + 1;
            res[i] = shift_right_unsigned(temp[i], 1);
        }
    } else { // normal form
        for (i = 0; i < 8; i++) {
            temp[i] = zero_ext(x[i], 8) + zero_ext(y[i], 8);
            res[i] = shift_right_unsigned(temp[i], 1) | (temp[i]{0});
        }
    }
    GR[r_j] = concatenate8(res[7], res[6], res[5], res[4],
                           res[3], res[2], res[1], res[0]);
} else { // two_byte_form
    x[0] = GR[r_2][15:0];  y[0] = GR[r_3][15:0];
    x[1] = GR[r_2][31:16]; y[1] = GR[r_3][31:16];

    if (raz_form) {
        for (i = 0; i < 4; i++) {
            temp[i] = zero_ext(x[i], 16) + zero_ext(y[i], 16) + 1;
            res[i] = shift_right_unsigned(temp[i], 1);
        }
    } else { // normal form
        for (i = 0; i < 4; i++) {
            temp[i] = zero_ext(x[i], 16) + zero_ext(y[i], 16);
            res[i] = shift_right_unsigned(temp[i], 1) | (temp[i]{0});
        }
    }
    GR[r_j] = concatenate4(res[3], res[2], res[1], res[0]);
} GR[r_j].nat = GR[r_2].nat || GR[r_3].nat;

Interruptions: Illegal Operation fault
Parallel Average Subtract

Format: 

\[ (qp) \text{ pavgsub1} \quad r_1 = r_2, r_3 \] 
\[ (qp) \text{ pavgsub2} \quad r_1 = r_2, r_3 \]

Description: The unsigned data elements of GR \( r_3 \) are subtracted from the unsigned data elements of GR \( r_2 \). The results of the subtraction are then each independently shifted to the right by one bit position. The high-order bits of each element are filled with the borrow bits of the subtraction (the complements of the ALU carries). To prevent cumulative round-off errors, an averaging is performed. The low-order bit of each result is set to 1 if at least one of the two least significant bits of the corresponding difference is 1. The signed results are placed in GR \( r_1 \).

Figure 2-32. Parallel Average Subtract Example
Operation:  
  if (PR[qp]) {
    check_target_register(r1);
    if (one_byte_form) {
      x[0] = GR[r2]{7:0}; y[0] = GR[r3]{7:0};
      for (i = 0; i < 8; i++) {
        temp[i] = zero_ext(x[i], 8) - zero_ext(y[i], 8);
        res[i] = (temp[i]{8:0} u>> 1) | (temp[i]{0});
      }
      GR[r1] = concatenate8(res[7], res[6], res[5], res[4],
        res[3], res[2], res[1], res[0]);
    } else { // two_byte_form
      x[0] = GR[r2]{15:0}; y[0] = GR[r3]{15:0};
      for (i = 0; i < 4; i++) {
        temp[i] = zero_ext(x[i], 16) - zero_ext(y[i], 16);
        res[i] = (temp[i]{16:0} u>> 1) | (temp[i]{0});
      }
      GR[r1] = concatenate4(res[3], res[2], res[1], res[0]);
    }
    GR[r1].nat = GR[r2].nat || GR[r3].nat;
  } else { // two_byte_form
    x[0] = GR[r2]{15:0}; y[0] = GR[r3]{15:0};
    for (i = 0; i < 4; i++) {
      temp[i] = zero_ext(x[i], 16) - zero_ext(y[i], 16);
      res[i] = (temp[i]{16:0} u>> 1) | (temp[i]{0});
    }
    GR[r1] = concatenate4(res[3], res[2], res[1], res[0]);
    GR[r1].nat = GR[r2].nat || GR[r3].nat;
  }

Interruptions:  Illegal Operation fault
Parallel Compare

Format:  
- \((qp)\) pcmp1.prel \(r_1 = r_2, r_3\)  
- \((qp)\) pcmp2.prel \(r_1 = r_2, r_3\)  
- \((qp)\) pcmp4.prel \(r_1 = r_2, r_3\)

Description: The two source operands are compared for one of the two relations shown in Table 2-44. If the comparison condition is true for corresponding data elements of GR \(r_2\) and GR \(r_3\), then the corresponding data element in GR \(r_1\) is set to all ones. If the comparison condition is false, then the corresponding data element in GR \(r_1\) is set to all zeros. For the ‘\(>\)’ relation, both operands are interpreted as signed.

Table 2-44. Pcmp Relations

<table>
<thead>
<tr>
<th>prel</th>
<th>Compare Relation ((r_2 \text{ prel} r_3))</th>
</tr>
</thead>
<tbody>
<tr>
<td>eq</td>
<td>(r_2 == r_3)</td>
</tr>
<tr>
<td>gt</td>
<td>(r_2 &gt; r_3) (signed)</td>
</tr>
</tbody>
</table>

Figure 2-33. Parallel Compare Example
pcmp

Operation:

```c
if (PR[qp]) {
    check_target_register(rj);
    if (one_byte_form) {
        // one-byte elements
        x[0] = GR[r2][7:0];  y[0] = GR[r3][7:0];
        x[1] = GR[r2][15:8]; y[1] = GR[r3][15:8];
        x[5] = GR[r2][47:40]; y[5] = GR[r3][47:40];
        x[7] = GR[r2][63:56]; y[7] = GR[r3][63:56];
        for (i = 0; i < 8; i++) {
            if (prel == 'eq')
                tmp_rel = x[i] == y[i];
            else if (two_byte_form) { // two-byte elements
                x[0] = GR[r2][15:0]; y[0] = GR[r3][15:0];
                x[1] = GR[r2][31:16]; y[1] = GR[r3][31:16];
                for (i = 0; i < 4; i++) {
                    if (prel == 'eq')
                        tmp_rel = x[i] == y[i];
                    else if (gt)
                        tmp_rel = greater_signed(sign_ext(x[i], 8),
                                                sign_ext(y[i], 8));
                    if (tmp_rel)
                        res[i] = 0xff;
                    else
                        res[i] = 0x00;
                }
                GR[rj] = concatenate8(res[7], res[6], res[5], res[4],
                                      res[3], res[2], res[1], res[0]);
            } else if (four-byte elements) { // four-byte elements
                x[0] = GR[r2][31:0]; y[0] = GR[r3][31:0];
                for (i = 0; i < 2; i++) {
                    if (prel == 'eq')
                        tmp_rel = x[i] == y[i];
                    else if (gt)
                        tmp_rel = greater_signed(sign_ext(x[i], 32),
                                                sign_ext(y[i], 32));
                    if (tmp_rel)
                        res[i] = 0xffffffff;
                    else
                        res[i] = 0x00000000;
                }
                GR[rj] = concatenate2(res[1], res[0]);
            }
        }
    }
    else if (two_byte_form) { // two-byte elements
        x[0] = GR[r2][15:0]; y[0] = GR[r3][15:0];
        x[1] = GR[r2][31:16]; y[1] = GR[r3][31:16];
        for (i = 0; i < 4; i++) {
            if (prel == 'eq')
                tmp_rel = x[i] == y[i];
            else if (gt)
                tmp_rel = greater_signed(sign_ext(x[i], 16),
                                        sign_ext(y[i], 16));
            if (tmp_rel)
                res[i] = 0xffff;
            else
                res[i] = 0x0000;
        }
        GR[rj] = concatenate4(res[3], res[2], res[1], res[0]);
    } else { // four-byte elements
        x[0] = GR[r2][31:0]; y[0] = GR[r3][31:0];
        for (i = 0; i < 2; i++) {
            if (prel == 'eq')
                tmp_rel = x[i] == y[i];
            else if (gt)
                tmp_rel = greater_signed(sign_ext(x[i], 32),
                                        sign_ext(y[i], 32));
            if (tmp_rel)
                res[i] = 0xffffffff;
            else
                res[i] = 0x00000000;
        }
        GR[rj] = concatenate2(res[1], res[0]);
    }
    GR[rj].nat = GR[r2].nat || GR[r3].nat;
}
```

Interruptions: Illegal Operation fault
Parallel Maximum

Format:  
\[(qp)\ pmax1.u\ r_1 = r_2, r_3\]  \hspace{1cm} \text{one_byte_form}  \hspace{1cm} 12  
\[(qp)\ pmax2\ r_1 = r_2, r_3\]  \hspace{1cm} \text{two_byte_form}  \hspace{1cm} 12

Description:  
The maximum of the two source operands is placed in the result register. In the one_byte_form, each unsigned 8-bit element of GR \(r_2\) is compared with the corresponding unsigned 8-bit element of GR \(r_3\) and the greater of the two is placed in the corresponding 8-bit element of GR \(r_1\). In the two_byte_form, each signed 16-bit element of GR \(r_2\) is compared with the corresponding signed 16-bit element of GR \(r_3\) and the greater of the two is placed in the corresponding 16-bit element of GR \(r_1\).

Figure 2-34. Parallel Maximum Example

<table>
<thead>
<tr>
<th>GR (r_2)</th>
<th>GR (r_1)</th>
<th>GR (r_3)</th>
</tr>
</thead>
<tbody>
<tr>
<td>&lt; &lt; &lt; &lt; &lt; &lt; &lt; &lt;</td>
<td>t</td>
<td>t</td>
</tr>
</tbody>
</table>

Operation:  
if (PR[qp]) {
    check_target_register(r_1);
    if (one_byte_form) {  // one-byte elements
        x[0] = GR[r_2]{7:0};  y[0] = GR[r_3]{7:0};
        for (i = 0; i < 8; i++) {
            res[i] = (zero_ext(x[i],8) < zero_ext(y[i],8)) ? y[i] : x[i];
        }
        GR[r_1] = concatenate8(res[7], res[6], res[5], res[4],
                                res[3], res[2], res[1], res[0]);
    } else {  // two-byte elements
        x[0] = GR[r_2]{15:0};  y[0] = GR[r_3]{15:0};
        for (i = 0; i < 4; i++) {
            res[i] = (sign_ext(x[i],16) < sign_ext(y[i],16)) ? y[i] : x[i];
        }
        GR[r_1] = concatenate4(res[3], res[2], res[1], res[0]);
    }
    GR[r_1].nat = GR[r_2].nat || GR[r_3].nat;
}

Interruptions:  
Illegal Operation fault
Parallel Minimum

Format:  
\[(qp)\text{ pmin1.u  } r_1 = r_2, r_3\]  
\[(qp)\text{ pmin2  } r_1 = r_2, r_3\]  
\text{one_byte_form  } I2  
\text{two_byte_form  } I2

Description:  
The minimum of the two source operands is placed in the result register. In the one_byte_form, each unsigned 8-bit element of GR \(r_2\) is compared with the corresponding unsigned 8-bit element of GR \(r_3\) and the smaller of the two is placed in the corresponding 8-bit element of GR \(r_1\). In the two_byte_form, each signed 16-bit element of GR \(r_2\) is compared with the corresponding signed 16-bit element of GR \(r_3\) and the smaller of the two is placed in the corresponding 16-bit element of GR \(r_1\).

**Figure 2-35. Parallel Minimum Example**

Operation:  
\text{if (PR[qp])} \{  
\text{check_target_register(r1);}  
\text{if (one_byte_form)} \{  
\text{\hspace{1cm} // one-byte elements}\}  
\text{x[0] = GR[r2](7:0);  \hspace{1cm} y[0] = GR[r3](7:0);}  
\text{x[1] = GR[r2](15:8);  \hspace{1cm} y[1] = GR[r3](15:8);}  
\text{x[2] = GR[r2](23:16);  \hspace{1cm} y[2] = GR[r3](23:16);}  
\text{x[3] = GR[r2](31:24);  \hspace{1cm} y[3] = GR[r3](31:24);}  
\text{x[4] = GR[r2](39:32);  \hspace{1cm} y[4] = GR[r3](39:32);}  
\text{x[5] = GR[r2](47:40);  \hspace{1cm} y[5] = GR[r3](47:40);}  
\text{x[6] = GR[r2](55:48);  \hspace{1cm} y[6] = GR[r3](55:48);}  
\text{x[7] = GR[r2](63:56);  \hspace{1cm} y[7] = GR[r3](63:56);}  
\text{\hspace{1cm} for (i = 0; i < 8; i++) \{}  
\text{\hspace{1cm} \hspace{1cm} res[i] = (zero_ext(x[i],8) < zero_ext(y[i],8)) ? x[i] : y[i];}  
\text{\hspace{1cm} \}}  
\text{GR[r1] = concatenate8(res[7], res[6], res[5], res[4], \hspace{1cm}}  
\text{res[3], res[2], res[1], res[0]);}  
\text{\}}  
\text{else} \{  
\text{\hspace{1cm} // two-byte elements}\}  
\text{x[0] = GR[r2](15:0);  \hspace{1cm} y[0] = GR[r3](15:0);}  
\text{x[1] = GR[r2](31:16);  \hspace{1cm} y[1] = GR[r3](31:16);}  
\text{x[2] = GR[r2](47:32);  \hspace{1cm} y[2] = GR[r3](47:32);}  
\text{x[3] = GR[r2](63:48);  \hspace{1cm} y[3] = GR[r3](63:48);}  
\text{\hspace{1cm} for (i = 0; i < 4; i++) \{}  
\text{\hspace{1cm} \hspace{1cm} res[i] = (sign_ext(x[i],16) < sign_ext(y[i],16)) ? x[i] : y[i];}  
\text{\hspace{1cm} \}}  
\text{GR[r1] = concatenate4(res[3], res[2], res[1], res[0]);}  
\text{\}}  
\text{GR[r1].nat = GR[r2].nat || GR[r3].nat;}  
\text{\}}  
\text{\}

Interruptions:  
Illegal Operation fault
Parallel Multiply

Format:  
\[(qp) \ pmpy2.r \ r_1 = r_2, r_3 \]  \(\text{right_form} \quad 12\)  
\[(qp) \ pmpy2.l \ r_1 = r_2, r_3 \]  \(\text{left_form} \quad 12\)

Description:  
Two signed 16-bit data elements of GR \(r_2\) are multiplied by the corresponding two signed 16-bit data elements of GR \(r_3\) as shown in Figure 2-36. The two 32-bit results are placed in GR \(r_1\).

**Figure 2-36. Parallel Multiply Operation**

![Parallel Multiply Diagram](image)

Operation:  
```c
if (PR[qp]) {
    check_target_register(r1);
    if (right_form) {
        GR[r1]{31:0} = sign_ext(GR[r2]{15:0}, 16) * sign_ext(GR[r3]{15:0}, 16);
        GR[r1]{63:32} = sign_ext(GR[r2]{47:32}, 16) * sign_ext(GR[r3]{47:32}, 16);
    } else { // left_form
        GR[r1]{31:0} = sign_ext(GR[r2]{31:16}, 16) * sign_ext(GR[r3]{31:16}, 16);
        GR[r1]{63:32} = sign_ext(GR[r2]{63:48}, 16) * sign_ext(GR[r3]{63:48}, 16);
    }
    GR[r1].nat = GR[r2].nat || GR[r3].nat;
}
```

Interruptions:  
Illegal Operation fault
Parallel Multiply and Shift Right

Format:

\[
(qp) \text{pmpyshr2} \ r_1 = r_2, r_3, count_2 \\
(qp) \text{pmpyshr2,u} \ r_1 = r_2, r_3, count_2
\]

signed_form \ I1 \quad \text{unsigned_form} \ I1

Description:
The four 16-bit data elements of GR \(r_2\) are multiplied by the corresponding four 16-bit data elements of GR \(r_3\) as shown in Figure 2-37. This multiplication can either be signed (pmpyshr2), or unsigned (pmpyshr2.u). Each product is then shifted to the right \(count_2\) bits, and the least-significant 16-bits of each shifted product form 4 16-bit results, which are placed in GR \(r_1\). A \(count_2\) of 0 gives the 16 low bits of the results, a \(count_2\) of 16 gives the 16 high bits of the results. The allowed values for \(count_2\) are given in Table 2-45.

Table 2-45. PMPYSHR Shift Options

<table>
<thead>
<tr>
<th>(count_2)</th>
<th>Selected Bit Field from Each 32-bit Product</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>15:0</td>
</tr>
<tr>
<td>7</td>
<td>22:7</td>
</tr>
<tr>
<td>15</td>
<td>30:15</td>
</tr>
<tr>
<td>16</td>
<td>31:16</td>
</tr>
</tbody>
</table>

Figure 2-37. Parallel Multiply and Shift Right Operation

![Diagram of Parallel Multiply and Shift Right Operation](image)
Operation:

```
if (PR[qp]) {
    check_target_register(r1);
    x[0] = GR[r2]{15:0};  y[0] = GR[r3]{15:0};
    for (i = 0; i < 4; i++) {
        if (unsigned_form) // unsigned multiplication
            temp[i] = zero_ext(x[i], 16) * zero_ext(y[i], 16);
        else // signed multiplication
            temp[i] = sign_ext(x[i], 16) * sign_ext(y[i], 16);
        res[i] = temp[i]{(count2 + 15):count2};
    }
    GR[r1] = concatenate4(res[3], res[2], res[1], res[0]);
    GR[r1].nat = GR[r2].nat || GR[r3].nat;
}
```

Interruptions: Illegal Operation fault
Population Count

Format: \((qp)\) popcnt \(r_1 = r_3\)

Description: The number of bits in GR \(r_3\) having the value 1 is counted, and the resulting sum is placed in GR \(r_1\).

Operation:
```c
if (PR[qp]) {
    check_target_register(r1);
    res = 0;
    // Count up all the one bits
    for (i = 0; i < 64; i++) {
        res += GR[r3][i];
    }
    GR[r1] = res;
    GR[r1].nat = GR[r3].nat;
}
```

Interruptions: Illegal Operation fault
Probe Access

**Format:**

- \((qp)\) probe.r \(r_1 = r_3, r_2\) \hspace{1cm} \text{read form, register form} \hspace{1cm} M38
- \((qp)\) probe.w \(r_1 = r_3, r_2\) \hspace{1cm} \text{write form, register form} \hspace{1cm} M38
- \((qp)\) probe.r \(r_1 = r_3, \text{imm}_2\) \hspace{1cm} \text{read form, immediate form} \hspace{1cm} M39
- \((qp)\) probe.w \(r_1 = r_3, \text{imm}_2\) \hspace{1cm} \text{write form, immediate form} \hspace{1cm} M39
- \((qp)\) probe.r.fault \(r_3, \text{imm}_2\) \hspace{1cm} \text{fault form, read form, immediate form} \hspace{1cm} M40
- \((qp)\) probe.w.fault \(r_3, \text{imm}_2\) \hspace{1cm} \text{fault form, write form, immediate form} \hspace{1cm} M40
- \((qp)\) probe.rw.fault \(r_3, \text{imm}_2\) \hspace{1cm} \text{fault form, read_write form, immediate form} \hspace{1cm} M40

**Description:**

This instruction determines whether read or write access, with a specified privilege level, to a given virtual address is permitted. GR \(r_1\) is set to 1 if the specified access is allowed and to 0 otherwise. In the fault form, if the specified access is allowed this instruction does nothing; if the specified access is not allowed, a fault is taken.

When PSR.dt is 1, the DTLB and the VHPT are queried for present translations to determine if access to the virtual address specified by GR \(r_3\) bits \(60:0\) and the region register indexed by GR \(r_3\) bits \(63:61\), is permitted at the privilege level given by either GR \(r_2\) bits \(1:0\) or \(\text{imm}_2\). If PSR.pk is 1, protection key checks are also performed. The read or write form specifies whether the instruction checks for read or write access, or both.

When PSR.dt is 0, a non-faulting probe uses its address operand as a virtual address to query the DTLB only, because the VHPT walker is disabled. If the probed address is found in the DTLB, the non-faulting probe returns the appropriate value, if not an Alternate Data TLB fault is raised.

When PSR.dt is 0, a faulting probe treats its address operand as a physical address, and takes no TLB related faults.

A non-faulting probe to an unimplemented virtual address returns 0. A faulting probe to an unimplemented virtual address (when PSR.dt is 1) or unimplemented physical address (when PSR.dt is 0) takes an Unimplemented Data Address fault.

If this instruction faults, then it will set the non-access bit in the ISR and set the ISR read or write bits depending on the completer. The following faults are taken by the faulting form of the probe instruction only (the non-faulting form of the instruction does not take them): Unimplemented Data Address fault, Data Key Permissions fault, Data Access Rights fault, Data Dirty Bit fault, Data Access Bit fault, and Data Debug fault.

This instruction can only probe with equal or lower privilege levels. If the specified privilege level is higher (lower number), then the probe is performed with the current privilege level.
Operation:

```c
if (PR[qp]) {
    itype = NON_ACCESS;
    itype |= (read_write_form) ? READ|WRITE : ((write_form) ? WRITE : READ);
    itype |= (fault_form) ? PROBE_FAULT : PROBE;

    if (!fault_form)
        check_target_register(r1);

    if (GR[r3].nat || (register_form ? GR[r2].nat : 0))
        register_nat_consumption_fault(itype);

    tmp_pl = (register_form) ? GR[r2][1:0] : imm;
    if (tmp_pl < PSR.cpl)
        tmp_pl = PSR.cpl;

    if (fault_form) {
        tlb_translate(GR[r3], 1, itype, tmp_pl, &mattr, &defer);
    } else {
        GR[r1] = tlb_grant_permission(GR[r3], itype, tmp_pl);
        GR[r1].nat = 0;
    }
}
```

Interruptions:

- Illegal Operation fault
- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Dirty Bit fault
- Data Access Bit fault
- Data Debug fault
Parallel Sum of Absolute Difference

**Format:**  
\[(qp) \text{psad} \quad r_1 = r_2, r_3\]

**Description:**  
The unsigned 8-bit elements of GR \(r_2\) are subtracted from the unsigned 8-bit elements of GR \(r_3\). The absolute value of each difference is accumulated across the elements and placed in GR \(r_1\).

**Figure 2-38. Parallel Sum of Absolute Difference Example**

**Operation:**  
if \((\text{PR}[qp])\) {
    check_target_register\(r_1\);
    x[0] = GR\[r_2\]{7:0}; y[0] = GR\[r_3\]{7:0};
    x[1] = GR\[r_2\]{15:8}; y[1] = GR\[r_3\]{15:8};
    GR\[r_3\] = 0;
    for \((i = 0; i < 8; i++)\) {
        temp\[i\] = zero_ext\(x[i]\), 8\) - zero_ext\(y[i], 8\);  
        if \((\text{temp}[i] < 0)\)
            temp\[i\] = -temp\[i\];
        GR\[r_1\] += temp\[i\];
    }
    GR\[r_1\].nat = GR\[r_2\].nat || GR\[r_3\].nat;
}

**Interruptions:**  Illegal Operation fault
Parallel Shift Left

Format:  
- \((qp)\) pshl2 \( r_1 = r_2, r_3 \)
- \((qp)\) pshl2 \( r_1 = r_2, \text{count}_5 \)
- \((qp)\) pshl4 \( r_1 = r_2, r_3 \)
- \((qp)\) pshl4 \( r_1 = r_2, \text{count}_5 \)

Description: The data elements of GR \( r_2 \) are each independently shifted to the left by the scalar shift count in GR \( r_3 \), or in the immediate field \( \text{count}_5 \). The low-order bits of each element are filled with zeros. The shift count is interpreted as unsigned. Shift counts greater than 15 (for 16-bit quantities) or 31 (for 32-bit quantities) yield all zero results. The results are placed in GR \( r_1 \).

Figure 2-39. Parallel Shift Left Example

Operation:  
```c
if (PR[qp]) {
    check_target_register(r1);
    shift_count = (variable_form ? GR[r3] : count5);
    tmp_nat = (variable_form ? GR[r3].nat : 0);
    if (two_byte_form) { // two_byte_form
        if (shift_count > 16)
            shift_count = 16;
        GR[r1]{15:0} = GR[r2]{15:0} << shift_count;
        GR[r1]{31:16} = GR[r2]{31:16} << shift_count;
        GR[r1]{47:32} = GR[r2]{47:32} << shift_count;
        GR[r1]{63:48} = GR[r2]{63:48} << shift_count;
    } else { // four_byte_form
        if (shift_count > 32)
            shift_count = 32;
        GR[r1]{31:0} = GR[r2]{31:0} << shift_count;
        GR[r1]{63:32} = GR[r2]{63:32} << shift_count;
    }
    GR[r1].nat = GR[r2].nat || tmp_nat;
}
```

Interruptions: Illegal Operation fault
Parallel Shift Left and Add

Format: \((qp)\) pshladd2 \(r_1 = r_2, count_2, r_3\)

Description: The four signed 16-bit data elements of GR \(r_2\) are each independently shifted to the left by \(count_2\) bits (shifting zeros into the low-order bits), and added to the four signed 16-bit data elements of GR \(r_3\). Both the left shift and the add operations are saturating: if the result of either the shift or the add is not representable as a signed 16-bit value, the final result is saturated. The four signed 16-bit results are placed in GR \(r_1\). The first operand can be shifted by 1, 2 or 3 bits.

Operation: 

\[
\begin{align*}
\text{if (PR[qp])} & \{ \\
& \text{check_target_register}(r_1); \\
& \quad x[0] = \text{GR}[r_2]\{15:0\}; \quad y[0] = \text{GR}[r_3]\{15:0\}; \\
& \quad x[1] = \text{GR}[r_2]\{31:16\}; \quad y[1] = \text{GR}[r_3]\{31:16\}; \\
& \quad x[2] = \text{GR}[r_2]\{47:32\}; \quad y[2] = \text{GR}[r_3]\{47:32\}; \\
& \quad x[3] = \text{GR}[r_2]\{63:48\}; \quad y[3] = \text{GR}[r_3]\{63:48\}; \\
& \quad \text{max} = \text{sign_ext}(0x7fff, 16); \\
& \quad \text{min} = \text{sign_ext}(0x8000, 16); \\
& \quad \text{for (i = 0; i < 4; i++)} \{ \\
& \quad \quad \text{temp}[i] = \text{sign_ext}(x[i], 16) \ll count_2; \\
& \quad \quad \text{if (temp}[i] > \text{max}) \\
& \quad \quad \quad \text{res}[i] = \text{max}; \\
& \quad \quad \text{else if (temp}[i] < \text{min}) \\
& \quad \quad \quad \text{res}[i] = \text{min}; \\
& \quad \quad \text{else} \{ \\
& \quad \quad \quad \text{res}[i] = \text{temp}[i] + \text{sign_ext}(y[i], 16); \\
& \quad \quad \quad \text{if (res}[i] > \text{max}) \\
& \quad \quad \quad \quad \text{res}[i] = \text{max}; \\
& \quad \quad \quad \text{if (res}[i] < \text{min}) \\
& \quad \quad \quad \quad \text{res}[i] = \text{min}; \\
& \quad \quad \} \\
& \quad \} \\
& \text{GR}[r_1] = \text{concatenate4}([\text{res}[3], \text{res}[2], \text{res}[1], \text{res}[0]]); \\
& \text{GR}[r_1].\text{nat} = \text{GR}[r_2].\text{nat} \lor \text{GR}[r_3].\text{nat}; \\
\}
\]

Interruptions: Illegal Operation fault
Parallel Shift Right

Format:

\begin{align*}
(qp) \text{ pshr2} & \quad r_j = r_3, r_2 \\
(qp) \text{ pshr2} & \quad r_j = r_3, \text{ count}_5 \\
(qp) \text{ pshr2.u} & \quad r_j = r_3, r_2 \\
(qp) \text{ pshr2.u} & \quad r_j = r_3, \text{ count}_5 \\
(qp) \text{ pshr4} & \quad r_j = r_3, r_2 \\
(qp) \text{ pshr4} & \quad r_j = r_3, \text{ count}_5 \\
(qp) \text{ pshr4.u} & \quad r_j = r_3, r_2 \\
(qp) \text{ pshr4.u} & \quad r_j = r_3, \text{ count}_5
\end{align*}

Description: The data elements of GR $r_3$ are each independently shifted to the right by the scalar shift count in GR $r_2$, or in the immediate field \textit{count}_5. The high-order bits of each element are filled with either the initial value of the sign bits of the data elements in GR $r_3$ (arithmetic shift) or zeros (logical shift). The shift count is interpreted as unsigned. Shift counts greater than 15 (for 16-bit quantities) or 31 (for 32-bit quantities) yield all zero or all one results depending on the initial values of the sign bits of the data elements in GR $r_3$ and whether a signed or unsigned shift is done. The results are placed in GR $r_j$. 
Operation: 

    if (PR[qp]) {
        check_target_register(r);
    
        shift_count = (variable_form ? GR[r2] : count);
        tmp_nat = (variable_form ? GR[r2].nat : 0);
    
        if (two_byte_form) { // two_byte_form
            shift_count = 16;
            if (unsigned_form) { // unsigned shift
                GR[r][15:0] = shift_right_unsigned(zero_ext(GR[r3][15:0], 16), shift_count);
                GR[r][31:16] = shift_right_unsigned(zero_ext(GR[r3][31:16], 16), shift_count);
                GR[r][47:32] = shift_right_unsigned(zero_ext(GR[r3][47:32], 16), shift_count);
                GR[r][63:48] = shift_right_unsigned(zero_ext(GR[r3][63:48], 16), shift_count);
                
            } else { // signed shift
                GR[r][15:0] = shift_right_signed(sign_ext(GR[r3][15:0], 16), shift_count);
                GR[r][31:16] = shift_right_signed(sign_ext(GR[r3][31:16], 16), shift_count);
                GR[r][47:32] = shift_right_signed(sign_ext(GR[r3][47:32], 16), shift_count);
                GR[r][63:48] = shift_right_signed(sign_ext(GR[r3][63:48], 16), shift_count);
            }
        } else { // four_byte_form
            shift_count = 32;
            if (unsigned_form) { // unsigned shift
                GR[r][31:0] = shift_right_unsigned(zero_ext(GR[r3][31:0], 32), shift_count);
                GR[r][63:32] = shift_right_unsigned(zero_ext(GR[r3][63:32], 32), shift_count);
            } else { // signed shift
                GR[r][31:0] = shift_right_signed(sign_ext(GR[r3][31:0], 32), shift_count);
                GR[r][63:32] = shift_right_signed(sign_ext(GR[r3][63:32], 32), shift_count);
            }
        }
    
    GR[r].nat = GR[r3].nat || tmp_nat;
}

Interruptions: Illegal Operation fault
Parallel Shift Right and Add

Format: \((qp)\) pshradd2 \(r_1 = r_2, count_2, r_3\)

Description: The four signed 16-bit data elements of GR \(r_2\) are each independently shifted to the right by \(count_2\) bits, and added to the four signed 16-bit data elements of GR \(r_3\). The right shift operation fills the high-order bits of each element with the initial value of the sign bits of the data elements in GR \(r_2\). The add operation is performed with signed saturation. The four signed 16-bit results of the add are placed in GR \(r_1\). The first operand can be shifted by 1, 2 or 3 bits.

Operation:  
\[
\text{if (PR[qp])} \{ \\
\quad \text{check_target_register}(r_1); \\
\quad x[0] = GR[r_2]{15:0}; \quad y[0] = GR[r_3]{15:0}; \\
\quad x[1] = GR[r_2]{31:16}; \quad y[1] = GR[r_3]{31:16}; \\
\quad max = \text{sign_ext}(0x7fff, 16); \\
\quad min = \text{sign_ext}(0x8000, 16); \\
\quad \text{for (i = 0; i < 4; i++)} \{ \\
\quad \quad \text{temp[i] = shift_right_signed(sign_ext(x[i], 16), count_2);} \\
\quad \quad \text{res[i] = temp[i] + sign_ext(y[i], 16);} \\
\quad \quad \text{if (res[i] > max)} \\
\quad \quad \quad \text{res[i] = max;} \\
\quad \quad \text{if (res[i] < min)} \\
\quad \quad \quad \text{res[i] = min;} \\
\quad \}\n\]

\[
\text{GR[r_1] = concatenate4(res[3], res[2], res[1], res[0]);} \\
\text{GR[r_1].nat = GR[r_2].nat || GR[r_3].nat;} \\
\}
\]

Interruptions: Illegal Operation fault
Parallel Subtract

Format:

- $(qp) \ psub1 \ r_1 = r_2, r_3$
  - one_byte_form, modulo_form
  - A9
- $(qp) \ psub1.sss \ r_1 = r_2, r_3$
  - one_byte_form, sss_saturation_form
  - A9
- $(qp) \ psub1.uus \ r_1 = r_2, r_3$
  - one_byte_form, uus_saturation_form
  - A9
- $(qp) \ psub1.uuu \ r_1 = r_2, r_3$
  - one_byte_form, uuu_saturation_form
  - A9
- $(qp) \ psub2 \ r_1 = r_2, r_3$
  - two_byte_form, modulo_form
  - A9
- $(qp) \ psub2.sss \ r_1 = r_2, r_3$
  - two_byte_form, sss_saturation_form
  - A9
- $(qp) \ psub2.uus \ r_1 = r_2, r_3$
  - two_byte_form, uus_saturation_form
  - A9
- $(qp) \ psub2.uuu \ r_1 = r_2, r_3$
  - two_byte_form, uuu_saturation_form
  - A9
- $(qp) \ psub4 \ r_1 = r_2, r_3$
  - four_byte_form, modulo_form
  - A9

Description: The sets of elements from the two source operands are subtracted, and the results placed in GR $r_1$. If the difference between two elements cannot be represented in the result element and a saturation completer is specified, then saturation clamping is performed. The saturation can either be signed or unsigned, as given in Table 2-46. If the difference of two elements is larger than the upper limit value, the result is the upper limit value. If it is smaller than the lower limit value, the result is the lower limit value. The saturation limits are given in Table 2-47.

Table 2-46. Parallel Subtract Saturation Completers

<table>
<thead>
<tr>
<th>Completer</th>
<th>Result $r_1$ treated as</th>
<th>Source $r_2$ treated as</th>
<th>Source $r_3$ treated as</th>
</tr>
</thead>
<tbody>
<tr>
<td>sss</td>
<td>signed</td>
<td>signed</td>
<td>signed</td>
</tr>
<tr>
<td>uus</td>
<td>unsigned</td>
<td>unsigned</td>
<td>signed</td>
</tr>
<tr>
<td>uuu</td>
<td>unsigned</td>
<td>unsigned</td>
<td>unsigned</td>
</tr>
</tbody>
</table>

Table 2-47. Parallel Subtract Saturation Limits

<table>
<thead>
<tr>
<th>Size</th>
<th>Element Width</th>
<th>Result $r_1$ signed</th>
<th>Result $r_1$ unsigned</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td>Upper Limit</td>
<td>Lower Limit</td>
</tr>
<tr>
<td>1</td>
<td>8 bit</td>
<td>0x7f</td>
<td>0x80</td>
</tr>
<tr>
<td>2</td>
<td>16 bit</td>
<td>0x7fff</td>
<td>0x8000</td>
</tr>
</tbody>
</table>

Figure 2-40. Parallel Subtract Example

```
GR $r_3$:                  GR $r_3$:            
GR $r_2$:                  GR $r_2$:            
GR $r_1$:                  GR $r_1$:            

psub1                      psub2
```
Operation:
  if (PR[gp]) {
    check_target_register(r1);  // one-byte elements
    if (one_byte_form) {
      x[0] = GR[r2]{7:0};  y[0] = GR[r3]{7:0};
      if (sss_saturation_form) {  // sss_saturation_form
        max = sign_ext(0x7f, 8);
        min = sign_ext(0x80, 8);
        for (i = 0; i < 8; i++) {
          temp[i] = sign_ext(x[i], 8) - sign_ext(y[i], 8);
        }
      } else if (uus_saturation_form) {  // uus_saturation_form
        max = 0xff;
        min = 0x00;
        for (i = 0; i < 8; i++) {
          temp[i] = zero_ext(x[i], 8) - sign_ext(y[i], 8);
        }
      } else if (uuu_saturation_form) {  // uuu_saturation_form
        max = 0xff;
        min = 0x00;
        for (i = 0; i < 8; i++) {
          temp[i] = zero_ext(x[i], 8) - zero_ext(y[i], 8);
        }
      } else {  // modulo_form
        for (i = 0; i < 8; i++) {
          temp[i] = zero_ext(x[i], 8) - zero_ext(y[i], 8);
        }
      }
    }
  }

GR[r1] = concatenate8(temp[7], temp[6], temp[5], temp[4],
                        temp[3], temp[2], temp[1], temp[0]);

  } else if (two_byte_form) {  // two-byte elements
    x[0] = GR[r2]{15:0};  y[0] = GR[r3]{15:0};
    if (sss_saturation_form) {  // sss_saturation_form
      max = sign_ext(0x7fff, 16);
      min = sign_ext(0x8000, 16);
      for (i = 0; i < 4; i++) {
        temp[i] = sign_ext(x[i], 16) - sign_ext(y[i], 16);
      }
    } else if (uus_saturation_form) {  // uus_saturation_form

  }
max = 0xffffffff;
min = 0x00000000;
for (i = 0; i < 4; i++) {
    temp[i] = zero_ext(x[i], 16) - sign_ext(y[i], 16);
} else if (uuu_saturation_form) { // uuu_saturation_form
    max = 0xffffffff;
    min = 0x00000000;
    for (i = 0; i < 4; i++) {
        temp[i] = zero_ext(x[i], 16) - zero_ext(y[i], 16);
    }
} else { // modulo_form
    for (i = 0; i < 4; i++) {
        temp[i] = zero_ext(x[i], 16) - zero_ext(y[i], 16);
    }
}

if (ssss_saturation_form || uus_saturation_form ||
    uuu_saturation_form) {
    for (i = 0; i < 4; i++) {
        if (temp[i] > max)
            temp[i] = max;
        if (temp[i] < min)
            temp[i] = min;
    }
}

GR[r1] = concatenate4(temp[3], temp[2], temp[1], temp[0]);
} else { // four-byte elements
    x[0] = GR[r2]{31:0};
    y[0] = GR[r3]{31:0};
    x[1] = GR[r2]{63:32};
    y[1] = GR[r3]{63:32};
    for (i = 0; i < 2; i++) { // modulo_form
        temp[i] = zero_ext(x[i], 32) - zero_ext(y[i], 32);
    }
    GR[r1] = concatenate2(temp[1], temp[0]);
}

GR[r1].nat = GR[r2].nat || GR[r3].nat;

**Interruptions:** Illegal Operation fault
Purge Translation Cache Entry

Format: \((qp)\) \texttt{ptc.e} \(r_3\)

Description: One or more translation entries are purged from the local processor’s instruction and data translation cache. Translation Registers and the VHPT are not modified.

The number of translation cache entries purged is implementation specific. Some implementations may purge all levels of the translation cache hierarchy with one iteration of \texttt{PTC.e}, while other implementations may require several iterations to flush all levels, sets and associativities of both instruction and data translation caches. \(GR\ r_3\) specifies an implementation specific parameter associated with each iteration.

The following loop is defined to flush the entire translation cache for all processor models. Software can acquire parameters through a processor dependent layer that is accessed through a procedural interface. The selected region registers must remain unchanged during the loop.

```c
disable_interrupts();
addr = base;
for (i = 0; i < count1; i++) {
  for (j = 0; j < count2; j++) {
    ptc.e(addr);
    addr += stride2;
  }
  addr += stride1;
}
enable_interruptions();
```

Operation: if \((PR[qp])\) {
  if \((PSR.cpl! = 0)\)
    privileged_operation_fault(0);
  if \((GR[r_3].nat)\)
    register_nat_consumption_fault(0);
  tlb_purge_translation_cache(GR[r_3]);
}

Interruptions: Privileged Operation fault

Register NaT Consumption fault

Serialization: Software must issue a data serialization operation to ensure the purge is complete before issuing a data access or non-access reference dependent upon the purge. Software must issue instruction serialize operation before fetching an instruction dependent upon the purge.
Purge Global Translation Cache

Format:

\[(qp) \textit{ptc.g} \ r_3, r_2 \quad \text{global form} \quad \text{M45}\]
\[(qp) \textit{ptc.ga} \ r_3, r_2 \quad \text{global_alat form} \quad \text{M45}\]

Description:

The instruction and data translation cache for each processor in the local TLB coherence domain are searched for all entries whose virtual address and page size partially or completely overlap the specified purge virtual address and purge address range. These entries are removed.

The purge virtual address is specified by GR\[r_3\] bits\{60:0\} and the purge region identifier is selected by GR\[r_3\] bits\{63:61\}. GR\[r_2\] specifies the address range of the purge as \(1 \leq \text{GR}[r_2][(7:2)]\) bytes in size.

Based on the processor model, the translation cache may be also purged of more translations than specified by the purge parameters up to and including removal of all entries within the translation cache.

\textit{ptc.g} has release semantics and is guaranteed to be made visible after all previous data memory accesses are made visible. The memory fence instruction forces all processors to complete the purge prior to any subsequent memory operations. Serialization is still required to observe the side-effects of a translation being removed.

\textit{ptc.g} must be the last instruction in an instruction group; otherwise, its behavior (including its ordering semantics) is undefined.

The behavior of the \textit{ptc.ga} instruction is similar to \textit{ptc.g}. In addition to the behavior specified for \textit{ptc.g} the \textit{ptc.ga} instruction encodes an extra bit of information in the broadcast transaction. This information specifies the purge is due to a page remapping as opposed to a protection change or page tear down. The remote processors within the coherence domain will then take whatever additional action is necessary to make their ALAT consistent. The local ALAT is not purged.

This instruction can only be executed at the most privileged level.

Only one global purge transaction may be issued at a time by all processors, the operation is undefined otherwise. Software is responsible for enforcing this restriction.

Propagation of \textit{ptc.g} between multiple local TLB coherence domains is platform dependent, and must be handled by software. It is expected that the local TLB coherence domain covers at least the processors on the same local bus.

Operation:

\[
\text{if (PR[qp]) \{}
\quad \text{if (!followed_by_stop())}
\quad \text{undefined_behavior();}
\quad \text{if (PSR.cpl != 0)}
\quad \text{privileged_operation_fault(0);}\]
\[
\quad \text{if (GR[r_3].nat || GR[r_2].nat)}
\quad \text{register_nat_consumption_fault(0);}\]
\[
\quad \text{if (unimplemented_virtual_address(GR[r_3]))}
\quad \text{unimplemented_data_address_fault(0);}\]
\[
\quad \text{tmp_rid = RR[GR[r_3][63:61]].rid;}
\quad \text{tmp_va = GR[r_3][60:0];}
\quad \text{tmp_size = GR[r_2][7:2];}
\quad \text{tmp_va = align_to_size_boundary(tmp_va, tmp_size);}\]
\[
\quad \text{tlb_must_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);}
\quad \text{tlb_must_purge_itc_entries(tmp_rid, tmp_va, tmp_size);}
\quad \text{if (global_alat_form) tmp_ptc_type = GLOBAL_ALAT_FORM;}
\quad \text{else tmp_ptc_type = GLOBAL_FORM;}\]
\[
\quad \text{tlb_broadcast_purge(tmp_rid, tmp_va, tmp_size, tmp_ptc_type);}
\}
\]
**Interruptions:**
- Machine Check abort
- Privileged Operation fault
- Register NaT Consumption fault
- Unimplemented Data Address fault

**Serialization:**
The broadcast purge TC is not synchronized with the instruction stream on a remote processor. Software cannot depend on any such synchronization with the instruction stream. Hardware on the remote machine cannot reload an instruction from memory or cache after acknowledging a broadcast purge TC without first retranslating the I-side access in the TLB. Hardware may continue to use a valid private copy of the instruction stream data (possibly in an I-buffer) obtained prior to acknowledging a broadcast purge TC to a page containing the i-stream data. Hardware must retranslate access to an instruction page upon an interruption or any explicit or implicit instruction serialization event (e.g., `srlz.i`, `rfi`).

Software must issue the appropriate data and/or instruction serialization operation to ensure the purge is completed before a local data access, non-access reference, or local instruction fetch access dependent upon the purge.
Purge Local Translation Cache

Format:  
\[(qp) \ ptc.l \ r_3, r_2\]

Description:  
The instruction and data translation cache of the local processor is searched for all entries whose
virtual address and page size partially or completely overlap the specified purge virtual address and
purge address range. All these entries are removed.

The purge virtual address is specified by GR \[r_3\] bits \{60:0\} and the purge region identifier is
selected by GR \[r_3\] bits \{63:61\}. GR \[r_2\] specifies the address range of the purge as \(1 \times 2^{GR[r_2][7:2]}\)
bytes in size.

The processor ensures that all entries matching the purging parameters are removed. However,
based on the processor model, the translation cache may be also purged of more translations than
specified by the purge parameters up to and including removal of all entries within the translation
cache.

This instruction can only be executed at the most privileged level.

This is a local operation, no purge broadcast to other processors occurs in a multiprocessor system.

Operation:  
\[\begin{align*}
&\text{if (PR[qp])} \\
&\quad \text{if (PSR.cpl} ! = 0) \\
&\quad \quad \text{privileged-operation-fault(0);} \\
&\quad \text{if (GR[r_3].nat || GR[r_2].nat)} \\
&\quad \quad \text{register-nat-consumption-fault(0);} \\
&\quad \text{if (unimplemented-virtual-address(GR[r_3]))} \\
&\quad \quad \text{unimplemented-data-address-fault(0);} \\
&\quad \text{tmp_rid = RR[GR[r_3][63:61]].rid;} \\
&\quad \text{tmp_va = GR[r_3][60:0];} \\
&\quad \text{tmp_size = GR[r_2][7:2];} \\
&\quad \text{tmp_va = align_to_size_boundary(tmp_va, tmp_size);} \\
&\quad \text{tlb_must_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);} \\
&\quad \text{tlb_must_purge_itc_entries(tmp_rid, tmp_va, tmp_size);} \\
\end{align*}\]

Interuptions:  
Machine Check abort  
Register NaT Consumption fault  
Privileged Operation fault  
Unimplemented Data Address fault

Serialization:  
Software must issue the appropriate data and/or instruction serialization operation to ensure the
purge is completed before a data access, non-access reference, or instruction fetch access dependent
upon the purge.
**Purge Translation Register**

**Format:**

\[(qp) \text{ ptr.d } r_3, r_2\]

\[(qp) \text{ ptr.i } r_3, r_2\]

\text{data form M45}

\text{instruction form M45}

**Description:**

In the data form of this instruction, the data translation registers and caches are searched for all entries whose virtual address and page size partially or completely overlap the specified purge virtual address and purge address range. All these entries are removed. Entries in the instruction translation registers are unaffected by the data form of the purge.

In the instruction form, the instruction translation registers and caches are searched for all entries whose virtual address and page size partially or completely overlap the specified purge virtual address and purge address range. All these entries are removed. Entries in the data translation registers are unaffected by the instruction form of the purge.

In addition, in both forms, the instruction and data translation cache may be purged of more translations than specified by the purge parameters up to and including removal of all entries within the translation cache.

The purge virtual address is specified by GR \( r_3 \) bits\{60:0\} and the purge region identifier is selected by GR \( r_3 \) bits \{63:61\}. GR \( r_2 \) specifies the address range of the purge as \(1 << \text{GR}[r_2][7:2]\) bytes in size.

This instruction can only be executed at the most privileged level.

This is a local operation, no purge broadcast to other processors occurs in a multiprocessor system.

As described in “Translation Cache (TC)” on page 1:40 in *Volume 2: System Architecture*, the processor may use the translation caches to cache virtual address mappings held by translation registers. The \texttt{ptr.i} and \texttt{ptr.d} instructions purge the processor’s translation registers as well as cached translation register copies that may be contained in the respective translation caches.
Operation: if (PR qp) {
  if (PSR.cpl != 0)
    privileged_operation_fault(0);
  if (GR r3.nat || GR r2.nat)
    register_nat_consumption_fault(0);
  if (unimplemented_virtual_address(GR r3))
    unimplemented_data_address_fault(0);

tmp_rid = RR[GR r3][63:61].rid;
tmp_va = GR r3[60:0];
tmp_size = GR r2[7:2];
tmp_va = align_to_size_boundary(tmp_va, tmp_size);

  if (data_form) {
    tlb_must_purge_dtr_entries(tmp_rid, tmp_va, tmp_size);
    tlb_must_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);
    tlb_may_purge_itc_entries(tmp_rid, tmp_va, tmp_size);
  } else { // instruction_form
    tlb_must_purge_itr_entries(tmp_rid, tmp_va, tmp_size);
    tlb_must_purge_itc_entries(tmp_rid, tmp_va, tmp_size);
    tlb_may_purge_dtc_entries(tmp_rid, tmp_va, tmp_size);
  }
}

Interruptions: Privileged Operation fault Unimplemented Data Address fault
Register NaT Consumption fault

Serialization: For the data form, software must issue a data serialization operation to ensure the purge is completed before issuing an instruction dependent upon the purge. For the instruction form, software must issue an instruction serialization operation to ensure the purge is completed before fetching an instruction dependent on that purge.
Return From Interruption

Format: \texttt{rfi}

Description: The machine context prior to an interruption is restored. PSR is restored from IPSR, IPSR is unmodified, and IP is restored from IIP. Execution continues at the bundle address loaded into the IP, and the instruction slot loaded into PSR.ri.

This instruction must be immediately followed by a stop. Otherwise, an Illegal Operation fault is taken. This instruction switches to the register bank specified by IPSR.bn. Instructions in the same instruction group that access GR16 to GR31 reference the previous register bank. Subsequent instruction groups reference the new register bank.

This instruction performs instruction serialization, which ensures:

- prior modifications to processor register resources that affect fetching of subsequent instruction groups are observed.
- prior modifications to processor register resources that affect subsequent execution or data memory accesses are observed.
- prior memory synchronization (\texttt{sync.i}) operations have taken effect on the local processor instruction cache.
- subsequent instruction group fetches (including the target instruction group) are re-initiated after \texttt{rfi} completes.

The \texttt{rfi} instruction must be in an instruction group after the instruction group containing the operation that is to be serialized.

This instruction can only be executed at the most privileged level. This instruction can not be predicated.

Execution of this instruction is undefined if PSR.ic or PSR.i are 1. Software must ensure that an interruption cannot occur that could modify IIP, IPSR, or IFS between when they are written and the subsequent \texttt{rfi}.

This instruction does not take Lower Privilege Transfer, Taken Branch or Single Step traps.

If this instruction sets PSR.ri to 2 and the target is an MLX bundle, then an Illegal Operation fault will be taken on the target bundle.

If IPSR.is is 1, control is resumed in the IA-32 instruction set at the virtual linear address specified by IIP[31:0]. PSR.di does not inhibit instruction set transitions for this instruction. If PSR.dfh is 1 after \texttt{rfi} completes execution, a Disabled FP Register fault is raised on the target IA-32 instruction.

If IPSR.is is 1 and an Unimplemented Instruction Address trap is taken, IIP will contain the original 64-bit target IP. (The value will not have been zero extended from 32 bits.)

When entering the IA-32 instruction set, the size of the current stack frame is set to zero, and all stacked general registers are left in an undefined state. Software can not rely on the value of these registers across an instruction set transition. Software must ensure that BSPSTORE==BSP on entry to the IA-32 instruction set, otherwise undefined behavior may result.

Software must issue a \texttt{mf} instruction before this instruction if memory ordering is required between IA-32 processor-consistent and Itanium unordered memory references. The processor does not ensure Itanium-instruction-set-generated writes into the instruction stream are seen by subsequent IA-32 instructions.
Software must ensure the code segment descriptor and selector are loaded before issuing this instruction. If the target EIP value exceeds the code segment limit or has a code segment privilege violation, an IA-32_Exception(GPFault) exception is raised on the target IA-32 instruction. For entry into 16-bit IA-32 code, if IIP is not within 64K-bytes of CSD.base a GPFault is raised on the target instruction.

EFLAGS.rf and PSR.id are unmodified until the successful completion of the target IA-32 instruction. PSR.da, PSR.dd, PSR.ia and PSR.ed are cleared to zero before the target IA-32 instruction begins execution.

IA-32 instruction set execution leaves the contents of the ALAT undefined. Software can not rely on ALAT state across an instruction set transition. On entry to IA-32 code, existing entries in the ALAT are ignored.

**Operation:**

```c
if (!followed_by_stop())
    illegal_operation_fault();

unimplemented_address = 0;
if (PSR.cpl != 0)
    privileged_operation_fault(0);

taken_rfi = 1;

PSR = CR[IPSR];
if (CR[IPSR].is == 1) { //resume IA-32 instruction set
    tmp_IP = CR[IIP];
    if (CR[IPSR].it && unimplemented_virtual_address(tmp_IP))
        //compute effective instruction pointer
        EIP{31:0} = CR[IIP]{31:0} - AR[CSD].Base;
        //force zero-sized restored frame
        rse_restore_frame(0, 0, CFM.sof);
    CFM.sof = 0;
    CFM.sol = 0;
    CFM.sor = 0;
    CFM.rrb.gr = 0;
    CFM.rrb.fr = 0;
    CFM.rrb.pr = 0;
    rse_invalidate_non_current_regs();
    //The register stack engine is disabled during IA-32
    //instruction set execution.
} else { //return to Itanium instruction set
    tmp_IP = CR[IIP] & ~0xf;
    slot = CR[IPSR].ri;
    if (CR[IPSR].it && unimplemented_virtual_address(tmp_IP))
        //compute effective instruction pointer
        tmp_growth = -CFM.sof;
        alat_frame_update(-CR[IFS].ifm.sof, 0);
        rse_restore_frame(CR[IFS].ifm.sof, tmp_growth, CFM.sof);
        CFM = CR[IFS].ifm;
    } 
    rse_enable_current_frame_load();
} 
IP = tmp_IP;
instructionserialize();
if (unimplemented_address)
    unimplemented_instruction_address_trap(0, tmp_IP);```
**Interruptions:**
- Illegal Operation fault
- Privileged Operation fault
- Unimplemented Instruction Address trap
- Additional Faults on IA-32 target instructions
- IA-32_Exception(GPFault)
- Disabled FP Reg Fault if PSR.df=1

**Serialization:**
An implicit instruction and data serialization operation is performed.
Reset System Mask

Format: \((q)p\) \texttt{rsm imm}_{24}\hfill \text{M44}

Description: The complement of the \texttt{imm}_{24} operand is ANDed with the system mask (PSR\{23:0\}) and the result is placed in the system mask. See “Processor Status Register (PSR)” on page 1:18 in \textit{Volume 2: System Architecture}.

The PSR system mask can only be written at the most privileged level.

When the current privilege level is zero (PSR.cpl is 0), an \texttt{rsm} instruction whose mask includes PSR.i may cause external interrupts to be disabled for an implementation-dependent number of instructions, even if the qualifying predicate for the \texttt{rsm} instruction is false. Architecturally, the extents of this external interrupt disabling “window” are defined as follows:

- External interrupts may be disabled for any instructions in the same instruction group as the \texttt{rsm}, including those that precede the \texttt{rsm} in sequential program order, regardless of the value of the qualifying predicate of the \texttt{rsm} instruction.
- If the qualifying predicate of the \texttt{rsm} is true, then external interrupts are disabled immediately following the \texttt{rsm} instruction.
- If the qualifying predicate of the \texttt{rsm} is false, then external interrupts may be disabled until the next data serialization operation that follows the \texttt{rsm} instruction.

The external interrupt disable window is guaranteed to be no larger than defined by the above criteria, but it may be smaller, depending on the processor implementation.

When the current privilege level is non-zero (PSR.cpl is not 0), an \texttt{rsm} instruction whose mask includes PSR.i may briefly disable external interrupts, regardless of the value of the qualifying predicate of the \texttt{rsm} instruction. However, processor implementations guarantee that non-privileged code cannot lock out external interrupts indefinitely (e.g., via an arbitrarily long sequence of \texttt{rsm} instructions with zero-valued qualifying predicates).

Operation:

\begin{verbatim}
if (PR[qp]) {
  if (PSR.cpl != 0)
    privileged_operation_fault(0);
  if (is_reserved_field(PSR_TYPE, PSR_SM, imm_{24}))
    reserved_register_field_fault();
  if (imm_{24}{1}) PSR{1} = 0; // be
  if (imm_{24}{2}) PSR{2} = 0; // up
  if (imm_{24}{3}) PSR{3} = 0; // ac
  if (imm_{24}{4}) PSR{4} = 0; // mfl
  if (imm_{24}{5}) PSR{5} = 0; // mfh
  if (imm_{24}{13}) PSR{13} = 0; // ic
  if (imm_{24}{14}) PSR{14} = 0; // i
  if (imm_{24}{15}) PSR{15} = 0; // pk
  if (imm_{24}{17}) PSR{17} = 0; // dt
  if (imm_{24}{18}) PSR{18} = 0; // dfl
  if (imm_{24}{19}) PSR{19} = 0; // dfh
  if (imm_{24}{20}) PSR{20} = 0; // sp
  if (imm_{24}{21}) PSR{21} = 0; // pp
  if (imm_{24}{22}) PSR{22} = 0; // di
  if (imm_{24}{23}) PSR{23} = 0; // si
}
\end{verbatim}

Interruptions: Privileged Operation fault \hspace{1cm} Reserved Register/Field fault
Serialization: Software must use a data serialize or instruction serialize operation before issuing instructions dependent upon the altered PSR bits – except the PSR.i bit. The PSR.i bit is implicitly serialized and the processor ensures that external interrupts are masked by the time the next instruction executes.
Reset User Mask

Format: \((qp)\) rum \(imm_{24}\)

Description: The complement of the \(imm_{24}\) operand is ANDed with the user mask \((PSR\{5:0}\)) and the result is placed in the user mask. See “Processor Status Register (PSR)” on page 1:18 in Volume 2: System Architecture.

PSR.up is only cleared if the secure performance monitor bit \((PSR.sp)\) is zero. Otherwise PSR.up is not modified.

Operation:

\[
\text{if (PR[qp]) } \{ \\
\quad \text{if (is_reserved_field\(\text{PSR\_TYPE, PSR\_UM, imm_{24}\}))} \\
\quad \quad \text{reserved_register_field_fault();} \\
\quad \text{if (imm_{24}(1)) PSR\{1\} = 0;} // be \\
\quad \text{if (imm_{24}(2) \&\& PSR.sp == 0) } //\text{non-secure perf monitor} \\
\quad \quad \text{PSR\{2\} = 0;} // up \\
\quad \text{if (imm_{24}(3)) PSR\{3\} = 0;} // \text{ac} \\
\quad \text{if (imm_{24}(4)) PSR\{4\} = 0;} // \text{mf1} \\
\quad \text{if (imm_{24}(5)) PSR\{5\} = 0;} // \text{mfh} \\
\}\]

Interruptions: Reserved Register/Field fault

Serialization: All user mask modifications are observed by the next instruction group.
Set Floating-point Value, Exponent, or Significand

**Format:**

- \((qp)\) setf.s \(f_1 = r_2\) \hspace{1cm} \text{single\_form} \hspace{0.5cm} \text{M18}
- \((qp)\) setf.d \(f_1 = r_2\) \hspace{1cm} \text{double\_form} \hspace{0.5cm} \text{M18}
- \((qp)\) setf.exp \(f_1 = r_2\) \hspace{1cm} \text{exponent\_form} \hspace{0.5cm} \text{M18}
- \((qp)\) setf.sig \(f_1 = r_2\) \hspace{1cm} \text{significand\_form} \hspace{0.5cm} \text{M18}

**Description:**

In the single and double forms, \(GR_r_2\) is treated as a single precision (in the single\_form) or double precision (in the double\_form) memory representation, converted into floating-point register format, and placed in \(FR_f_1\), as shown in Figure 5-7 on page 1:87 and Figure 5-8 on page 1:87 in *Volume 1: Application Architecture*, respectively.

In the exponent\_form, bits 16:0 of \(GR_r_2\) are copied to the exponent field of \(FR_f_1\) and bit 17 of \(GR_r_2\) is copied to the sign bit of \(FR_f_1\). The significand field of \(FR_f_1\) is set to one (0x800...000).

**Figure 2-41. Function of setf.exp**

![Diagram showing the function of setf.exp](image)

In the significand\_form, the value in \(GR_r_2\) is copied to the significand field of \(FR_f_1\).

The exponent field of \(FR_f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of \(FR_f_1\) is set to positive (0).

**Figure 2-42. Function of setf.sig**

![Diagram showing the function of setf.sig](image)

For all forms, if the NaT bit corresponding to \(r_2\) is equal to 1, \(FR_f_1\) is set to NaTVal instead of the computed result.
Operation:  \[\text{if (PR[qp])} \{\]
\[\text{fp\_check\_target\_register}(f_1);\]
\[\text{if (tmp\_isrcode = fp\_reg\_disabled}(f_1, 0, 0))\]
\[\text{disabled\_fp\_register\_fault(tmp\_isrcode, 0);}\]
\[\text{if (!GR[r_2].nat) }\{\]
\[\text{if (single\_form)}\]
\[\text{FR[f_1] = fp\_mem\_to\_fr\_format}(GR[r_2], 4, 0);\]
\[\text{else if (double\_form)}\]
\[\text{FR[f_1] = fp\_mem\_to\_fr\_format}(GR[r_2], 8, 0);\]
\[\text{else if (significand\_form) }\{\]
\[\text{FR[f_1].significand = GR[r_2];}\]
\[\text{FR[f_1].exponent = FP\_INTEGER\_EXP;}\]
\[\text{FR[f_1].sign = 0;}\]
\[\} \text{else } \{ \text{\textbf{// exponent\_form}}\]
\[\text{FR[f_1].significand = 0x8000000000000000;}\]
\[\text{FR[f_1].exp = GR[r_2]\{16:0\};}\]
\[\text{FR[f_1].sign = GR[r_2]\{17\};}\]
\[\}\}
\[\} \text{else}\]
\[\text{FR[f_1] = NATVAL;}\]
\[\text{fp\_update\_psr}(f_1);\]
\[\}\]

Interruptions:  Illegal Operation fault \hspace{2cm} \text{Disabled Floating-point Register fault}
Shift Left

Format: \[(qp) \text{ shl } r_j = r_2, r_3\]  
 pseudo-op of: \[(qp) \text{ dep.z } r_j = r_2, \text{ count}_6, 64-\text{count}_6\]

Description: The value in GR \(r_2\) is shifted to the left, with the vacated bit positions filled with zeroes, and placed in GR \(r_j\). The number of bit positions to shift is specified by the value in GR \(r_3\) or by an immediate value \(\text{count}_6\). The shift count is interpreted as an unsigned number. If the value in GR \(r_3\) is greater than 63, then the result is all zeroes.

See “Deposit” on page 1:46 for the immediate form.

Operation:

\[
\begin{align*}
\text{if } (\text{PR}[qp]) \{
\qquad & \text{check_target_register}(r_j); \\
\qquad & \text{count} = \text{GR}[r_3]; \\
\qquad & \text{GR}[r_j] = (\text{count} > 63) \ ? \ 0 : \ \text{GR}[r_2] \ll \text{count}; \\
\qquad & \text{GR}[r_j].\text{nat} = \text{GR}[r_2].\text{nat} \ || \ \text{GR}[r_3].\text{nat}; \\
\}
\end{align*}
\]

Interruptions: Illegal Operation fault
Shift Left and Add

Format: \((qp)\) shladd \(r_1 = r_2, count_2, r_3\)

Description: The first source operand is shifted to the left by \(count_2\) bits and then added to the second source operand and the result placed in GR \(r_1\). The first operand can be shifted by 1, 2, 3, or 4 bits.

Operation: 

\[
\text{if (PR[qp]) } \\
\quad \text{check_target_register}(r_1); \quad \text{GR}[r_1] = (\text{GR}[r_2] \ll count_2) + \text{GR}[r_3]; \quad \text{GR}[r_1].nat = \text{GR}[r_2].nat \lor \text{GR}[r_3].nat; \]

Interruptions: Illegal Operation fault
Shift Left and Add Pointer

Format: \((qp)\) shladdp4 \(r_1 = r_2, count_2, r_3\)

Description: The first source operand is shifted to the left by \(count_2\) bits and then is added to the second source operand. The upper 32 bits of the result are forced to zero, and then bits \(31:30\) of GR \(r_3\) are copied to bits \(62:61\) of the result. This result is placed in GR \(r_1\). The first operand can be shifted by 1, 2, 3, or 4 bits.

Operation:

```plaintext
if (PR[qp]) {
    check_target_register(r1);
    tmp_res = (GR[r2] << count_2) + GR[r3];
    tmp_res = zero_ext(tmp_res{31:0}, 32);
    tmp_res{62:61} = GR[r3]{31:30};
    GR[r1] = tmp_res;
    GR[r1].nat = GR[r2].nat || GR[r3].nat;
}
```

Interruptions: Illegal Operation fault
**Shift Right**

**Format:**

(qp) shr \( r_1 = r_3, r_2 \)

(qp) shr.u \( r_1 = r_3, r_2 \)

(qp) shr \( r_1 = r_3, \text{count}_6 \)

(qp) shr.u \( r_1 = r_3, \text{count}_6 \)

**Description:** The value in GR \( r_3 \) is shifted to the right and placed in GR \( r_1 \). In the signed_form the vacated bit positions are filled with bit 63 of GR \( r_3 \); in the unsigned_form the vacated bit positions are filled with zeroes. The number of bit positions to shift is specified by the value in GR \( r_2 \) or by an immediate value \( \text{count}_6 \). The shift count is interpreted as an unsigned number. If the value in GR \( r_2 \) is greater than 63, then the result is all zeroes (for the unsigned_form, or if bit 63 of GR \( r_3 \) was 0) or all ones (for the signed_form if bit 63 of GR \( r_3 \) was 1).

If the .u completer is specified, the shift is unsigned (logical), otherwise it is signed (arithmetic).

See “Extract” on page 1:49 for the immediate forms.

**Operation:**

if \((\text{PR}[qp])\) {
    check_target_register(\( r_1 \));

    if (signed_form) {
        count = (GR[r_2] > 63) ? 63 : GR[r_2];
        GR[r_1] = shift_right_signed(GR[r_3], count);  // Type cast to int
    } else {
        count = GR[r_2] <= 63 ? 0 : count > 63 ? 63 : GR[r_3];
        GR[r_1] = shift_right_unsigned(GR[r_3], count);
    }

    GR[r_1].nat = GR[r_2].nat || GR[r_3].nat;
}

**Interruptions:** Illegal Operation fault
Shift Right Pair

Format: \[(qp) \text{ shrp } r_1 = r_2, r_3, count_6\]

Description: The two source operands, GR\(r_2\) and GR\(r_3\), are concatenated to form a 128-bit value and shifted to the right \(count_6\) bits. The least-significant 64 bits of the result are placed in GR\(r_1\).

The immediate value \(count_6\) can be any number in the range 0 to 63.

Figure 2-44. Shift Right Pair

```
if (PR[qp]) {
    check_target_register(r1);
    temp1 = shift_right_unsigned(GR[r3], count_6);
    temp2 = GR[r2] << (64 - count_6);
    GR[r1] = zero_ext(temp1, 64 - count_6) | temp2;
    GR[r1].nat = GR[r2].nat || GR[r3].nat;
}
```

Interruptions: Illegal Operation fault
Serialize

Format:

\[(qp) \text{srlz.i}\] \quad \text{instruction_form} \quad \text{M24}
\[(qp) \text{srlz.d}\] \quad \text{data_form} \quad \text{M24}

Description: Instruction serialization (\text{srlz.i}) ensures:

- prior modifications to processor register resources that affect fetching of subsequent instruction groups are observed,
- prior modifications to processor register resources that affect subsequent execution or data memory accesses are observed,
- prior memory synchronization (\text{sync.i}) operations have taken effect on the local processor instruction cache,
- subsequent instruction group fetches are re-initiated after \text{srlz.i} completes.

The \text{srlz.i} instruction must be in an instruction group after the instruction group containing the operation that is to be serialized. Operations dependent on the serialization must be in an instruction group after the instruction group containing the \text{srlz.i}.

Data serialization (\text{srlz.d}) ensures:

- prior modifications to processor register resources that affect subsequent execution or data memory accesses are observed.

The \text{srlz.d} instruction must be in an instruction group after the instruction group containing the operation that is to be serialized. Operations dependent on the serialization must follow the \text{srlz.d}, but they can be in the same instruction group as the \text{srlz.d}.

A \text{srlz} cannot be used to stall processor data memory references until prior data memory references, or memory fences are visible or “accepted” by the external platform.

The following processor resources require a serialize to ensure side-effects are observed; CRs, PSR, DBRs, IBRs, PMDs, PMCs, RRs, PKRs, TRs and TCs (refer to “Serialization” on page 1:13 in Volume 2: System Architecture for details).

Operation:

\[
\text{if } (PR[qp]) \{ \\
\quad \text{if } (\text{instruction_form}) \\
\quad \quad \text{instruction_serialize();} \\
\quad \text{else } \quad \text{// data_form} \\
\quad \quad \text{data_serialize();} \\
\}\n\]

Interruptions: None
Set System Mask

Format: \( (qp) \text{ ssm imm}_{24} \)

Description: The \( \text{imm}_{24} \) operand is ORed with the system mask (PSR\{23:0\}) and the result is placed in the system mask. See “Processor Status Register (PSR)” on page 1:18 in Volume 2: System Architecture.

The PSR system mask can only be written at the most privileged level.

The contents of the interruption resources (that are overwritten when the PSR.ic bit is 1), are undefined if an interruption occurs between the enabling of the PSR.ic bit and a subsequent instruction serialize operation.

Operation:

\[
\text{if } (PR[qp]) \{
\quad \text{if } (PSR.cpl != 0) \{
\quad \quad \text{privileged_operation_fault}(0);
\quad \}\}
\]

\[
\text{if } (\text{is_reserved_field}(PSR\_TYPE, \text{PSR\_SM}, \text{imm}_{24})) \{
\quad \text{reserved_register_field_fault();}
\}
\]

\[
\begin{align*}
\text{if } (\text{imm}_{24}[1]) & \quad \text{PSR}[1] = 1; & \quad \text{be} \\
\text{if } (\text{imm}_{24}[2]) & \quad \text{PSR}[2] = 1; & \quad \text{up} \\
\text{if } (\text{imm}_{24}[3]) & \quad \text{PSR}[3] = 1; & \quad \text{ac} \\
\text{if } (\text{imm}_{24}[4]) & \quad \text{PSR}[4] = 1; & \quad \text{mfl} \\
\text{if } (\text{imm}_{24}[5]) & \quad \text{PSR}[5] = 1; & \quad \text{mfh} \\
\text{if } (\text{imm}_{24}[13]) & \quad \text{PSR}[13] = 1; & \quad \text{ic} \\
\text{if } (\text{imm}_{24}[14]) & \quad \text{PSR}[14] = 1; & \quad \text{i} \\
\text{if } (\text{imm}_{24}[15]) & \quad \text{PSR}[15] = 1; & \quad \text{pk} \\
\text{if } (\text{imm}_{24}[17]) & \quad \text{PSR}[17] = 1; & \quad \text{dt} \\
\text{if } (\text{imm}_{24}[18]) & \quad \text{PSR}[18] = 1; & \quad \text{dfl} \\
\text{if } (\text{imm}_{24}[19]) & \quad \text{PSR}[19] = 1; & \quad \text{dfh} \\
\text{if } (\text{imm}_{24}[20]) & \quad \text{PSR}[20] = 1; & \quad \text{sp} \\
\text{if } (\text{imm}_{24}[21]) & \quad \text{PSR}[21] = 1; & \quad \text{pp} \\
\text{if } (\text{imm}_{24}[22]) & \quad \text{PSR}[22] = 1; & \quad \text{di} \\
\text{if } (\text{imm}_{24}[23]) & \quad \text{PSR}[23] = 1; & \quad \text{si}
\end{align*}
\]

Interruptions: Privileged Operation fault Reserved Register/Field fault

Serialization: Software must issue a data serialize or instruction serialize operation before issuing instructions dependent upon the altered PSR bits from the \text{ssm} instruction. Unlike with the \text{rsm} instruction, setting the PSR.i bit is not treated specially. Refer to “Serialization” on page 1:13 in Volume 2: System Architecture for a description of serialization.
Store

Format:  

\[(qp) \text{ stsz, sttype, sthint } [r_3] = r_2\]  

(\text{normal_form, no_base_update_form}) \quad \text{M4}

\[(qp) \text{ stsz, sttype, sthint } [r_3] = r_2, \text{ imm}_9\]  

(\text{normal_form, imm_base_update_form}) \quad \text{M5}

\[(qp) \text{ st8.spill, sthint } [r_3] = r_2\]  

(\text{spill_form, no_base_update_form}) \quad \text{M4}

\[(qp) \text{ st8.spill, sthint } [r_3] = r_2, \text{ imm}_9\]  

(\text{spill_form, imm_base_update_form}) \quad \text{M5}

Description:  

A value consisting of the least significant sz bytes of the value in GR \(r_2\) is written to memory starting at the address specified by the value in GR \(r_3\). The values of the sz completer are given in Table 2-31 on page 3:135. The sttype completer specifies special store operations, which are described in Table 2-48. If the NaT bit corresponding to GR \(r_3\) is 1 (or in the normal_form, if the NaT bit corresponding to GR \(r_2\) is 1), a Register NaT Consumption fault is taken.

In the spill_form, an 8-byte value is stored, and the NaT bit corresponding to GR \(r_2\) is copied to a bit in the UNAT application register. This instruction is used for spilling a register/NaT pair. See “Control Speculation” on page 1:53 in Volume 1: Application Architecture for details.

In the imm_base_update form, the value in GR \(r_3\) is added to a signed immediate value (immm) and the result is placed back in GR \(r_3\). This base register update is done after the store, and does not affect the store address, nor the value stored (for the case where \(r_2\) and \(r_3\) specify the same register).

For more details on ordered stores see “Memory Access Ordering” on page 1:65 in Volume 1: Application Architecture.

The ALAT is queried using the physical memory address and the access size, and all overlapping entries are invalidated.

The value of the sthint completer specifies the locality of the memory access. The values of the sthint completer are given in Table 2-49. A prefetch hint is implied in the base update forms. The address specified by the value in GR \(r_3\) after the base update acts as a hint to prefetch the indicated cache line. This prefetch uses the locality hints specified by sthint. See Section 4.4.6, “Memory Hierarchy Control and Consistency” on page 1:62 in Volume 1: Application Architecture.

Table 2-48. Store Types

<table>
<thead>
<tr>
<th>sttype Completer</th>
<th>Interpretation</th>
<th>Special Store Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Normal store</td>
<td></td>
</tr>
<tr>
<td>rel</td>
<td>Ordered store</td>
<td>An ordered store is performed with release semantics.</td>
</tr>
</tbody>
</table>

Table 2-49. Store Hints

<table>
<thead>
<tr>
<th>sthint Completer</th>
<th>Interpretation</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>Temporal locality, level 1</td>
</tr>
<tr>
<td>nta</td>
<td>Non-temporal locality, all levels</td>
</tr>
</tbody>
</table>
Operation:

if (PR[gp]) {
    size = spill_form ? 8 : sz;
    otype = (sttype == 'rel') ? RELEASE : UNORDERED;

    if (imm_base_update_form)
        check_target_register(r3);
    if (GR[r3].nat || (normal_form && GR[r2].nat))
        register_nat_consumption_fault(WRITE);

    paddr = tlb_translate(GR[r3], size, WRITE, PSR.cpl, &mattr, &tmp_unused);
    if (spill_form && GR[r2].nat)
        natd_gr_write(GR[r2], paddr, size, UM.be, mattr, otype, sthint);
    else
        mem_write(GR[r2], paddr, size, UM.be, mattr, otype, sthint);

    if (spill_form) {
        bit_pos = GR[r3]{8:3};
        AR[UNAT]{bit_pos} = GR[r2].nat;
    }

    alat_inval_multiple_entries(paddr, size);

    if (imm_base_update_form) {
        GR[r3] = GR[r3] + sign_ext(imm, 9);
        GR[r3].nat = 0;
        mem_implicit_prefetch(GR[r3], sthint, WRITE);
    }
}

Interruptions:

Illegal Operation fault
Register NaT Consumption fault
Unimplemented Data Address fault
Data Nested TLB fault
Alternate Data TLB fault
VHPT Data fault
Data TLB fault
Data Page Not Present fault
Data NaT Page Consumption fault
Data Key Miss fault
Data Key Permission fault
Data Access Rights fault
Data Dirty Bit fault
Data Access Bit fault
Data Debug fault
Unaligned Data Reference fault
Floating-point Store

Format:

(\(qp\)) \texttt{stf\_s\_stf \[r_3\] = f_2} \quad \text{normal form, no_base_update_form} \quad M9  
(\(qp\)) \texttt{stf\_s\_stf \[r_3\] = f_2, \textit{imm}_9} \quad \text{normal form, imm_base_update_form} \quad M10  
(\(qp\)) \texttt{stf\_s\_stf \[r_3\] = f_2} \quad \text{integer form, no_base_update_form} \quad M9  
(\(qp\)) \texttt{stf\_s\_stf \[r_3\] = f_2, \textit{imm}_9} \quad \text{integer form, imm_base_update_form} \quad M10  
(\(qp\)) \texttt{stf\_spill\_stf \[r_3\] = f_2} \quad \text{spill form, no_base_update_form} \quad M9  
(\(qp\)) \texttt{stf\_spill\_stf \[r_3\] = f_2, \textit{imm}_9} \quad \text{spill form, imm_base_update_form} \quad M10

Description: A value, consisting of \(fsz\) bytes, is generated from the value in \(FR\ f_2\) and written to memory starting at the address specified by the value in \(GR\ r_3\). In the normal_form, the value in FR \(f_2\) is converted to the memory format and then stored. In the integer_form, the significand of FR \(f_2\) is stored. The values of the \(fsz\) completer are given in Table 2-34 on page 3:139. In the normal_form or the integer_form, if the NaT bit corresponding to GR \(r_3\) is 1 or if FR \(f_2\) contains NaTVal, a Register NaT Consumption fault is taken. See “Data Types and Formats” on page 1:77 in Volume 1: Application Architecture for details on conversion from floating-point-register format.

In the spill_form, a 16-byte value from FR \(f_2\) is stored without conversion. This instruction is used for spilling a register. See “Control Speculation” on page 1:53 in Volume 1: Application Architecture for details.

In the imm_base_update form, the value in GR \(r_3\) is added to a signed immediate value (\(\textit{imm}_9\)) and the result is placed back in GR \(r_3\). This base register update is done after the store, and does not affect the store address.

The ALAT is queried using the physical memory address and the access size, and all overlapping entries are invalidated.

The value of the \(stf\_stf\) completer specifies the locality of the memory access. The values of the \(stf\_stf\) completer are given in Table 2-49 on page 3:223. A prefetch hint is implied in the base update forms. The address specified by the value in GR \(r_3\) after the base update acts as a hint to prefetch the indicated cache line. This prefetch uses the locality hints specified by \(stf\_stf\). See “Memory Hierarchy Control and Consistency” on page 1:62 in Volume 1: Application Architecture.

Hardware support for \texttt{stfe} (10-byte) instructions that reference a page that is neither a cacheable page with write-back policy nor a NaTPage is optional. On processor models that do not support such \texttt{stfe} accesses, an Unsupported Data Reference fault is raised when an unsupported reference is attempted.

Operation:

\[
\text{if } (\text{PR}\[q\]) \{ \\
\quad \text{if } (\text{imm_base_update_form}) \\
\quad \quad \text{check_target_register}(r_3); \\
\quad \quad \text{if } (\text{tmp_isrcode} = \text{fp_reg_disabled}(f_2, 0, 0, 0)) \\
\quad \quad \quad \text{disabled_fp_register_fault}(\text{tmp_isrcode}, \text{WRITE}); \\
\quad \quad \text{if } (\text{GR}[r_3].\text{nat} \quad | \quad | \quad (\text{spill_form} \&\& \text{(FR}[f_2] \quad == \quad \text{NATVAL})) \\
\quad \quad \quad \text{register_nat_consumption_fault}(\text{WRITE}); \\
\quad \quad \text{size} = \text{spill_form} \quad ? \quad 16 \quad : \quad (\text{integer_form} \quad ? \quad 8 \quad : \quad \text{fsz}); \\
\quad \quad \text{paddr} = \text{tlb_translate}(\text{GR}[r_3], \text{size}, \text{WRITE}, \text{PSR.cpl}, \&\text{mattr}, \&\text{tmp_unused}); \\
\quad \quad \text{val} = \text{fp_fr_to_mem_format}(\text{FR}[f_2], \text{size}, \text{integer_form}); \\
\quad \quad \text{mem_write}(\text{val}, \text{paddr}, \text{size}, \text{UM.be}, \text{mattr}, \text{UNORDERED}, \text{sthint}); \\
\quad \quad \text{alat_inval_multiple_entries}(\text{paddr}, \text{size}); \\
\quad \quad \text{if } (\text{imm_base_update_form}) \{ \\
\quad \quad \quad \text{GR}[r_3] = \text{GR}[r_3] + \text{sign_ext}(\text{imm}_9, 9); \\
\quad \quad \quad \text{GR}[r_3].\text{nat} = 0; \\
\quad \quad \quad \text{mem_implicit_prefetch}(\text{GR}[r_3], \text{sthint}, \text{WRITE}); \\
\quad \} \\
\}
\]
<table>
<thead>
<tr>
<th>Interruptions:</th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>Illegal Operation fault</td>
<td>Data NaT Page Consumption fault</td>
</tr>
<tr>
<td>Disabled Floating-point Register fault</td>
<td>Data Key Miss fault</td>
</tr>
<tr>
<td>Register NaT Consumption fault</td>
<td>Data Key Permission fault</td>
</tr>
<tr>
<td>Unimplemented Data Address fault</td>
<td>Data Access Rights fault</td>
</tr>
<tr>
<td>Data Nested TLB fault</td>
<td>Data Dirty Bit fault</td>
</tr>
<tr>
<td>Alternate Data TLB fault</td>
<td>Data Access Bit fault</td>
</tr>
<tr>
<td>VHPT Data fault</td>
<td>Data Debug fault</td>
</tr>
<tr>
<td>Data TLB fault</td>
<td>Unaligned Data Reference fault</td>
</tr>
<tr>
<td>Data Page Not Present fault</td>
<td>Unsupported Data Reference fault</td>
</tr>
</tbody>
</table>
Subtract

Format: 

(qp) sub \( r_j = r_2, r_3 \) \hspace{1cm} \text{register\_form A1}

(qp) sub \( r_j = r_2, r_3, 1 \) \hspace{1cm} \text{minus1\_form, register\_form A1}

(qp) sub \( r_j = \text{imm}_8, r_3 \) \hspace{1cm} \text{imm8\_form A3}

Description: The second source operand (and an optional constant 1) are subtracted from the first operand and the result placed in GR \( r_j \). In the register form the first operand is GR \( r_2 \); in the immediate form the first operand is taken from the sign-extended \text{imm}_8 encoding field.

The minus1\_form is available only in the register\_form (although the equivalent effect can be achieved by adjusting the immediate).

Operation: 

if (PR[qp]) {
    check_target_register(r_j);

    tmp_src = (register\_form ? GR[r_2] : sign_ext(\text{imm}_8, 8));
    tmp_nat = (register\_form ? GR[r_2].nat : 0);

    if (minus1\_form)
        GR[r_j] = tmp_src - GR[r_3] - 1;
    else
        GR[r_j] = tmp_src - GR[r_3];

    GR[r_j].nat = tmp_nat || GR[r_3].nat;
}

Interruptions: Illegal Operation fault
Set User Mask

Format: \((qp)\ sum \ imm_{24}\)  

Description: The \(imm_{24}\) operand is ORed with the user mask (PSR\([5:0]\)) and the result is placed in the user mask. See “Processor Status Register (PSR)” on page 1:18 in Volume 2: System Architecture. PSR.up can only be set if the secure performance monitor bit (PSR.sp) is zero. Otherwise PSR.up is not modified.

Operation:
```c
if (PR[qp]) {
    if (is_reserved_field(PSR_TYPE, PSR_UM, \ imm_{24}\))
        reserved_register_field_fault();
    if (imm_{24}[1]) PSR[1] = 1; \ \ // be
    if (imm_{24}[2] && PSR.sp == 0) // non-secure perf monitor
        PSR[2] = 1; \ \ // up
    if (imm_{24}[3]) PSR[3] = 1; \ \ // ac
    if (imm_{24}[4]) PSR[4] = 1; \ \ // mfl
    if (imm_{24}[5]) PSR[5] = 1; \ \ // mfh
}
```

Interruptions: Reserved Register/Field fault

Serialization: All user mask modifications are observed by the next instruction group.
**Sign Extend**

**Format:** \( (qp) \text{sxt} xsz \ r_1 = r_3 \)

**Description:** The value in GR \( r_3 \) is sign extended from the bit position specified by \( xsz \) and the result is placed in GR \( r_1 \). The mnemonic values for \( xsz \) are given in Table 2-50.

**Table 2-50. xsz Mnemonic Values**

<table>
<thead>
<tr>
<th>xsz Mnemonic</th>
<th>Bit Position</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>7</td>
</tr>
<tr>
<td>2</td>
<td>15</td>
</tr>
<tr>
<td>4</td>
<td>31</td>
</tr>
</tbody>
</table>

**Operation:**

```c
if (PR[qp]) {
    check_target_register(r_1);
    GR[r_1] = sign_ext(GR[r_3], xsz * 8);
    GR[r_1].nat = GR[r_3].nat;
}
```

**Interruptions:** Illegal Operation fault
Memory Synchronization

**Format:**  
\[(qp) \text{ sync.i}\]

**Description:**  
\text{sync.i} ensures that when previously initiated Flush Cache (\text{fc}) operations issued by the local processor become visible to local data memory references, prior Flush Cache operations are also observed by the local processor instruction fetch stream. \text{sync.i} also ensures that at the time previously initiated Flush Cache (\text{fc}) operations are observed on a remote processor by data memory references they are also observed by instruction memory references on the remote processor. \text{sync.i} is ordered with respect to all cache flush operations as observed by another processor. A \text{sync.i} and a previous \text{fc} must be in separate instruction groups. If semantically required, the programmer must explicitly insert ordered data references (acquire, release or fence type) to appropriately constrain \text{sync.i} (and hence \text{fc}) visibility to the data stream on other processors.

\text{sync.i} is used to maintain an ordering relationship between instruction and data caches on local and remote processors. An instruction serialize operation must be used to ensure synchronization initiated by \text{sync.i} on the local processor has been observed by a given point in program execution.

An example of self-modifying code (local processor):

```
    st [L1] = data  //store into local instruction stream
    fc L1           //flush stale datum from instruction/data cache
    ;;                //require instruction boundary between fc and sync.i
    sync.i          //ensure local and remote data/inst caches
                    //are synchronized
    ;;                //ensure sync has been observed by the local processor,
    srlz.i          //ensure subsequent instructions observe
                    //modified memory
L1: target        //instruction modified
```

**Operation:**  
\[
    \text{if (PR[gp])} \{ \\
      \text{instruction\_synchronize();} \\
    \}
\]

**Interruptions:**  
None
Translation Access Key

Format: \((qp)\) tak \( r_1 = r_3 \)

Description: The protection key for a given virtual address is obtained and placed in GR \( r_1 \).

When PSR.dt is 1, the DTLB and the VHPT are searched for the virtual address specified by GR \( r_3 \) and the region register indexed by GR \( r_3 \) bits \{63:61\}. If a matching present translation is found the protection key of the translation is placed in GR \( r_1 \). If a matching present translation is not found or if an unimplemented virtual address is specified by GR \( r_3 \), the value 1 is returned.

When PSR.dt is 0, only the DTLB is searched, because the VHPT walker is disabled. If no matching present translation is found in the DTLB, the value 1 is returned.

A translation with the NaTPage attribute is not treated differently and returns its key field.

This instruction can only be executed at the most privileged level.

Operation:

```c
if (PR[qp]) {
    itype = NON_ACCESS|TAK;
    check_target_register(r1);
    if (PSR.cpl != 0)
        privileged_operation_fault(itype);
    if (GR[r3].nat)
        register_nat_consumption_fault(itype);
    GR[r1] = tlb_access_key(GR[r3], itype);
    GR[r1].nat = 0;
}
```

Interruptions: Illegal Operation fault Register NaT Consumption fault

Privileged Operation fault
Test Bit

Format: \((qp)\) tbit.ctype  \(p_1, p_2 = r_3, pos_6\)

Description: The bit specified by the \(pos_6\) immediate is selected from GR \(r_3\). The selected bit forms a single bit result either complemented or not depending on the \(trel\) completer. This result is written to the two predicate register destinations \(p_1\) and \(p_2\). The way the result is written to the destinations is determined by the compare type specified by \(ctype\). See the Compare instruction and Table 2-15 on page 3:35.

The \(trel\) completer values .nz and .z indicate non-zero and zero sense of the test. For normal and unc types, only the .z value is directly implemented in hardware; the .nz value is actually a pseudo-op. For it, the assembler simply switches the predicate target specifiers and uses the implemented relation. For the parallel types, both relations are implemented in hardware.

Table 2-51. Test Bit Relations for Normal and unc tbits

<table>
<thead>
<tr>
<th>(trel)</th>
<th>Test Relation</th>
<th>Pseudo-op of</th>
</tr>
</thead>
<tbody>
<tr>
<td>nz</td>
<td>selected bit == 1</td>
<td>(z)</td>
</tr>
<tr>
<td>z</td>
<td>selected bit == 0</td>
<td>(p_1 \leftrightarrow p_2)</td>
</tr>
</tbody>
</table>

Table 2-52. Test Bit Relations for Parallel tbits

<table>
<thead>
<tr>
<th>(trel)</th>
<th>Test Relation</th>
</tr>
</thead>
<tbody>
<tr>
<td>nz</td>
<td>selected bit == 1</td>
</tr>
<tr>
<td>z</td>
<td>selected bit == 0</td>
</tr>
</tbody>
</table>

If the two predicate register destinations are the same (\(p_1\) and \(p_2\) specify the same predicate register), the instruction will take an Illegal Operation fault, if the qualifying predicate is set, or if the compare type is unc.
Operation:

```c
if (PR[qp]) {
    if (p1 == p2)
        illegal_operation_fault();

    if (trel == 'nz')  // 'nz' - test for 1
        tmp_rel = GR[r3][pos6];
    else  // 'z' - test for 0
        tmp_rel = !GR[r3][pos6];

    switch (ctype) {
        case 'and':  // and-type compare
            if (GR[r3].nat || !tmp_rel) {
                PR[p1] = 0;
                PR[p2] = 0;
            }
            break;
        case 'or':  // or-type compare
            if (!GR[r3].nat && tmp_rel) {
                PR[p1] = 1;
                PR[p2] = 1;
            }
            break;
        case 'or.andcm':  // or.andcm-type compare
            if (!GR[r3].nat && tmp_rel) {
                PR[p1] = 1;
                PR[p2] = 0;
            }
            break;
        case 'unc':  // unc-type compare
            default:  // normal-type compare
                if (GR[r3].nat) {
                    PR[p1] = 0;
                    PR[p2] = 0;
                } else {
                    PR[p1] = tmp_rel;
                    PR[p2] = !tmp_rel;
                }
            break;
        }
    }
else {
    if (ctype == 'unc') {
        if (p1 == p2)
            illegal_operation_fault();
        PR[p1] = 0;
        PR[p2] = 0;
    }
}
```

Interruptions: Illegal Operation fault
Translation Hashed Entry Address

**Format:**

\[(qp) \text{ thash } r_1 = r_3\]

**Description:**

A Virtual Hashed Page Table (VHPT) entry address is generated based upon the specified virtual address and the result is placed in GR \(r_1\). The virtual address is specified by GR \(r_3\) and the region register selected by GR \(r_3\) bits \(63:61\).

If \text{thash} is given a NaT input argument or an unimplemented virtual address as an input, the resulting target register value is undefined, and its NaT bit is set to one.

When the processor is configured to use the region-based short format VHPT (PTA.vf=0), the value returned by \text{thash} is defined by the architected short format hash function. See “Region-based VHPT Short Format” on page 1:52 in *Volume 2: System Architecture*.

When the processor is configured to use the long format VHPT (PTA.vf=1), \text{thash} performs an implementation-specific long format hash function on the virtual address to generate a hash index into the long format VHPT.

In the long format, a translation in the VHPT must be uniquely identified by its hash index generated by this instruction and the hash tag produced from the \text{ttag} instruction.

The hash function must use all implemented region bits and only virtual address bits \(60:0\) to determine the offset into the VHPT. Virtual address bits \(63:61\) are used only by the short format hash to determine the region of the VHPT.

This instruction must be implemented on all processor models, even processor models that do not implement a VHPT walker.

**Operation:**

```c
if (PR[qp]) {
    check_target_register(r1);

    if (GR[r3].nat || unimplemented_virtual_address(GR[r3])) {
        GR[r1] = undefined();
        GR[r1].nat = 1;
    } else {
        tmp_vr = GR[r3]{63:61};
        tmp_va = GR[r3]{60:0};
        GR[r1] = tlb_vhpt_hash(tmp_vr, tmp_va, RR[tmp_vr].rid, RR[tmp_vr].ps);
        GR[r1].nat = 0;
    }
}
```

**Interruptions:** Illegal Operation fault
Test NaT

Format: \( (qp) \text{tnat.trel.ctype} \ p_1, p_2 = r_3 \)

Description: The NaT bit from GR \( r_3 \) forms a single bit result, either complemented or not depending on the \( trel \) completer. This result is written to the two predicate register destinations, \( p_1 \) and \( p_2 \). The way the result is written to the destinations is determined by the compare type specified by \( ctype \). See the Compare instruction and Table 2-15 on page 3:35.

The \( trel \) completer values \( .nz \) and \( .z \) indicate non-zero and zero sense of the test. For normal and unc types, only the \( .z \) value is directly implemented in hardware; the \( .nz \) value is actually a pseudo-op. For it, the assembler simply switches the predicate target specifiers and uses the implemented relation. For the parallel types, both relations are implemented in hardware.

### Table 2-53. Test NaT Relations for Normal and unc tnats

<table>
<thead>
<tr>
<th>( trel )</th>
<th>Test Relation</th>
<th>Pseudo-op of</th>
</tr>
</thead>
<tbody>
<tr>
<td>( nz )</td>
<td>selected bit == 1</td>
<td>( z ) ( p_1 \leftrightarrow p_2 )</td>
</tr>
<tr>
<td>( z )</td>
<td>selected bit == 0</td>
<td></td>
</tr>
</tbody>
</table>

### Table 2-54. Test NaT Relations for Parallel tnats

<table>
<thead>
<tr>
<th>( trel )</th>
<th>Test Relation</th>
</tr>
</thead>
<tbody>
<tr>
<td>( nz )</td>
<td>selected bit == 1</td>
</tr>
<tr>
<td>( z )</td>
<td>selected bit == 0</td>
</tr>
</tbody>
</table>

If the two predicate register destinations are the same (\( p_1 \) and \( p_2 \) specify the same predicate register), the instruction will take an Illegal Operation fault, if the qualifying predicate is set, or if the compare type is unc.
Operation:

```c
if (PR[qp]) {
    if (p1 == p2)
        illegal_operation_fault();

    if (trel == 'nz')
        tmp_rel = GR[r3].nat; // 'nz' - test for 1
    else
        tmp_rel = !GR[r3].nat; // 'z' - test for 0

    switch (ctype) {
        case 'and': // and-type compare
            if (!tmp_rel) {
                PR[p1] = 0;
                PR[p2] = 0;
            }
            break;
        case 'or': // or-type compare
            if (tmp_rel) {
                PR[p1] = 1;
                PR[p2] = 1;
            }
            break;
        case 'or.andcm': // or.andcm-type compare
            if (tmp_rel) {
                PR[p1] = 1;
                PR[p2] = 0;
            }
            break;
        case 'unc': // unc-type compare
            default: // normal compare
                PR[p1] = tmp_rel;
                PR[p2] = !tmp_rel;
                break;
    }
} else {
    if (ctype == 'unc') {
        if (p1 == p2)
            illegal_operation_fault();
        PR[p1] = 0;
        PR[p2] = 0;
    }
}
```

Interruptions: Illegal Operation fault
Translate to Physical Address

Format:  \((qp)\) tpa  \(r_j = r_3\)

Description:  The physical address for the virtual address specified by GR \(r_3\) is obtained and placed in GR \(r_j\).

When PSR.dt is 1, the DTLB and the VHPT are searched for the virtual address specified by GR \(r_3\) and the region register indexed by GR \(r_3\) bits \{63:61\}. If a matching present translation is found the physical address of the translation is placed in GR \(r_j\). If a matching present translation is not found the appropriate TLB fault is taken.

When PSR.dt is 0, only the DTLB is searched, because the VHPT walker is disabled. If no matching present translation is found in the DTLB, an Alternate Data TLB fault is raised if psr.ic is one or a Data Nested TLB fault is raised if psr.ic is zero.

If this instruction faults, then it will set the non-access bit in the ISR. The ISR read and write bits are not set.

This instruction can only be executed at the most privileged level.

Operation:

\[
\text{if} (PR[qp]) \{
\text{itype} = \text{NON\_ACCESS}|\text{TPA};
\text{check_target_register}(r_j);
\text{if} (PSR.cpl != 0)
\text{privileged_operation_fault}(itype);
\text{if} (GR[r_3].nat)
\text{register_nat_consumption_fault}(itype);
GR[r_j] = \text{tlb_translate_nonaccess}(GR[r_3], itype);
GR[r_j].nat = 0;
\}
\]

Interruptions:

- Illegal Operation fault
- Privileged Operation fault
- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
Translation Hashed Entry Tag

Format: \((qp) \ ttag \ r_I = r_3\)

Description: A tag used for matching during searches of the long format Virtual Hashed Page Table (VHPT) is generated and placed in \(GR_r_1\). The virtual address is specified by \(GR_r_3\) and the region register selected by \(GR_r_3\) bits \(\{63:61\}\).

If \(ttag\) is given a NaT input argument or an unimplemented virtual address as an input, the resulting target register value is undefined, and its NaT bit is set to one.

The tag generation function generates an implementation-specific long format VHPT tag. The tag generation function must use all implemented region bits and only virtual address bits \(\{60:0\}\). PTA.vf is ignored by this instruction.

A translation in the long format VHPT must be uniquely identified by its hash index generated by the \(\text{thash}\) instruction and the tag produced from this instruction.

This instruction must be implemented on all processor models, even processor models that do not implement a VHPT walker.

Operation:
```c
if (PR[qp]) {
    check_target_register(r_I);
    if (GR[r_3].nat || unimplemented_virtual_address(GR[r_3])) {
        GR[r_3] = undefined();
        GR[r_3].nat = 1;
    } else {
        tmp_vr = GR[r_3]{63:61};
        tmp_va = GR[r_3]{60:0};
        GR[r_1] = tlb_vhpt_tag(tmp_va, RR[tmp_vr].rid, RR[tmp_vr].ps);
        GR[r_1].nat = 0;
    }
}
```

Interruptions: Illegal Operation fault
Unpack

**Format:**

- `(qp) unpack1.h r_1 = r_2, r_3`  
  one_byte_form, high_form  
  12
- `(qp) unpack2.h r_1 = r_2, r_3`  
  two_byte_form, high_form  
  12
- `(qp) unpack4.h r_1 = r_2, r_3`  
  four_byte_form, high_form  
  12
- `(qp) unpack1.l r_1 = r_2, r_3`  
  one_byte_form, low_form  
  12
- `(qp) unpack2.l r_1 = r_2, r_3`  
  two_byte_form, low_form  
  12
- `(qp) unpack4.l r_1 = r_2, r_3`  
  four_byte_form, low_form  
  12

**Description:**
The data elements of GR \(r_2\) and \(r_3\) are unpacked, and the result placed in GR \(r_1\). In the high_form, the most significant elements of each source register are selected, while in the low_form the least significant elements of each source register are selected. Elements are selected alternately from the source registers.
Figure 2-45. Unpack Operation
Operation:
if (PR[qp]) {
    check_target_register(r1);
    if (one_byte_form) { // one-byte elements
        x[0] = GR[r2]{7:0};  y[0] = GR[r3]{7:0};
        if (high_form)
            GR[r1] = concatenate8(x[7], y[7], x[6], y[6],
                                x[5], y[5], x[4], y[4]);
        else // low_form
            GR[r1] = concatenate8(x[3], y[3], x[2], y[2],
                                x[1], y[1], x[0], y[0]);
    } else if (two_byte_form) { // two-byte elements
        x[0] = GR[r2]{15:0};  y[0] = GR[r3]{15:0};
        if (high_form)
            GR[r1] = concatenate4(x[3], y[3], x[2], y[2]);
        else // low_form
            GR[r1] = concatenate4(x[1], y[1], x[0], y[0]);
    } else { // four-byte elements
        x[0] = GR[r2]{31:0};  y[0] = GR[r3]{31:0};
        if (high_form)
            GR[r1] = concatenate2(x[1], y[1]);
        else // low_form
            GR[r1] = concatenate2(x[0], y[0]);
    }
    GR[r1].nat = GR[r2].nat | | GR[r3].nat;
}

Interruptions: Illegal Operation fault
**Exchange**

**Format:** \[(qp)\] \textit{xchg} \textit{sz ldhint} \( r_j = [r_3], r_2 \) 

**Description:** A value consisting of \( sz \) bytes is read from memory starting at the address specified by the value in GR \( r_3 \). The least significant \( sz \) bytes of the value in GR \( r_2 \) are written to memory starting at the address specified by the value in GR \( r_3 \). The value read from memory is then zero extended and placed in GR \( r_j \) and the NaT bit corresponding to GR \( r_j \) is cleared. The values of the \( sz \) completer are given in Table 2-55.

If the address specified by the value in GR \( r_3 \) is not naturally aligned to the size of the value being accessed in memory, an Unaligned Data Reference fault is taken independent of the state of the User Mask alignment checking bit, UM.ac (PSR.ac in the Processor Status Register).

Both read and write access privileges for the referenced page are required.

**Table 2-55. Memory Exchange Size**

<table>
<thead>
<tr>
<th>( sz ) Completer</th>
<th>Bytes Accessed</th>
</tr>
</thead>
<tbody>
<tr>
<td>1</td>
<td>1 byte</td>
</tr>
<tr>
<td>2</td>
<td>2 bytes</td>
</tr>
<tr>
<td>4</td>
<td>4 bytes</td>
</tr>
<tr>
<td>8</td>
<td>8 bytes</td>
</tr>
</tbody>
</table>

The exchange is performed with acquire semantics, i.e., the memory read/write is made visible prior to all subsequent data memory accesses. See “Sequentiality Attribute and Ordering” on page 1:69 in Volume 2: System Architecture for details on memory ordering.

The memory read and write are guaranteed to be atomic.

This instruction is only supported to cacheable pages with write-back write policy. Accesses to NaTPages cause a Data NaT Page Consumption fault. Accesses to pages with other memory attributes cause an Unsupported Data Reference fault.

The value of the \textit{ldhint} completer specifies the locality of the memory access. The values of the \textit{ldhint} completer are given in Table 2-33 on page 3:136. Locality hints do not affect program functionality and may be ignored by the implementation. See “Memory Hierarchy Control and Consistency” on page 1:62 in Volume 1: Application Architecture for details.

**Operation:**

```c
if (PR[qp]) {
    check_target_register(r_j);
    if (GR[r_3].nat || GR[r_3].nat)
        register_nat_consumption_fault(SEMAPHORE);
    paddr = tlb_translate(GR[r_3], sz, SEMAPHORE, PSR.cpl, &mattr, &tmp_unused);
    if (!ma_supports_semaphores(mattr))
        unsupported_data_reference_fault(SEMAPHORE, GR[r_3]);
    val = mem_xchg(GR[r_2], paddr, sz, UM.be, mattr, ACQUIRE, ldhint);
    alat_inval_multiple_entries(paddr, sz);
    GR[r_j] = zero_ext(val, sz * 8);
    GR[r_j].nat = 0;
}
```
**Interruptions:**
- Illegal Operation fault
- Register NaT Consumption fault
- Unimplemented Data Address fault
- Data Nested TLB fault
- Alternate Data TLB fault
- VHPT Data fault
- Data TLB fault
- Data Page Not Present fault
- Data NaT Page Consumption fault
- Data Key Miss fault
- Data Key Permission fault
- Data Access Rights fault
- Data Dirty Bit fault
- Data Access Bit fault
- Data Debug fault
- Unaligned Data Reference fault
- Unsupported Data Reference fault
Fixed-Point Multiply Add

**Format:**

- `(qp) xma.l f_1 = f_3, f_4, f_2`  
  `format: (qp) `  
  `pseudo-op of: (qp) xma.l f_1 = f_3, f_4, f_2`  
  `high_form F2`  
- `(qp) xma.lu f_1 = f_3, f_4, f_2`  
  `pseudo-op of: (qp) xma.lu f_1 = f_3, f_4, f_2`  
  `low_form F2`  
- `(qp) xma.h f_1 = f_3, f_4, f_2`  
  `pseudo-op of: (qp) xma.h f_1 = f_3, f_4, f_2`  
  `high_form F2`  
- `(qp) xma.hu f_1 = f_3, f_4, f_2`  
  `pseudo-op of: (qp) xma.hu f_1 = f_3, f_4, f_2`  
  `high_unsigned_form F2`

**Description:**

Two source operands (FR \(f_3\) and FR \(f_4\)) are treated as either signed or unsigned integers and multiplied. The third source operand (FR \(f_2\)) is zero extended and added to the product. The upper or lower 64 bits of the resultant sum are selected and placed in FR \(f_1\).

In the high\_unsigned\_form, the significand fields of FR \(f_3\) and FR \(f_4\) are treated as unsigned integers and multiplied to produce a full 128-bit unsigned result. The significand field of FR \(f_2\) is zero extended and added to the product. The most significant 64-bits of the resultant sum are placed in the significand field of FR \(f_1\).

In the high\_form, the significand fields of FR \(f_3\) and FR \(f_4\) are treated as signed integers and multiplied to produce a full 128-bit signed result. The significand field of FR \(f_2\) is zero extended and added to the product. The most significant 64-bits of the resultant sum are placed in the significand field of FR \(f_1\).

In the other forms, the significand fields of FR \(f_3\) and FR \(f_4\) are treated as signed integers and multiplied to produce a full 128-bit signed result. The significand field of FR \(f_2\) is zero extended and added to the product. The least significant 64-bits of the resultant sum are placed in the significand field of FR \(f_1\).

In all forms, the exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0).

**Note:** f1 as an operand is not an integer 1; it is just the register file format’s 1.0 value.

In all forms, if any of FR \(f_3\), FR \(f_4\), or FR \(f_2\) is a NaTVal, FR \(f_1\) is set to NaTVal instead of the computed result.

**Operation:**

```c
if (PR[qp]) {
    fp_check_target_register(f_1);
    if (tmp_isrcode = fp_reg_disabled(f_1, f_2, f_3, f_4))
        disabled_fp_register_fault(tmp_isrcode, 0);

    if (fp_is_natval(FR[f_3]) || fp_is_natval(FR[f_3]) ||
        fp_is_natval(FR[f_4]) ||
        FR[f_1] = NATVAL)
    ) else {
        if (low_form || high_form)
            tmp_res_128 = fp_I64_x_I64_to_I128(FR[f_3].significand, FR[f_4].significand);
        else // high_unsigned_form
            tmp_res_128 = fp_U64_x_U64_to_U128(FR[f_3].significand, FR[f_4].significand);

        if (high_form || high_unsigned_form)
            FR[f_1].significand = tmp_res_128.hi;
        else // low_form
            FR[f_1].significand = tmp_res_128.lo;

        FR[f_1].exponent = FP_INTEGER_EXP;
        FR[f_1].sign = FP_SIGN_POSITIVE;
    }

    fp_update_psr(f_1);
}
```

**Interruptions:** Disabled Floating-point Register fault
Fixed-Point Multiply

Format:

- \((qp)\ xmpy.l \ f_1 = f_3.f_4\)
- \((qp)\ xmpy.lu \ f_1 = f_3.f_4\)
- \((qp)\ xmpy.h \ f_1 = f_3.f_4\)
- \((qp)\ xmpy.hu \ f_1 = f_3.f_4\)

pseudo-op of: \((qp)\ xma.l \ f_1 = f_3.f_4, f_0\)
pseudo-op of: \((qp)\ xma.l \ f_1 = f_3.f_4, f_0\)
pseudo-op of: \((qp)\ xma.h \ f_1 = f_3.f_4, f_0\)
pseudo-op of: \((qp)\ xma.hu \ f_1 = f_3.f_4, f_0\)

Description: Two source operands (FR \(f_3\) and FR \(f_4\)) are treated as either signed or unsigned integers and multiplied. The upper or lower 64 bits of the resultant product are selected and placed in FR \(f_1\).

In the high_unsigned_form, the significand fields of FR \(f_3\) and FR \(f_4\) are treated as unsigned integers and multiplied to produce a full 128-bit unsigned result. The most significant 64-bits of the resultant product are placed in the significand field of FR \(f_1\).

In the high_form, the significand fields of FR \(f_3\) and FR \(f_4\) are treated as signed integers and multiplied to produce a full 128-bit signed result. The most significant 64-bits of the resultant product are placed in the significand field of FR \(f_1\).

In the other forms, the significand fields of FR \(f_3\) and FR \(f_4\) are treated as signed integers and multiplied to produce a full 128-bit signed result. The least significant 64-bits of the resultant product are placed in the significand field of FR \(f_1\).

In all forms, the exponent field of FR \(f_1\) is set to the biased exponent for \(2.0^{63}\) (0x1003E) and the sign field of FR \(f_1\) is set to positive (0). Note: \(f_1\) as an operand is not an integer 1; it is just the register file format’s 1.0 value.

Operation: See "Fixed-Point Multiply Add" on page 3:244.
**Exclusive Or**

**Format:**

\[
(qp) \text{ xor } r_j = r_2, r_3
\]

\[
(qp) \text{ xor } r_j = \text{imm}_8, r_3
\]

**register_form** \( A1 \)

**imm8_form** \( A3 \)

**Description:** The two source operands are logically XORed and the result placed in GR \( r_j \). In the register_form the first operand is GR \( r_2 \); in the imm8_form the first operand is taken from the \( \text{imm}_8 \) encoding field.

**Operation:**

\[
\text{if} \ (PR[qp]) \ {
\begin{align*}
\text{check_target_register}(r_j); \\
\text{tmp}_\text{src} &= (\text{register_form} \ ? \ \text{GR}[r_2] : \text{sign_ext}(\text{imm}_8, 8)); \\
\text{tmp}_\text{nat} &= (\text{register_form} \ ? \ \text{GR}[r_2].\text{nat} : 0); \\
\text{GR}[r_j] &= \text{tmp}_\text{src} \ ^\land \ \text{GR}[r_3]; \\
\text{GR}[r_j].\text{nat} &= \text{tmp}_\text{nat} \ | \ \text{GR}[r_3].\text{nat};
\end{align*}
\}
\]

**Interruptions:** Illegal Operation fault
Zero Extend

Format: \( (qp) \, zxtxsz \, r_1 = r_3 \)

Description: The value in GR \( r_3 \) is zero extended above the bit position specified by \( xsz \) and the result is placed in GR \( r_1 \). The mnemonic values for \( xsz \) are given in Table 2-50 on page 3:229.

Operation:

```c
if (PR[qp]) { check_target_register(r_1);
    GR[r_1] = zero_ext(GR[r_3], xsz * 8);
    GR[r_1].nat = GR[r_3].nat;
}
```

Interruptions: Illegal Operation fault
This chapter contains a table of all pseudo-code functions used on the Itanium instruction pages.

Table 3-1. Pseudo-Code Functions

<table>
<thead>
<tr>
<th>Function</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>xxx_fault(parameters ...)</td>
<td>There are several fault functions. Each fault function accepts parameters specific to the fault, e.g., exception code values, virtual addresses, etc. If the fault is deferred for speculative load exceptions the fault function will return with a deferral indication. Otherwise, fault routines do not return and terminate the instruction sequence.</td>
</tr>
<tr>
<td>xxx_trap(parameters ...)</td>
<td>There are several trap functions. Each trap function accepts parameters specific to the trap, e.g., trap code values, virtual addresses, etc. Trap routines do not return.</td>
</tr>
<tr>
<td>acceptance_fence()</td>
<td>Ensures prior data memory references to uncached ordered-sequential memory pages are &quot;accepted&quot;, before subsequent data memory references are performed by the processor.</td>
</tr>
<tr>
<td>alat_cmp(rtype, raddr)</td>
<td>Returns a one if the implementation finds an ALAT entry which matches the register type specified by rtype and the register address specified by raddr, else returns zero. This function is implementation specific. Note that an implementation may optionally choose to return zero (indicating no match) even if a matching entry exists in the ALAT. This provides implementation flexibility in designing fast ALAT lookup circuits.</td>
</tr>
<tr>
<td>alat_frame_update(delta_bof, delta_sof)</td>
<td>Notifies the ALAT of a change in the bottom of frame and/or size of frame. This allows management of the ALAT’s tag bits or other management functions it might need.</td>
</tr>
<tr>
<td>alat_inval()</td>
<td>Invalidate all entries in the ALAT.</td>
</tr>
<tr>
<td>alat_inval_multiple_entries(paddr, size)</td>
<td>The ALAT is queried using the physical memory address specified by paddr and the access size specified by size. All matching ALAT entries are invalidated. No value is returned.</td>
</tr>
<tr>
<td>alat_inval_single_entry(rtype, rega)</td>
<td>The ALAT is queried using the register type specified by rtype and the register address specified by rega. At most one matching ALAT entry is invalidated. No value is returned.</td>
</tr>
<tr>
<td>alat_write(rtype, raddr, paddr, size)</td>
<td>Allocates a new ALAT entry using the register type specified by rtype, the register address specified by raddr, the physical memory address specified by paddr, and the access size specified by size. No value is returned. This function guarantees that only one ALAT entry exists for a given raddr. If a ld.c.nc, ldf.c.nc, or ldfp.c.nc instruction’s raddr matches an existing ALAT entry’s register tag, but the instruction’s size and/or paddr are different than that of the existing entry’s; then this function may either preserve the existing entry, or invalidate it and write a new entry with the instruction’s specified size and paddr.</td>
</tr>
<tr>
<td>align_to_size_boundary(vaddr, size)</td>
<td>Returns vaddr aligned to the boundary specified by size.</td>
</tr>
<tr>
<td>branch_predict(wh, ih, ret, target, tag)</td>
<td>Implementation-dependent routine which updates the processor’s branch prediction structures.</td>
</tr>
<tr>
<td>check_branch_implemented(check_type)</td>
<td>Implementation-dependent routine which returns TRUE or FALSE, depending on whether a failing check instruction causes a branch (TRUE), or a Speculative Operation fault (FALSE). The result may be different for different types of check instructions: CHKS_GENERAL, CHKS_FLOAT, CHKA_GENERAL, CHKA_FLOAT. In addition, the result may depend on other implementation-dependent parameters.</td>
</tr>
<tr>
<td>check_target_register(r1)</td>
<td>If r1 targets an out-of-frame stacked register (as defined by CFM), an illegal operation fault is delivered, and this function does not return.</td>
</tr>
<tr>
<td>check_target_register_sof(r1, newsof)</td>
<td>If r1 targets an out-of-frame stacked register (as defined by the newsof parameter), an illegal operation fault is delivered, and this function does not return.</td>
</tr>
<tr>
<td>concatenate2(x1, x2)</td>
<td>Concatenates the lower 32 bits of the 2 arguments, and returns the 64-bit result.</td>
</tr>
</tbody>
</table>
### Table 3-1. Pseudo-Code Functions (Continued)

<table>
<thead>
<tr>
<th>Function</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>concatenate4(x1, x2, x3, x4)</code></td>
<td>Concatenates the lower 16 bits of the 4 arguments, and returns the 64-bit result.</td>
</tr>
<tr>
<td><code>concatenate8(x1, x2, x3, x4, x5, x6, x7, x8)</code></td>
<td>Concatenates the lower 8 bits of the 8 arguments, and returns the 64-bit result.</td>
</tr>
<tr>
<td><code>data_serialize()</code></td>
<td>Ensures all prior register updates with side-effects are observed before subsequent execution and data memory references are performed.</td>
</tr>
<tr>
<td><code>deliver_unmasked_pending_interrupt()</code></td>
<td>This implementation-specific function checks whether any unmasked external interrupts are pending, and if so, transfers control to the external interrupt vector.</td>
</tr>
<tr>
<td><code>fadd(fp_dp, fr2)</code></td>
<td>Adds a floating-point register value to the infinitely precise product and return the infinitely precise sum, ready for rounding.</td>
</tr>
<tr>
<td><code>fcmp_exception_fault_check(f2, f3, frel, sf, *tmp_fp_env)</code></td>
<td>Checks for all floating-point faulting conditions for the <code>fcmp</code> instruction.</td>
</tr>
<tr>
<td><code>fcvt_fx_exception_fault_check(fr2, signed_form, trunc_form, sf, *tmp_fp_env)</code></td>
<td>Checks for all floating-point faulting conditions for the <code>fcvt.fx</code>, <code>fcvt.fxu</code>, <code>fcvt.fx.trunc</code> and <code>fcvt.fxu.trunc</code> instructions. It propagates NaNs.</td>
</tr>
<tr>
<td><code>fma_exception_fault_check(f2, f3, f4, pc, sf, *tmp_fp_env)</code></td>
<td>Checks for all floating-point faulting conditions for the <code>fma</code> instruction. It propagates NaNs and special IEEE results.</td>
</tr>
<tr>
<td><code>fminmax_exception_fault_check(f2, f3, sf, *tmp_fp_env)</code></td>
<td>Checks for all floating-point faulting conditions for the <code>famin</code>, <code>fmin</code>, <code>fmax</code>, and <code>fmin</code> instructions.</td>
</tr>
<tr>
<td><code>fms_fnma_exception_fault_check(f2, f3, f4, pc, sf, *tmp_fp_env)</code></td>
<td>Checks for all floating-point faulting conditions for the <code>fms</code> and <code>fnma</code> instructions. It propagates NaNs and special IEEE results.</td>
</tr>
<tr>
<td><code>fmul(fr3, fr4)</code></td>
<td>Performs an infinitely precise multiply of two floating-point register values.</td>
</tr>
<tr>
<td><code>followed_by_stop()</code></td>
<td>Returns TRUE if the current instruction is followed by a stop; otherwise, returns FALSE.</td>
</tr>
<tr>
<td><code>fp_check_target_register(f1)</code></td>
<td>If the specified floating-point register identifier is 0 or 1, this function causes an illegal operation fault.</td>
</tr>
<tr>
<td><code>fp_decode_fault(tmp_fp_env)</code></td>
<td>Returns floating-point exception fault code values for ISR.code.</td>
</tr>
<tr>
<td><code>fp_decode_traps(tmp_fp_env)</code></td>
<td>Returns floating-point trap code values for ISR.code.</td>
</tr>
<tr>
<td><code>fp_is_nan_or_inf(freg)</code></td>
<td>Returns true if the floating-point exception_fault_check functions returned a IEEE fault disabled default result or a propagated NaN.</td>
</tr>
<tr>
<td><code>fp_equal(fr1, fr2)</code></td>
<td>IEEE standard equality relationship test.</td>
</tr>
<tr>
<td><code>fp_ieee_recip(num, den)</code></td>
<td>Returns the true quotient for special sets of operands, or an approximation to the reciprocal of the divisor to be used in the software divide algorithm.</td>
</tr>
<tr>
<td><code>fp_ieee_recip_sqrt(root)</code></td>
<td>Returns the true square root result for special operands, or an approximation to the reciprocal square root to be used in the software square root algorithm.</td>
</tr>
<tr>
<td><code>fp_is_nan(freg)</code></td>
<td>Returns true when floating register contains a NaN.</td>
</tr>
<tr>
<td><code>fp_is_natval(freg)</code></td>
<td>Returns true when floating register contains a NaTVal.</td>
</tr>
<tr>
<td><code>fp_is_normal(freg)</code></td>
<td>Returns true when floating register contains a normal number.</td>
</tr>
<tr>
<td><code>fp_is_pos_inf(freg)</code></td>
<td>Returns true when floating register contains a positive infinity.</td>
</tr>
<tr>
<td><code>fp_is_qnan(freg)</code></td>
<td>Returns true when floating register contains a quiet NaN.</td>
</tr>
<tr>
<td><code>fp_is_snan(freg)</code></td>
<td>Returns true when floating register contains a signalling NaN.</td>
</tr>
<tr>
<td><code>fp_is_unorm(freg)</code></td>
<td>Returns true when floating register contains an unnormalized number.</td>
</tr>
<tr>
<td><code>fp_is_unsupported(freg)</code></td>
<td>Returns true when floating register contains an unsupported format.</td>
</tr>
<tr>
<td><code>fp_less_than(fr1, fr2)</code></td>
<td>IEEE standard less-than relationship test.</td>
</tr>
<tr>
<td><code>fp_less_or_equal(fr1, fr2)</code></td>
<td>IEEE standard less-than or equal-to relationship test.</td>
</tr>
<tr>
<td><code>fp_normalize(fr1)</code></td>
<td>Normalizes an unnormalized fp value. This function flushes to zero any unnormal values which can not be represented in the register file.</td>
</tr>
<tr>
<td><code>fp_raise_fault(tmp_fp_env)</code></td>
<td>Checks the local instruction state for any faulting conditions which require an interruption to be raised.</td>
</tr>
</tbody>
</table>
Table 3-1. Pseudo-Code Functions

<table>
<thead>
<tr>
<th>Function</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>fp_raise_traps(tmp_fp_env)</td>
<td>Checks the local instruction state for any trapping conditions which require an interruption to be raised.</td>
</tr>
<tr>
<td>fp_reg_bank_conflict(f1, f2)</td>
<td>Returns true if the two specified FRs are in the same bank.</td>
</tr>
<tr>
<td>fp_reg_disabled(f1, f2, f3, f4)</td>
<td>Check for possible disabled floating-point register faults.</td>
</tr>
<tr>
<td>fp_reg_read(freg)</td>
<td>Reads the FR and gives canonical double-extended denormals (and pseudo-denormals) their true mathematical exponent. Other classes of operands are unaltered.</td>
</tr>
<tr>
<td>fp_unordered(f1, f2)</td>
<td>IEEE standard unordered relationship</td>
</tr>
<tr>
<td>fp_fr_to_mem_format(freg, size)</td>
<td>Converts a floating-point value in register format to floating-point memory format. It assumes that the floating-point value in the register has been previously rounded to the correct precision which corresponds with the size parameter.</td>
</tr>
<tr>
<td>fp_cmp_exception_fault_check(f2, f3, frel, sf, *tmp_fp_env)</td>
<td>Checks for all floating-point faulting conditions for the fp_cmp instruction.</td>
</tr>
<tr>
<td>fp_cvt_exception_fault_check(f2, signed_form, trunc_form, sf, *tmp_fp_env)</td>
<td>Checks for all floating-point faulting conditions for the fp_cvt.fx, fp_cvt.fxu, fp_cvt.fx.trunc, and fp_cvt.fxu.trunc instructions. It propagates NaNs.</td>
</tr>
<tr>
<td>fp_ma_exception_fault_check(f2, f3, f4, sf, *tmp_fp_env)</td>
<td>Checks for all floating-point faulting conditions for the fp_ma instruction. It propagates NaNs and special IEEE results.</td>
</tr>
<tr>
<td>fp_minmax_exception_fault_check(f2, f3, sf, *tmp_fp_env)</td>
<td>Checks for all floating-point faulting conditions for the fp_min, fp_max, fp_min and fp_max instructions.</td>
</tr>
<tr>
<td>fp_m_fpm_a_exception_fault_check(f2, f3, f4, sf, *tmp_fp_env)</td>
<td>Checks for all floating-point faulting conditions for the fp_m and fp_m_a instructions. It propagates NaNs and special IEEE results.</td>
</tr>
<tr>
<td>fp Rica_exception_fault_check(f2, f3, sf, *tmp_fp_env, *limits_check)</td>
<td>Checks for all floating-point faulting conditions for the fp_rica instruction. It propagates NaNs and special IEEE results. It also indicates operand limit violations.</td>
</tr>
<tr>
<td>fp rsaqrt_exception_fault_check(f3, sf, *tmp_fp_env, *limits_check)</td>
<td>Checks for all floating-point faulting conditions for the fp_rsaqrt instruction. It propagates NaNs and special IEEE results. It also indicates operand limit violations.</td>
</tr>
<tr>
<td>f_rsqrt_a_exception_fault_check(f3, sf, *tmp_fp_env)</td>
<td>Checks for all floating-point faulting conditions for the f_rsqrt_a instruction. It propagates NaNs and special IEEE results.</td>
</tr>
<tr>
<td>ignored_field_mask(regclass, reg, value)</td>
<td>Boolean function that returns value with bits cleared to 0 corresponding to ignored bits for the specified register and register type.</td>
</tr>
<tr>
<td>impl_itir_cwi_mask()</td>
<td>Implementation-specific function that either returns the value passed to it or the value passed to it masked with zeros in bit positions [63:32] and/or [1:0].</td>
</tr>
<tr>
<td>instruction_serialize()</td>
<td>Ensures all prior register updates with side-effects are observed before subsequent instruction and data memory references are performed. Also ensures prior SYNC.i operations have been observed by the instruction cache.</td>
</tr>
<tr>
<td>instruction_synchronize</td>
<td>Synchronizes the instruction and data stream for Flush Cache operations. This function ensures that when prior FC operations are observed by the local data cache they are observed by the local instruction cache, and when prior FC operations are observed by another processor’s data cache they are observed within the same processor’s instruction cache.</td>
</tr>
<tr>
<td>is_finite(freg)</td>
<td>Returns true when floating register contains a finite number.</td>
</tr>
<tr>
<td>is_ignored_reg(regnum)</td>
<td>Boolean function that returns true if regnum is an ignored application register, otherwise false.</td>
</tr>
<tr>
<td>is_inf(freg)</td>
<td>Returns true when floating register contains an infinite number.</td>
</tr>
<tr>
<td>is_interruption_or(regnum)</td>
<td>Boolean function that returns true if regnum is one of the Interruption Control registers (see “Interruption Control Registers” on page 2:29 in Volume 2), otherwise false.</td>
</tr>
<tr>
<td>is_kernel_reg(ar_addr)</td>
<td>Returns a one if ar_addr is the address of a kernel register application register</td>
</tr>
<tr>
<td>Function</td>
<td>Operation</td>
</tr>
<tr>
<td>--------------------------------</td>
<td>----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------</td>
</tr>
<tr>
<td>is_read_only_reg(rtype, raddr)</td>
<td>Returns a one if the register addressed by raddr in the register bank of type rtype is a read only register.</td>
</tr>
<tr>
<td>is_reserved_field(regclass, arg2, arg3)</td>
<td>Returns true if the specified data would write a one in a reserved field.</td>
</tr>
<tr>
<td>is_reserved_reg(regclass, regnum)</td>
<td>Returns true if register regnum is reserved in the regclass register file.</td>
</tr>
<tr>
<td>long_branch_implemented()</td>
<td>Implementation-dependent routine which returns TRUE or FALSE, depending on whether long branches are implemented.</td>
</tr>
<tr>
<td>mem_flush(paddr)</td>
<td>The line addressed by the physical address paddr is invalidated in all levels of the memory hierarchy above memory and written back to memory if it is inconsistent with memory.</td>
</tr>
<tr>
<td>mem_flush_pending_stores()</td>
<td>The processor is instructed to start draining pending stores in write coalescing and write buffers. This operation is a “hint”. There is no indication when prior stores have actually been drained.</td>
</tr>
<tr>
<td>mem_implicit_prefetch(vaddr, hint, type)</td>
<td>Moves the line addressed by vaddr to the location of the memory hierarchy specified by hint. This function is implementation dependent and can be ignored. The type allows the implementation to distinguish prefetches for different instruction types.</td>
</tr>
<tr>
<td>mem_promote(paddr, mtype, hint)</td>
<td>Moves the line addressed by paddr to the highest level of the memory hierarchy conditioned by the access hints specified by hint. Implementation dependent and can be ignored.</td>
</tr>
<tr>
<td>mem_read(paddr, size, border, mattr, otype, hint)</td>
<td>Returns the size bytes starting at the physical memory location specified by paddr with byte order specified by border, memory attributes specified by mattr, and access hint specified by hint. otype specifies the memory ordering attribute of this access, and must be UNORDERED or ACQUIRE.</td>
</tr>
<tr>
<td>fp_mem_to_fr_format(mem, size)</td>
<td>Converts a floating-point value in memory format to floating-point register format.</td>
</tr>
<tr>
<td>mem_write(value, paddr, size, border, mattr, otype, hint)</td>
<td>Writes the least significant size bytes of value into memory starting at the physical memory address specified by paddr with byte order specified by border, memory attributes specified by mattr, and access hint specified by hint. otype specifies the memory ordering attribute of this access, and must be UNORDERED or RELEASE.</td>
</tr>
<tr>
<td>mem_xchg(data, paddr, size, byte_order, mattr, otype, hint)</td>
<td>Returns size bytes from memory starting at the physical address specified by paddr. The read is conditioned by the locality hint specified by hint. After the read, the least significant size bytes of data are written to size bytes in memory starting at the physical address specified by paddr. The read and write are performed atomically. Both the read and the write are conditioned by the memory attribute specified by mattr and the byte ordering in memory is specified by byte_order. otype specifies the memory ordering attribute of this access, and must be ACQUIRE.</td>
</tr>
<tr>
<td>mem_xchg_add(add_val, paddr, size, byte_order, mattr, otype, hint)</td>
<td>Returns size bytes from memory starting at the physical address specified by paddr. The read is conditioned by the locality hint specified by hint. The least significant size bytes of the sum of the value read from memory and add_val is then written to size bytes in memory starting at the physical address specified by paddr. The read and write are performed atomically. Both the read and the write are conditioned by the memory attribute specified by mattr and the byte ordering in memory is specified by byte_order. otype specifies the memory ordering attribute of this access, and has the value ACQUIRE or RELEASE.</td>
</tr>
<tr>
<td>mem_xchg_cond(cmp_val, data, paddr, size, byte_order, mattr, otype, hint)</td>
<td>Returns size bytes from memory starting at the physical address specified by paddr. The read is conditioned by the locality hint specified by hint. If the value read from memory is equal to cmp_val, then the least significant size bytes of data are written to size bytes in memory starting at the physical address specified by paddr. If the write is performed, the read and write are performed atomically. Both the read and the write are conditioned by the memory attribute specified by mattr and the byte ordering in memory is specified by byte_order. otype specifies the memory ordering attribute of this access, and has the value ACQUIRE or RELEASE.</td>
</tr>
</tbody>
</table>
Table 3-1. Pseudo-Code Functions (Continued)

<table>
<thead>
<tr>
<th>Function</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>ordering_fence()</td>
<td>Ensures prior data memory references are made visible before future data memory references are made visible by the processor.</td>
</tr>
<tr>
<td>pr_phys_to_virt(phys_id)</td>
<td>Returns the virtual register id of the predicate from the physical register id, phys_id of the predicate.</td>
</tr>
<tr>
<td>rotate_regs()</td>
<td>Decrements the Register Rename Base registers, effectively rotating the register files. CFM.rrb.gr is decremented only if CFM.sor is non-zero.</td>
</tr>
<tr>
<td>rse_enable_current_frame_load()</td>
<td>If the RSE load pointer (RSE.BspLoad) is greater than AR[BSP], the RSE.CFLE bit is set to indicate that mandatory RSE loads are allowed to restore registers in the current frame (in no other case does the RSE spill or fill registers in the current frame). This function does not perform mandatory RSE loads. This procedure does not cause any interruptions.</td>
</tr>
<tr>
<td>rse_ensure_regs_loaded(number_of_bytes)</td>
<td>All registers and NaT collections between AR[BSP] and (AR[BSP]+number_of_bytes) which are not already in stacked registers are loaded into the register stack with mandatory RSE loads. If the number of registers to be loaded is greater than RSE.N_STACK_PHYS an Illegal Operation fault is raised. All registers starting with backing store address (AR[BSP] - 8) and decrementing down to and including backing store address (AR[BSP] - number_of_bytes) are made part of the dirty partition. With exception of the current frame, all other stacked registers are made part of the invalid partition. Note that number_of_bytes may be zero. The resulting sequence of RSE loads may be interrupted. Mandatory RSE loads may cause an interruption; see Table 6-6 on page 2:128 in Volume 2.</td>
</tr>
<tr>
<td>rse_invalidate_non_current_regs()</td>
<td>All registers outside the current frame are invalidated.</td>
</tr>
<tr>
<td>rse_load(type)</td>
<td>Restores a register or NaT collection from the backing store (load_address = RSE.BspLoad - 8). If load_address(8:3) is equal to 0x3f then a NaT collection is loaded into a NaT dispersal register. (dispersal register may not be the same as AR[RNAT].) If load_address(8:3) is not equal to 0x3f then the register RSE.LoadReg.1 is loaded and the NaT bit for that register is set to dispersal_register(load_address(8:3)). If the load is successful RSE.BspLoad is decremented by 8. If the load is successful and a register was loaded RSE.LoadReg is decremented by 1 (possibly wrapping in the stacked registers). The load moves a register from the invalid partition to the current frame if RSE.CFLE is 1, or to the clean partition if RSE.CFLE is 0. For mandatory RSE loads, type is MANDATORY. Mandatory RSE loads may cause interruptions. See Table 6-6 on page 2:128 in Volume 2.</td>
</tr>
<tr>
<td>rse_new_frame(current_frame_size, new_frame_size)</td>
<td>A new frame is defined without changing any register renaming. The new frame size is completely defined by the new_frame_size parameter (successive calls are not cumulative). If new_frame_size is larger than current_frame_size and the number of registers in the invalid and clean partitions is less than the size of frame growth then mandatory RSE stores are issued until enough registers are available. The resulting sequence of RSE stores may be interrupted. Mandatory RSE stores may cause interruptions; see Table 6-6 on page 2:128 in Volume 2.</td>
</tr>
<tr>
<td>rse_preserve_frame(preserved_frame_size)</td>
<td>The number of registers specified by preserved_frame_size are marked to be preserved by the RSE. Register renaming causes the preserved_frame_size registers after GR[32] to be renamed to GR[32]. AR[BSP] is updated to contain the backing store address where the new GR[32] will be stored.</td>
</tr>
</tbody>
</table>
### Table 3-1. Pseudo-Code Functions (Continued)

<table>
<thead>
<tr>
<th>Function</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>rse_restore_frame(preserved_sol, growth, current_frame_size)</td>
<td>The first two parameters define how the current frame is about to be updated by a branch return or rfi: preserved_sol defines how many registers need to be restored below RSE.BOF; growth defines how by many registers the top of the current frame will grow (growth will generally be negative). The number of registers specified by preserved_sol are marked to be restored. Register renaming causes the preserved_sol registers before GR[32] to be renamed to GR[32]. AR[BSP] is updated to contain the backing store address where the new GR[32] will be stored. If the number of dirty and clean registers is less than preserved_sol then mandatory RSE loads must be issued before the new current frame is considered valid. This function does not perform mandatory RSE loads. This function returns TRUE if the preserved frame grows beyond the invalid and clean regions into the dirty region. In this case the third argument, current_frame_size, is used to force the returned to frame to zero (see “Bad PFS used by Branch Return” on page 2:126 in Volume 2).</td>
</tr>
<tr>
<td>rse_store(type)</td>
<td>Saves a register or NaT collection to the backing store (store_address = AR[BSPSTORE]). If store_address(8:3) is equal to 0x3f then the NaT collection AR[RNAT] is stored. If store_address(8:3) is not equal to 0x3f then the register RSE.StoreReg is stored and the NaT bit from that register is deposited in AR[RNAT][store_address(8:3)]. If the store is successful AR[BSPSTORE] is incremented by 8. If the store is successful and a register was stored RSE.StoreReg is incremented by 1 (possibly wrapping in the stacked registers). This store moves a register from the dirty partition to the clean partition. For mandatory RSE stores, type is MANDATORY. Mandatory RSE stores may cause interruptions. See Table 6-6 on page 2:128 in Volume 2.</td>
</tr>
<tr>
<td>rse_update_internal_stack_pointers(new_store_pointer)</td>
<td>Given a new value for AR[BSPSTORE] (new_store_pointer) this function computes the new value for AR[BSP]. This value is equal to new_store_pointer plus the number of dirty registers plus the number of intervening NaT collections. This means that the size of the dirty partition is the same before and after a write to AR[BSPSTORE]. All clean registers are moved to the invalid partition.</td>
</tr>
<tr>
<td>sign_ext(value, pos)</td>
<td>Returns a 64 bit number with bits pos-1 through 0 taken from value and bit pos-1 of value replicated in bit positions pos through 63. If pos is greater than or equal to 64, value is returned.</td>
</tr>
<tr>
<td>tlb_access_key(vaddr)</td>
<td>This function returns the access key from the TLB for the entry corresponding to vaddr.</td>
</tr>
<tr>
<td>tlb_broadcast_purge(rid, vaddr, size, type)</td>
<td>Sends a broadcast purge DTC and ITC transaction to other processors in the multi-processor coherency domain, where the region identifier (rid), virtual address (vaddr) and page size (size) specify the translation entry to purge. The operation waits until all processors that receive the purge have completed the purge operation. The purge type (type) specifies whether the ALAT on other processors should also be purged in conjunction with the TC.</td>
</tr>
<tr>
<td>tlb_enter_privileged_code()</td>
<td>This function determines the new privilege level for epc from the TLB entry for the page containing this instruction. If the page containing the epc instruction has execute-only page access rights and the privilege level assigned to the page is higher than (numerically less than) the current privilege level, then the current privilege level is set to the privilege level field in the translation for the page containing the epc instruction.</td>
</tr>
<tr>
<td>tlb_grant_permission(vaddr, type, pl)</td>
<td>Returns a boolean indicating if read, write access is granted for the specified virtual memory address (vaddr) and privilege level (pl). The access type (type) specifies either read or write. The following faults are checked; VHPT Translation, TLB Miss, Nested TLB, Page Not Present, NaT Page Consumption, and Key Miss. If a fault is generated, this function does not return.</td>
</tr>
<tr>
<td>tlb_insert_data(slot, pte0, pte1, vaddr, rid, tr)</td>
<td>Inserts an entry into the DTB, at the specified slot number. pte0, pte1 compose the translation. vaddr and rid specify the virtual address and region identifier for the translation. If tr is true the entry is placed in the TR section, otherwise the TC section.</td>
</tr>
</tbody>
</table>
Table 3-1. Pseudo-Code Functions (Continued)

<table>
<thead>
<tr>
<th>Function</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>tlb_insert_inst(slot, pte0, pte1, vaddr, rid, tr)</td>
<td>Inserts an entry into the ITLB, at the specified slot number. pte0, pte1 compose the translation. vaddr and rid specify the virtual address and region identifier for the translation. If tr is true, the entry is placed in the TR section, otherwise the TC section.</td>
</tr>
<tr>
<td>tlb_may_purge_dtc_entries(rid, vaddr, size)</td>
<td>May locally purge DTC entries that match the specified virtual address (vaddr), region identifier (rid) and page size (size). May also invalidate entries that partially overlap the parameters. The extent of purging is implementation dependent. If the purge size is not supported, an implementation may generate a machine check abort or over purge the translation cache up to and including removal of all entries from the translation cache.</td>
</tr>
<tr>
<td>tlb_may_purge_itc_entries(rid, vaddr, size)</td>
<td>May locally purge ITC entries that match the specified virtual address (vaddr), region identifier (rid) and page size (size). May also invalidate entries that partially overlap the parameters. The extent of purging is implementation dependent. If the purge size is not supported, an implementation may generate a machine check abort or over purge the translation cache up to and including removal of all entries from the translation cache.</td>
</tr>
<tr>
<td>tlb_must_purge_dtc_entries(rid, vaddr, size)</td>
<td>Purges all local, possibly overlapping, DTC entries matching the specified region identifier (rid), virtual address (vaddr) and page size (size). vaddr{63:61} (VRN) is ignored in the purge, ie all entries that match vaddr{60:0} must be purged regardless of the VRN bits. If the purge size is not supported, an implementation may generate a machine check abort or over purge the translation cache up to and including removal of all entries from the translation cache. If the specified purge values overlap with an existing DTR translation, an implementation may generate a machine check abort.</td>
</tr>
<tr>
<td>tlb_must_purge_itc_entries(rid, vaddr, size)</td>
<td>Purges all local, possibly overlapping, ITC entry matching the specified region identifier (rid), virtual address (vaddr) and page size (size). vaddr{63:61} (VRN) is ignored in the purge, ie all entries that match vaddr{60:0} must be purged regardless of the VRN bits. If the purge size is not supported, an implementation may generate a machine check abort or over purge the translation cache up to and including removal of all entries from the translation cache. If the specified purge values overlap with an existing ITR translation, an implementation may generate a machine check abort.</td>
</tr>
<tr>
<td>tlb_must_purge_dtr_entries(rid, vaddr, size)</td>
<td>Purges all local, possibly overlapping, DTR entries matching the specified region identifier (rid), virtual address (vaddr) and page size (size). vaddr{63:61} (VRN) is ignored in the purge, ie all entries that match vaddr{60:0} must be purged regardless of the VRN bits. If the purge size is not supported, an implementation may generate a machine check abort or over purge the translation cache up to and including removal of all entries from the translation cache.</td>
</tr>
<tr>
<td>tlb_must_purge_itr_entries(rid, vaddr, size)</td>
<td>Purges all local, possibly overlapping, ITR entry matching the specified region identifier (rid), virtual address (vaddr) and page size (size). vaddr{63:61} (VRN) is ignored in the purge, ie all entries that match vaddr{60:0} must be purged regardless of the VRN bits. If the purge size is not supported, an implementation may generate a machine check abort or over purge the translation cache up to and including removal of all entries from the translation cache.</td>
</tr>
<tr>
<td>tlb_purge_translation_cache(loop)</td>
<td>Removes 1 to N translations from the local processor’s ITC and DTC. The number of entries removed is implementation specific. The parameter loop is used to generate an implementation specific purge parameter.</td>
</tr>
<tr>
<td>tlb_replacement_algorithm(tlb)</td>
<td>Returns the next ITC or DTC slot number to replace. Replacement algorithms are implementation specific. tlb specifies to perform the algorithm on the ITC or DTC.</td>
</tr>
<tr>
<td>tlb_search_pkr(key)</td>
<td>Searches for a valid protection key register with a matching protection key. The search algorithm is implementation specific. Returns the PKR register slot number if found, otherwise returns Not Found.</td>
</tr>
</tbody>
</table>
**Table 3-1. Pseudo-Code Functions (Continued)**

<table>
<thead>
<tr>
<th>Function</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<tr>
<td>tlb_translate(vaddr, size, type, cpl, <em>attr</em>, <em>defer</em>)</td>
<td>Returns the translated data physical address for the specified virtual memory address (vaddr) when translation enabled; otherwise, returns vaddr.size specifies the size of the access, type specifies the type of access (e.g., read, write, advance, spec). cpl specifies the privilege level for access checking purposes. <em>attr</em> returns the mapped physical memory attribute. If any fault conditions are detected and deferred, tlb_translate returns with *defer set. If a fault is generated but the fault is not deferred, tlb_translate does not return. tlb_translate checks the following faults:</td>
</tr>
<tr>
<td></td>
<td>• VHPT Data fault</td>
</tr>
<tr>
<td></td>
<td>• Data Nested TLB fault</td>
</tr>
<tr>
<td></td>
<td>• Data TLB fault</td>
</tr>
<tr>
<td></td>
<td>• Alternate Data TLB fault</td>
</tr>
<tr>
<td></td>
<td>• Data page not present fault</td>
</tr>
<tr>
<td></td>
<td>• Data NaT page consumption fault</td>
</tr>
<tr>
<td></td>
<td>• Data key miss fault</td>
</tr>
<tr>
<td></td>
<td>• Data key permission fault</td>
</tr>
<tr>
<td></td>
<td>• Data access rights fault</td>
</tr>
<tr>
<td></td>
<td>• Data dirty bit fault</td>
</tr>
<tr>
<td></td>
<td>• Data access bit fault</td>
</tr>
<tr>
<td></td>
<td>• Data debug fault</td>
</tr>
<tr>
<td>tlb_translate_nonaccess(vaddr, type)</td>
<td>Returns the translated data physical address for the specified virtual memory address (vaddr). type specifies the type of access (e.g., FC, TPA). If a fault is generated, tlb_translate_nonaccess does not return. The following faults are checked:</td>
</tr>
<tr>
<td></td>
<td>• VHPT data fault</td>
</tr>
<tr>
<td></td>
<td>• Data TLB fault</td>
</tr>
<tr>
<td></td>
<td>• Data Nested TLB fault</td>
</tr>
<tr>
<td></td>
<td>• Data page not present fault</td>
</tr>
<tr>
<td></td>
<td>• Data NaT page consumption fault</td>
</tr>
<tr>
<td>tlb_vhpt_hash(vrn, vaddr61, rid, size)</td>
<td>Generates a VHPT entry address for the specified virtual region number (vrn) and 61-bit virtual offset (vaddr61), region identifier (rid) and page size (size). Tlb_vhpt_hash hashes vaddr, rid and size parameters to produce a hash index. The hash index is then masked based on PTA.size and concatenated with PTA.base to generate the VHPT entry address. The long format hash is implementation specific.</td>
</tr>
<tr>
<td>tlb_vhpt_tag(vaddr, rid, size)</td>
<td>Generates a VHPT tag identifier for the specified virtual address (vaddr), region identifier (rid) and page size (size). Tlb_vhpt_tag hashes the vaddr, rid and size parameters to produce translation identifier. The tag in conjunction with the hash index is used to uniquely identify translations in the VHPT. Tag generation is implementation specific. All processor models tag function must guarantee that bit 63 of the generated tag is zero (ti bit).</td>
</tr>
<tr>
<td>unimplemented_physical_address(paddr)</td>
<td>Return TRUE if the presented physical address is unimplemented on this processor model; FALSE otherwise. This function is model-specific.</td>
</tr>
<tr>
<td>undefined()</td>
<td>Returns an undefined 64-bit value.</td>
</tr>
<tr>
<td>undefined_behavior()</td>
<td>Causes undefined processor behavior. Extent of undefined behavior is described in “Undefined Behavior” on page 2:39 in Volume 1.</td>
</tr>
<tr>
<td>unimplemented_virtual_address(vaddr)</td>
<td>Return TRUE if the presented virtual address is unimplemented on this processor model; FALSE otherwise. This function is model-specific.</td>
</tr>
<tr>
<td>fp_update_fpsr(sf, tmp_fp_env)</td>
<td>Copies a floating-point instruction’s local state into the global FPSR.</td>
</tr>
<tr>
<td>fp_update_psr(dest_freg)</td>
<td>Conditionally sets PSR.mfl or PSR.mfh based on dest_freg.</td>
</tr>
<tr>
<td>zero_ext(value, pos)</td>
<td>Returns a 64 bit unsigned number with bits pos-1 through 0 taken from value and zeroes in bit positions pos through 63. If pos is greater than or equal to 64, value is returned.</td>
</tr>
</tbody>
</table>
Each Itanium instruction is categorized into one of six types; each instruction type may be executed on one or more execution unit types. Table 4-1 lists the instruction types and the execution unit type on which they are executed:

**Table 4-1. Relationship between Instruction Type and Execution Unit Type**

<table>
<thead>
<tr>
<th>Instruction Type</th>
<th>Description</th>
<th>Execution Unit Type</th>
</tr>
</thead>
<tbody>
<tr>
<td>A</td>
<td>Integer ALU</td>
<td>I-unit or M-unit</td>
</tr>
<tr>
<td>I</td>
<td>Non-ALU integer</td>
<td>I-unit</td>
</tr>
<tr>
<td>M</td>
<td>Memory</td>
<td>M-unit</td>
</tr>
<tr>
<td>F</td>
<td>Floating-point</td>
<td>F-unit</td>
</tr>
<tr>
<td>B</td>
<td>Branch</td>
<td>B-unit</td>
</tr>
<tr>
<td>L+X</td>
<td>Extended</td>
<td>I-unit/B-unit&lt;sup&gt;a&lt;/sup&gt;</td>
</tr>
</tbody>
</table>

<sup>a</sup> L+X Major Opcodes 0 - 7 execute on an I-unit. L+X Major Opcodes 8 - F execute on a B-unit.

Three instructions are grouped together into 128-bit sized and aligned containers called **bundles**. Each bundle contains three 41-bit **instruction slots** and a 5-bit template field. The format of a bundle is depicted in Figure 4-1.

**Figure 4-1. Bundle Format**

```
+----------------+----------------+----------------+----------------+----------------+
|                |                |                |                |                |
| instruction slot 2 | instruction slot 1 | instruction slot 0 | template |
+----------------+----------------+----------------+----------------+
| 41             | 41             | 41             | 5              |
| 87             | 86             | 46             | 45             |
+----------------+----------------+----------------+----------------+
```

The template field specifies two properties: stops within the current bundle, and the mapping of instruction slots to execution unit types. Not all combinations of these two properties are allowed - Table 4-2 indicates the defined combinations. The three rightmost columns correspond to the three instruction slots in a bundle; listed within each column is the execution unit type controlled by that instruction slot for each encoding of the template field. A double line to the right of an instruction slot indicates that a stop occurs at that point within the current bundle. See “Instruction Encoding Overview” on page 3:32 for the definition of a stop. Within a bundle, execution order proceeds from slot 0 to slot 2. Unused template values (appearing as empty rows in Table 4-2) are reserved and cause an Illegal Operation fault.

Extended instructions, used for long immediate integer and long branch instructions, occupy two instruction slots. Depending on the major opcode, extended instructions execute on a B-unit (long branch/call) or an I-unit (all other L+X instructions).
4.1 Format Summary

All instructions in the instruction set are 41 bits in length. The leftmost 4 bits (40:37) of each instruction are the major opcode. Table 4-3 shows the major opcode assignments for each of the 5 instruction types — ALU (A), Integer (I), Memory (M), Floating-point (F), and Branch (B). Bundle template bits are used to distinguish among the 4 columns, so the same major op values can be reused in each column.

Unused major ops (appearing as blank entries in Table 4-3) behave in one of four ways:
• Ignored major ops (white entries in Table 4-3) execute as \texttt{nop} instructions.
• Reserved major ops (light gray in the gray scale version of Table 4-3, brown in the color version) cause an Illegal Operation fault.
• Reserved if PR\[qp\] is 1 major ops (dark gray in the gray scale version of Table 4-3, purple in the color version) cause an Illegal Operation fault if the predicate register specified by the qp field of the instruction (bits 5:0) is 1 and execute as a \texttt{nop} instruction if 0.
• Reserved if PR\[qp\] is 1 B-unit major ops (medium gray in the gray scale version of Table 4-3, cyan in the color version) cause an Illegal Operation fault if the predicate register specified by the qp field of the instruction (bits 5:0) is 1 and execute as a \texttt{nop} instruction if 0. These differ from the Reserved if PR\[qp\] is 1 major ops (purple) only in their RAW dependency behavior (see “RAW Dependency Table” on page 3:338).

Table 4-3. Major Opcode Assignments

<table>
<thead>
<tr>
<th>Major Op (Bits 40:37)</th>
<th>Instruction Type</th>
<th>I/A</th>
<th>M/A</th>
<th>F</th>
<th>B</th>
<th>L+X</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>Misc 0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>1</td>
<td>Sys/Mem Mgmt 1</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>2</td>
<td>Sys/Mem Mgmt 2</td>
<td>2</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>3</td>
<td>Sys/Mem Mgmt 3</td>
<td>3</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>4</td>
<td>Deposit 4</td>
<td>4</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>5</td>
<td>Shift/Test Bit 5</td>
<td>5</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>6</td>
<td>Shift/Test Bit 6</td>
<td>6</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>7</td>
<td>Shift/Test Bit 7</td>
<td>7</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>8</td>
<td>ALU/MM ALU 8</td>
<td>8</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>9</td>
<td>Add Imm 22</td>
<td>9</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>A</td>
<td>Compare C</td>
<td>A</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>B</td>
<td>Compare D</td>
<td>B</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>C</td>
<td>Compare E</td>
<td>C</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>D</td>
<td>Compare F</td>
<td>D</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>E</td>
<td>Compare G</td>
<td>E</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>F</td>
<td>Compare H</td>
<td>F</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>

Table 4-4 on page 3:260 summarizes all the instruction formats. The instruction fields are color-coded for ease of identification, as described in Table 4-5 on page 3:262. A color version of this chapter is available for those heavily involved in working with the instruction encodings.

The instruction field names, used throughout this chapter, are described in Table 4-6 on page 3:262. The set of special notations (such as whether an instruction is privileged) are listed in Table 4-7 on page 3:263. These notations appear in the “Instruction” column of the opcode tables.

Most instruction containing immediates encode those immediates in more than one instruction field. For example, the 14-bit immediate in the Add Imm 14 instruction (format A4) is formed from the imm7b, imm6d, and s fields. Table 4-70 on page 3:333 shows how the immediates are formed from the instruction fields for each instruction which has an immediate.
Table 4-4. Instruction Format Summary

<table>
<thead>
<tr>
<th>Format</th>
<th>Example</th>
</tr>
</thead>
<tbody>
<tr>
<td>ALU</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Shift L and Add</td>
<td><code>x2a y2b x4 x2b</code></td>
</tr>
<tr>
<td>ALU Imm31</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Add Imm14</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Add Imm22</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Compare</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Compare to Zero</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Compare Imm31</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>MM ALU</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>MM Shift and Add</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>MM Multiply Shift</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>MM Mpy/Mix/Pack</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>MM Mux1</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>MM Mux2</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Shift R Variable</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Shift L Variable</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Shift L Fixed</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>MM Shift L Fixed</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Popcount</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Shift Right Pair</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Extract</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Dep.Z</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Dep.Z Imm31</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Deposit Imm14</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Deposit</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Test Bit</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Test NaT</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Break/Nop</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Int Spec Check</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move to BR</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move from BR</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move to Pred</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move from Pred</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move to Pred Imm31</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move from Pred/IP</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move to AR</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move to AR Imm31</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Move from AR</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Sxt/Zxt/Cxz</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Int Load</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Int Load +Reg</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Int Load +Imm</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Int Store +Imm</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Load</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Load +Reg</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Load +Imm</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Store</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Store +Imm</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Load Pair</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Load Pair +Imm</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Line Prefetch</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Line Prefetch +Reg</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Line Prefetch +Imm</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>(Cmp &amp; Exch)</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Fetch &amp; Add</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Set FR</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Get FR</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>Int Spec Check</td>
<td><code>x2a y2b x4</code></td>
</tr>
<tr>
<td>FP Spec Check</td>
<td><code>x2a y2b x4</code></td>
</tr>
</tbody>
</table>
### Table 4-4. Instruction Format Summary (Continued)

<table>
<thead>
<tr>
<th>Instruction Type</th>
<th>Format</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>Int ALAT Check M22</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>FP ALAT Check M23</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Sync/Srlz/ALAT M24</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>RSE Control M25</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Int ALAT Inval M26</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>FP ALAT Inval M27</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Flush Cache/Ptc.e M28</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Move to AR M29</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Move to AR Imm8 M30</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Move from AR M31</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Move to CR M32</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Move from CR M33</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Alloc M34</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Move to PSR M35</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Move from PSR M36</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Break/Nop M37</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Probe M38</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Probe Imm2 M39</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Probe Fault Imm2 M40</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>TC Insert M41</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Mv to Ind/TR Ins M42</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Mv from Ind M43</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Set/Reset Mask M44</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Translation Purge M45</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>Translation Access M46</td>
<td>0 1 2</td>
<td>imm20b  r1  q0</td>
</tr>
<tr>
<td>IP-Relative Branch B1</td>
<td>4 0 3</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Counted Branch B2</td>
<td>4 0 3</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>IP-Relative Call B3</td>
<td>5 0 3</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Indirect Branch B4</td>
<td>0 1 2</td>
<td>imm20b  b2  p  blype  q0</td>
</tr>
<tr>
<td>Indirect Call B5</td>
<td>0 1 2</td>
<td>imm20b  b2  p  blype  q0</td>
</tr>
<tr>
<td>IP-Relative Predict B6</td>
<td>7 0 3</td>
<td>imm20b  imm7b  wh  imm7a  wh</td>
</tr>
<tr>
<td>Indirect Predict B7</td>
<td>2 0 3</td>
<td>imm20b  b2  imm7b  wh  imm7a  wh</td>
</tr>
<tr>
<td>Misc B8</td>
<td>0 1 2</td>
<td>imm20b  b2  imm7b  wh  imm7a  wh</td>
</tr>
<tr>
<td>Break/Nop B9</td>
<td>0/2 1</td>
<td>imm20b  b2  imm7b  wh  imm7a  wh</td>
</tr>
<tr>
<td>FP Arithmetic F1</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Fixed Multiply Add F2</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Select F3</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Compare F4</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Class F5</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Recip Approx F6</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Recip Sqr App F7</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Min/Max/Pcmp F8</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Merge/Logical F9</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Convert FP to Fixed F10</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Convert Fixed to FP F11</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Set Controls F12</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Clear Flags F13</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>FP Check Flags F14</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Break/Nop F15</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Break/Nop X1</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Move Imm64 X2</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Long Branch X3</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
<tr>
<td>Move Imm32 X4</td>
<td>0 1 6</td>
<td>imm20b  btype  p  blype  q0</td>
</tr>
</tbody>
</table>
### Table 4-5. Instruction Field Color Key

<table>
<thead>
<tr>
<th>Field &amp; Color</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>ALU Instruction</td>
<td>Opcode Extension</td>
</tr>
<tr>
<td>Integer Instruction</td>
<td>Opcode Hint Extension</td>
</tr>
<tr>
<td>Memory Instruction</td>
<td>Immediate</td>
</tr>
<tr>
<td>Branch Instruction</td>
<td>Indirect Source</td>
</tr>
<tr>
<td>Floating-point Instruction</td>
<td>Predicate Destination</td>
</tr>
<tr>
<td>Integer Source</td>
<td>Integer Destination</td>
</tr>
<tr>
<td>Memory Source</td>
<td>Memory Source &amp; Destination</td>
</tr>
<tr>
<td>Shift Source</td>
<td>Shift Immediate</td>
</tr>
<tr>
<td>Special Register Source</td>
<td>Special Register Destination</td>
</tr>
<tr>
<td>Floating-point Source</td>
<td>Floating-point Destination</td>
</tr>
<tr>
<td>Branch Source</td>
<td>Branch Destination</td>
</tr>
<tr>
<td>Address Source</td>
<td>Branch Tag Immediate</td>
</tr>
<tr>
<td>Qualifying Predicate</td>
<td>Reserved Instruction</td>
</tr>
<tr>
<td>Ignored Field/Instruction</td>
<td>Reserved Inst if PR[qp] is 1</td>
</tr>
<tr>
<td></td>
<td>Reserved B-type Inst if PR[qp] is 1</td>
</tr>
</tbody>
</table>

### Table 4-6. Instruction Field Names

<table>
<thead>
<tr>
<th>Field Name</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>ar₃</td>
<td>application register source/target</td>
</tr>
<tr>
<td>b₁, b₂</td>
<td>branch register source/target</td>
</tr>
<tr>
<td>btype</td>
<td>branch type opcode extension</td>
</tr>
<tr>
<td>c</td>
<td>complement compare relation opcode extension</td>
</tr>
<tr>
<td>count₅c</td>
<td>multimedia shift left complemented shift count immediate</td>
</tr>
<tr>
<td>count₆d, count₆d</td>
<td>multimedia shift right/shift right pair shift count immediate</td>
</tr>
<tr>
<td>cposₓ</td>
<td>deposit complemented bit position immediate</td>
</tr>
<tr>
<td>cr₃</td>
<td>control register source/target</td>
</tr>
<tr>
<td>ct₂d, mht₈c</td>
<td>multimedia mux1/mux2 immediate</td>
</tr>
<tr>
<td>d</td>
<td>branch cache deallocation hint opcode extension</td>
</tr>
<tr>
<td>f₀</td>
<td>floating-point register source/target</td>
</tr>
<tr>
<td>fc₂, fclass₇c</td>
<td>floating-point class immediate</td>
</tr>
<tr>
<td>hint</td>
<td>memory reference hint opcode extension</td>
</tr>
<tr>
<td>i, i₂b, i₂d, immₓ</td>
<td>immediate of length 1, 2, or x</td>
</tr>
<tr>
<td>ih</td>
<td>branch importance hint opcode extension</td>
</tr>
<tr>
<td>len₄d, len₆d</td>
<td>extract/deposit length immediate</td>
</tr>
<tr>
<td>m</td>
<td>memory reference post-modify opcode extension</td>
</tr>
<tr>
<td>maskₓ</td>
<td>predicate immediate mask</td>
</tr>
<tr>
<td>mb₁₂c, mhf₆c</td>
<td>multimedia mux1/mux2 immediate</td>
</tr>
<tr>
<td>p</td>
<td>sequential prefetch hint opcode extension</td>
</tr>
<tr>
<td>p₁₁, P₂</td>
<td>predicate register target</td>
</tr>
<tr>
<td>pos₂b</td>
<td>test bit/exact bit position immediate</td>
</tr>
<tr>
<td>q</td>
<td>floating-point reciprocal/reciprocal square-root opcode extension</td>
</tr>
<tr>
<td>qp</td>
<td>qualifying predicate register source</td>
</tr>
<tr>
<td>r₀</td>
<td>general register source/target</td>
</tr>
<tr>
<td>s</td>
<td>immediate sign bit</td>
</tr>
<tr>
<td>sf</td>
<td>floating-point status field opcode extension</td>
</tr>
</tbody>
</table>
The remaining sections of this chapter present the detailed encodings of all instructions. The “A-Unit Instruction encodings” are presented first, followed by the “I-Unit Instruction Encodings” on page 3:274, “M-Unit Instruction Encodings” on page 3:287, “B-Unit Instruction Encodings” on page 3:315, “F-Unit Instruction Encodings” on page 3:322, and “X-Unit Instruction Encodings” on page 3:330.

Within each section, the instructions are grouped by function, and appear with their instruction format in the same order as in Table 4-4 “Instruction Format Summary” on page 3:260. The opcode extension fields are briefly described and tables present the opcode extension assignments. Unused instruction encodings (appearing as blank entries in the opcode extensions tables) behave in one of four ways:

- Ignored instructions (white color entries in the tables) execute as **nop** instructions.
- Reserved instructions (light gray color in the gray scale version of the tables, brown color in the color version) cause an Illegal Operation fault.
- Reserved if PR[qp] is 1 instructions (dark gray in the gray scale version of the tables, purple in the color version) cause an Illegal Operation fault if the predicate register specified by the qp field of the instruction (bits 5:0) is 1 and execute as a **nop** instruction if 0.
- Reserved if PR[qp] is 1 B-unit instructions (medium gray in the gray scale version of the tables, cyan in the color version) cause an Illegal Operation fault if the predicate register specified by the qp field of the instruction (bits 5:0) is 1 and execute as a **nop** instruction if 0. These differ from the Reserved if PR[qp] is 1 instructions (purple) only in their RAW dependency behavior (see “RAW Dependency Table” on page 3:338).

Some processors may implement the Reserved if PR[qp] is 1 (purple) and Reserved if PR[qp] is 1 B-unit (cyan) encodings in the L+X opcode space as Reserved (brown). These encodings appear in the L+X column of Table 4-3 on page 3:259, and in Table 4-66 on page 3:330, Table 4-67 on

---

**Table 4-6. Instruction Field Names (Continued)**

<table>
<thead>
<tr>
<th>Field Name</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>sof, sol, sor</td>
<td>alloc size of frame, size of locals, size of rotating immediates</td>
</tr>
<tr>
<td>l_a, l_b</td>
<td>compare type opcode extension</td>
</tr>
<tr>
<td>l_ge, limm_k</td>
<td>branch predict tag immediate</td>
</tr>
<tr>
<td>v_x</td>
<td>reserved opcode extension field</td>
</tr>
<tr>
<td>wh</td>
<td>branch whether hint opcode extension</td>
</tr>
<tr>
<td>x, x_n</td>
<td>opcode extension of length 1 or n</td>
</tr>
<tr>
<td>y</td>
<td>extract/deposit/test bit/test NaT opcode extension</td>
</tr>
<tr>
<td>z_a, z_b</td>
<td>multimedia operand size opcode extension</td>
</tr>
</tbody>
</table>

**Table 4-7. Special Instruction Notations**

<table>
<thead>
<tr>
<th>Notation</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>e</td>
<td>instruction ends an instruction group when taken, or for Reserved if PR[qp] is 1 (cyan) encodings and non-branch instructions with a qualifying predicate, when its PR[qp] is 1, or for Reserved (brown) encodings, unconditionally</td>
</tr>
<tr>
<td>f</td>
<td>instruction must be the first instruction in an instruction group and must either be in instruction slot 0 or in instruction slot 1 of a template having a stop after slot 0</td>
</tr>
<tr>
<td>i</td>
<td>instruction is allowed in the I slot of an MLI template</td>
</tr>
<tr>
<td>l</td>
<td>instruction must be the last in an instruction group</td>
</tr>
<tr>
<td>p</td>
<td>privileged instruction</td>
</tr>
<tr>
<td>t</td>
<td>instruction is only allowed in instruction slot 2</td>
</tr>
</tbody>
</table>
On processors which implement these encodings as Reserved (brown), the operating system is required to provide an Illegal Operation fault handler which emulates them as Reserved if PR[qp] is 1 (cyan/purple) by decoding the reserved opcodes, checking the qualifying predicate, and returning to the next instruction if PR[qp] is 0.

Constant 0 fields in instructions must be 0 or undefined operation results. The undefined operation may include checking that the constant field is 0 and causing an Illegal Operation fault if it is not. If an instruction having a constant 0 field also has a qualifying predicate (qp field), the fault or other undefined operation must not occur if PR[qp] is 0. For constant 0 fields in instruction bits 5:0 (normally used for qp), the fault or other undefined operation may or may not depend on the PR addressed by those bits.

Ignored (white space) fields in instructions should be coded as 0. Although ignored in this revision of the architecture, future architecture revisions may define these fields as hint extensions. These hint extensions will be defined such that the 0 value in each field corresponds to the default hint. It is expected that assemblers will automatically set these fields to zero by default.

## 4.2 A-Unit Instruction Encodings

### 4.2.1 Integer ALU

All integer ALU instructions are encoded within major opcode 8 using a 2-bit opcode extension field in bits 35:34 ($x_{2a}$) and most have a second 2-bit opcode extension field in bits 28:27 ($x_{2b}$), a 4-bit opcode extension field in bits 32:29 ($x_4$), and a 1-bit reserved opcode extension field in bit 33 ($v_e$). Table 4-8 shows the 2-bit $x_{2a}$ and 1-bit $v_e$ assignments, Table 4-9 shows the integer ALU 4-bit+2-bit assignments, and Table 4-12 on page 3:270 shows the multimedia ALU 1-bit+2-bit assignments (which also share major opcode 8).

### Table 4-8. Integer ALU 2-bit+1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_{2a}$ Bits 35:34</th>
<th>$v_e$ Bit 33</th>
</tr>
</thead>
<tbody>
<tr>
<td>8</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
</tr>
</tbody>
</table>

- **8**: Integer ALU 4-bit+2-bit Ext (Table 4-9)
- **1**: Multimedia ALU 1-bit+2-bit Ext (Table 4-12)
- **2**: adds − imm$_{14}$ A4
- **3**: addp4 − imm$_{14}$ A4
Table 4-9. Integer ALU 4-bit+2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>x2a Bits 35:34</th>
<th>Ve Bits 33</th>
<th>x4 Bits 32:29</th>
<th>x2b Bits 28:27</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>2</td>
<td>3</td>
</tr>
<tr>
<td>0 add A1</td>
<td>8</td>
<td>0</td>
<td>0</td>
<td></td>
</tr>
<tr>
<td>1 sub -1 A1</td>
<td></td>
<td>1</td>
<td></td>
<td></td>
</tr>
<tr>
<td>2 addp4 A1</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>3 and A1</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>4 shladd A2</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5 shladdp4 A2</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>8 sub imm8 A3</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B and imm8 A3</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.2.1.1 Integer ALU – Register-Register

#### A1

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>add</td>
<td>r1 = r2, r3</td>
<td>8</td>
<td>0 0 0 0</td>
</tr>
<tr>
<td>sub</td>
<td>r1 = r2, r3, 1</td>
<td></td>
<td>1 1 0 0</td>
</tr>
<tr>
<td>addp4</td>
<td>r1 = r2, r3, 1</td>
<td></td>
<td>2 0</td>
</tr>
<tr>
<td>and</td>
<td>r1 = r2, r3</td>
<td></td>
<td>3 0 0 0</td>
</tr>
<tr>
<td>or</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>xor</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.2.1.2 Shift Left and Add

#### A2

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>shladd</td>
<td>r1 = r2, count2, r3</td>
<td>8</td>
<td>0 0 4 6</td>
</tr>
<tr>
<td>shladdp4</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
4.2.1.3 Integer ALU – Immediate Register

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>sub</td>
<td></td>
<td>8</td>
<td>9 1</td>
</tr>
<tr>
<td>and</td>
<td></td>
<td>8</td>
<td>0 0 B 1 2 3</td>
</tr>
<tr>
<td>andcm</td>
<td></td>
<td>8</td>
<td>0 0 B 1 2 3</td>
</tr>
<tr>
<td>or</td>
<td></td>
<td>8</td>
<td>0 0 B 1 2 3</td>
</tr>
<tr>
<td>xor</td>
<td></td>
<td>8</td>
<td>0 0 B 1 2 3</td>
</tr>
</tbody>
</table>

4.2.1.4 Add Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>adds</td>
<td></td>
<td>8</td>
<td>2 0</td>
</tr>
<tr>
<td>addp4</td>
<td></td>
<td>8</td>
<td>2 0</td>
</tr>
</tbody>
</table>

4.2.1.5 Add Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
</tr>
</thead>
<tbody>
<tr>
<td>addl</td>
<td></td>
<td>9</td>
</tr>
</tbody>
</table>
4.2.2 Integer Compare

The integer compare instructions are encoded within major opcodes C – E using a 2-bit opcode extension field \( (x_2) \) in bits 35:34 and three 1-bit opcode extension fields in bits 33 \( (t_a) \), 36 \( (t_b) \), and 12 \( (c) \), as shown in Table 4-10. The integer compare immediate instructions are encoded within major opcodes C – E using a 2-bit opcode extension field \( (x_2) \) in bits 35:34 and two 1-bit opcode extension fields in bits 33 \( (t_a) \) and 12 \( (c) \), as shown in Table 4-11.

Table 4-10. Integer Compare Opcode Extensions

<table>
<thead>
<tr>
<th>( x_2 ) Bits 35:34</th>
<th>( t_b ) Bit 36</th>
<th>( t_a ) Bit 33</th>
<th>( c ) Bit 12</th>
<th>Opcode Bits 40:37</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>D</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>E</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>cmp.lt A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.ltu A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.eq A6</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>cmp.lt.unc A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.ltu.unc A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.eq.unc A6</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>cmp.ne.and A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.ne.or A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.ne.or.andcm A6</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>cmp.gt.and A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.gt.or A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.gt.or.andcm A6</td>
</tr>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>cmp.ge.and A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.ge.or A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.ge.or.andcm A7</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>cmp.lt.and A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.lt.or A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp.lt.or.andcm A7</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>cmp4.lt A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.ltu A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.eq A6</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>cmp4.lt.unc A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.ltu.unc A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.eq.unc A6</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>cmp4.ne.and A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.ne.or A6</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.ne.or.andcm A6</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>cmp4.gt.and A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.gt.or A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.gt.or.andcm A7</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>cmp4.ge.and A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.ge.or A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.ge.or.andcm A7</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>cmp4.lt.and A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.lt.or A7</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>cmp4.lt.or.andcm A7</td>
</tr>
</tbody>
</table>

Table 4-11. Integer Compare Immediate Opcode Extensions

<table>
<thead>
<tr>
<th>( x_2 ) Bits 35:34</th>
<th>( t_a ) Bit 33</th>
<th>( c ) Bit 12</th>
<th>Opcode Bits 40:37</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>C</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>D</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>E</td>
</tr>
<tr>
<td>2</td>
<td>0</td>
<td>0</td>
<td>cmp.lt – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp.ltu – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp.eq – immg A8</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>cmp.eq.unc – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp.eq.or – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp.eq.or.andcm – immg A8</td>
</tr>
<tr>
<td>1</td>
<td>0</td>
<td>0</td>
<td>cmp.ne.and – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp.ne.or – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp.ne.or.andcm – immg A8</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>cmp4.lt – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.ltu – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.eq – immg A8</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>cmp4.eq.unc – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.eq.or – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.eq.or.andcm – immg A8</td>
</tr>
<tr>
<td>3</td>
<td>0</td>
<td>0</td>
<td>cmp4.ne.and – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.ne.or – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.ne.or.andcm – immg A8</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>cmp4.eq.unc – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.eq.or – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.eq.or.andcm – immg A8</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>cmp4.ne.and – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.ne.or – immg A8</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>cmp4.ne.or.andcm – immg A8</td>
</tr>
</tbody>
</table>
### Integer Compare – Register-Register

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>cmp.lt</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp.lt.u</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp.eq</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td><code>cmp.lt.unc</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp.lt.u.unc</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp.eq.unc</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td><code>cmp.eq.and</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp.eq.or</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp.eq.or.andcm</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td><code>cmp.ne.and</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp.ne.or</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp.ne.or.andcm</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.lt</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.lt.u</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.eq</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.lt.unc</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.lt.u.unc</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.eq.unc</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.eq.and</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.eq.or</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.eq.or.andcm</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.ne.and</code></td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.ne.or</code></td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td><code>cmp4.ne.or.andcm</code></td>
<td></td>
<td>E</td>
<td></td>
</tr>
</tbody>
</table>

$p_1, p_2 = r_2, r_3$
### Integer Compare to Zero – Register

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>cmp.gt.and</td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td>cmp.gt.or</td>
<td></td>
<td>D</td>
<td></td>
</tr>
<tr>
<td>cmp.gt.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>cmp.le.and</td>
<td></td>
<td>C</td>
<td>0</td>
</tr>
<tr>
<td>cmp.le.or</td>
<td></td>
<td>D</td>
<td>1</td>
</tr>
<tr>
<td>cmp.le.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>cmp.ge.and</td>
<td></td>
<td>C</td>
<td>0</td>
</tr>
<tr>
<td>cmp.ge.or</td>
<td></td>
<td>D</td>
<td>0</td>
</tr>
<tr>
<td>cmp.ge.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>cmp.lt.and</td>
<td></td>
<td>C</td>
<td>1</td>
</tr>
<tr>
<td>cmp.lt.or</td>
<td></td>
<td>D</td>
<td>1</td>
</tr>
<tr>
<td>cmp.lt.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>cmp4.gt.and</td>
<td></td>
<td>C</td>
<td></td>
</tr>
<tr>
<td>cmp4.gt.or</td>
<td></td>
<td>D</td>
<td>0</td>
</tr>
<tr>
<td>cmp4.gt.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>cmp4.le.and</td>
<td></td>
<td>C</td>
<td>0</td>
</tr>
<tr>
<td>cmp4.le.or</td>
<td></td>
<td>D</td>
<td>1</td>
</tr>
<tr>
<td>cmp4.le.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>cmp4.ge.and</td>
<td></td>
<td>C</td>
<td>0</td>
</tr>
<tr>
<td>cmp4.ge.or</td>
<td></td>
<td>D</td>
<td>1</td>
</tr>
<tr>
<td>cmp4.ge.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>cmp4.lt.and</td>
<td></td>
<td>C</td>
<td>1</td>
</tr>
<tr>
<td>cmp4.lt.or</td>
<td></td>
<td>D</td>
<td>1</td>
</tr>
<tr>
<td>cmp4.lt.or.andcm</td>
<td></td>
<td>E</td>
<td></td>
</tr>
</tbody>
</table>

$p_1, p_2 = r_0, r_3$
4.2.2.3 Integer Compare – Immediate-Register

### Table 4-12. Multimedia ALU 2-bit+1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_{2a}$ Bits 35:34</th>
<th>$z_a$ Bit 36</th>
<th>$z_b$ Bit 33</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>8</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>Multimedia ALU Size 1 (Table 4-13)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1</td>
<td>1</td>
<td>Multimedia ALU Size 2 (Table 4-14)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1</td>
<td>0</td>
<td>Multimedia ALU Size 4 (Table 4-15)</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1</td>
<td>1</td>
<td></td>
</tr>
</tbody>
</table>

4.2.3 Multimedia

All multimedia ALU instructions are encoded within major opcode 8 using two 1-bit opcode extension fields in bits 36 ($z_a$) and 33 ($z_b$) and a 2-bit opcode extension field in bits 35:34 ($x_{2a}$) as shown in Table 4-12. The multimedia ALU instructions also have a 4-bit opcode extension field in bits 32:29 ($x_4$), and a 2-bit opcode extension field in bits 28:27 ($x_{2b}$) as shown in Table 4-13 on page 3:271.
### Table 4-13. Multimedia ALU Size 1 4-bit+2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_{2a}$ Bits 35:34</th>
<th>$z_a$ Bit 36</th>
<th>$z_b$ Bit 33</th>
<th>$x_4$ Bits 32:29</th>
<th>$x_{2b}$ Bits 28:27</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>8</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>padd1 A9</td>
<td>padd1.sss A9</td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
<td>padd2 A9</td>
<td>padd2.sss A9</td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td>psb1 A9</td>
<td>psb1.sss A9</td>
</tr>
<tr>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td>pavg1 A9</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td>pavgsub1 A9</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
<td>pcmp1.eq A9</td>
<td>pcmp1.gt A9</td>
</tr>
</tbody>
</table>

### Table 4-14. Multimedia ALU Size 2 4-bit+2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_{2a}$ Bits 35:34</th>
<th>$z_a$ Bit 36</th>
<th>$z_b$ Bit 33</th>
<th>$x_4$ Bits 32:29</th>
<th>$x_{2b}$ Bits 28:27</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>8</td>
<td>1</td>
<td>0</td>
<td>1</td>
<td>padd2 A9</td>
<td>padd2.sss A9</td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
<td>padd2 A9</td>
<td>padd2.sss A9</td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td>psb2 A9</td>
<td>psb2.sss A9</td>
</tr>
<tr>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td>pavg2 A9</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td>pavgsub2 A9</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
<td>pcmp2.eq A9</td>
<td>pcmp2.gt A9</td>
</tr>
</tbody>
</table>

Volume 3: Instruction Formats
Table 4-15. Multimedia ALU Size 4 4-bit+2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_{2a}$ Bits 35:34</th>
<th>$z_a$ Bit 36</th>
<th>$z_b$ Bit 33</th>
<th>$x_4$ Bits 32:29</th>
<th>$x_{2b}$ Bits 28:27</th>
</tr>
</thead>
<tbody>
<tr>
<td>8</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>padd4 A9</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>psub4 A9</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
<td></td>
<td>padd4.eq A9</td>
<td></td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
<td>pcmp4.gt A9</td>
<td></td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
### 4.2.3.1 Multimedia ALU

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>padd1</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd2</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd4</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd1.sss</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd2.sss</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd1.uuu</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd2.uuu</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd1.uus</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>padd2.uus</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub1</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub2</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub4</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub1.sss</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub2.sss</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub1.uuu</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub2.uuu</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub1.uus</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>psub2.uus</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pavg1</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pavg2</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pavg1.raz</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pavg2.raz</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pavgsub1</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pavgsub2</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pcmpt1.eq</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pcmpt2.eq</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pcmpt4.eq</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pcmpt1.gt</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pcmpt2.gt</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>pcmpt4.gt</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>

### 4.2.3.2 Multimedia Shift and Add

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>pshlladd2</td>
<td></td>
<td>8</td>
<td>1 0 1 4</td>
</tr>
<tr>
<td>pshrradd2</td>
<td></td>
<td>8</td>
<td>1 0 1 6</td>
</tr>
</tbody>
</table>
4.3 I-Unit Instruction Encodings

4.3.1 Multimedia and Variable Shifts

All multimedia multiply/shift/max/min/mix/mux/pack/unpack and variable shift instructions are encoded within major opcode 7 using two 1-bit opcode extension fields in bits 36 \( (z_a) \) and 33 \( (z_b) \) and a 1-bit reserved opcode extension in bit 32 \( (v_e) \) as shown in Table 4-16. They also have a 2-bit opcode extension field in bits 35:34 \( (x_{2a}) \) and a 2-bit field in bits 29:28 \( (x_{2b}) \) and most have a 2-bit field in bits 31:30 \( (x_{2c}) \) as shown in Table 4-17.

Table 4-16. Multimedia and Variable Shift 1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>( z_a ) Bit 36</th>
<th>( z_b ) Bit 33</th>
<th>( v_e ) Bit 32</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
<td>0</td>
</tr>
</tbody>
</table>

Multimedia Size 1 (Table 4-17)
Multimedia Size 2 (Table 4-18)
Multimedia Size 4 (Table 4-19)
Variable Shift (Table 4-20)

Table 4-17. Multimedia Opcode 7 Size 1 2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>( z_a ) Bit 36</th>
<th>( z_b ) Bit 33</th>
<th>( v_e ) Bit 32</th>
<th>( x_{2a} ) Bits 35:34</th>
<th>( x_{2b} ) Bits 29:28</th>
<th>( x_{2c} ) Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td>7</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>

unpack1.h i2      mix1.r i2
pmin1.u i2        pmax1.u i2
unpack1.l i2      mix1.l i2
psad1 l2         psad1 i2

| 2                 | 0         | 0         | 0         | 0               | 0               | 0               |
|                   | 1         | 1         | 0         | 1               | 1               | 1               |

| 3                 | 0         | 1         | 0         | 1               | 1               | 0               |
|                   | 2         | 2         | 0         | 2               | 2               | 0               |

mux1 l3           mux1 i3

### Table 4-18. Multimedia Opcode 7 Size 2 2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>Z_a Bit 36</th>
<th>Z_b Bit 33</th>
<th>V_a Bit 32</th>
<th>X_{2a} Bits 35:34</th>
<th>X_{2b} Bits 29:28</th>
<th>X_{2c} Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7 0 1 0</td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### Table 4-19. Multimedia Opcode 7 Size 4 2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>Z_a Bit 36</th>
<th>Z_b Bit 33</th>
<th>V_a Bit 32</th>
<th>X_{2a} Bits 35:34</th>
<th>X_{2b} Bits 29:28</th>
<th>X_{2c} Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7 1 0 0</td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
### 4.3.1.1 Multimedia Multiply and Shift

#### Table 4-20. Variable Shift Opcode 7 2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>z_a</th>
<th>z_b</th>
<th>v_e</th>
<th>x_{2a}</th>
<th>x_{2b}</th>
<th>x_{2c} Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>shru – var I5</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>shl – var I7</td>
</tr>
<tr>
<td>2</td>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>shru – var I5</td>
</tr>
<tr>
<td>3</td>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

#### Instruction Operands Opcode Extension

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>z_a</th>
<th>z_b</th>
<th>v_e</th>
<th>x_{2a}</th>
<th>x_{2b}</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>pmpyshr2</td>
<td>r_1, r_2, r_3, count_2</td>
<td>7</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>3</td>
<td>1</td>
</tr>
<tr>
<td>pmpyshr2.u</td>
<td>r_1 = r_2, r_3, count_2</td>
<td>7</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>3</td>
<td>1</td>
</tr>
</tbody>
</table>
### 4.3.1.2 Multimedia Multiply/Mix/Pack/Unpack

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>pmpy2.r</td>
<td></td>
<td>40</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pmpy2.l</td>
<td></td>
<td>37</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>mix1.r</td>
<td></td>
<td>36</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>mix2.r</td>
<td></td>
<td>35</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>mix4.r</td>
<td></td>
<td>34</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>mix1.l</td>
<td></td>
<td>33</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>mix2.l</td>
<td></td>
<td>32</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>mix4.l</td>
<td></td>
<td>31</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pack2.uss</td>
<td></td>
<td>30</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pack2.sss</td>
<td></td>
<td>29</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pack4.sss</td>
<td></td>
<td>28</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>unpack1.h</td>
<td></td>
<td>27</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>unpack2.h</td>
<td></td>
<td>26</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>unpack4.h</td>
<td></td>
<td>25</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>unpack1.l</td>
<td></td>
<td>24</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>unpack2.l</td>
<td></td>
<td>23</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>unpack4.l</td>
<td></td>
<td>22</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pmin1.u</td>
<td></td>
<td>21</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pmax1.u</td>
<td></td>
<td>20</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pmin2</td>
<td></td>
<td>19</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>pmax2</td>
<td></td>
<td>18</td>
<td>7 0 1 3 3</td>
</tr>
<tr>
<td>psad1</td>
<td></td>
<td>17</td>
<td>7 0 1 3 3</td>
</tr>
</tbody>
</table>

### 4.3.1.3 Multimedia Mux1

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mu1</td>
<td></td>
<td>0</td>
<td>7 0 0 3 2 2</td>
</tr>
</tbody>
</table>

### 4.3.1.4 Multimedia Mux2

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mu2</td>
<td></td>
<td>0</td>
<td>7 0 1 3 2 2</td>
</tr>
</tbody>
</table>
### 4.3.1.5 Shift Right – Variable

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>pshr2</td>
<td></td>
<td>7</td>
<td>0 1 0 0 2 0</td>
</tr>
<tr>
<td>pshr4</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>shr</td>
<td>( r_f = r_3, r_2 )</td>
<td>7 1 0 0 3 0</td>
<td></td>
</tr>
<tr>
<td>pshr2.u</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>pshr4.u</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>shr.u</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.1.6 Multimedia Shift Right – Fixed

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>pshr2</td>
<td></td>
<td>7</td>
<td>0 1 0 0 3 0</td>
</tr>
<tr>
<td>pshr4</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>shr</td>
<td>( r_f = r_3, ) count5</td>
<td>7 1 0 0 3 0</td>
<td></td>
</tr>
<tr>
<td>pshr2.u</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>pshr4.u</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.1.7 Shift Left – Variable

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>pshl2</td>
<td></td>
<td>7</td>
<td>0 1 0 0 0 1</td>
</tr>
<tr>
<td>pshl4</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>shl</td>
<td>( r_f = r_2, r_3 )</td>
<td>7 1 0 0 0 1</td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.1.8 Multimedia Shift Left – Fixed

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>pshl2</td>
<td></td>
<td>7</td>
<td>0 1 0 0 3 1</td>
</tr>
<tr>
<td>pshl4</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>r_f = r_2, count5</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

---

Volume 3: Instruction Formats
4.3.1.9 Population Count

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>popcnt</td>
<td>$r_1 = r_3$</td>
<td>7</td>
<td>0 1 0 1 1 2</td>
</tr>
</tbody>
</table>

4.3.2 Integer Shifts

The integer shift, test bit, and test NaT instructions are encoded within major opcode 5 using a 2-bit opcode extension field in bits 35:34 ($x_2$) and a 1-bit opcode extension field in bit 33 ($x$). The extract and test bit instructions also have a 1-bit opcode extension field in bit 13 ($y$). Table 4-21 shows the test bit, extract, and shift right pair assignments.

Table 4-21. Integer Shift/Test Bit/Test NaT 2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_2$ Bits 35:34</th>
<th>$x$ Bit 33</th>
<th>$y$ Bit 13</th>
</tr>
</thead>
<tbody>
<tr>
<td>4</td>
<td>0 0</td>
<td>0</td>
<td>0 Test Bit (Table 4-23)</td>
</tr>
<tr>
<td>5</td>
<td>0 1</td>
<td>0</td>
<td>0 Test NaT (Table 4-23)</td>
</tr>
<tr>
<td>6</td>
<td>0 2</td>
<td>0</td>
<td>extr $u$ I11</td>
</tr>
<tr>
<td>7</td>
<td>0 3</td>
<td>0</td>
<td>extr I11</td>
</tr>
<tr>
<td>8</td>
<td>0 0</td>
<td>0</td>
<td>shrp I10</td>
</tr>
</tbody>
</table>

Most deposit instructions also have a 1-bit opcode extension field in bit 26 ($y$). Table 4-22 shows these assignments.

Table 4-22. Deposit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_2$ Bits 35:34</th>
<th>$x$ Bit 33</th>
<th>$y$ Bit 26</th>
</tr>
</thead>
<tbody>
<tr>
<td>4</td>
<td>0 0</td>
<td>0</td>
<td>0 Test Bit/Test NaT (Table 4-23)</td>
</tr>
<tr>
<td>5</td>
<td>0 1</td>
<td>0</td>
<td>0 dep $z$ I12</td>
</tr>
<tr>
<td>6</td>
<td>0 2</td>
<td>0</td>
<td>dep $z$ – imm $e$ I13</td>
</tr>
<tr>
<td>7</td>
<td>0 3</td>
<td>0</td>
<td>dep $z$ – imm $e$ I14</td>
</tr>
</tbody>
</table>

4.3.2.1 Shift Right Pair

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>shrp</td>
<td>$r_1 = r_2, r_3, count_6$</td>
<td>5</td>
<td>3 0</td>
</tr>
</tbody>
</table>

Volume 3: Instruction Formats
### 4.3.2.2 Extract

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>extr.u</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>extr</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.2.3 Zero and Deposit

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>dep.z</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.2.4 Zero and Deposit Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>dep.z</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.2.5 Deposit Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>dep</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.2.6 Deposit

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
</tr>
</thead>
<tbody>
<tr>
<td>dep</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
4.3.3 Test Bit

All test bit instructions are encoded within major opcode 5 using a 2-bit opcode extension field in bits 35:34 \((x_2)\) plus four 1-bit opcode extension fields in bits 33 \((t_a)\), 36 \((t_b)\), 12 \((c)\), and 19 \((y)\). Table 4-23 summarizes these assignments.

### Table 4-23. Test Bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>(x_2) Bits 35:34</th>
<th>(t_a) Bit 33</th>
<th>(t_b) Bit 36</th>
<th>c Bit 12</th>
<th>y Bit 13</th>
</tr>
</thead>
<tbody>
<tr>
<td>5</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>tbit.z I16</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1</td>
<td>0</td>
<td>tbit.z.unc I16</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1</td>
<td>0</td>
<td>tbit.z.and I16</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1</td>
<td>0</td>
<td>tbit.nz.and I16</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>tbit.z.or I16</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1</td>
<td>0</td>
<td>tbit.nz.or I16</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td>1</td>
<td>0</td>
<td>tbit.nz.or.andcm I16</td>
</tr>
</tbody>
</table>

4.3.3.1 Test Bit

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>tbit.z</td>
<td></td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>tbit.z.unc</td>
<td></td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>tbit.z.and</td>
<td></td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>tbit.nz.and</td>
<td></td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>tbit.z.or</td>
<td></td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>tbit.nz.or</td>
<td></td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>tbit.z.or.andcm</td>
<td></td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>tbit.nz.or.andcm</td>
<td></td>
<td>5</td>
<td></td>
</tr>
</tbody>
</table>

Instruction Operands Opcode Extension

\(p_1, p_2 = r_3, pos_6\)
4.3.3.2 Test NaT

The miscellaneous I-unit instructions are encoded in major opcode 0 using a 3-bit opcode extension field \((x_3)\) in bits 35:33. Some also have a 6-bit opcode extension field \((x_6)\) in bits 32:27. Table 4-24 shows the 3-bit assignments and Table 4-25 summarizes the 6-bit assignments.

Table 4-24. Misc I-Unit 3-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>tnat.z</td>
<td></td>
<td>5</td>
<td>(x_2) (t_a) (t_b) (y) (c)</td>
</tr>
<tr>
<td>tnat.z.unc</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.z.and</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.nz.and</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.z.or</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.nz.or</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.z.or.andcm</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.nz.or.andcm</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

\(p_1, p_2 = r_3\)

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>tnat.z</td>
<td></td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>tnat.z.unc</td>
<td></td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>tnat.z.and</td>
<td></td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>tnat.nz.and</td>
<td></td>
<td>1</td>
<td>0</td>
</tr>
<tr>
<td>tnat.z.or</td>
<td></td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>tnat.nz.or</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.z.or.andcm</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>tnat.nz.or.andcm</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

4.3.4 Miscellaneous I-Unit Instructions

The miscellaneous I-unit instructions are encoded in major opcode 0 using a 3-bit opcode extension field \((x_3)\) in bits 35:33. Some also have a 6-bit opcode extension field \((x_6)\) in bits 32:27. Table 4-24 shows the 3-bit assignments and Table 4-25 summarizes the 6-bit assignments.

Table 4-24. Misc I-Unit 3-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>(x_3) Bits 35:33</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>6-bit Ext (Table 4-25)</td>
</tr>
<tr>
<td>1</td>
<td>chk.s.i – int I20</td>
</tr>
<tr>
<td>2</td>
<td>mov pr rot – imm44 I24</td>
</tr>
<tr>
<td>3</td>
<td>mov pr I23</td>
</tr>
<tr>
<td>4</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>mov to b I21</td>
</tr>
</tbody>
</table>
### Table 4-25. Misc I-Unit 6-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_3$ Bits 35:33</th>
<th>$x_6$ Bits 32:31</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>break.i I19</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>nop.i I19</td>
</tr>
<tr>
<td></td>
<td>2</td>
<td>zxt1 I29</td>
</tr>
<tr>
<td></td>
<td>3</td>
<td>mov from ip I25</td>
</tr>
<tr>
<td>4</td>
<td>4</td>
<td>sxt1 I29</td>
</tr>
<tr>
<td></td>
<td>5</td>
<td>mov from b I22</td>
</tr>
<tr>
<td>6</td>
<td>6</td>
<td>mov from ar I28</td>
</tr>
<tr>
<td></td>
<td>7</td>
<td>mov from pr I25</td>
</tr>
<tr>
<td>8</td>
<td>8</td>
<td>mov to ar – imm3 I27</td>
</tr>
<tr>
<td></td>
<td>9</td>
<td>mov to ar I26</td>
</tr>
<tr>
<td>A</td>
<td>8</td>
<td>mov to ar I26</td>
</tr>
<tr>
<td>B</td>
<td>8</td>
<td>mov to ar I26</td>
</tr>
<tr>
<td>C</td>
<td>8</td>
<td>mov to ar I26</td>
</tr>
<tr>
<td>D</td>
<td>8</td>
<td>mov to ar I26</td>
</tr>
<tr>
<td>E</td>
<td>8</td>
<td>mov to ar I26</td>
</tr>
<tr>
<td>F</td>
<td>8</td>
<td>mov to ar I26</td>
</tr>
</tbody>
</table>

#### 4.3.4.1 Break/Nop (I-Unit)

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>break.i</td>
<td>imm21</td>
<td>0</td>
<td>00</td>
</tr>
<tr>
<td>nop.i</td>
<td></td>
<td>0</td>
<td>01</td>
</tr>
</tbody>
</table>

#### 4.3.4.2 Integer Speculation Check (I-Unit)

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>chk.s.i</td>
<td>r2, target25</td>
<td>0</td>
<td>1</td>
</tr>
</tbody>
</table>
4.3.5 GR/BR Moves

The GR/BR move instructions are encoded in major opcode 0. See “Miscellaneous I-Unit Instructions” on page 3:282 for a summary of the opcode extensions. The mov to BR instruction uses a 2-bit “whether” prediction hint field in bits 21:20 (wh) as shown in Table 4-26.

Table 4-26. Move to BR Whether Hint Completer

<table>
<thead>
<tr>
<th>wh Bits 21:20</th>
<th>mwh</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>.sptk</td>
</tr>
<tr>
<td>1</td>
<td>none</td>
</tr>
<tr>
<td>2</td>
<td>.dptk</td>
</tr>
<tr>
<td>3</td>
<td></td>
</tr>
</tbody>
</table>

The mov to BR instruction also uses a 1-bit opcode extension field (x) in bit 22 to distinguish the return form from the normal form, and a 1-bit hint extension in bit 23 (ih) (see Table 4-54 on page 3:320).

4.3.5.1 Move to BR

| Instruction Operands Opcode Extension x3 x ih wh |
|----------------|------|-----|-----|
| mov.mwh.ih     | b_j = r_2, tag_{1,3} & b1 = 0 |
| mov.ret.mwh.ih | b_j = r_2, tag_{1,3} & b1 = 0 |

4.3.5.2 Move from BR

| Instruction Operands Opcode Extension x3 x6 |
|----------------|------|
| mov            | r_j = b_2 |


4.3.6 GR/Predicate/IP Moves

The GR/Predicate/IP move instructions are encoded in major opcode 0. See “Miscellaneous I-Unit Instructions” on page 3:282 for a summary of the opcode extensions.

4.3.6.1 Move to Predicates – Register

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov</td>
<td>pr = r₂, mask₁₇</td>
<td>0</td>
<td>3</td>
</tr>
</tbody>
</table>

4.3.6.2 Move to Predicates – Immediate₄₄

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov</td>
<td>pr.rot = imm₄₄</td>
<td>0</td>
<td>2</td>
</tr>
</tbody>
</table>

4.3.6.3 Move from Predicates/IP

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov</td>
<td>r₁ = ip</td>
<td>0</td>
<td>30</td>
</tr>
<tr>
<td></td>
<td>r₁ = pr</td>
<td>0</td>
<td>33</td>
</tr>
</tbody>
</table>

4.3.7 GR/AR Moves (I-Unit)

The I-Unit GR/AR move instructions are encoded in major opcode 0. (Some ARs are accessed using system/memory management instructions on the M-unit. See “GR/AR Moves (M-Unit)” on page 3:307.) See “Miscellaneous I-Unit Instructions” on page 3:282 for a summary of the I-Unit GR/AR opcode extensions.
### 4.3.7.1 Move to AR – Register (I-Unit)

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov.i</td>
<td>ar3 = r2</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.7.2 Move to AR – Immediate8 (I-Unit)

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov.i</td>
<td>ar3 = imm8</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.7.3 Move from AR (I-Unit)

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov.i</td>
<td>r1 = ar3</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.3.8 Sign/Zero Extend/Compute Zero Index

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>zxt1</td>
<td>r1 = r3</td>
<td></td>
<td></td>
</tr>
<tr>
<td>zxt2</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>zxt4</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>sxt1</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>sxt2</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>sxt4</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>czx1.l</td>
<td>r1</td>
<td>0</td>
<td></td>
</tr>
<tr>
<td>czx2.l</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>czx1.r</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>czx2.r</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>czx1.r</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>czx2.r</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Extension:
- 10
- 11
- 12
- 14
- 15
- 16
- 18
- 19
- 1C
- 1D
4.4 M-Unit Instruction Encodings

4.4.1 Loads and Stores

All load and store instructions are encoded within major opcodes 4, 5, 6, and 7 using a 6-bit opcode extension field in bits 35:30 (x₆). Instructions in major opcode 4 (integer load/store, semaphores, and get FR) use two 1-bit opcode extension fields in bit 36 (m) and bit 27 (x) as shown in Table 4-27. Instructions in major opcode 6 (floating-point load/store, load pair, and set FR) use two 1-bit opcode extension fields in bit 36 (m) and bit 27 (x) as shown in Table 4-28.

Table 4-27. Integer Load/Store/Semaphore/Get FR 1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>0</td>
<td>Load/Store (Table 4-29)</td>
</tr>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>Semaphore/get FR (Table 4-32)</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>Load +Reg (Table 4-30)</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
<td></td>
</tr>
</tbody>
</table>

Table 4-28. Floating-point Load/Store/Load Pair/Set FR 1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>0</td>
<td>FP Load/Store (Table 4-33)</td>
</tr>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>FP Load Pair/set FR (Table 4-36)</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>0</td>
<td>FP Load +Reg (Table 4-34)</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1</td>
<td>FP Load Pair +Imm (Table 4-37)</td>
</tr>
</tbody>
</table>

The integer load/store opcode extensions are summarized in Table 4-29 on page 3:288, Table 4-30 on page 3:288, and Table 4-31 on page 3:289, and the semaphore and get FR opcode extensions in Table 4-32 on page 3:289. The floating-point load/store opcode extensions are summarized in Table 4-33 on page 3:290, Table 4-34 on page 3:290, and Table 4-35 on page 3:291, the floating-point load pair and set FR opcode extensions in Table 4-36 on page 3:291 and Table 4-37 on page 3:292.
### Table 4-29. Integer Load/Store Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>Bits 35:32</th>
<th>( x_0 )</th>
<th>Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>1</td>
<td>2</td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td>ld1 M1</td>
<td>ld2 M1</td>
<td>ld4 M1</td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td>ld1.s M1</td>
<td>ld2.s M1</td>
<td>ld4.s M1</td>
</tr>
<tr>
<td>2</td>
<td></td>
<td></td>
<td>ld1.a M1</td>
<td>ld2.a M1</td>
<td>ld4.a M1</td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
<td>ld1.sa M1</td>
<td>ld2.sa M1</td>
<td>ld4.sa M1</td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td>ld1.bias M1</td>
<td>ld2.bias M1</td>
<td>ld4.bias M1</td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td>ld1.acq M1</td>
<td>ld2.acq M1</td>
<td>ld4.acq M1</td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>ld8.fill M1</td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
<td>ld1.c.clr M1</td>
<td>ld2.c.clr M1</td>
<td>ld4.c.clr M1</td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
<td>ld1.c.clr M1</td>
<td>ld2.c.clr M1</td>
<td>ld4.c.clr M1</td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td>ld1.c.nc M1</td>
<td>ld2.c.nc M1</td>
<td>ld4.c.nc M1</td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
<td>ld1.c.clr.acq M1</td>
<td>ld2.c.clr.acq M1</td>
<td>ld4.c.clr.acq M1</td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td>st1 M4</td>
<td>st2 M4</td>
<td>st4 M4</td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td>st1.rel M4</td>
<td>st2.rel M4</td>
<td>st4.rel M4</td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### Table 4-30. Integer Load +Reg Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>Bits 35:32</th>
<th>( x_0 )</th>
<th>Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>1</td>
<td>2</td>
</tr>
<tr>
<td>0</td>
<td></td>
<td></td>
<td>ld1 M2</td>
<td>ld2 M2</td>
<td>ld4 M2</td>
</tr>
<tr>
<td>1</td>
<td></td>
<td></td>
<td>ld1.s M2</td>
<td>ld2.s M2</td>
<td>ld4.s M2</td>
</tr>
<tr>
<td>2</td>
<td></td>
<td></td>
<td>ld1.a M2</td>
<td>ld2.a M2</td>
<td>ld4.a M2</td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
<td>ld1.sa M2</td>
<td>ld2.sa M2</td>
<td>ld4.sa M2</td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td>ld1.bias M2</td>
<td>ld2.bias M2</td>
<td>ld4.bias M2</td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td>ld1.acq M2</td>
<td>ld2.acq M2</td>
<td>ld4.acq M2</td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>ld8.fill M2</td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
<td>ld1.c.clr M2</td>
<td>ld2.c.clr M2</td>
<td>ld4.c.clr M2</td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
<td>ld1.c.clr M2</td>
<td>ld2.c.clr M2</td>
<td>ld4.c.clr M2</td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td>ld1.c.nc M2</td>
<td>ld2.c.nc M2</td>
<td>ld4.c.nc M2</td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
<td>ld1.c.clr.acq M2</td>
<td>ld2.c.clr.acq M2</td>
<td>ld4.c.clr.acq M2</td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
### Table 4-31. Integer Load/Store +Imm Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>x6</th>
<th>Bits 35:32</th>
<th>Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>ld1 M3</td>
<td>ld2 M3</td>
<td>ld4 M3</td>
</tr>
<tr>
<td>1</td>
<td>ld1.s M3</td>
<td>ld2.s M3</td>
<td>ld4.s M3</td>
</tr>
<tr>
<td>2</td>
<td>ld1.a M3</td>
<td>ld2.a M3</td>
<td>ld4.a M3</td>
</tr>
<tr>
<td>3</td>
<td>ld1.sa M3</td>
<td>ld2.sa M3</td>
<td>ld4.sa M3</td>
</tr>
<tr>
<td>4</td>
<td>ld1.bias M3</td>
<td>ld2.bias M3</td>
<td>ld4.bias M3</td>
</tr>
<tr>
<td>5</td>
<td>ld1.acq M3</td>
<td>ld2.acq M3</td>
<td>ld4.acq M3</td>
</tr>
<tr>
<td>6</td>
<td>ld8.fill M3</td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>ld1.c.clr M3</td>
<td>ld2.c.clr M3</td>
<td>ld4.c.clr M3</td>
</tr>
<tr>
<td>8</td>
<td>ld1.c.nc M3</td>
<td>ld2.c.nc M3</td>
<td>ld4.c.nc M3</td>
</tr>
<tr>
<td>A</td>
<td>ld1.c.clr.acq M3</td>
<td>ld2.c.clr.acq M3</td>
<td>ld4.c.clr.acq M3</td>
</tr>
<tr>
<td>B</td>
<td>st1 M5</td>
<td>st2 M5</td>
<td>st4 M5</td>
</tr>
<tr>
<td>C</td>
<td>st1.rel M5</td>
<td>st2.rel M5</td>
<td>st4.rel M5</td>
</tr>
<tr>
<td>D</td>
<td>st8.spill M5</td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td>getf.sig M19</td>
<td>getf.exp M19</td>
<td>getf.s M19</td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### Table 4-32. Semaphore/Get FR Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>x6</th>
<th>Bits 35:32</th>
<th>Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>cmpxchg1.acq M16</td>
<td>cmpxchg2.acq M16</td>
<td>cmpxchg4.acq M16</td>
<td>cmpxchg8.acq M16</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>cmpxchg1.rel M16</td>
<td>cmpxchg2.rel M16</td>
<td>cmpxchg4.rel M16</td>
<td>cmpxchg8.rel M16</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>xchg1 M16</td>
<td>xchg2 M16</td>
<td>xchg4 M16</td>
<td>xchg8 M16</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>getf.sig M19</td>
<td>getf.exp M19</td>
<td>getf.s M19</td>
<td>getf.d M19</td>
<td></td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
### Table 4-33. Floating-point Load/Store/Lfetch Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>Bits 35:32</th>
<th>Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>ldfe M6</td>
<td>ldfl M6</td>
<td>ldfr M6</td>
<td>ldfd M6</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>ldfe.s M6</td>
<td>ldfl.s M6</td>
<td>ldfr.s M6</td>
<td>ldfd.s M6</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>ldfe.a M6</td>
<td>ldfl.a M6</td>
<td>ldfr.a M6</td>
<td>ldfd.a M6</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>ldfe.sa M6</td>
<td>ldfl.sa M6</td>
<td>ldfr.sa M6</td>
<td>ldfd.sa M6</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td>ldfe.c.clr M6</td>
<td>ldfl.c.clr M6</td>
<td>ldfr.c.clr M6</td>
<td>ldfd.c.clr M6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>ldfe.c.clr M6</td>
<td>ldfl.c.clr M6</td>
<td>ldfr.c.clr M6</td>
<td>ldfd.c.clr M6</td>
<td></td>
</tr>
<tr>
<td>8</td>
<td>ldfe.c.nc M6</td>
<td>ldfl.c.nc M6</td>
<td>ldfr.c.nc M6</td>
<td>ldfd.c.nc M6</td>
<td></td>
</tr>
<tr>
<td>9</td>
<td>ldfe.c.nc M6</td>
<td>ldfl.c.nc M6</td>
<td>ldfr.c.nc M6</td>
<td>ldfd.c.nc M6</td>
<td></td>
</tr>
<tr>
<td>A</td>
<td>lfetch M13</td>
<td>lfetch.excl M13</td>
<td>lfetch.fault M13</td>
<td>lfetch.fault.excl M13</td>
<td></td>
</tr>
<tr>
<td>B</td>
<td>stfe M9</td>
<td>stfl M9</td>
<td>stfs M9</td>
<td>stfd M9</td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### Table 4-34. Floating-point Load/Lfetch +Reg Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>Bits 35:32</th>
<th>Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>ldfe M7</td>
<td>ldfl M7</td>
<td>ldfr M7</td>
<td>ldfd M7</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>ldfe.s M7</td>
<td>ldfl.s M7</td>
<td>ldfr.s M7</td>
<td>ldfd.s M7</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>ldfe.a M7</td>
<td>ldfl.a M7</td>
<td>ldfr.a M7</td>
<td>ldfd.a M7</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>ldfe.sa M7</td>
<td>ldfl.sa M7</td>
<td>ldfr.sa M7</td>
<td>ldfd.sa M7</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td>ldfe.c.clr M7</td>
<td>ldfl.c.clr M7</td>
<td>ldfr.c.clr M7</td>
<td>ldfd.c.clr M7</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>ldfe.c.clr M7</td>
<td>ldfl.c.clr M7</td>
<td>ldfr.c.clr M7</td>
<td>ldfd.c.clr M7</td>
<td></td>
</tr>
<tr>
<td>8</td>
<td>ldfe.c.nc M7</td>
<td>ldfl.c.nc M7</td>
<td>ldfr.c.nc M7</td>
<td>ldfd.c.nc M7</td>
<td></td>
</tr>
<tr>
<td>9</td>
<td>ldfe.c.nc M7</td>
<td>ldfl.c.nc M7</td>
<td>ldfr.c.nc M7</td>
<td>ldfd.c.nc M7</td>
<td></td>
</tr>
<tr>
<td>A</td>
<td>lfetch M14</td>
<td>lfetch.excl M14</td>
<td>lfetch.fault M14</td>
<td>lfetch.fault.excl M14</td>
<td></td>
</tr>
<tr>
<td>B</td>
<td>stfe M9</td>
<td>stfl M9</td>
<td>stfs M9</td>
<td>stfd M9</td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
Table 4-35. Floating-point Load/Store/Lfetch +Imm Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>Bits 35:32</th>
<th>(x_6)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>ldfe M8</td>
<td>ldfl8 M8</td>
</tr>
<tr>
<td>1</td>
<td>ldfe.s M8</td>
<td>ldfl8.s M8</td>
</tr>
<tr>
<td>2</td>
<td>ldfe.a M8</td>
<td>ldfl8.a M8</td>
</tr>
<tr>
<td>3</td>
<td>ldfe.sa M8</td>
<td>ldfl8.sa M8</td>
</tr>
</tbody>
</table>

Table 4-36. Floating-point Load Pair/Set FR Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>Bits 35:32</th>
<th>(x_6)</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>2</td>
<td>3</td>
</tr>
<tr>
<td>0</td>
<td>ldfp8 M11</td>
<td>ldfps M11</td>
<td>ldfpd M11</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>ldfp8.s M11</td>
<td>ldfps.s M11</td>
<td>ldfpd.s M11</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>ldfp8.a M11</td>
<td>ldfps.a M11</td>
<td>ldfpd.a M11</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>ldfp8.sa M11</td>
<td>ldfps.sa M11</td>
<td>ldfpd.sa M11</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>setf.sig M18</td>
<td>setf.exp M18</td>
<td>setf.s M18</td>
<td>setf.d M18</td>
</tr>
<tr>
<td>8</td>
<td>ldfp8.c.clr M11</td>
<td>ldfps.c.clr M11</td>
<td>ldfpd.c.clr M11</td>
<td></td>
</tr>
<tr>
<td>9</td>
<td>ldfp8.c.nc M11</td>
<td>ldfps.c.nc M11</td>
<td>ldfpd.c.nc M11</td>
<td></td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
The load and store instructions all have a 2-bit opcode extension field in bits 29:28 (hint) which encodes locality hint information. Table 4-38 and Table 4-39 summarize these assignments.

### Table 4-37. Floating-point Load Pair +Imm Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>m Bit 36</th>
<th>x Bit 27</th>
<th>$x_6$ Bits 35:32</th>
<th>$x_6$ Bits 31:30</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>0</td>
<td>1</td>
<td>1</td>
<td>0</td>
</tr>
<tr>
<td>ldfp8 M12</td>
<td>ldfps M12</td>
<td>ldfpd M12</td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfp8.s M12</td>
<td>ldfps.s M12</td>
<td>ldfpd.s M12</td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfp8.a M12</td>
<td>ldfps.a M12</td>
<td>ldfpd.a M12</td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfp8.sa M12</td>
<td>ldfps.sa M12</td>
<td>ldfpd.sa M12</td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfp8.c.clr M12</td>
<td>ldfps.c.clr M12</td>
<td>ldfpd.c.clr M12</td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfp8.c.nc M12</td>
<td>ldfps.c.nc M12</td>
<td>ldfpd.c.nc M12</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### Table 4-38. Load Hint Completer

<table>
<thead>
<tr>
<th>hint Bits 29:28</th>
<th>ldhint</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>none</td>
</tr>
<tr>
<td>1</td>
<td>.nt1</td>
</tr>
<tr>
<td>2</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>.nta</td>
</tr>
</tbody>
</table>

### Table 4-39. Store Hint Completer

<table>
<thead>
<tr>
<th>hint Bits 29:28</th>
<th>sthint</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>none</td>
</tr>
<tr>
<td>1</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>.nta</td>
</tr>
</tbody>
</table>
### 4.4.1.1 Integer Load

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>ld1.ldhint</td>
<td></td>
<td>m x 6</td>
<td>hint</td>
</tr>
<tr>
<td>ld2.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.fill.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.fill.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.fill.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.fill.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Operands: \( r_f = [r_3] \)

See Table 4-38 on page 3:292.
### 4.4.1.2 Integer Load – Increment by Register

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>ld1.ldhint</td>
<td></td>
<td>m x 6</td>
<td>hint</td>
</tr>
<tr>
<td>ld2.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.bias.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.fill.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld1.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld2.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld4.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ld8.c.clr.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

- \( r_1 = [r_3], r_2 \)
- See Table 4-38 on page 3:292.
### Integer Load – Increment by Immediate

#### Instruction Formats

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>ld1.idhint</code></td>
<td></td>
<td>00</td>
<td></td>
</tr>
<tr>
<td><code>ld2.idhint</code></td>
<td></td>
<td>01</td>
<td></td>
</tr>
<tr>
<td><code>ld4.idhint</code></td>
<td></td>
<td>02</td>
<td></td>
</tr>
<tr>
<td><code>ld8.idhint</code></td>
<td></td>
<td>03</td>
<td></td>
</tr>
<tr>
<td><code>ld1.s.idhint</code></td>
<td></td>
<td>04</td>
<td></td>
</tr>
<tr>
<td><code>ld2.s.idhint</code></td>
<td></td>
<td>05</td>
<td></td>
</tr>
<tr>
<td><code>ld4.s.idhint</code></td>
<td></td>
<td>06</td>
<td></td>
</tr>
<tr>
<td><code>ld8.s.idhint</code></td>
<td></td>
<td>07</td>
<td></td>
</tr>
<tr>
<td><code>ld1.a.idhint</code></td>
<td></td>
<td>08</td>
<td></td>
</tr>
<tr>
<td><code>ld2.a.idhint</code></td>
<td></td>
<td>09</td>
<td></td>
</tr>
<tr>
<td><code>ld4.a.idhint</code></td>
<td></td>
<td>0A</td>
<td></td>
</tr>
<tr>
<td><code>ld8.a.idhint</code></td>
<td></td>
<td>0B</td>
<td></td>
</tr>
<tr>
<td><code>ld1.sahint</code></td>
<td></td>
<td>0C</td>
<td></td>
</tr>
<tr>
<td><code>ld2.sahint</code></td>
<td></td>
<td>0D</td>
<td></td>
</tr>
<tr>
<td><code>ld4.sahint</code></td>
<td></td>
<td>0E</td>
<td></td>
</tr>
<tr>
<td><code>ld8.sahint</code></td>
<td></td>
<td>0F</td>
<td></td>
</tr>
<tr>
<td><code>ld1.bias.idhint</code></td>
<td></td>
<td>10</td>
<td></td>
</tr>
<tr>
<td><code>ld2.bias.idhint</code></td>
<td></td>
<td>11</td>
<td></td>
</tr>
<tr>
<td><code>ld4.bias.idhint</code></td>
<td></td>
<td>12</td>
<td></td>
</tr>
<tr>
<td><code>ld8.bias.idhint</code></td>
<td></td>
<td>13</td>
<td></td>
</tr>
<tr>
<td><code>ld1.acq.idhint</code></td>
<td></td>
<td>14</td>
<td></td>
</tr>
<tr>
<td><code>ld2.acq.idhint</code></td>
<td></td>
<td>15</td>
<td></td>
</tr>
<tr>
<td><code>ld4.acq.idhint</code></td>
<td></td>
<td>16</td>
<td></td>
</tr>
<tr>
<td><code>ld8.acq.idhint</code></td>
<td></td>
<td>17</td>
<td></td>
</tr>
<tr>
<td><code>ld1.fill.idhint</code></td>
<td></td>
<td>18</td>
<td></td>
</tr>
<tr>
<td><code>ld1.c.clr.idhint</code></td>
<td></td>
<td>20</td>
<td></td>
</tr>
<tr>
<td><code>ld2.c.clr.idhint</code></td>
<td></td>
<td>21</td>
<td></td>
</tr>
<tr>
<td><code>ld4.c.clr.idhint</code></td>
<td></td>
<td>22</td>
<td></td>
</tr>
<tr>
<td><code>ld8.c.clr.idhint</code></td>
<td></td>
<td>23</td>
<td></td>
</tr>
<tr>
<td><code>ld1.c.clr.acq.idhint</code></td>
<td></td>
<td>24</td>
<td></td>
</tr>
<tr>
<td><code>ld2.c.clr.acq.idhint</code></td>
<td></td>
<td>25</td>
<td></td>
</tr>
<tr>
<td><code>ld4.c.clr.acq.idhint</code></td>
<td></td>
<td>26</td>
<td></td>
</tr>
<tr>
<td><code>ld8.c.clr.acq.idhint</code></td>
<td></td>
<td>27</td>
<td></td>
</tr>
<tr>
<td><code>ld1.c.clr.acq.idhint</code></td>
<td></td>
<td>28</td>
<td></td>
</tr>
<tr>
<td><code>ld2.c.clr.acq.idhint</code></td>
<td></td>
<td>29</td>
<td></td>
</tr>
<tr>
<td><code>ld4.c.clr.acq.idhint</code></td>
<td></td>
<td>2A</td>
<td></td>
</tr>
<tr>
<td><code>ld8.c.clr.acq.idhint</code></td>
<td></td>
<td>2B</td>
<td></td>
</tr>
</tbody>
</table>

The extension is `x_6 hint`, where `r_f = [r_3], imm_9`.

See Table 4-38 on page 3:292.
### 4.4.1.4 Integer Store

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>st1.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st2.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st4.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st8.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st1.rel.sthint</td>
<td>$[r_3] = r_2$</td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st2.rel.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st4.rel.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st8.rel.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
<tr>
<td>st8.spill.sthint</td>
<td></td>
<td>4</td>
<td>0 0</td>
</tr>
</tbody>
</table>

See Table 4-39 on page 3:292.

### 4.4.1.5 Integer Store – Increment by Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>st1.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st2.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st4.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st8.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st1.rel.sthint</td>
<td>$[r_3] = r_2$, imm_9</td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st2.rel.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st4.rel.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st8.rel.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
<tr>
<td>st8.spill.sthint</td>
<td></td>
<td>5</td>
<td>30 31 32 33</td>
</tr>
</tbody>
</table>

See Table 4-39 on page 3:292.
### 4.4.1.6 Floating-point Load

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>ldfs.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.sa.ldhint</code></td>
<td></td>
<td>6</td>
<td>00</td>
</tr>
<tr>
<td><code>ldfd.sa.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.sa.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.sa.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf.fill.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>ldfs.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.s.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.a.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.sa.ldhint</code></td>
<td></td>
<td>6</td>
<td>00</td>
</tr>
<tr>
<td><code>ldfd.sa.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.sa.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.sa.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf.fill.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.c.clr.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfs.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfd.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldf8.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td><code>ldfe.c.nc.ldhint</code></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

$ f_j = [r_j] $
### 4.4.1.7 Floating-point Load – Increment by Register

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>ldfs.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldd.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf8.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf.e.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfs.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldd.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf8.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf.e.s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfs.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldd.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf8.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf.e.a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfs.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldd.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf8.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf.e.sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfs.fillldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldd.fillldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf8.fillldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf.e.fillldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfs.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldd.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf8.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf.e.c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldfs.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldd.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf8.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldf.e.c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

- $f_j = [r_3], r_2$
- See Table 4-38 on page 3:292.
### 4.4.1.8 Floating-point Load – Increment by Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>ldfs.ldhint</td>
<td></td>
<td></td>
<td>02</td>
</tr>
<tr>
<td>ldfd.ldhint</td>
<td></td>
<td></td>
<td>03</td>
</tr>
<tr>
<td>ldfs.s.ldhint</td>
<td></td>
<td></td>
<td>01</td>
</tr>
<tr>
<td>ldfe.ldhint</td>
<td></td>
<td></td>
<td>00</td>
</tr>
<tr>
<td>ldfs.a.ldhint</td>
<td></td>
<td></td>
<td>06</td>
</tr>
<tr>
<td>ldfd.a.ldhint</td>
<td></td>
<td></td>
<td>07</td>
</tr>
<tr>
<td>ldfs.a.s.ldhint</td>
<td></td>
<td></td>
<td>05</td>
</tr>
<tr>
<td>ldfe.a.ldhint</td>
<td></td>
<td></td>
<td>04</td>
</tr>
<tr>
<td>ldfs.a.ldhint</td>
<td></td>
<td></td>
<td>08</td>
</tr>
<tr>
<td>ldfe.a.ldhint</td>
<td></td>
<td></td>
<td>08</td>
</tr>
<tr>
<td>ldfs.a.s.ldhint</td>
<td></td>
<td></td>
<td>00</td>
</tr>
<tr>
<td>ldfe.a.ldhint</td>
<td></td>
<td></td>
<td>00</td>
</tr>
</tbody>
</table>

\( f_1 = [r_3], \text{imm}_9 \)

See Table 4-38 on page 3:292.

### 4.4.1.9 Floating-point Store

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>stfs.sthint</td>
<td></td>
<td>6</td>
<td>32</td>
</tr>
<tr>
<td>std.s.sthint</td>
<td></td>
<td></td>
<td>33</td>
</tr>
<tr>
<td>stfs.s.sthint</td>
<td>([r_3] = f_2)</td>
<td>0</td>
<td>31</td>
</tr>
<tr>
<td>stfe.s.sthint</td>
<td></td>
<td>0</td>
<td>30</td>
</tr>
<tr>
<td>stfs.spill.sthint</td>
<td></td>
<td>0</td>
<td>3B</td>
</tr>
</tbody>
</table>

See Table 4-39 on page 3:292.
### 4.4.1.10 Floating-point Store – Increment by Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>stfs. sthint</td>
<td></td>
<td>7</td>
<td></td>
</tr>
<tr>
<td>std. sthint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>stfs8. sthint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>stfe. sthint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>stfs.spill. sthint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>stfs. sthint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

- \([r_3] = f_2 \cdot \text{imm}_9\]
- X6
- Hint

See Table 4-39 on page 3:292.

### 4.4.1.11 Floating-point Load Pair

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>ldfps. ldhint</td>
<td></td>
<td>6</td>
<td></td>
</tr>
<tr>
<td>ldpd. ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldpd. s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. s.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldpd. a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. a.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldpd. sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. sa.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldpd. c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. c.clr.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldpd. c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>ldp8. c.nc.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

- \(f_1, f_2 = [r_3]\]
- X6
- Hint

See Table 4-38 on page 3:292.
4.4.1.12 Floating-point Load Pair – Increment by Immediate

The line prefetch instructions are encoded in major opcodes 6 and 7 along with the floating-point load/store instructions. See “Loads and Stores” on page 3:287 for a summary of the opcode extensions.

The line prefetch instructions all have a 2-bit opcode extension field in bits 29:28 (hint) which encodes locality hint information as shown in Table 4-40.

Table 4-40. Line Prefetch Hint Completer

<table>
<thead>
<tr>
<th>hint Bits 29:28</th>
<th>lhint</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>none</td>
</tr>
<tr>
<td>1</td>
<td>.nt1</td>
</tr>
<tr>
<td>2</td>
<td>.nt2</td>
</tr>
<tr>
<td>3</td>
<td>.nta</td>
</tr>
</tbody>
</table>
### 4.4.2.1 Line Prefetch

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>lfetch.fshint</td>
<td>m x 6 hint x r3</td>
<td>6 0 0</td>
<td>2C 2D 2E 2F</td>
</tr>
<tr>
<td>lfetch.excl.fshint</td>
<td>[r3]</td>
<td></td>
<td></td>
</tr>
<tr>
<td>lfetch.fault.fshint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>lfetch.fault.excl.fshint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

See Table 4-40 on page 3:301.

### 4.4.2.2 Line Prefetch – Increment by Register

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>lfetch.fshint</td>
<td>m x 6 hint x r3, r2</td>
<td>6 1 0</td>
<td>2C 2D 2E 2F</td>
</tr>
<tr>
<td>lfetch.excl.fshint</td>
<td>[r3], r2</td>
<td></td>
<td></td>
</tr>
<tr>
<td>lfetch.fault.fshint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>lfetch.fault.excl.fshint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

See Table 4-40 on page 3:301.

### 4.4.2.3 Line Prefetch – Increment by Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>lfetch.fshint</td>
<td>m x 6 hint i imm7b</td>
<td>7</td>
<td>2C 2D 2E 2F</td>
</tr>
<tr>
<td>lfetch.excl.fshint</td>
<td>[r3], imm9</td>
<td></td>
<td></td>
</tr>
<tr>
<td>lfetch.fault.fshint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>lfetch.fault.excl.fshint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

See Table 4-40 on page 3:301.
4.4.3 Semaphores

The semaphore instructions are encoded in major opcode 4 along with the integer load/store instructions. See “Loads and Stores” on page 3:287 for a summary of the opcode extensions.

4.4.3.1 Exchange/Compare and Exchange

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>cmpxchg1.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>cmpxchg2.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>cmpxchg4.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>cmpxchg8.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>cmpxchg1.rel.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>cmpxchg2.rel.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>cmpxchg4.rel.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>cmpxchg8.rel.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>xchg1.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>xchg2.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>xchg4.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>xchg8.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Instruction Operands Opcode Extension

\[ r_1 = [r_3], r_2, \text{ ar.ccv} \]

4
0
1
00
01
02
03
04
05
06
07
08
09
0A
0B

See Table 4-38 on page 3:292.

4.4.3.2 Fetch and Add – Immediate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fetchadd4.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>fetchadd8.acq.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>fetchadd4.rel.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>fetchadd8.rel.ldhint</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Instruction Operands Opcode Extension

\[ r_1 = [r_3], \text{ inc}_3 \]

4
0
1
12
13
16
17

See Table 4-38 on page 3:292.
4.4.4 Set/Get FR

The set FR instructions are encoded in major opcode 6 along with the floating-point load/store instructions. The get FR instructions are encoded in major opcode 4 along with the integer load/store instructions. See “Loads and Stores” on page 3:287 for a summary of the opcode extensions.

### 4.4.4.1 Set FR

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>setf.sig</td>
<td>mxx</td>
<td>6</td>
<td>0 01</td>
</tr>
<tr>
<td>setf.exp</td>
<td></td>
<td></td>
<td>0C 1D 1E</td>
</tr>
<tr>
<td>setf.s</td>
<td></td>
<td></td>
<td>1F</td>
</tr>
<tr>
<td>setf.d</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

\[ f_1 = r_2 \]

### 4.4.4.2 Get FR

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>getf.sig</td>
<td>mxx</td>
<td>4</td>
<td>0 01</td>
</tr>
<tr>
<td>getf.exp</td>
<td></td>
<td></td>
<td>0C 1D 1E</td>
</tr>
<tr>
<td>getf.s</td>
<td></td>
<td></td>
<td>1F</td>
</tr>
<tr>
<td>getf.d</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

\[ r_1 = f_2 \]
4.4.5 Speculation and Advanced Load Checks

The speculation and advanced load check instructions are encoded in major opcodes 0 and 1 along with the system/memory management instructions. See “System/Memory Management” on page 3:310 for a summary of the opcode extensions.

4.4.5.1 Integer Speculation Check (M-Unit)

4.4.5.2 Floating-point Speculation Check

4.4.5.3 Integer Advanced Load Check

4.4.5.4 Floating-point Advanced Load Check
### 4.4.6 Cache/Synchronization/RSE/ALAT

The cache/synchronization/RSE/ALAT instructions are encoded in major opcode 0 along with the memory management instructions. See “System/Memory Management” on page 3:310 for a summary of the opcode extensions.

#### 4.4.6.1 Sync/Fence/Serialize/ALAT Control

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>invala</td>
<td>0</td>
<td>x3, x4, x2</td>
</tr>
<tr>
<td>fwb</td>
<td>0</td>
<td>0, 1</td>
</tr>
<tr>
<td>mf</td>
<td>0</td>
<td>0, 2, 3</td>
</tr>
<tr>
<td>mf.a</td>
<td>0</td>
<td>0, 2, 3</td>
</tr>
<tr>
<td>srlz.d</td>
<td>0</td>
<td>0, 1, 3</td>
</tr>
<tr>
<td>srlz.i</td>
<td>0</td>
<td>0, 1, 3</td>
</tr>
<tr>
<td>sync.i</td>
<td>0</td>
<td>0, 1, 3</td>
</tr>
</tbody>
</table>

#### 4.4.6.2 RSE Control

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>flushrs f</td>
<td>0</td>
<td>x3, x4, x2</td>
</tr>
<tr>
<td>loadrs f</td>
<td>0</td>
<td>C, A, 0</td>
</tr>
</tbody>
</table>

#### 4.4.6.3 Integer ALAT Entry Invalidate

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>invala.e</td>
<td>r_f</td>
<td>0</td>
<td>x3, x4, x2</td>
</tr>
</tbody>
</table>

---

Volume 3: Instruction Formats
4.4.6.4 Floating-point ALAT Entry Invalidate

M27

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>invala.e</td>
<td>f1</td>
<td>0</td>
<td>0 3 1</td>
</tr>
</tbody>
</table>

4.4.6.5 Flush Cache/Purge Translation Cache Entry

M28

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fc</td>
<td>r3</td>
<td>1</td>
<td>0 30</td>
</tr>
<tr>
<td>ptc.e p</td>
<td>r3</td>
<td>1</td>
<td>0 34</td>
</tr>
</tbody>
</table>

4.4.7 GR/AR Moves (M-Unit)

The M-Unit GR/AR move instructions are encoded in major opcode 0 along with the system/memory management instructions. (Some ARs are accessed using system control instructions on the I-unit. See “GR/AR Moves (I-Unit)” on page 3:285.) See “System/Memory Management” on page 3:310 for a summary of the M-Unit GR/AR opcode extensions.

4.4.7.1 Move to AR – Register (M-Unit)

M29

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov.m</td>
<td>ar3 = r2</td>
<td>1</td>
<td>0 2A</td>
</tr>
</tbody>
</table>

4.4.7.2 Move to AR – Immediate8 (M-Unit)

M30

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov.m</td>
<td>ar3 = imm8</td>
<td>0</td>
<td>0 8 2</td>
</tr>
</tbody>
</table>
4.4.7.3 Move from AR (M-Unit)

M31

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov.m</td>
<td>fr = ar3</td>
<td>1</td>
<td>0</td>
</tr>
</tbody>
</table>

4.4.8 GR/CR Moves

The GR/CR move instructions are encoded in major opcode 0 along with the system/memory management instructions. See “System/Memory Management” on page 3:310 for a summary of the opcode extensions.

4.4.8.1 Move to CR

M32

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>movp</td>
<td>cr3 = r2</td>
<td>1</td>
<td>0</td>
</tr>
</tbody>
</table>

4.4.8.2 Move from CR

M33

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>movp</td>
<td>r1 = cr3</td>
<td>1</td>
<td>0</td>
</tr>
</tbody>
</table>
4.4.9 Miscellaneous M-Unit Instructions

The miscellaneous M-unit instructions are encoded in major opcode 0 along with the system/memory management instructions. See “System/Memory Management” on page 3:310 for a summary of the opcode extensions.

4.4.9.1 Allocate Register Stack Frame

Note: The three immediates in the instruction encoding are formed from the operands as follows:

\[
\text{sof} = i + l + o \\
\text{sol} = i + l \\
\text{sor} = r >> 3
\]

4.4.9.2 Move to PSR

4.4.9.3 Move from PSR
### 4.4.9.4 Break/Nop (M-Unit)

```plaintext
M37
```

### 4.4.10 System/Memory Management

All system/memory management instructions are encoded within major opcodes 0 and 1 using a 3-bit opcode extension field ($x_3$) in bits 35:33. Some instructions also have a 4-bit opcode extension field ($x_4$) in bits 30:27, or a 6-bit opcode extension field ($x_6$) in bits 32:27. Most of the instructions having a 4-bit opcode extension field also have a 2-bit extension field ($x_2$) in bits 32:31. Table 4-41 shows the 3-bit assignments for opcode 0, Table 4-42 summarizes the 4-bit+2-bit assignments for opcode 0, Table 4-43 shows the 3-bit assignments for opcode 1, and Table 4-44 summarizes the 6-bit assignments for opcode 1.

**Table 4-41. Opcode 0 System/Memory Management 3-bit Opcode Extensions**

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_3$ Bits 35:33</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>System/Memory Management 4-bit+2-bit Ext (Table 4-42)</td>
</tr>
<tr>
<td>1</td>
<td>chk.a.nc – int M22</td>
</tr>
<tr>
<td>2</td>
<td>chk.a.clr – int M22</td>
</tr>
<tr>
<td>3</td>
<td>chk.a.nc – fp M23</td>
</tr>
<tr>
<td>4</td>
<td>chk.a.clr – fp M23</td>
</tr>
<tr>
<td>5</td>
<td>chk.a.clr – int M22</td>
</tr>
<tr>
<td>6</td>
<td>chk.a.clr – fp M23</td>
</tr>
<tr>
<td>7</td>
<td>chk.a.clr – fp M23</td>
</tr>
</tbody>
</table>

---

*Volume 3: Instruction Formats*
### Table 4-42. Opcode 0 System/Memory Management 4-bit+2-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_3$ Bits 35:33</th>
<th>$x_4$ Bits 30:27</th>
<th>$x_2$ Bits 32:31</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>break.m M37</td>
<td>invala M24</td>
<td>fwb M24</td>
</tr>
<tr>
<td>1</td>
<td>nop.m M37</td>
<td></td>
<td>srlz.d M24</td>
</tr>
<tr>
<td>2</td>
<td>invala.e – int M26</td>
<td>mf M24</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>invala.e – fp M27</td>
<td>mf.a M24</td>
<td>sync.i M24</td>
</tr>
<tr>
<td>4</td>
<td></td>
<td>sum M44</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td>rum M44</td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td>ssm M44</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
<td>rsm M44</td>
<td></td>
</tr>
<tr>
<td>8</td>
<td></td>
<td>mov.m to ar – imm_8 M30</td>
<td></td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>A</td>
<td>loadrs M25</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td>flushrs M25</td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### Table 4-43. Opcode 1 System/Memory Management 3-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_3$ Bits 35:33</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>System/Memory Management 6-bit Ext (Table 4-44)</td>
</tr>
<tr>
<td>1</td>
<td>chk.s.m – int M20</td>
</tr>
<tr>
<td>2</td>
<td>chk.s – fp M21</td>
</tr>
<tr>
<td>3</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td>alloc M34</td>
</tr>
<tr>
<td>6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
</tr>
</tbody>
</table>
Table 4-44. Opcode 1 System/Memory Management 6-bit Opcode Extensions

<table>
<thead>
<tr>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>mov to rr M42</td>
<td>mov from rr M43</td>
<td>fc M28</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>mov to dbrr M42</td>
<td>mov from dbrr M43</td>
<td>probe.rw.fault – imm&lt;sub&gt;2&lt;/sub&gt; M40</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>mov to ibr M42</td>
<td>mov from ibr M43</td>
<td>probe.r.fault – imm&lt;sub&gt;2&lt;/sub&gt; M40</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>mov to pkr M42</td>
<td>mov from pkr M43</td>
<td>probe.w.fault – imm&lt;sub&gt;2&lt;/sub&gt; M40</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>4</td>
<td>mov to pmc M42</td>
<td>mov from pmc M43</td>
<td>mov from cr M33</td>
<td>ptc.e M28</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td>mov to pmd M42</td>
<td>mov from pmd M43</td>
<td>mov from cr M33</td>
<td>ptc.e M28</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td>mov to ibr M42</td>
<td>mov from ibr M43</td>
<td>mov to ar</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>mov from cpuid M43</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>8</td>
<td>probe.r – imm&lt;sub&gt;2&lt;/sub&gt; M39</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>9</td>
<td>ptc.l M45</td>
<td>probe.w – imm&lt;sub&gt;2&lt;/sub&gt; M39</td>
<td>mov to cr M33</td>
<td>probe.w M38</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>A</td>
<td>ptc.g M45</td>
<td>thash M46</td>
<td>mov to m to ar M29</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B</td>
<td>ptc.ga M45</td>
<td>ttag M46</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td>ptr.d M45</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td>ptr.i M45</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td>itr.d M42</td>
<td>tpa M46</td>
<td>itc.d M41</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td>itr.i M42</td>
<td>tak M46</td>
<td>itc.i M41</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

4.4.10.1 Probe – Register

M38

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>probe.r</td>
<td>r&lt;sub&gt;1&lt;/sub&gt; = r&lt;sub&gt;3&lt;/sub&gt;, r&lt;sub&gt;2&lt;/sub&gt;</td>
<td>1</td>
<td>x&lt;sub&gt;3&lt;/sub&gt; x&lt;sub&gt;6&lt;/sub&gt;</td>
</tr>
<tr>
<td>probe.w</td>
<td>r&lt;sub&gt;1&lt;/sub&gt; = r&lt;sub&gt;3&lt;/sub&gt;, imm&lt;sub&gt;2&lt;/sub&gt;</td>
<td>1</td>
<td>x&lt;sub&gt;3&lt;/sub&gt; x&lt;sub&gt;6&lt;/sub&gt;</td>
</tr>
</tbody>
</table>

4.4.10.2 Probe – Immediate<sub>2</sub>

M39

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>probe.r</td>
<td>r&lt;sub&gt;1&lt;/sub&gt; = r&lt;sub&gt;3&lt;/sub&gt;, imm&lt;sub&gt;2&lt;/sub&gt;</td>
<td>1</td>
<td>x&lt;sub&gt;3&lt;/sub&gt; x&lt;sub&gt;6&lt;/sub&gt;</td>
</tr>
<tr>
<td>probe.w</td>
<td>r&lt;sub&gt;1&lt;/sub&gt; = r&lt;sub&gt;3&lt;/sub&gt;, imm&lt;sub&gt;2&lt;/sub&gt;</td>
<td>1</td>
<td>x&lt;sub&gt;3&lt;/sub&gt; x&lt;sub&gt;6&lt;/sub&gt;</td>
</tr>
</tbody>
</table>
### 4.4.10.3 Probe

**Fault – Immediate**

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>probe.rw.fault</code></td>
<td><code>r3, imm2</code></td>
<td>1</td>
<td>31</td>
</tr>
<tr>
<td><code>probe.r.fault</code></td>
<td></td>
<td>0</td>
<td>32</td>
</tr>
<tr>
<td><code>probe.w.fault</code></td>
<td></td>
<td>0</td>
<td>33</td>
</tr>
</tbody>
</table>

### 4.4.10.4 Translation Cache Insert

**Instruction Operands Opcode Extension**

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>ltc.d</code></td>
<td><code>r2</code></td>
<td>1</td>
<td>2E</td>
</tr>
<tr>
<td><code>ltc.i</code></td>
<td><code>r2</code></td>
<td>0</td>
<td>2F</td>
</tr>
</tbody>
</table>

### 4.4.10.5 Move to Indirect Register/Translation Register Insert

**Instruction Operands Opcode Extension**

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td><code>mov</code></td>
<td><code>r[r3] = r2</code></td>
<td>1</td>
<td>00</td>
</tr>
<tr>
<td></td>
<td><code>dbr[r3] = r2</code></td>
<td></td>
<td>01</td>
</tr>
<tr>
<td></td>
<td><code>ibr[r3] = r2</code></td>
<td></td>
<td>02</td>
</tr>
<tr>
<td></td>
<td><code>pkr[r3] = r2</code></td>
<td></td>
<td>03</td>
</tr>
<tr>
<td></td>
<td><code>pmc[r3] = r2</code></td>
<td></td>
<td>04</td>
</tr>
<tr>
<td></td>
<td><code>pmd[r3] = r2</code></td>
<td></td>
<td>05</td>
</tr>
<tr>
<td><code>itr.d</code></td>
<td><code>dtr[r3] = r2</code></td>
<td></td>
<td>0E</td>
</tr>
<tr>
<td><code>itr.i</code></td>
<td><code>itr[r3] = r2</code></td>
<td></td>
<td>0F</td>
</tr>
</tbody>
</table>
### 4.4.10.6 Move from Indirect Register

#### M43

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov&lt;sup&gt;P&lt;/sup&gt;</td>
<td>( r_1 = \text{rr}[r_3] )</td>
<td>1</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td>( r_1 = \text{db}[r_3] )</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>( r_1 = \text{ib}[r_3] )</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>( r_1 = \text{pk}[r_3] )</td>
<td></td>
<td></td>
</tr>
<tr>
<td></td>
<td>( r_1 = \text{pm}[r_3] )</td>
<td></td>
<td></td>
</tr>
<tr>
<td>mov&lt;sup&gt;P&lt;/sup&gt;</td>
<td>( r_1 = \text{pm}[r_3] )</td>
<td>1</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td>( r_1 = \text{cp}[r_3] )</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

### 4.4.10.7 Set/Reset User/System Mask

#### M44

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>sum</td>
<td>( \text{imm}_{24} )</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>rum</td>
<td></td>
<td></td>
<td>4</td>
</tr>
<tr>
<td>ssm&lt;sup&gt;P&lt;/sup&gt;</td>
<td></td>
<td></td>
<td>5</td>
</tr>
<tr>
<td>rsm&lt;sup&gt;P&lt;/sup&gt;</td>
<td>( \text{imm}_{24} )</td>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>

### 4.4.10.8 Translation Purge

#### M45

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>ptc&lt;sup&gt;i&lt;/sup&gt;&lt;sub&gt;&lt;sup&gt;P&lt;/sup&gt;&lt;/sub&gt;</td>
<td>( r_3, r_2 )</td>
<td>1</td>
<td>0</td>
</tr>
<tr>
<td>ptc&lt;sup&gt;i&lt;/sup&gt;&lt;sub&gt;&lt;sup&gt;G&lt;/sup&gt;&lt;/sub&gt;&lt;sup&gt;P&lt;/sup&gt;</td>
<td></td>
<td></td>
<td>09</td>
</tr>
<tr>
<td>ptc&lt;sup&gt;i&lt;/sup&gt;&lt;sub&gt;&lt;sup&gt;G&lt;/sup&gt;&lt;/sub&gt;&lt;sup&gt;A&lt;/sup&gt;&lt;sup&gt;P&lt;/sup&gt;</td>
<td></td>
<td></td>
<td>0A</td>
</tr>
<tr>
<td>ptr&lt;sup&gt;i&lt;/sup&gt;&lt;sub&gt;&lt;sup&gt;D&lt;/sup&gt;&lt;/sub&gt;&lt;sup&gt;P&lt;/sup&gt;</td>
<td></td>
<td></td>
<td>0B</td>
</tr>
<tr>
<td>ptr&lt;sup&gt;i&lt;/sup&gt;&lt;sup&gt;P&lt;/sup&gt;</td>
<td></td>
<td></td>
<td>0C</td>
</tr>
<tr>
<td>ptr&lt;sup&gt;i&lt;/sup&gt;</td>
<td></td>
<td></td>
<td>0D</td>
</tr>
</tbody>
</table>
4.4.10.9 Translation Access

4.5 B-Unit Instruction Encodings

The branch-unit includes branch, predict, and miscellaneous instructions.

4.5.1 Branches

Opcode 0 is used for indirect branch, opcode 1 for indirect call, opcode 4 for IP-relative branch, and opcode 5 for IP-relative call.

The IP-relative branch instructions encoded within major opcode 4 use a 3-bit opcode extension field in bits 8:6 (btype) to distinguish the branch types as shown in Table 4-45.

Table 4-45. IP-Relative Branch Types

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>btype Bits 8:6</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>br.cond B1</td>
</tr>
<tr>
<td>1</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>br.wexit B1</td>
</tr>
<tr>
<td>3</td>
<td>br.wtop B1</td>
</tr>
<tr>
<td>4</td>
<td>br.cloop B2</td>
</tr>
<tr>
<td>5</td>
<td>br.cexit B2</td>
</tr>
<tr>
<td>6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
</tr>
</tbody>
</table>

The indirect branch, indirect return, and miscellaneous branch-unit instructions are encoded within major opcode 0 using a 6-bit opcode extension field in bits 32:27 (x6). Table 4-46 summarizes these assignments.
The indirect branch instructions encoded within major opcodes 0 use a 3-bit opcode extension field in bits 8:6 (btype) to distinguish the branch types as shown in Table 4-47.

### Table 4-46. Indirect/Miscellaneous Branch Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Bits 40:37</th>
<th>x₆</th>
<th>Bits 32:31</th>
<th>btype</th>
<th>Bits 8:6</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>break.b B9</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>1</td>
<td>e</td>
<td>e</td>
<td>Indirect Branch (Table 4-47)</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>2</td>
<td>cover B8</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>3</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>4</td>
<td>clrrb B8</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>5</td>
<td>clrrb.pr B8</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>6</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>7</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>8</td>
<td>rfi B8</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>9</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>A</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>B</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>C</td>
<td>bsw.0 B8</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>D</td>
<td>bsw.1 B8</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>E</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>F</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
</tbody>
</table>

The indirect return branch instructions encoded within major opcodes 0 use a 3-bit opcode extension field in bits 8:6 (btype) to distinguish the branch types as shown in Table 4-48.

### Table 4-47. Indirect Branch Types

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Bits 40:37</th>
<th>x₆</th>
<th>btype</th>
<th>Bits 8:6</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>20</td>
<td>0</td>
<td>br.cond B4</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>1</td>
<td>br ia B4</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>3</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>4</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>5</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>6</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
<tr>
<td>7</td>
<td>e</td>
<td>e</td>
<td>e</td>
<td>e</td>
</tr>
</tbody>
</table>
All of the branch instructions have a 1-bit opcode extension field, $p$, in bit 12 which provides a sequential prefetch hint. Table 4-49 summarizes these assignments.

Table 4-48. Indirect Return Branch Types

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$x_6$ Bits 32:27</th>
<th>btype Bits 8:6</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>21</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td></td>
<td>1</td>
</tr>
<tr>
<td></td>
<td></td>
<td>2</td>
</tr>
<tr>
<td></td>
<td></td>
<td>3</td>
</tr>
<tr>
<td>4</td>
<td></td>
<td>4, br.ret B4</td>
</tr>
<tr>
<td>5</td>
<td></td>
<td>5</td>
</tr>
<tr>
<td>6</td>
<td></td>
<td>6</td>
</tr>
<tr>
<td>7</td>
<td></td>
<td>7</td>
</tr>
</tbody>
</table>

Table 4-49. Sequential Prefetch Hint Completer

<table>
<thead>
<tr>
<th>p Bit 12</th>
<th>ph</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>.few</td>
</tr>
<tr>
<td>1</td>
<td>.many</td>
</tr>
</tbody>
</table>

The IP-relative and indirect branch instructions all have a 2-bit opcode extension field in bits 34:33 (wh) which encodes branch prediction “whether” hint information as shown in Table 4-50. Indirect call instructions have a 3-bit opcode extension field in bits 34:32 (wh) for “whether” hint information as shown in Table 4-51.

Table 4-50. Branch Whether Hint Completer

<table>
<thead>
<tr>
<th>wh Bits 34:33</th>
<th>bwh</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>.sptk</td>
</tr>
<tr>
<td>1</td>
<td>.spnt</td>
</tr>
<tr>
<td>2</td>
<td>.dptk</td>
</tr>
<tr>
<td>3</td>
<td>.dpnt</td>
</tr>
</tbody>
</table>

Table 4-51. Indirect Call Whether Hint Completer

<table>
<thead>
<tr>
<th>wh Bits 34:32</th>
<th>bwh</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>.sptk</td>
</tr>
<tr>
<td>1</td>
<td>.spnt</td>
</tr>
<tr>
<td>2</td>
<td>.dptk</td>
</tr>
<tr>
<td>3</td>
<td>.dpnt</td>
</tr>
<tr>
<td>4</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
</tr>
</tbody>
</table>
The branch instructions also have a 1-bit opcode extension field in bit 35 (d) which encodes a branch cache deallocation hint as shown in Table 4-52.

### Table 4-52. Branch Cache Deallocation Hint Completer

<table>
<thead>
<tr>
<th>Bit 35</th>
<th>dh</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>none</td>
</tr>
<tr>
<td>1</td>
<td>.clr</td>
</tr>
</tbody>
</table>

#### 4.5.1.1 IP-Relative Branch

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>br.cond.bwh.ph.dh</td>
<td>s d wh</td>
<td>imm&lt;sub&gt;20b&lt;/sub&gt;</td>
<td>p btype qp</td>
</tr>
<tr>
<td>br.wexit.bwh.ph.dh</td>
<td>s d wh</td>
<td>imm&lt;sub&gt;20b&lt;/sub&gt;</td>
<td>p btype qp</td>
</tr>
<tr>
<td>br.wtop.bwh.ph.dh</td>
<td>s d wh</td>
<td>imm&lt;sub&gt;20b&lt;/sub&gt;</td>
<td>p btype qp</td>
</tr>
</tbody>
</table>

#### 4.5.1.2 IP-Relative Counted Branch

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>br.cloop.bwh.ph.dh</td>
<td>s d wh</td>
<td>imm&lt;sub&gt;20b&lt;/sub&gt;</td>
<td>p btype qp</td>
</tr>
<tr>
<td>br.cexit.bwh.ph.dh</td>
<td>s d wh</td>
<td>imm&lt;sub&gt;20b&lt;/sub&gt;</td>
<td>p btype qp</td>
</tr>
<tr>
<td>br.ctop.bwh.ph.dh</td>
<td>s d wh</td>
<td>imm&lt;sub&gt;20b&lt;/sub&gt;</td>
<td>p btype qp</td>
</tr>
</tbody>
</table>

#### 4.5.1.3 IP-Relative Call

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>br.call.bwh.ph.dh</td>
<td>s d wh</td>
<td>imm&lt;sub&gt;20b&lt;/sub&gt;</td>
<td>p btype qp</td>
</tr>
</tbody>
</table>

See Table 4-49 on page 3:317.

See Table 4-50 on page 3:317.

See Table 4-52 on page 3:318.
### 4.5.1.4 Indirect Branch

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>br.cond.bwh.ph.dh</td>
<td>b2</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>br.ia.bwh.ph.dh</td>
<td></td>
<td></td>
<td>1</td>
</tr>
<tr>
<td>br.ret.bwh.ph.dh</td>
<td></td>
<td></td>
<td>4</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>br.call.bwh.ph.dh</td>
<td>b1 = b2</td>
<td>1</td>
<td></td>
</tr>
</tbody>
</table>

#### 4.5.2 Branch Predict and Nop

The branch predict and nop instructions are encoded in major opcodes 2 (Indirect Predict/Nop) and 7 (IP-relative Predict). The indirect predict and nop instructions in major opcode 2 use a 6-bit opcode extension field in bits 32:27 (x6). Table 4-53 summarizes these assignments.

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>x6 Bits 30:27</th>
<th>x6 Bits 32:31</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>2</td>
<td>2</td>
<td>2</td>
</tr>
<tr>
<td>3</td>
<td>3</td>
<td>3</td>
</tr>
<tr>
<td>4</td>
<td>4</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>6</td>
<td>6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td>7</td>
<td></td>
</tr>
<tr>
<td>8</td>
<td>8</td>
<td></td>
</tr>
<tr>
<td>A</td>
<td>A</td>
<td></td>
</tr>
<tr>
<td>B</td>
<td>B</td>
<td></td>
</tr>
<tr>
<td>C</td>
<td>C</td>
<td></td>
</tr>
<tr>
<td>D</td>
<td>D</td>
<td></td>
</tr>
<tr>
<td>E</td>
<td>E</td>
<td></td>
</tr>
<tr>
<td>F</td>
<td>F</td>
<td></td>
</tr>
</tbody>
</table>
The branch predict instructions all have a 1-bit opcode extension field in bit 35 (ih) which encodes a branch importance hint. The mov to BR instruction (page 3:284) also has this hint in bit 23. Table 4-54 shows these assignments.

**Table 4-54. Branch Importance Hint Completer**

<table>
<thead>
<tr>
<th>ih Bit 23 or Bit 35</th>
<th>ih</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>none</td>
</tr>
<tr>
<td>1</td>
<td>.imp</td>
</tr>
</tbody>
</table>

The IP-relative branch predict instructions have a 2-bit opcode extension field in bits 4:3 (wh) which encodes branch prediction “whether” hint information as shown in Table 4-55. Note that the combination of the .loop or .exit whether hint completer with the none importance hint completer is undefined.

**Table 4-55. IP-Relative Predict Whether Hint Completer**

<table>
<thead>
<tr>
<th>wh Bits 4:3</th>
<th>ipwh</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>.sptk</td>
</tr>
<tr>
<td>1</td>
<td>.loop</td>
</tr>
<tr>
<td>2</td>
<td>.dptk</td>
</tr>
<tr>
<td>3</td>
<td>.exit</td>
</tr>
</tbody>
</table>

The indirect branch predict instructions have a 2-bit opcode extension field in bits 4:3 (wh) which encodes branch prediction “whether” hint information as shown in Table 4-56.

**Table 4-56. Indirect Predict Whether Hint Completer**

<table>
<thead>
<tr>
<th>wh Bits 4:3</th>
<th>indwh</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>.sptk</td>
</tr>
<tr>
<td>1</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td>.dptk</td>
</tr>
<tr>
<td>3</td>
<td></td>
</tr>
</tbody>
</table>

### 4.5.2.1 IP-Relative Predict

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>brp.ipwh.ih</td>
<td>target25, tag13</td>
<td>7</td>
<td>See Table 4-54 on page 3:320. See Table 4-55 on page 3:320.</td>
</tr>
</tbody>
</table>
### 4.5.2.2 Indirect Predict

![Instruction Format Diagram]

#### Table 4-46

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>brp.indwh.ih</td>
<td>( b_2, \text{tag}_{13} )</td>
<td>2</td>
<td>10 ( \text{See Table 4-54 on page 3:320.} )</td>
</tr>
<tr>
<td>brp.ret.indwh.ih</td>
<td></td>
<td></td>
<td>11 ( \text{See Table 4-56 on page 3:320.} )</td>
</tr>
</tbody>
</table>

### 4.5.3 Miscellaneous B-Unit Instructions

The miscellaneous branch-unit instructions include a number of instructions encoded within major opcode 0 using a 6-bit opcode extension field in bits 32:27 (\( x_6 \)) as described in Table 4-46 on page 3:316.

#### 4.5.3.1 Miscellaneous (B-Unit)

![Instruction Format Diagram]

#### Table 4-54

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>cover (^1)</td>
<td>0 ( x_6 )</td>
<td>02</td>
</tr>
<tr>
<td>clrrb (^1)</td>
<td>0</td>
<td>04</td>
</tr>
<tr>
<td>clrrb.pr (^1)</td>
<td>0</td>
<td>05</td>
</tr>
<tr>
<td>rfi (^{e,p})</td>
<td>0</td>
<td>08</td>
</tr>
<tr>
<td>bsw.0 (^{p})</td>
<td>0</td>
<td>0C</td>
</tr>
<tr>
<td>bsw.1 (^{p})</td>
<td>0</td>
<td>0D</td>
</tr>
<tr>
<td>epc</td>
<td>0</td>
<td>10</td>
</tr>
</tbody>
</table>

#### 4.5.3.2 Break/Nop (B-Unit)

![Instruction Format Diagram]

#### Table 4-56

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>break.b (^*)</td>
<td>( \text{imm}_{21} )</td>
<td>0</td>
<td>00</td>
</tr>
<tr>
<td>nop.b</td>
<td></td>
<td>2</td>
<td>00</td>
</tr>
</tbody>
</table>
4.6 F-Unit Instruction Encodings

The floating-point instructions are encoded in major opcodes 8 – E for floating-point and fixed-point arithmetic, opcode 4 for floating-point compare, opcode 5 for floating-point class, and opcodes 0 and 1 for miscellaneous floating-point instructions.

The miscellaneous and reciprocal approximation floating-point instructions are encoded within major opcodes 0 and 1 using a 1-bit opcode extension field (x) in bit 33 and either a second 1-bit extension field in bit 36 (q) or a 6-bit opcode extension field (x6) in bits 32-27. Table 4-57 shows the 1-bit x assignments, Table 4-60 shows the additional 1-bit q assignments for the reciprocal approximation instructions; Table 4-58 and Table 4-59 summarize the 6-bit x6 assignments.

Table 4-57. Miscellaneous Floating-point 1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Bits 40:37</th>
<th>x</th>
<th>Bit 33</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td></td>
<td>0</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>1</td>
<td>Reciprocal Approximation (Table 4-60)</td>
</tr>
<tr>
<td>1</td>
<td></td>
<td>0</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>1</td>
<td>Reciprocal Approximation (Table 4-60)</td>
</tr>
</tbody>
</table>

Table 4-58. Opcode 0 Miscellaneous Floating-point 6-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Bits 40:37</th>
<th>x</th>
<th>Bits 30:27</th>
<th>x6</th>
<th>Bits 32:31</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td></td>
<td>0</td>
<td></td>
<td>0</td>
<td></td>
</tr>
<tr>
<td></td>
<td></td>
<td>1</td>
<td></td>
<td>1</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td></td>
<td>0</td>
<td></td>
<td>2</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td></td>
<td></td>
<td></td>
<td>3</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Table 4-59. Opcode 1 Miscellaneous Floating-point 6-bit Opcode Extensions
Most floating-point instructions have a 2-bit opcode extension field in bits 35:34 (sf) which encodes the FPSR status field to be used. Table 4-61 summarizes these assignments.

Table 4-61. Floating-point Status Field Completer

<table>
<thead>
<tr>
<th>sf</th>
<th>Bits 35:34</th>
<th>sf</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td></td>
<td>.s0</td>
</tr>
<tr>
<td>1</td>
<td></td>
<td>.s1</td>
</tr>
<tr>
<td>2</td>
<td></td>
<td>.s2</td>
</tr>
<tr>
<td>3</td>
<td></td>
<td>.s3</td>
</tr>
</tbody>
</table>
4.6.1 Arithmetic

The floating-point arithmetic instructions are encoded within major opcodes 8 – D using a 1-bit opcode extension field (x) in bit 36 and a 2-bit opcode extension field (sf) in bits 35:34. The opcode and x assignments are shown in Table 4-62.

Table 4-62. Floating-point Arithmetic 1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>x Bit 36</th>
<th>Opcode Bits 40:37</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>8 fma F1, fma.d F1, fms F1, fms.d F1, fnma F1, fnma.d F1</td>
</tr>
<tr>
<td>1</td>
<td>9 fma.s F1, fpga F1, fms.s F1, fpga.s F1, fnms F1, fnms.s F1, fpga.m F1</td>
</tr>
</tbody>
</table>

The fixed-point arithmetic and parallel floating-point select instructions are encoded within major opcode E using a 1-bit opcode extension field (x) in bit 36. The fixed-point arithmetic instructions also have a 2-bit opcode extension field (x2) in bits 35:34. These assignments are shown in Table 4-63.

Table 4-63. Fixed-point Multiply Add and Select Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>x Bit 36</th>
<th>x2 Bits 35:34</th>
</tr>
</thead>
<tbody>
<tr>
<td>E</td>
<td>0</td>
<td>0 fselect F3</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>1 xma.l F2, xma.h F2, xma.h F2</td>
</tr>
</tbody>
</table>

4.6.1.1 Floating-point Multiply Add

The floating-point multiply add instructions are encoded within major opcode F1 using a 1-bit opcode extension field (x) in bit 36 and a 2-bit opcode extension field (sf) in bits 35:34. The fma instructions have a 1-bit extension field (f) in bit 39 and a 1-bit extension field (p) in bit 38. The fixed-point multiply add instructions have a 2-bit extension field (x) in bit 36. The assignments are shown in Table 4-61 on page 3:323.
4.6.1.2 Fixed-point Multiply Add

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>xma.l</td>
<td></td>
<td>E</td>
<td>1</td>
</tr>
<tr>
<td>xma.h</td>
<td></td>
<td></td>
<td>3</td>
</tr>
<tr>
<td>xma.hu</td>
<td></td>
<td></td>
<td>2</td>
</tr>
</tbody>
</table>

The floating-point class instructions are encoded within major opcode 5 using a 1-bit opcode extension field in bit 12 (t_a) as shown in Table 4-65.

Table 4-65. Floating-point Class 1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>t_a Bit 12</th>
</tr>
</thead>
<tbody>
<tr>
<td>5</td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>fclass.m F5</td>
</tr>
<tr>
<td>1</td>
<td>fclass.m.unc F5</td>
</tr>
</tbody>
</table>

4.6.2 Parallel Floating-point Select

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fselect</td>
<td></td>
<td>E</td>
<td>0</td>
</tr>
</tbody>
</table>

4.6.3 Compare and Classify

The predicate setting floating-point compare instructions are encoded within major opcode 4 using three 1-bit opcode extension fields in bits 33 (r_a), 36 (r_b), and 12 (t_a), and a 2-bit opcode extension field (sf) in bits 35:34. The opcode, r_a, r_b, and t_a assignments are shown in Table 4-64. The sf assignments are shown in Table 4-61 on page 3:323.

The parallel floating-point compare instructions are described on page 3:327.

Table 4-64. Floating-point Compare Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>r_a Bit 33</th>
<th>r_b Bit 36</th>
<th>t_a Bit 12</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>fcmp.eq F4</td>
<td>fcmp.eq.unc F4</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>fcmp.lt F4</td>
<td>fcmp.lt.unc F4</td>
</tr>
<tr>
<td>4</td>
<td>0</td>
<td>fcmp.le F4</td>
<td>fcmp.le.unc F4</td>
</tr>
<tr>
<td></td>
<td>1</td>
<td>fcmp.unord F4</td>
<td>fcmp.unord.unc F4</td>
</tr>
</tbody>
</table>

The parallel floating-point compare instructions are described on page 3:327.
4.6.3.1 Floating-point Compare

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fcmp.eq.sf</td>
<td></td>
<td>4</td>
<td></td>
</tr>
<tr>
<td>fcmp.lt.sf</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>fcmp.le.sf</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>fcmp.unord.sf</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>p1, p2 = f2, f3</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

See Table 4-61 on page 3:323.

4.6.3.2 Floating-point Class

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fclass.m</td>
<td>p1, p2 = f2, fclass9</td>
<td>5</td>
<td></td>
</tr>
<tr>
<td>fclass.m.unc</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

4.6.4 Approximation

4.6.4.1 Floating-point Reciprocal Approximation

There are two Reciprocal Approximation instructions. The first, in major op 0, encodes the full register variant. The second, in major op 1, encodes the parallel variant.
4.6.4.2 Floating-point Reciprocal Square Root Approximation

There are two Reciprocal Square Root Approximation instructions. The first, in major op 0, encodes the full register variant. The second, in major op 1, encodes the parallel variant.

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>frsqrta.sf</td>
<td>$f_1, p_2 = f_3$</td>
<td>x, q, sf</td>
</tr>
<tr>
<td>fprsqrta.sf</td>
<td>$f_1, p_2 = f_3$</td>
<td>x, q, sf</td>
</tr>
</tbody>
</table>

4.6.5 Minimum/Maximum and Parallel Compare

There are two groups of Minimum/Maximum instructions. The first group, in major op 0, encodes the full register variants. The second group, in major op 1, encodes the parallel variants. The parallel compare instructions are all encoded in major op 1.

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fmin.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>fmax.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>famin.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>famax.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>fpmin.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>fpmax.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>fpamin.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>fpamax.sf</td>
<td>$f_1 = f_2, f_3$</td>
<td>x, $x_6$, sf</td>
</tr>
<tr>
<td>fpcmp.eq.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 14</td>
</tr>
<tr>
<td>fpcmp.lt.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 15</td>
</tr>
<tr>
<td>fpcmp.le.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 16</td>
</tr>
<tr>
<td>fpcmp.unord.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 17</td>
</tr>
<tr>
<td>fpcmp.neq.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 18</td>
</tr>
<tr>
<td>fpcmp.nlt.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 19</td>
</tr>
<tr>
<td>fpcmp.nle.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 20</td>
</tr>
<tr>
<td>fpcmp.ord.sf</td>
<td>$f_1, f_2 = f_3$</td>
<td>0, 21</td>
</tr>
</tbody>
</table>

See Table 4-61 on page 3:323.
### 4.6.6 Merge and Logical

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fmerge.s</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fmerge.ns</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fmerge.se</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fmix.lr</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fmix.r</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fmix.i</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fsxt.r</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fsxt.i</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fpack</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fswap</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fswap.nl</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fswap.nr</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fand</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fandcm</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>for</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fxor</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fpmerge.s</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fpmerge.ns</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
<tr>
<td>fpmerge.se</td>
<td>x</td>
<td>f1=f2,f3</td>
<td>0</td>
</tr>
</tbody>
</table>

### 4.6.7 Conversion

#### 4.6.7.1 Convert Floating-point to Fixed-point

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fcvt.fx.sf</td>
<td>x</td>
<td>f1=f2</td>
<td>0</td>
</tr>
<tr>
<td>fcvt.fxu.sf</td>
<td>x</td>
<td>f1=f2</td>
<td>0</td>
</tr>
<tr>
<td>fcvt.fx.trunc.sf</td>
<td>1</td>
<td>f1=f2</td>
<td>0</td>
</tr>
<tr>
<td>fpcvt.fx.sf</td>
<td>x</td>
<td>f1=f2</td>
<td>0</td>
</tr>
<tr>
<td>fpcvt.fxu.sf</td>
<td>x</td>
<td>f1=f2</td>
<td>0</td>
</tr>
<tr>
<td>fpcvt.fx.trunc.sf</td>
<td>1</td>
<td>f1=f2</td>
<td>0</td>
</tr>
</tbody>
</table>

See Table 4-61 on page 3:323.
### 4.6.7.2 Convert Fixed-point to Floating-point

- **F11**
  - Instruction: convert fixed-point to floating-point
  - Operands: 40 37 36 34 33 32 27 26 20 19 13 12 6 5 0
  - Opcode: 0, 0, 1C

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fcvt.xf</td>
<td>x</td>
<td>x_6</td>
<td></td>
</tr>
</tbody>
</table>

### 4.6.8 Status Field Manipulation

#### 4.6.8.1 Floating-point Set Controls

- **F12**
  - Instruction: floating-point set controls
  - Operands: omask_7c, amask_7b
  - Opcode: 0, 0, 04

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fsetc.sf</td>
<td>amask_7, omask_7</td>
<td>0</td>
<td>04</td>
</tr>
</tbody>
</table>

See Table 4-61 on page 3:323.

#### 4.6.8.2 Floating-point Clear Flags

- **F13**
  - Instruction: floating-point clear flags
  - Operands: x, x_6
  - Opcode: 0, 0, 05

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fclr.fsf</td>
<td>0</td>
<td>05</td>
</tr>
</tbody>
</table>

See Table 4-61 on page 3:323.

#### 4.6.8.3 Floating-point Check Flags

- **F14**
  - Instruction: floating-point check flags
  - Operands: target_25
  - Opcode: 0, 0, 08

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>fchkf.sf</td>
<td>target_25</td>
<td>0</td>
<td>08</td>
</tr>
</tbody>
</table>

See Table 4-61 on page 3:323.
4.6.9 Miscellaneous F-Unit Instructions

4.6.9.1 Break/Nop (F-Unit)

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>break.f</td>
<td>imm_{21}</td>
<td>0</td>
<td>00</td>
</tr>
<tr>
<td>nop.f</td>
<td></td>
<td>0</td>
<td>01</td>
</tr>
</tbody>
</table>

4.7 X-Unit Instruction Encodings

The X-unit instructions occupy two instruction slots, L+X. The major opcode, opcode extensions and hints, qp, and small immediate fields occupy the X instruction slot. For movl, break.x, and nop.x, the imm_{41} field occupies the L instruction slot. For brl, the imm_{39} field and a 2-bit Ignored field occupy the L instruction slot.

4.7.1 Miscellaneous X-Unit Instructions

The miscellaneous X-unit instructions are encoded in major opcode 0 using a 3-bit opcode extension field (x_3) in bits 35:33 and a 6-bit opcode extension field (x_6) in bits 32:27. Table 4-66 shows the 3-bit assignments and Table 4-67 summarizes the 6-bit assignments. These instructions are executed by an I-unit.

Table 4-66. Misc X-Unit 3-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>x_3 Bits 35:33</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>6-bit Ext (Table 4-67)</td>
</tr>
<tr>
<td>1</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
</tr>
</tbody>
</table>
### Table 4-67. Misc X-Unit 6-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>( x_3 ) Bits 35:33</th>
<th>( x_6 ) Bits 32:31</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>---</td>
<td>---</td>
<td>---</td>
</tr>
<tr>
<td>0</td>
<td>break.x X1</td>
<td></td>
</tr>
<tr>
<td>1</td>
<td>nop.x X1</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td></td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
<td></td>
</tr>
<tr>
<td>8</td>
<td></td>
<td></td>
</tr>
<tr>
<td>9</td>
<td></td>
<td></td>
</tr>
<tr>
<td>A</td>
<td></td>
<td></td>
</tr>
<tr>
<td>B</td>
<td></td>
<td></td>
</tr>
<tr>
<td>C</td>
<td></td>
<td></td>
</tr>
<tr>
<td>D</td>
<td></td>
<td></td>
</tr>
<tr>
<td>E</td>
<td></td>
<td></td>
</tr>
<tr>
<td>F</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

#### 4.7.1.1 Break/Nop (X-Unit)

```
40 37 36 35 33 32 27 26 25
0 1 3 x6
```

### Instruction Operands Opcode Extension

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>break.x</td>
<td>( imm_{62} )</td>
<td>0</td>
<td>00</td>
</tr>
<tr>
<td>nop.x</td>
<td>( imm_{62} )</td>
<td>0</td>
<td>01</td>
</tr>
</tbody>
</table>
### 4.7.2 Move Long Immediate\textsubscript{64}

The move long immediate instruction is encoded within major opcode 6 using a 1-bit reserved opcode extension in bit 20 ($v_c$) as shown in Table 4-68. This instruction is executed by an I-unit.

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>$v_c$ Bit 20</th>
</tr>
</thead>
<tbody>
<tr>
<td>6</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td>1</td>
</tr>
</tbody>
</table>

#### Table 4-68. Move Long 1-bit Opcode Extensions

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td>movl X2</td>
<td></td>
<td>6</td>
<td>0</td>
</tr>
</tbody>
</table>

### 4.7.3 Long Branches

Long branches are executed by a B-unit. Opcode C is used for long branch and opcode D for long call.

The long branch instructions encoded within major opcode C use a 3-bit opcode extension field in bits 8:6 (btype) to distinguish the branch types as shown in Table 4-69.

<table>
<thead>
<tr>
<th>Opcode Bits 40:37</th>
<th>btype Bits 8:6</th>
</tr>
</thead>
<tbody>
<tr>
<td>C</td>
<td></td>
</tr>
<tr>
<td>0</td>
<td>brl.cond X3</td>
</tr>
<tr>
<td>1</td>
<td></td>
</tr>
<tr>
<td>2</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td></td>
</tr>
<tr>
<td>4</td>
<td></td>
</tr>
<tr>
<td>5</td>
<td></td>
</tr>
<tr>
<td>6</td>
<td></td>
</tr>
<tr>
<td>7</td>
<td></td>
</tr>
</tbody>
</table>

#### Table 4-69. Long Branch Types

The long branch instructions have the same opcode hint fields in bit 12 (p), bits 34:33 (wh), and bit 35 (d) as normal IP-relative branches. These are shown in Table 4-49 on page 3:317, Table 4-50 on page 3:317, and Table 4-52 on page 3:318.
4.7.3.1 Long Branch

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>btpe</td>
</tr>
<tr>
<td>brl.cond.bwh.ph.dh *1</td>
<td>target64</td>
<td>C</td>
<td>0</td>
</tr>
</tbody>
</table>

4.7.3.2 Long Call

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Operands</th>
<th>Opcode</th>
<th>Extension</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>btpe</td>
</tr>
<tr>
<td>brl.call.bwh.ph.dh *1</td>
<td>( b_1 = \text{target}_{64} )</td>
<td>D</td>
<td>See Table 4-49 on page 3:317.</td>
</tr>
</tbody>
</table>

4.8 Immediate Formation

Table 4-70 shows, for each instruction format that has one or more immediates, how those immediates are formed. In each equation, the symbol to the left of the equals is the assembly language name for the immediate. The symbols to the right are the field names in the instruction encoding.

Table 4-70. Immediate Formation

<table>
<thead>
<tr>
<th>Instruction Format</th>
<th>Immediate Formation</th>
</tr>
</thead>
<tbody>
<tr>
<td>A2</td>
<td>( \text{count}<em>2 = \text{ct}</em>{2d} + 1 )</td>
</tr>
<tr>
<td>A3 A8 I27 M30</td>
<td>( \text{imm}<em>8 = \text{sign_ext}(s &lt;&lt; 7 \mid \text{imm}</em>{7b}, 8) )</td>
</tr>
<tr>
<td>A4</td>
<td>( \text{imm}<em>{14} = \text{sign_ext}(s &lt;&lt; 13 \mid \text{imm}</em>{6d} &lt;&lt; 7 \mid \text{imm}_{7b}, 14) )</td>
</tr>
<tr>
<td>A5</td>
<td>( \text{imm}<em>{22} = \text{sign_ext}(s &lt;&lt; 21 \mid \text{imm}</em>{5c} &lt;&lt; 16 \mid \text{imm}<em>{9d} &lt;&lt; 7 \mid \text{imm}</em>{7b}, 22) )</td>
</tr>
<tr>
<td>A10</td>
<td>( \text{count}<em>2 = (\text{ct}</em>{2d} &gt; 2) \wedge \text{reservedQP}^b : \text{ct}_{6d} + 1 )</td>
</tr>
<tr>
<td>I1</td>
<td>( \text{count}<em>2 = (\text{ct}</em>{2d} = 0) \wedge (\text{ct}<em>{2d} = 1) \wedge (\text{ct}</em>{2d} = 2) \wedge 15 : 16 )</td>
</tr>
<tr>
<td>I3</td>
<td>( \text{mbtype}<em>4 = (\text{mbt}</em>{4c} = 0) \wedge \text{brsct} : (\text{mbt}<em>{4c} = 8) \wedge \text{mix} : (\text{mbt}</em>{4c} = 9) \wedge \text{shuf} : (\text{mbt}<em>{4c} = 0A) \wedge \text{alt} : (\text{mbt}</em>{4c} = 0B) \wedge \text{rev} : \text{reservedQP}^a )</td>
</tr>
<tr>
<td>I4</td>
<td>( \text{mhtype}<em>6 = \text{mht}</em>{6c} )</td>
</tr>
<tr>
<td>I6</td>
<td>( \text{count}<em>5 = \text{count}</em>{5b} )</td>
</tr>
<tr>
<td>I8</td>
<td>( \text{count}<em>5 = 31 - \text{count}</em>{5c} )</td>
</tr>
<tr>
<td>I10</td>
<td>( \text{count}<em>6 = \text{count}</em>{6d} )</td>
</tr>
<tr>
<td>I11</td>
<td>( \text{len}<em>6 = \text{len}</em>{6d} + 1 )</td>
</tr>
<tr>
<td></td>
<td>( \text{pos}<em>6 = \text{pos}</em>{6b} )</td>
</tr>
<tr>
<td>I12</td>
<td>( \text{len}<em>6 = \text{len}</em>{6d} + 1 )</td>
</tr>
<tr>
<td></td>
<td>( \text{pos}<em>6 = 63 - \text{cpos}</em>{6c} )</td>
</tr>
</tbody>
</table>
### Table 4-70. Immediate Formation (Continued)

<table>
<thead>
<tr>
<th>Instruction Format</th>
<th>Immediate Formation</th>
</tr>
</thead>
<tbody>
<tr>
<td>I13</td>
<td>len₉ = len₆d + 1</td>
</tr>
<tr>
<td></td>
<td>imm₈ = sign_ext(s &lt;&lt; 7</td>
</tr>
<tr>
<td>I14</td>
<td>len₉ = len₆d + 1</td>
</tr>
<tr>
<td></td>
<td>imm₁ = sign_ext(s, 1)</td>
</tr>
<tr>
<td>I15</td>
<td>len₄ = len₆d + 1</td>
</tr>
<tr>
<td>I16</td>
<td>pos₆ = pos₆b</td>
</tr>
<tr>
<td>I19 M37</td>
<td>imm₂₁ = i &lt;&lt; 20</td>
</tr>
<tr>
<td>I21</td>
<td>tag₁₃ = IP + (sign_ext(timm₉c, 9) &lt;&lt; 4)</td>
</tr>
<tr>
<td>I23</td>
<td>mask₁₇ = sign_ext(s &lt;&lt; 16</td>
</tr>
<tr>
<td>I24</td>
<td>imm₄₄ = sign_ext(s &lt;&lt; 43</td>
</tr>
<tr>
<td>M3 M8 M15</td>
<td>imm₉ = sign_ext(s &lt;&lt; 8</td>
</tr>
<tr>
<td>M5 M10</td>
<td></td>
</tr>
<tr>
<td>M17</td>
<td>inc₃ = sign_ext((s ? –1 : 1) * ((l₂b = 3) ? 1 : 1 &lt;&lt; (4 – l₂b)), 6)</td>
</tr>
<tr>
<td>I20 M20 M21</td>
<td>target₂₅ = IP + (sign_ext(s &lt;&lt; 20</td>
</tr>
<tr>
<td>M22 M23</td>
<td>target₂₅ = IP + (sign_ext(s &lt;&lt; 20</td>
</tr>
<tr>
<td>M34</td>
<td>il = sol</td>
</tr>
<tr>
<td></td>
<td>r = sor &lt;&lt; 3</td>
</tr>
<tr>
<td>M39 M40</td>
<td>imm₂ = l₂₀</td>
</tr>
<tr>
<td>M44</td>
<td>imm₂₄ = i &lt;&lt; 23</td>
</tr>
<tr>
<td>B1 B2 B3</td>
<td>target₂₅ = IP + (sign_ext(s &lt;&lt; 20</td>
</tr>
<tr>
<td>B6</td>
<td>target₂₅ = IP + (sign_ext(s &lt;&lt; 20</td>
</tr>
<tr>
<td></td>
<td>tag₁₃ = IP + (sign_ext(l₂ₑ &lt;&lt; 7</td>
</tr>
<tr>
<td>B7</td>
<td>tag₁₃ = IP + (sign_ext(l₂ₑ &lt;&lt; 7</td>
</tr>
<tr>
<td>B9</td>
<td>imm₂₁ = i &lt;&lt; 20</td>
</tr>
<tr>
<td>F5</td>
<td>fclass₉ = fclass₇c &lt;&lt; 2</td>
</tr>
<tr>
<td>F12</td>
<td>amask₇ = amask₇b</td>
</tr>
<tr>
<td>F14</td>
<td>target₂₅ = IP + (sign_ext(s &lt;&lt; 20</td>
</tr>
<tr>
<td>F15</td>
<td>imm₂₁ = i &lt;&lt; 20</td>
</tr>
<tr>
<td>X1</td>
<td>imm₆₀₂ = imm₄₁ &lt;&lt; 21</td>
</tr>
<tr>
<td>X2</td>
<td>imm₆₄ = i &lt;&lt; 63</td>
</tr>
<tr>
<td>X3 X4</td>
<td>target₉₄ = IP + (i &lt;&lt; 59</td>
</tr>
</tbody>
</table>

a. This encoding causes an Illegal Operation fault if the value of the qualifying predicate is 1.
5.1 Reading and Writing Resources

An Itanium instruction is said to be a reader of a resource if the instruction’s qualifying predicate is 1 or it has no qualifying predicate or is one of the instructions that reads a resource even when its qualifying predicate is 0, and the execution of the instruction depends on that resource.

An Itanium instruction is said to be an writer of a resource if the instruction’s qualifying predicate is 1 or it has no qualifying predicate or writes the resource even when the qualifying predicate is 0, and the execution of the instruction writes that resource.

An Itanium instruction is said to be a reader or writer of a resource even if it only sometimes depends on that resource and it cannot be determined statically whether the resource will be read or written. For example, `cover` only writes CR[IFS] when PSR.ic is 0, but for purposes of dependency, it is treated as if it always writes the resource since this condition cannot be determined statically. On the other hand, `rsm` conditionally writes several bits in the PSR depending on a mask which is encoded as an immediate in the instruction. Since the PSR bits to be written can be determined by examining the encoded instruction, the instruction is treated as only writing those bits which have a corresponding mask bit set. All exceptions to these general rules are described in this appendix.

5.2 Dependencies and Serialization

A RAW (Read-After-Write) dependency is a sequence of two events where the first is a writer of a resource and the second is a reader of the same resource. Events may be instructions, interruptions, or other ‘uses’ of the resource such as instruction stream fetches and VHPT walks. Table 5-2 covers only dependencies based on instruction readers and writers.

A WAW (Write-After-Write) dependency is a sequence of two events where both events write the resource in question. Events may be instructions, interruptions, or other ‘updates’ of the resource. Table 5-3 covers only dependencies based on instruction writers.

A WAR (Write-After-Read) dependency is a sequence of two instructions, where the first is a reader of a resource and the second is a writer of the same resource. Such dependencies are always allowed except as indicated in Table 5-4 and only those related to instruction readers and writers are included.

A RAR (Read-After-Read) dependency is a sequence of two instructions where both are readers of the same resource. Such dependencies are always allowed.

RAW and WAW dependencies are generally not allowed without some type of serialization event (an implied, data, or instruction serialization after the first writing instruction. (See “Serialization” on page 3:13 in Volume 2 for details on serialization.) The tables and associated rules in this appendix provide a comprehensive list of readers and writers of resources and describe the
serialization required for the dependency to be observed and possible outcomes if the required serialization is not met. Even when targeting code for machines which do not check for particular disallowed dependencies, such code sequences are considered architecturally undefined and may cause code to behave differently across processors, operating systems, or even separate executions of the code sequence during the same program run. In some cases, different serializations may yield different, but well-defined results.

The serialization of application level (non-privileged) resources is always implied. This means that if a writer of that resource and a subsequent read of that same resource are in different instruction groups, then the reader will see the value written. In addition, for dependencies on PRs and BRs, where the writer is a non-branch instruction and the reader is a branch instruction, the writer and reader may be in the same instruction group.

System resources generally require explicit serialization, i.e., the use of a \texttt{srлиз.i} or \texttt{srлиз.d} instruction, between the writing and the reading of that resource. Note that RAW accesses to CRs are not exceptional - they require explicit data or instruction serialization. However, in some cases (other than CRs) where pairs of instructions explicitly encode the same resource, serialization is implied.

There are cases where it is architecturally allowed to omit a serialization, and that the response from the CPU must be atomic (act as if either the old or the new state were fully in place). The tables in this appendix indicate dependency requirements under the assumption that the desired result is for the dependency to always be observed. In some such cases, the programmer may not care if the old or new state is used; such situations are allowed, but the value seen is not deterministic.

On the other hand, if an \textit{IMPLIEDF} dependency is violated, then the program is incorrectly coded and the processor’s behavior is undefined.

### 5.3 Resource and Dependency Table Format Notes

- The “Writers” and “Readers” columns of the dependency tables contain instruction class names and instruction mnemonic prefixes as given in the format section of each instruction page. To avoid ambiguity, instruction classes are shown in bold, while instruction mnemonic prefixes are in regular font. For instruction mnemonic prefixes, all instructions that exactly match the name specified or those that begin with the specified text and are followed by a `.`, and then followed by any other text will match.

- The dependency on a listed instruction is in effect no matter what values are encoded in the instruction or what dynamic values occur in operands, unless a superscript is present or one of the special case instruction rules in Section Section 5.3.1 applies. Instructions listed are still subject to rules regarding qualifying predicates.

- Instruction classes are groups of related instructions. Such names appear in boldface for clarity. The list of all instruction classes is contained in Table 5-5. Note that an instruction may appear in multiple instruction classes, instruction classes may expand to contain other classes, and that when fully expanded, a set of classes (e.g., the readers of some resource) may contain the same instruction multiple times.

- The syntax \texttt{xy} where \texttt{x} and \texttt{y} are both instruction classes, indicates an unnamed instruction class that includes all instructions in instruction class \texttt{x} but that are not in instruction class \texttt{y}.
Similarly, the notation ‘\(x'y'z'\)’ means all instructions in instruction class \(x\), but that are not in either instruction class \(y\) or instruction class \(z\).

- Resources on separate rows of a table are independent resources. This means that there are no serialization requirements for an event which references one of them followed by an event which uses a different resource. In cases where resources are broken into subrows, dependencies only apply between instructions within a subrow. Instructions that do not appear in a subrow together have no dependencies (reader/writer or writer/writer dependencies) for the resource in question, although they may still have dependencies on some other resource.

- The dependencies listed for pairs of instructions on each resource are not unique -- the same pair of instructions might also have a dependency on some other resource with a different semantics of dependency. In cases where there are multiple resource dependencies for the same pair of instructions, the most stringent semantics are assumed: instr overrides data which overrides impliedF which overrides implied which overrides none.

- Arrays of numbered resources are represented in a single row of a table using the `%` notation as a substitute for the number of the resource. In such cases, the semantics of the table are as if each numbered resource had its own row in that table and is thus an independent resource. The range of values that the `%` can take are given in the “Resource Name” column.

- An asterisk ‘*’ in the “Resource Name” column indicates that this resource may not have a physical resource associated with it, but is added to enforce special dependencies.

- A pound sign ‘#’ in the “Resource Name” column indicates that this resource is an array of resources that are indexed by a value in a GR. The number of individual elements in the array is described in the detailed description of each resource.

- The “Semantics of Dependency” column describes the outcome given various serialization and instruction group boundary conditions. The exact definition for each keyword is given in Table 5-1.

### Table 5-1. Semantics of Dependency Codes

<table>
<thead>
<tr>
<th>Semantics of Dependency Code</th>
<th>Serialization Type Required</th>
<th>Effects of Serialization Violation</th>
</tr>
</thead>
<tbody>
<tr>
<td>instr</td>
<td>Instruction Serialization (See “Instruction Serialization” on page 3:14 in Volume 2).</td>
<td>Atomic: Any attempt to read a resource after one or more insufficiently serialized writes is either the value previously in the register (before any of the unserialized writes) or the value of one of any unserialized writes. Which value is returned is unpredictable and multiple insufficiently serialized reads may see different results. No fault will be caused by the insufficient serialization.</td>
</tr>
<tr>
<td>data</td>
<td>Data Serialization (See “Data Serialization” on page 3:14 in Volume 2)</td>
<td>An undefined value is returned, or an Illegal Operation fault may be taken. If no fault is taken, the value returned is unpredictable, and may be unrelated to past writes, but will not be data which could not be accessed by the current process (e.g., if PSR.cpl != 0, the undefined value to return cannot be read from some control register).</td>
</tr>
<tr>
<td>implied</td>
<td>Instruction Group Break. Writer and reader must be in separate instruction groups. (See “Instruction Sequencing Considerations” on page 3:34 in Volume 1).</td>
<td>Stop. Writer and reader must must be separated by a stop.</td>
</tr>
<tr>
<td>impliedF</td>
<td>Instruction Group Break (same as above).</td>
<td>Stop. Writer and reader must must be separated by a stop.</td>
</tr>
<tr>
<td>none</td>
<td>None</td>
<td>N/A</td>
</tr>
<tr>
<td>specific</td>
<td>Implementation Specific</td>
<td>N/A</td>
</tr>
<tr>
<td>SC</td>
<td>Special Case</td>
<td>Described elsewhere in book, see referenced section in the entry.</td>
</tr>
</tbody>
</table>
5.3.1 Special Case Instruction Rules

The following rules apply to the specified instructions when they appear in Table 5-2, Table 5-3, Table 5-4, or Table 5-5:

- An instruction always reads a given resource if its qualifying predicate is 1 and it appears in the “Reader” column of the table (except as noted). An instruction always writes a given resource if its qualifying predicate is 1 and it appears in the “Writer” column of the table (except as noted). An instruction never reads or writes the specified resource if its qualifying predicate is 0 (except as noted). These rules include branches and their qualifying predicate. Instructions in the unpredicatable-instructions class have no qualifying predicate and thus always read or write their resources (except as noted).
- An instruction of type mov-from-PR reads all PRs if its PR[qp] is true. If the PR[qp] is false, then only the PR[qp] is read.
- An instruction of type mov-to-PR writes only those PRs as indicated by the immediate mask encoded in the instruction.
- A st8.spill only writes AR[UNAT]{X} where X equals the value in bits 8:3 of the store’s data address. A ld8.fill instruction only reads AR[UNAT]{Y} where Y equals the value in bits 8:3 of the load’s data address.
- Instructions of type mod-sched-brs always read AR[EC] and the rotating register base registers in CFM, and always write AR[EC], the rotating register bases in CFM, and PR[63] even if they do not change their values or if their PR[qp] is false.
- Instructions of type mod-sched-brs-counted always read and write AR[LC], even if they do not change its value.
- For instructions of type pr-or-writers or pr-and-writers, if their completer is or.orcm, then only the first target predicate is an or-compare and the second target predicate is an and-compare. Similarly, if their completer is and.andcm, then only the second target predicate is an or-compare and the first target predicate is an and-compare.
- rum and sum only read PSR.sp when the bit corresponding to PSR.up (bit 2) is set in the immediate field of the instruction.

5.3.2 RAW Dependency Table

Table 5-2 architecturally defines the following information:

- A list of all architecturally-defined, independently-writable resources in the Itanium architecture. Each row represents an ‘atomic’ resource. Thus, for each row in the table, hardware will probably require a separate write-enable control signal.
- For each resource, a complete list of readers and writers.
- For each instruction, a complete list of all resources read and written. Such a list can be obtained by taking the union of all the rows in which each instruction appears.
<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Readers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>ALAT</td>
<td><code>chk.a.clr, mem-readers-alat, mem-writers, invala-all</code></td>
<td><code>mem-readers-alat, mem-writers, chk-a, invala.e</code></td>
<td>none</td>
</tr>
<tr>
<td>AR[BSP]</td>
<td><code>br.call, brl.call, br.ret, cover, mov-to-AR-BSPSTORE, rfi</code></td>
<td><code>br.call, brl.call, br.ai, br.ret, cover, flushrs, loadrs, mov-from-AR-BSP, rfi</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[BSPSTORE]</td>
<td><code>alloc, loadrs, flushrs, mov-to-AR-BSPSTORE</code></td>
<td><code>alloc, br.ai, flushrs, mov-from-AR-BSPSTORE</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[CCV]</td>
<td><code>mov-to-AR-CCV</code></td>
<td><code>br.ai, cmpxchg, mov-from-AR-CCV</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[EC]</td>
<td><code>mod-sched-brs, br.ret, mov-to-AR-EC</code></td>
<td><code>br.call, brl.call, br.ai, mod-sched-brs, mov-from-AR-EC</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf0.controls</td>
<td><code>mov-to-AR-FPSR, fsetc.s0</code></td>
<td><code>br.ai, fp-arith-s0, fcmp-s0, fpcmp-s0, fsetc, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf1.controls</td>
<td><code>mov-to-AR-FPSR, fsetc.s1</code></td>
<td><code>br.ai, fp-arith-s1, fcmp-s1, fpcmp-s1, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf2.controls</td>
<td><code>mov-to-AR-FPSR, fsetc.s2</code></td>
<td><code>br.ai, fp-arith-s2, fcmp-s2, fpcmp-s2, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf3.controls</td>
<td><code>mov-to-AR-FPSR, fsetc.s3</code></td>
<td><code>br.ai, fp-arith-s3, fcmp-s3, fpcmp-s3, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf0.flags</td>
<td><code>fp-arith-s0, fclrf.s0, fcmp-s0, fpcmp-s0, mov-to-AR-FPSR</code></td>
<td><code>br.ai, fchkf, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf1.flags</td>
<td><code>fp-arith-s1, fclrf.s1, fcmp-s1, fpcmp-s1, mov-to-AR-FPSR</code></td>
<td><code>br.ai, fchkf.s1, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf2.flags</td>
<td><code>fp-arith-s2, fclrf.s2, fcmp-s2, fpcmp-s2, mov-to-AR-FPSR</code></td>
<td><code>br.ai, fchkf.s2, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf3.flags</td>
<td><code>fp-arith-s3, fclrf.s3, fcmp-s3, fpcmp-s3, mov-to-AR-FPSR</code></td>
<td><code>br.ai, fchkf.s3, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].traps</td>
<td><code>mov-to-AR-FPSR</code></td>
<td><code>br.ai, fp-arith, fchkf, fcmp, fpcmp, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].rv</td>
<td><code>mov-to-AR-FPSR</code></td>
<td><code>br.ai, fp-arith, fchkf, fcmp, fpcmp, mov-from-AR-FPSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[ITC]</td>
<td><code>mov-to-AR-ITC</code></td>
<td><code>br.ai, mov-from-AR-ITC</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[K%], % in 0 - 7</td>
<td><code>mov-to-AR-K1</code></td>
<td><code>br.ai, mov-from-AR-K1</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[PFS]</td>
<td><code>br.call, brl.call</code></td>
<td><code>alloc, br.ai, br.ret, epc, mov-from-AR-PFS</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[RNAT]</td>
<td><code>alloc, flushrs, loadrs, mov-to-AR-RNAT, mov-to-AR-BSPSTORE</code></td>
<td><code>alloc, br.ai, flushrs, loadrs, mov-from-AR-RNAT</code></td>
<td>impliedF</td>
</tr>
</tbody>
</table>
Table 5-2. RAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Readers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>AR[UNAT][%], % in 0 - 63</td>
<td>mov-to-AR-UNAT, st8.spill</td>
<td>br.ia, ld8.fill, mov-from-AR-UNAT</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[%, % in 48-63, 112-127</td>
<td>mov-to-AR-ig</td>
<td>br.ia, mov-from-AR-ig</td>
<td>impliedF</td>
</tr>
<tr>
<td>BR[%, % in 0 - 7</td>
<td>br.call, brl.call</td>
<td>indirect-brs, indirect-brp, mov-from-BR</td>
<td>impliedF</td>
</tr>
<tr>
<td>CFM</td>
<td>mod-sched-brs</td>
<td>mod-sched-brs</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td></td>
<td>cover, alloc, rfi, loads, br.ret, br.call, brl.call</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td></td>
<td>cfm-readers</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td></td>
<td>br.call, brl.call, br.ret, clr.rb, cover, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td></td>
<td>alloc</td>
<td>impliedF</td>
</tr>
<tr>
<td>CPUI#</td>
<td>none</td>
<td>mov-from-IND-CPUID</td>
<td>specific</td>
</tr>
<tr>
<td>CR[EII]</td>
<td>mov-to-CR-EII</td>
<td>none</td>
<td>SC Section 5.8.3.4 in Volume 2</td>
</tr>
<tr>
<td>CR[IFA]</td>
<td>mov-to-CR-IFA</td>
<td>ilc.i, ilc.d, ilr.i, ilr.d</td>
<td>implied</td>
</tr>
<tr>
<td></td>
<td></td>
<td>mov-from-CR-IFA</td>
<td>data</td>
</tr>
<tr>
<td>CR[IFS]</td>
<td>mov-to-CR-IFS</td>
<td>mov-from-CR-IFS</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td></td>
<td>rfi</td>
<td>implied</td>
</tr>
<tr>
<td></td>
<td></td>
<td>rfi</td>
<td>implied</td>
</tr>
</tbody>
</table>
Table 5-2. RAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Readers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>CR[IRR%], % in 0 - 3</td>
<td>mov-from-CR-IVR</td>
<td>mov-from-CR-IRR¹</td>
<td>data</td>
</tr>
<tr>
<td>CR[IVR]</td>
<td>none</td>
<td>mov-from-CR-IVR</td>
<td>SC Section 5.8.3.2 in Volume 2</td>
</tr>
<tr>
<td>CR[LRR%], % in 0 - 1</td>
<td>mov-to-CR-LRR¹</td>
<td>mov-from-CR-LRR¹</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td></td>
<td>mov-to-PSR-l, rfi, rsm, ssm</td>
<td>SC Section 5.8.3.3 in Volume 2</td>
</tr>
<tr>
<td>CR% , % in 3-7, 10-15, 18, 26-63, 75-79, 82-127</td>
<td>none</td>
<td>mov-from-CR-rv¹</td>
<td>none</td>
</tr>
<tr>
<td>DBR#</td>
<td>mov-to-IND-DBR³</td>
<td>mov-from-IND-DBR³</td>
<td>impliedF</td>
</tr>
<tr>
<td>DTC</td>
<td>ptc.e, ptc.g, ptc.ga, ptc.i, ptr.i, ptr.d, itc.i, itc.d, itr.i, itr.d</td>
<td>mem-readers, mem-writers, fc, probe-all, tak, tpa</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>itc.i, itc.d, itr.i, itr.d</td>
<td>mem-readers, mem-writers, fc, probe-all, tak, tpa</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>ptc.e, ptc.g, ptc.ga, ptc.i, ptr.i, ptr.d, itc.i, itc.d, itr.i, itr.d</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ptc.e, ptc.g, ptc.ga, ptc.i, ptr.i, ptr.d</td>
<td>none</td>
<td></td>
</tr>
<tr>
<td></td>
<td>itc.i, itc.d, itr.i, itr.d</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td>DTR</td>
<td>ltr.d</td>
<td>mem-readers, mem-writers, fc, probe-all, tak, tpa</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>ptc.g, ptc.g, ptc.i, ptr.d, ltr.d</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ptr.d</td>
<td>mem-readers, mem-writers, fc, probe-all, tak, tpa</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>ptc.g, ptc.g, ptc.i, ptr.d</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ptc.g, ptc.g, ptc.i, ptr.d</td>
<td>none</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ltr.d, ltr.d</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td>FR% , % in 0 - 1</td>
<td>none</td>
<td>fr-readers¹</td>
<td>none</td>
</tr>
<tr>
<td>FR% , % in 2 - 127</td>
<td>fr-readers¹</td>
<td>impledF</td>
<td></td>
</tr>
<tr>
<td></td>
<td>ltr.c, ltr.f</td>
<td>fr-readers¹</td>
<td></td>
</tr>
</tbody>
</table>

volume 3: Resource and Dependency Semantics
Table 5-2. RAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Readers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>GR0</td>
<td>none</td>
<td>gr-readers&lt;sup&gt;1&lt;/sup&gt;</td>
<td>none</td>
</tr>
<tr>
<td>GR%&lt;sup&gt;%&lt;/sup&gt;, % in 1 - 127</td>
<td>ld-c&lt;sup&gt;1,13&lt;/sup&gt;</td>
<td>gr-readers&lt;sup&gt;1&lt;/sup&gt;</td>
<td>impliedF</td>
</tr>
<tr>
<td>IBR#</td>
<td>mov-to-IND-IBR&lt;sup&gt;3&lt;/sup&gt;</td>
<td>mov-from-IND-IBR&lt;sup&gt;3&lt;/sup&gt;</td>
<td>impliedF</td>
</tr>
<tr>
<td>InService*</td>
<td>mov-to-CR-EOI</td>
<td>mov-from-CR-IVR</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>mov-from-CR-IVR</td>
<td>mov-from-CR-IVR</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>mov-to-CR-EOI</td>
<td>mov-to-CR-EOI</td>
<td>impliedF</td>
</tr>
<tr>
<td>IP</td>
<td>all</td>
<td>all</td>
<td>none</td>
</tr>
<tr>
<td>ITC</td>
<td>ptc.e, ptc.g, ptc.ga, ptc.l, ptr.i, ptr.d</td>
<td>epc</td>
<td>instr</td>
</tr>
<tr>
<td></td>
<td>itc.i, itc.d, ltr.i, ltr.d</td>
<td>ptr.i, ptr.d, ptc.e, ptc.g, ptc.ga, ptc.l</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>itc.i, itc.d, ltr.i, ltr.d</td>
<td>epc</td>
<td>instr</td>
</tr>
<tr>
<td></td>
<td>itc.d, itc.i, ltr.d, ltr.i, ptr.d, ptr.i, ptr.g, ptc.ga, ptc.l</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td>ITC_LIMIT*</td>
<td>ptc.g, ptc.ga</td>
<td>ptc.g, ptc.ga</td>
<td>impliedF</td>
</tr>
<tr>
<td>ITR</td>
<td>ltr.i</td>
<td>ltr.i, itc.i, ptc.g, ptc.ga, ptc.l, ptr.i</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>epc</td>
<td>epc</td>
<td>instr</td>
</tr>
<tr>
<td></td>
<td>ptr.i</td>
<td>ltc.i, ptr.i</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>ltc.g, ptc.ga, ptc.l, ptr.i</td>
<td>none</td>
<td></td>
</tr>
<tr>
<td></td>
<td>epc</td>
<td>epc</td>
<td>instr</td>
</tr>
<tr>
<td>memory</td>
<td>mem-writers</td>
<td>mem-readers</td>
<td>none</td>
</tr>
<tr>
<td>PKR#</td>
<td>mov-to-IND-PKR&lt;sup&gt;3&lt;/sup&gt;</td>
<td>mem-readers, mem-writers, mov-from-IND-PKR&lt;sup&gt;4&lt;/sup&gt;, probe-all</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>mov-to-IND-PKR&lt;sup&gt;4&lt;/sup&gt;</td>
<td>none</td>
<td></td>
</tr>
<tr>
<td></td>
<td>mov-from-IND-PKR&lt;sup&gt;3&lt;/sup&gt;</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td></td>
<td>mov-to-IND-PKR&lt;sup&gt;3&lt;/sup&gt;</td>
<td>impliedF</td>
<td></td>
</tr>
<tr>
<td>PMC#</td>
<td>mov-to-IND-PMC&lt;sup&gt;3&lt;/sup&gt;</td>
<td>mov-from-IND-PMC&lt;sup&gt;3&lt;/sup&gt;</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>mov-from-IND-PMC&lt;sup&gt;3&lt;/sup&gt;</td>
<td>SC&lt;sup&gt;3&lt;/sup&gt; Section 7.1.1 in Volume 2</td>
<td></td>
</tr>
<tr>
<td>PMD#</td>
<td>mov-to-IND-PMD&lt;sup&gt;3&lt;/sup&gt;</td>
<td>mov-from-IND-PMD&lt;sup&gt;3&lt;/sup&gt;</td>
<td>impliedF</td>
</tr>
<tr>
<td>PR0</td>
<td>pr-writers&lt;sup&gt;1&lt;/sup&gt;</td>
<td>pr-readers-br&lt;sup&gt;1&lt;/sup&gt;, pr-readers-nobr-nomovpr&lt;sup&gt;1&lt;/sup&gt;, mov-from-PR&lt;sup&gt;12&lt;/sup&gt;, mov-to-PR&lt;sup&gt;12&lt;/sup&gt;</td>
<td>none</td>
</tr>
<tr>
<td>PR%&lt;sup&gt;%&lt;/sup&gt;, % in 1 - 15</td>
<td>pr-writers&lt;sup&gt;1&lt;/sup&gt;, mov-to-PR-allreg&lt;sup&gt;7&lt;/sup&gt;</td>
<td>pr-readers-nobr-nomovpr&lt;sup&gt;1&lt;/sup&gt;, mov-from-PR, mov-to-PR&lt;sup&gt;12&lt;/sup&gt;</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>pr-writers-fp&lt;sup&gt;1&lt;/sup&gt;</td>
<td>pr-readers-br&lt;sup&gt;1&lt;/sup&gt;</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>pr-writers-int&lt;sup&gt;1&lt;/sup&gt;, mov-to-PR-allreg&lt;sup&gt;7&lt;/sup&gt;</td>
<td>pr-readers-br&lt;sup&gt;1&lt;/sup&gt;</td>
<td>none</td>
</tr>
</tbody>
</table>
Table 5-2. RAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Readers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>PR%, % in 16 - 62</td>
<td>pr-writers(^1), mov-to-PR-allreg(^7), mov-to-PR-rotreg</td>
<td>pr-readers-nobr-nomovpr(^1), mov-from-PR, mov-to-PR(^{12})</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>pr-writers-fp(^1)</td>
<td>pr-readers-br(^1)</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>pr-writers-int(^1), mov-to-PR-allreg(^7), mov-to-PR-rotreg</td>
<td>pr-readers-br(^1)</td>
<td>none</td>
</tr>
<tr>
<td>PR63</td>
<td>mod-sched-brs, pr-writers(^1), mov-to-PR-allreg(^7), mov-to-PR-rotreg</td>
<td>pr-readers-nobr-nomovpr(^1), mov-from-PR, mov-to-PR(^{12})</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>pr-writers-fp(^1), mod-sched-brs</td>
<td>pr-readers-br(^1)</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>pr-writers-int(^1), mov-to-PR-allreg(^7), mov-to-PR-rotreg</td>
<td>pr-readers-br(^1)</td>
<td>none</td>
</tr>
<tr>
<td>PSR.ac</td>
<td>user-mask-writers-partial(^7), mov-to-PSR-um</td>
<td>mem-readers, mem-writers</td>
<td>implied</td>
</tr>
<tr>
<td></td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>mem-readers, mem-writers</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>user-mask-writers-partial(^7), mov-to-PSR-um, sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>mov-from-PSR, mov-from-PSR-um</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.be</td>
<td>user-mask-writers-partial(^7), mov-to-PSR-um</td>
<td>mem-readers, mem-writers</td>
<td>implied</td>
</tr>
<tr>
<td></td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>mem-readers, mem-writers</td>
<td>data</td>
</tr>
<tr>
<td></td>
<td>user-mask-writers-partial(^7), mov-to-PSR-um, sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>mov-from-PSR, mov-from-PSR-um</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.bn</td>
<td>bsw, rfi</td>
<td>gr-readers(^{10}), gr-writers(^{10})</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.da</td>
<td>rfi</td>
<td>mem-readers, lfetch-fault, mem-writers, probe-fault</td>
<td>data</td>
</tr>
</tbody>
</table>
### Table 5-2. RAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Readers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>PSR.db</td>
<td><code>mov-to-PSR-l</code></td>
<td><code>mem-readers, mem-writers, probe-fault</code></td>
<td>data</td>
</tr>
<tr>
<td></td>
<td><code>rfi</code></td>
<td><code>mem-readers, mem-writers, mov-from-PSR, probe-fault</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.dd</td>
<td><code>rfi</code></td>
<td><code>mem-readers, probe-fault, mem-writers, lfetch-fault</code></td>
<td>data</td>
</tr>
<tr>
<td>PSR.dfh</td>
<td><code>sys-mask-writers-partial, mov-to-PSR-I, rfi</code></td>
<td><code>fr-readers, fr-writers</code></td>
<td>data</td>
</tr>
<tr>
<td></td>
<td><code>rfi</code></td>
<td><code>mem-readers, probe-fault, mem-writers, lfetch-fault</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.dfl</td>
<td><code>sys-mask-writers-partial, mov-to-PSR-I, rfi</code></td>
<td><code>fr-writers, fr-readers</code></td>
<td>data</td>
</tr>
<tr>
<td></td>
<td><code>rfi</code></td>
<td><code>mem-readers, probe-fault, mem-writers, lfetch-fault</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.di</td>
<td><code>sys-mask-writers-partial, mov-to-PSR-I, rfi</code></td>
<td><code>br.ia</code></td>
<td>data</td>
</tr>
<tr>
<td>PSR.dt</td>
<td><code>sys-mask-writers-partial, mov-to-PSR-I, rfi</code></td>
<td><code>mem-readers, mem-writers</code></td>
<td>data</td>
</tr>
<tr>
<td></td>
<td><code>rfi</code></td>
<td><code>mov-from-PSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.ed</td>
<td><code>rfi</code></td>
<td><code>lfetch-all, mem-readers-spec</code></td>
<td>data</td>
</tr>
<tr>
<td>PSR.i</td>
<td><code>sys-mask-writers-partial, mov-to-PSR-I, rfi</code></td>
<td><code>mov-from-PSR</code></td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td><code>rfi</code></td>
<td><code>mov-from-PSR</code></td>
<td>data</td>
</tr>
<tr>
<td>PSR.ia</td>
<td><code>rfi</code></td>
<td><code>none</code></td>
<td>none</td>
</tr>
<tr>
<td>PSR.id</td>
<td><code>rfi</code></td>
<td><code>none</code></td>
<td>none</td>
</tr>
<tr>
<td>PSR.is</td>
<td><code>br.ia, rfi</code></td>
<td><code>none</code></td>
<td>none</td>
</tr>
<tr>
<td>PSR.it</td>
<td><code>rfi</code></td>
<td><code>branches, mov-from-PSR, chk, epc, fchkf</code></td>
<td>data</td>
</tr>
</tbody>
</table>
Table 5-2. RAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Readers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>PSR.lp</td>
<td>mov-to-PSR-I</td>
<td>mov-from-PSR</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>rfi</td>
<td>br.ret, mov-from-PSR</td>
<td>data</td>
</tr>
<tr>
<td>PSR.mc</td>
<td>rfi</td>
<td>mov-from-PSR</td>
<td>none</td>
</tr>
<tr>
<td>PSR.mfh</td>
<td>fr-writers”, user-mask-writers-partial”, mov-to-PSR-um, sys-mask-writers-partial”, mov-to-PSR-I, rfi</td>
<td>mov-from-PSR-um, mov-from-PSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.mfl</td>
<td>fr-writers”, user-mask-writers-partial”, mov-to-PSR-um, sys-mask-writers-partial”, mov-to-PSR-I, rfi</td>
<td>mov-from-PSR-um, mov-from-PSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.pk</td>
<td>sys-mask-writers-partial”, mov-to-PSR-I, rfi</td>
<td>mem-readers, mem-writers, probe-all</td>
<td>data</td>
</tr>
<tr>
<td>PSR.pp</td>
<td>sys-mask-writers-partial”, mov-to-PSR-I, rfi</td>
<td>mov-from-PSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.ri</td>
<td>rfi</td>
<td>none</td>
<td>none</td>
</tr>
<tr>
<td>PSR.rt</td>
<td>mov-to-PSR-I</td>
<td>mov-from-PSR</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>rfi</td>
<td>alloc, flushrs, loadsrs</td>
<td>data</td>
</tr>
<tr>
<td>PSR.si</td>
<td>sys-mask-writers-partial”, mov-to-PSR-I, rfi</td>
<td>mov-from-PSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.sp</td>
<td>sys-mask-writers-partial”, mov-to-PSR-I, rfi</td>
<td>mov-from-PSR-um, rum, sum</td>
<td>data</td>
</tr>
<tr>
<td>PSR.ss</td>
<td>rfi</td>
<td>all</td>
<td>data</td>
</tr>
<tr>
<td>PSR.tb</td>
<td>mov-to-PSR-I, rfi</td>
<td>branches, chk, fchkf</td>
<td>data</td>
</tr>
<tr>
<td>PSR.up</td>
<td>user-mask-writers-partial”, mov-to-PSR-um, sys-mask-writers-partial”, mov-to-PSR-I, rfi</td>
<td>mov-from-PSR-um, mov-from-PSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>RR#</td>
<td>mov-to-IND-RR</td>
<td>mem-readers, mem-writers, itc.i, itc.d, ltr.i, ltr.d, probe-all, ptc.g, ptc.ga, ptc.l, ptr.i, ptr.d, tak, thash, tpa, ttag</td>
<td>data</td>
</tr>
<tr>
<td>RSE</td>
<td>rse-writers</td>
<td>rse-readers</td>
<td>impliedF</td>
</tr>
</tbody>
</table>
### 5.3.3 WAW Dependency Table

General rules specific to the WAW table:

- All resources require at most an instruction group break to provide sequential behavior.
- Some resources require no instruction group break to provide sequential behavior.
- There are a few special cases that are described in greater detail elsewhere in the manual and are indicated with an SC (special case) result.
- Each sub-row of writers represents a group of instructions that when taken in pairs in any combination has the dependency result indicated. If the column is split in sub-columns, then the dependency semantics apply to any pair of instructions where one is chosen from left sub-column and one is chosen from the right sub-column.

#### Table 5-3. WAW Dependencies Organized by Resource

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>ALAT</td>
<td>mem-readers-alat, mem-writers, chk.a.clr, invala-all</td>
<td>none</td>
</tr>
<tr>
<td>AR[BSP]</td>
<td>br.call, brl.call, br.ret, cover, mov-to-AR-BSPSTORE, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[BSPSTORE]</td>
<td>alloc, loadrs, flushrs, mov-to-AR-BSPSTORE</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[CCV]</td>
<td>mov-to-AR-CCV</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[EC]</td>
<td>br.ret, mod-sched-brs, mov-to-AR-EC</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf0.controls</td>
<td>mov-to-AR-FPSR, fsetc.s0</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf1.controls</td>
<td>mov-to-AR-FPSR, fsetc.s1</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf2.controls</td>
<td>mov-to-AR-FPSR, fsetc.s2</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf3.controls</td>
<td>mov-to-AR-FPSR, fsetc.s3</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].sf0.flags</td>
<td>fp-arith-s0, fcmp-s0, fcmpeq-s0</td>
<td>none</td>
</tr>
<tr>
<td>AR[FPSR].sf1.flags</td>
<td>fp-arith-s1, fcmp-s1, fcmpeq-s1</td>
<td>none</td>
</tr>
<tr>
<td>AR[FPSR].sf2.flags</td>
<td>fp-arith-s2, fcmp-s2, fcmpeq-s2</td>
<td>none</td>
</tr>
<tr>
<td>AR[FPSR].sf3.flags</td>
<td>fp-arith-s3, fcmp-s3, fcmpeq-s3</td>
<td>none</td>
</tr>
<tr>
<td>AR[FPSR].rv</td>
<td>mov-to-AR-FPSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[FPSR].traps</td>
<td>mov-to-AR-FPSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[ITC]</td>
<td>mov-to-AR-ITC</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[K%], % in 0 - 7</td>
<td>mov-to-AR-K</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[LC]</td>
<td>mod-sched-brs-counted, mov-to-AR-LC</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[PS]</td>
<td>br.call, brl.call</td>
<td>none</td>
</tr>
<tr>
<td>AR[RSC]</td>
<td>mov-to-AR-RSC</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR[RNAT]</td>
<td>alloc, flushrs, loadrs, mov-to-AR-RNAT, mov-to-AR-BSPSTORE</td>
<td>impliedF</td>
</tr>
</tbody>
</table>
Table 5-3. WAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>AR(UNAT)[%], % in 0 - 63</td>
<td>mov-to-AR-UNAT, st8.spill</td>
<td>impliedF</td>
</tr>
<tr>
<td>AR%, % in 48 - 63, 112-127</td>
<td>mov-to-AR-lg¹</td>
<td>impliedF</td>
</tr>
<tr>
<td>BR%, % in 0 - 7</td>
<td>br.call¹, brl.call¹; mov-to-BR¹</td>
<td>impliedF</td>
</tr>
<tr>
<td>CFM</td>
<td>mod-sched-brs, br.call, brl.call, br.ret, alloc, clr.rb, cover, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>CPUID#</td>
<td>none</td>
<td>none</td>
</tr>
<tr>
<td>CR[CMCV]</td>
<td>mov-to-CR-CMCV</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[DCR]</td>
<td>mov-to-CR-DCR</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[EOI]</td>
<td>mov-to-CR-EOI</td>
<td>SC Section 5.8.3.4 in Volume 2</td>
</tr>
<tr>
<td>CR[GPTA]</td>
<td>mov-to-CR-GPTA</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IFA]</td>
<td>mov-to-CR-IFA</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IFS]</td>
<td>mov-to-CR-IFS, cover</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IHA]</td>
<td>mov-to-CR-IHA</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IIM]</td>
<td>mov-to-CR-IIM</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IIP]</td>
<td>mov-to-CR-IIP</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IIPA]</td>
<td>mov-to-CR-IIPA</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IPSR]</td>
<td>mov-to-CR-IPSR</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IRR%], % in 0 - 3</td>
<td>mov-from-CR-IVR</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[ISR]</td>
<td>mov-to-CR-ISR</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[ITIR]</td>
<td>mov-to-CR-ITIR</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[ITM]</td>
<td>mov-to-CR-ITM</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[ITV]</td>
<td>mov-to-CR-ITV</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IVA]</td>
<td>mov-to-CR-IVA</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[IVR]</td>
<td>none</td>
<td>SC</td>
</tr>
<tr>
<td>CR[LID]</td>
<td>mov-to-CR-LID</td>
<td>SC</td>
</tr>
<tr>
<td>CR[LRR%], % in 0 - 1</td>
<td>mov-to-CR-LRR¹</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[PMV]</td>
<td>mov-to-CR-PMV</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[PTA]</td>
<td>mov-to-CR-PTA</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR[TPR]</td>
<td>mov-to-CR-TPR</td>
<td>impliedF</td>
</tr>
<tr>
<td>CR%, % in 3-7, 10-15, 18, 26-63, 75-79, 82-127</td>
<td>none</td>
<td>none</td>
</tr>
<tr>
<td>DBR#</td>
<td>mov-to-IND-DBR³</td>
<td>impliedF</td>
</tr>
</tbody>
</table>
Table 5-3. WAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>DTC</td>
<td>ptc.e, ptc.g, ptc.ga, ptc.l, ptr.i, ptr.d</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>ptc.e, ptc.g, ptc.ga, ptc.l, ptr.i, ptr.d, itc.i, itc.d, ltr.i, ltr.d</td>
<td>itc.i, itc.d, ltr.i, ltr.d</td>
</tr>
<tr>
<td>DTC_LIMIT*</td>
<td>ptc.g, ptc.ga</td>
<td>impliedF</td>
</tr>
<tr>
<td>DTR</td>
<td>ltr.d</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>ltr.d</td>
<td>ptr.d</td>
</tr>
<tr>
<td></td>
<td>ptr.d</td>
<td>none</td>
</tr>
<tr>
<td>FR%, % in 0 - 1</td>
<td>none</td>
<td>none</td>
</tr>
<tr>
<td>FR%, % in 2 - 127</td>
<td>fr-writers', ldf-c', ldfp-c'</td>
<td>impliedF</td>
</tr>
<tr>
<td>GR0</td>
<td>none</td>
<td>none</td>
</tr>
<tr>
<td>GR%, % in 1 - 127</td>
<td>ld-c', gr-writers'</td>
<td>impliedF</td>
</tr>
<tr>
<td>IBR#</td>
<td>mov-to-IND-IBR^3</td>
<td>impliedF</td>
</tr>
<tr>
<td>InService*</td>
<td>mov-to-CR-EOI, mov-from-CR-IVR</td>
<td>SC</td>
</tr>
<tr>
<td>IP</td>
<td>all</td>
<td>none</td>
</tr>
<tr>
<td>ITC</td>
<td>ptc.e, ptc.g, ptc.ga, ptc.l, ptr.i, ptr.d</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>ptc.e, ptc.g, ptc.ga, ptc.l, ptr.i, ptr.d, itc.i, itc.d, ltr.i, ltr.d</td>
<td>itc.i, itc.d, ltr.i, ltr.d</td>
</tr>
<tr>
<td>ITR</td>
<td>ltr.i</td>
<td>ltr.i, ptr.i</td>
</tr>
<tr>
<td></td>
<td>ptr.i</td>
<td>none</td>
</tr>
<tr>
<td>memory</td>
<td>mem-writers</td>
<td>none</td>
</tr>
<tr>
<td>PKR#</td>
<td>mov-to-IND-PKR^4</td>
<td>mov-to-IND-PKR^4</td>
</tr>
<tr>
<td></td>
<td>mov-to-IND-PKR^d</td>
<td>impliedF</td>
</tr>
<tr>
<td>PMC#</td>
<td>mov-to-IND-PMC^3</td>
<td>impliedF</td>
</tr>
<tr>
<td>PMD#</td>
<td>mov-to-IND-PMD^3</td>
<td>impliedF</td>
</tr>
<tr>
<td>PR0</td>
<td>pr-writers'</td>
<td>none</td>
</tr>
<tr>
<td>PR%, % in 1 - 15</td>
<td>pr-and-writers'</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>pr-or-writers'</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>pr-unc-writers-fp^1</td>
<td>pr-unc-writers-fp^1</td>
</tr>
<tr>
<td></td>
<td>pr-unc-writers-int^1</td>
<td>pr-unc-writers-int^1</td>
</tr>
<tr>
<td></td>
<td>pr-norm-writers-fp^1</td>
<td>pr-norm-writers-fp^1</td>
</tr>
<tr>
<td></td>
<td>pr-norm-writers-int^1</td>
<td>pr-norm-writers-int^1</td>
</tr>
<tr>
<td></td>
<td>pr-and-writers'^1</td>
<td>pr-and-writers'^1</td>
</tr>
<tr>
<td></td>
<td>mov-to-PR-allreg^7</td>
<td>mov-to-PR-allreg^7</td>
</tr>
<tr>
<td>PR%, % in 16 - 62</td>
<td>pr-and-writers'</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>pr-or-writers'</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>pr-unc-writers-fp^1</td>
<td>pr-unc-writers-fp^1</td>
</tr>
<tr>
<td></td>
<td>pr-unc-writers-int^1</td>
<td>pr-unc-writers-int^1</td>
</tr>
<tr>
<td></td>
<td>pr-norm-writers-fp^1</td>
<td>pr-norm-writers-fp^1</td>
</tr>
<tr>
<td></td>
<td>pr-norm-writers-int^1</td>
<td>pr-norm-writers-int^1</td>
</tr>
<tr>
<td></td>
<td>pr-and-writers'</td>
<td>pr-and-writers'</td>
</tr>
<tr>
<td></td>
<td>mov-to-PR-allreg^7</td>
<td>mov-to-PR-allreg^7</td>
</tr>
<tr>
<td></td>
<td>mov-to-PR-rotreg</td>
<td>mov-to-PR-rotreg</td>
</tr>
</tbody>
</table>
Table 5-3. WAW Dependencies Organized by Resource (Continued)

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Writers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>PR63</td>
<td>pr-and-writers(^1)</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>pr-or-writers(^1)</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>mod-sched-brs,</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>pr-unc-writers-fp(^1),</td>
<td></td>
</tr>
<tr>
<td></td>
<td>pr-unc-writers-int(^1),</td>
<td></td>
</tr>
<tr>
<td></td>
<td>pr-norm-writers-fp(^1),</td>
<td></td>
</tr>
<tr>
<td></td>
<td>pr-norm-writers-int(^1),</td>
<td></td>
</tr>
<tr>
<td></td>
<td>pr-and-writers(^1),</td>
<td></td>
</tr>
<tr>
<td></td>
<td>mov-to-PR-allreg(^7),</td>
<td></td>
</tr>
<tr>
<td></td>
<td>mov-to-PR-rotreg</td>
<td></td>
</tr>
<tr>
<td>PSR.ac</td>
<td>user-mask-writers-partial(^7), mov-to-PSR-um, sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.be</td>
<td>user-mask-writers-partial(^7), mov-to-PSR-um, sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.bn</td>
<td>bsw, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.cpl</td>
<td>epc, br.ret, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.da</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.db</td>
<td>mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.dd</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.dfh</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.dfl</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.dli</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.dlt</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.ed</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.i</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.ia</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.ic</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.id</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.is</td>
<td>br,ia, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.it</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.lp</td>
<td>mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.mc</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.mfh</td>
<td>fr-writers(^9)</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>user-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.mfl</td>
<td>fr-writers(^9)</td>
<td>none</td>
</tr>
<tr>
<td></td>
<td>user-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td></td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.pk</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.pp</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.ri</td>
<td>rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.rt</td>
<td>mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
<tr>
<td>PSR.si</td>
<td>sys-mask-writers-partial(^7), mov-to-PSR-I, rfi</td>
<td>impliedF</td>
</tr>
</tbody>
</table>
5.3.4 WAR Dependency Table

A general rule specific to the WAR table:

1. WAR dependencies are always allowed within instruction groups except for the entry in Table 5-4 below. The readers and subsequent writers specified must be separated by a stop in order to have defined behavior.

Table 5-4. WAR Dependencies Organized by Resource

<table>
<thead>
<tr>
<th>Resource Name</th>
<th>Readers</th>
<th>Writers</th>
<th>Semantics of Dependency</th>
</tr>
</thead>
<tbody>
<tr>
<td>PR63</td>
<td>pr-readers-br(^1)</td>
<td>mod-sched-brs</td>
<td>stop</td>
</tr>
</tbody>
</table>

5.3.5 Listing of Rules Referenced in Dependency Tables

The following rules restrict the specific instances in which some of the instructions in the tables cause a dependency and must be applied where referenced to correctly interpret those entries. Rules only apply to the instance of the instruction class, or instruction mnemonic prefix where the rule is referenced as a superscript. If the rule is referenced in Table 5-5 where instruction classes are defined, then it applies to all instances of the instruction class.

Rule 1. These instructions only write a register when that register’s number is explicitly encoded as a target of the instruction and is only read when it is encoded as a source of the instruction (or encoded as its PR\([qp]\)).

Rule 2. These instructions only read CFM when they access a rotating GR, FR, or PR. mov-to-PR and mov-from-PR only access CFM when their qualifying predicate is in the rotating region.

Rule 3. These instructions use a general register value to determine the specific indirect register accessed. These instructions only access the register resource specified by the value in bits \(\{7:0\}\) of the dynamic value of the index register.

Rule 4. These instructions only read the given resource when bits \(\{7:0\}\) of the indirect index register value does not match the register number of the resource.

Rule 5. All rules are implementation specific.

Rule 6. There is a dependency only when both the index specified by the reader and the index specified by the writer have the same value in bits \(\{63:61\}\).
Rule 7. These instructions access the specified resource only when the corresponding mask bit is set.

Rule 8. PSR.dfh is only read when these instructions reference FR32-127. PSR.dfl is only read when these instructions reference FR2-31.

Rule 9. PSR.mfl is only written when these instructions write FR2-31. PSR.mfh is only written when these instructions write FR32-127.

Rule 10. The PSR.bn bit is only accessed when one of GR16-31 is specified in the instruction.

Rule 11. The target predicates are written independently of PR[qp], but source registers are only read if PR[qp] is true.

Rule 12. This instruction only reads the specified predicate register when that register is the PR[qp].

Rule 13. This reference to ld-c only applies to the GR whose value is loaded with data returned from memory, not the post-incremented address register. Thus, a stop is still required between a post-incrementing ld-c and a consumer that reads the post-incremented GR.

Rule 14. The RSE resource includes implementation-specific internal state. At least one (and possibly more) of these resources are read by each instruction listed in the rse-readers class. At least one (and possibly more) of these resources are written by each instruction listed in the rse-writers class. To determine exactly which instructions read or write each individual resource, see the corresponding instruction pages.

Rule 15. This class represents all instructions marked as Reserved if PR[qp] is 1 B-type instructions as described in “Format Summary” on page 3:258.

Rule 16. This class represents all instructions marked as Reserved if PR[qp] is 1 instructions as described in “Format Summary” on page 3:258.

5.4 Support Tables

Table 5-5. Instruction Classes

<table>
<thead>
<tr>
<th>Class</th>
<th>Events/Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>all</td>
<td>predicatable-instructions, unpredicticable-instructions</td>
</tr>
<tr>
<td>branches</td>
<td>indirect-brs, ip-rel-brs</td>
</tr>
<tr>
<td>cfm-readers</td>
<td>fr-readers, fr-writers, gr-readers, gr-writers, mod-sched-brs, predicatable-instructions, pr-writers, alloc, br.call, brl.call, br.ret, cover, loadrs, rfi, chk-a, invala.e</td>
</tr>
<tr>
<td>chk-a</td>
<td>chk.a.clr, chk.a.nc</td>
</tr>
<tr>
<td>cmpxchg</td>
<td>cmpxchg1, cmpxchg2, cmpxchg4, cmpxchg8</td>
</tr>
<tr>
<td>czx</td>
<td>czx1, czx2</td>
</tr>
<tr>
<td>fcmp-s0</td>
<td>fcmp[Field(sf)==s0]</td>
</tr>
<tr>
<td>fcmp-s1</td>
<td>fcmp[Field(sf)==s1]</td>
</tr>
<tr>
<td>fcmp-s2</td>
<td>fcmp[Field(sf)==s2]</td>
</tr>
<tr>
<td>fcmp-s3</td>
<td>fcmp[Field(sf)==s3]</td>
</tr>
<tr>
<td>fetchadd</td>
<td>fetchadd4, fetchadd8</td>
</tr>
</tbody>
</table>
Table 5-5. Instruction Classes (Continued)

<table>
<thead>
<tr>
<th>Class</th>
<th>Events/Instructions</th>
</tr>
</thead>
</table>
| fp-arith       | fadd, famax, famin, fcvt.fx, fcvt.fxu, fcvt.xuf, fma, fmax, fmin, fmpy, fnma, fnmpy, fnorm, fpmax, fpmmin, fpmx, fpmx, fpmx, fpmx, fpmx, fpmx, fnmnp, fpmnp, fpmnp, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnasc, fnas...
Table 5-5. Instruction Classes (Continued)

<table>
<thead>
<tr>
<th>Class</th>
<th>Events/Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>ldfp-c</td>
<td>ldfp-c-nc, ldfp-c-clr</td>
</tr>
<tr>
<td>ldfp-c-clr</td>
<td>ldfps.c.clr, ldfpd.c.clr, ldfp8.c.clr</td>
</tr>
<tr>
<td>ldfp-c-nc</td>
<td>ldfps.c.nc, ldfpd.c.nc, ldfp8.c.nc</td>
</tr>
<tr>
<td>ldfp-s</td>
<td>ldfps.s, ldfpd.s, ldfp8.s</td>
</tr>
<tr>
<td>ldfp-sa</td>
<td>ldfps.sa, ldfpd.sa, ldfp8.sa</td>
</tr>
<tr>
<td>llfetch-all</td>
<td>llfetch</td>
</tr>
<tr>
<td>llfetch-fault</td>
<td>llfetch[Field(lftype)==fault]</td>
</tr>
<tr>
<td>llfetch-nofault</td>
<td>llfetch[Field(lftype)==]</td>
</tr>
<tr>
<td>llfetch-postinc</td>
<td>llfetch[Format in {M14 M15}]</td>
</tr>
<tr>
<td>mem-readers</td>
<td>mem-readers-fp, mem-readers-int</td>
</tr>
<tr>
<td>mem-readers-alat</td>
<td>ld-a, ldf-a, ldf-a, ld-sa, ldfp-sa, ldfp-sa, ld-c, ldfp-c</td>
</tr>
<tr>
<td>mem-readers-fp</td>
<td>idf, ldfp</td>
</tr>
<tr>
<td>mem-readers-int</td>
<td>cmpxchg, fetchadd, xchg, ld</td>
</tr>
<tr>
<td>mem-readers-spec</td>
<td>ld-s, ld-sa, ldf-s, ldfp-s, ldfp-sa</td>
</tr>
<tr>
<td>mem-writers</td>
<td>mem-writers-fp, mem-writers-int</td>
</tr>
<tr>
<td>mem-writers-fp</td>
<td>stf</td>
</tr>
<tr>
<td>mem-writers-int</td>
<td>cmpxchg, fetchadd, xchg, st</td>
</tr>
<tr>
<td>mix</td>
<td>mix1, mix2, mix4</td>
</tr>
<tr>
<td>mod-sched-brs</td>
<td>br.cexit, br.ctop, br.wexit, br.wtop</td>
</tr>
<tr>
<td>mod-sched-brs-counted</td>
<td>br.cexit, br.cloop, br.ctop</td>
</tr>
<tr>
<td>mov-from-AR-BSP</td>
<td>mov-from-AR-M[Field(ar3) == BSP]</td>
</tr>
<tr>
<td>mov-from-AR-BSPSTORE</td>
<td>mov-from-AR-M[Field(ar3) == BSPSTORE]</td>
</tr>
<tr>
<td>mov-from-AR-CCV</td>
<td>mov-from-AR-M[Field(ar3) == CCV]</td>
</tr>
<tr>
<td>mov-from-AR-EC</td>
<td>mov-from-AR-M[Field(ar3) == EC]</td>
</tr>
<tr>
<td>mov-from-AR-FPSR</td>
<td>mov-from-AR-M[Field(ar3) == FPSR]</td>
</tr>
<tr>
<td>mov-from-AR-I</td>
<td>mov_ar[Format in {I28}]</td>
</tr>
<tr>
<td>mov-from-AR-ig</td>
<td>mov-from-AR-IM[Field(ar3) in {48-63 112-127}]</td>
</tr>
<tr>
<td>mov-from-AR-IM</td>
<td>mov_ar[Format in {I28 M31}]</td>
</tr>
<tr>
<td>mov-from-AR-ITC</td>
<td>mov-from-AR-M[Field(ar3) == ITC]</td>
</tr>
<tr>
<td>mov-from-AR-K</td>
<td>mov-from-AR-M[Field(ar3) in {K0 K1 K2 K3 K4 K5 K6 K7}]</td>
</tr>
<tr>
<td>mov-from-AR-LC</td>
<td>mov-from-AR-M[Field(ar3) == LC]</td>
</tr>
<tr>
<td>mov-from-AR-M</td>
<td>mov_ar[Format in {M31}]</td>
</tr>
<tr>
<td>mov-from-AR-PFS</td>
<td>mov-from-AR-M[Field(ar3) == PFS]</td>
</tr>
<tr>
<td>mov-from-AR-RNAT</td>
<td>mov-from-AR-M[Field(ar3) == RNAT]</td>
</tr>
<tr>
<td>mov-from-AR-RSC</td>
<td>mov-from-AR-M[Field(ar3) == RSC]</td>
</tr>
<tr>
<td>mov-from-AR-rv</td>
<td>none</td>
</tr>
<tr>
<td>mov-from-AR-UNAT</td>
<td>mov-from-AR-M[Field(ar3) == UNAT]</td>
</tr>
<tr>
<td>mov-from-BR</td>
<td>mov_br[Format in {I22}]</td>
</tr>
<tr>
<td>mov-from-CR</td>
<td>mov_cr[Format in {M33}]</td>
</tr>
<tr>
<td>mov-from-CR-CMCV</td>
<td>mov-from-CR[Field(cr3) == CMCV]</td>
</tr>
<tr>
<td>mov-from-CR-DCR</td>
<td>mov-from-CR[Field(cr3) == DCR]</td>
</tr>
<tr>
<td>mov-from-CR-EOI</td>
<td>mov-from-CR[Field(cr3) == EOI]</td>
</tr>
<tr>
<td>mov-from-CR-GPTA</td>
<td>mov-from-CR[Field(cr3) == GPTA]</td>
</tr>
<tr>
<td>mov-from-CR-IFA</td>
<td>mov-from-CR[Field(cr3) == IFA]</td>
</tr>
</tbody>
</table>
Table 5-5. Instruction Classes (Continued)

<table>
<thead>
<tr>
<th>Class</th>
<th>Events/Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov-from-CR-IFS</td>
<td>mov-from-CR[Field(cr3) == IFS]</td>
</tr>
<tr>
<td>mov-from-CR-IHA</td>
<td>mov-from-CR[Field(cr3) == IHA]</td>
</tr>
<tr>
<td>mov-from-CR-IIM</td>
<td>mov-from-CR[Field(cr3) == IIM]</td>
</tr>
<tr>
<td>mov-from-CR-IIP</td>
<td>mov-from-CR[Field(cr3) == IIP]</td>
</tr>
<tr>
<td>mov-from-CR-IIPA</td>
<td>mov-from-CR[Field(cr3) == IPA]</td>
</tr>
<tr>
<td>mov-from-CR-IPSR</td>
<td>mov-from-CR[Field(cr3) == IPSR]</td>
</tr>
<tr>
<td>mov-from-CR-IRR</td>
<td>mov-from-CR[Field(cr3) in IRR0 IRR1 IRR2 IRR3]</td>
</tr>
<tr>
<td>mov-from-CR-ISR</td>
<td>mov-from-CR[Field(cr3) == ISR]</td>
</tr>
<tr>
<td>mov-from-CR-ITIR</td>
<td>mov-from-CR[Field(cr3) == ITIR]</td>
</tr>
<tr>
<td>mov-from-CR-ITM</td>
<td>mov-from-CR[Field(cr3) == ITM]</td>
</tr>
<tr>
<td>mov-from-CR-ITV</td>
<td>mov-from-CR[Field(cr3) == ITV]</td>
</tr>
<tr>
<td>mov-from-CR-IVA</td>
<td>mov-from-CR[Field(cr3) == IVA]</td>
</tr>
<tr>
<td>mov-from-CR-IVR</td>
<td>mov-from-CR[Field(cr3) == IVR]</td>
</tr>
<tr>
<td>mov-from-CR-LID</td>
<td>mov-from-CR[Field(cr3) == LID]</td>
</tr>
<tr>
<td>mov-from-CR-LRR</td>
<td>mov-from-CR[Field(cr3) in LRR0 LRR1]</td>
</tr>
<tr>
<td>mov-from-CR-PMV</td>
<td>mov-from-CR[Field(cr3) == PMV]</td>
</tr>
<tr>
<td>mov-from-CR-PTA</td>
<td>mov-from-CR[Field(cr3) == PTA]</td>
</tr>
<tr>
<td>mov-from-CR-rv</td>
<td>none</td>
</tr>
<tr>
<td>mov-from-CR-TPR</td>
<td>mov-from-CR[Field(cr3) == TPR]</td>
</tr>
<tr>
<td>mov-from-IND</td>
<td>mov_indirect[Format in {M43}]</td>
</tr>
<tr>
<td>mov-from-IND-CPUID</td>
<td>mov-from-IND[Field(ireg) == cpuid]</td>
</tr>
<tr>
<td>mov-from-IND-DBR</td>
<td>mov-from-IND[Field(ireg) == dbr]</td>
</tr>
<tr>
<td>mov-from-IND-IBR</td>
<td>mov-from-IND[Field(ireg) == ibr]</td>
</tr>
<tr>
<td>mov-from-IND-PKR</td>
<td>mov-from-IND[Field(ireg) == prk]</td>
</tr>
<tr>
<td>mov-from-IND-PMC</td>
<td>mov-from-IND[Field(ireg) == pmc]</td>
</tr>
<tr>
<td>mov-from-IND-PMD</td>
<td>mov-from-IND[Field(ireg) == pmd]</td>
</tr>
<tr>
<td>mov-from-IND-priv</td>
<td>mov-from-IND[Field(ireg) in dbr ibr prk pmc ri]</td>
</tr>
<tr>
<td>mov-from-IND-RR</td>
<td>mov-from-IND[Field(ireg) == ri]</td>
</tr>
<tr>
<td>mov-from-PR</td>
<td>mov_pr[Format in {I25}]</td>
</tr>
<tr>
<td>mov-from-PSR</td>
<td>mov_psrs[Format in {M36}]</td>
</tr>
<tr>
<td>mov-from-PSR-um</td>
<td>mov_um[Format in {M36}]</td>
</tr>
<tr>
<td>mov-ip</td>
<td>mov_ip[Format in {I25}]</td>
</tr>
<tr>
<td>mov-to-AR</td>
<td>mov-to-AR-I, mov-to-AR-I</td>
</tr>
<tr>
<td>mov-to-AR-BSP</td>
<td>mov-to-AR-M[Field(ar3) == BSP]</td>
</tr>
<tr>
<td>mov-to-AR-BSPSTORE</td>
<td>mov-to-AR-M[Field(ar3) == BSPSTORE]</td>
</tr>
<tr>
<td>mov-to-AR-CCV</td>
<td>mov-to-AR-M[Field(ar3) == CCV]</td>
</tr>
<tr>
<td>mov-to-AR-EC</td>
<td>mov-to-AR-I[Field(ar3) == EC]</td>
</tr>
<tr>
<td>mov-to-AR-FPSR</td>
<td>mov-to-AR-M[Field(ar3) == FPCR]</td>
</tr>
<tr>
<td>mov-to-AR-gr</td>
<td>mov-to-AR-M[Format in {I26}, mov-to-AR-I[Format in {I26]}]</td>
</tr>
<tr>
<td>mov-to-AR-I</td>
<td>mov_ar[Format in {I26, I25}]</td>
</tr>
<tr>
<td>mov-to-AR-IM</td>
<td>mov-to-AR-I[Field(ar3) in {48-63 112-127}]</td>
</tr>
<tr>
<td>mov-to-AR-ITC</td>
<td>mov-to-AR-M[Field(ar3) == ITC]</td>
</tr>
<tr>
<td>mov-to-AR-K</td>
<td>mov-to-AR-M[Field(ar3) in {K0 K1 K2 K3 K4 K5 K6 K7}]</td>
</tr>
<tr>
<td>mov-to-AR-LC</td>
<td>mov-to-AR-I[Field(ar3) == LC]</td>
</tr>
</tbody>
</table>
### Table 5-5. Instruction Classes (Continued)

<table>
<thead>
<tr>
<th>Class</th>
<th>Events/Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>mov-to-AR-M</td>
<td>mov_ar[Format in {M29 M30}]</td>
</tr>
<tr>
<td>mov-to-AR-PFS</td>
<td>mov-to-AR-[Field(ar3) == PFS]</td>
</tr>
<tr>
<td>mov-to-AR-RNAT</td>
<td>mov-to-AR-M[Field(ar3) == RNAT]</td>
</tr>
<tr>
<td>mov-to-AR-RSC</td>
<td>mov-to-AR-M[Field(ar3) == RSC]</td>
</tr>
<tr>
<td>mov-to-AR-UNAT</td>
<td>mov-to-AR-M[Field(ar3) == UNAT]</td>
</tr>
<tr>
<td>mov-to-BR</td>
<td>mov_br[Format in {I21}]</td>
</tr>
<tr>
<td>mov-to-CR</td>
<td>mov_cr[Format in {M32}]</td>
</tr>
<tr>
<td>mov-to-CR-CMCV</td>
<td>mov-to-CR[Field(cr3) == CMCV]</td>
</tr>
<tr>
<td>mov-to-CR-DCR</td>
<td>mov-to-CR[Field(cr3) == DCR]</td>
</tr>
<tr>
<td>mov-to-CR-EOI</td>
<td>mov-to-CR[Field(cr3) == EOI]</td>
</tr>
<tr>
<td>mov-to-CR-GPTA</td>
<td>mov-to-CR[Field(cr3) == GPTA]</td>
</tr>
<tr>
<td>mov-to-CR-IFA</td>
<td>mov-to-CR[Field(cr3) == IFA]</td>
</tr>
<tr>
<td>mov-to-CR-IFS</td>
<td>mov-to-CR[Field(cr3) == IFS]</td>
</tr>
<tr>
<td>mov-to-CR-IHA</td>
<td>mov-to-CR[Field(cr3) == IHA]</td>
</tr>
<tr>
<td>mov-to-CR-IIM</td>
<td>mov-to-CR[Field(cr3) == IIM]</td>
</tr>
<tr>
<td>mov-to-CR-IIP</td>
<td>mov-to-CR[Field(cr3) == IIP]</td>
</tr>
<tr>
<td>mov-to-CR-IIPA</td>
<td>mov-to-CR[Field(cr3) == IIPA]</td>
</tr>
<tr>
<td>mov-to-CR-IPSR</td>
<td>mov-to-CR[Field(cr3) == IPSR]</td>
</tr>
<tr>
<td>mov-to-CR-IRR</td>
<td>mov-to-CR[Field(cr3) in {IRR0 IRR1 IRR2 IRR3}]</td>
</tr>
<tr>
<td>mov-to-CR-ISR</td>
<td>mov-to-CR[Field(cr3) == ISR]</td>
</tr>
<tr>
<td>mov-to-CR-ITIR</td>
<td>mov-to-CR[Field(cr3) == ITIR]</td>
</tr>
<tr>
<td>mov-to-CR-ITM</td>
<td>mov-to-CR[Field(cr3) == ITM]</td>
</tr>
<tr>
<td>mov-to-CR-ITV</td>
<td>mov-to-CR[Field(cr3) == ITV]</td>
</tr>
<tr>
<td>mov-to-CR-IVA</td>
<td>mov-to-CR[Field(cr3) == IVA]</td>
</tr>
<tr>
<td>mov-to-CR-IVR</td>
<td>mov-to-CR[Field(cr3) == IVR]</td>
</tr>
<tr>
<td>mov-to-CR-LID</td>
<td>mov-to-CR[Field(cr3) == LID]</td>
</tr>
<tr>
<td>mov-to-CR-LRR</td>
<td>mov-to-CR[Field(cr3) in {LRR0 LRR1}]</td>
</tr>
<tr>
<td>mov-to-CR-PMV</td>
<td>mov-to-CR[Field(cr3) == PMV]</td>
</tr>
<tr>
<td>mov-to-CR-PTA</td>
<td>mov-to-CR[Field(cr3) == PTA]</td>
</tr>
<tr>
<td>mov-to-CR-TPR</td>
<td>mov-to-CR[Field(cr3) == TPR]</td>
</tr>
<tr>
<td>mov-to-IND</td>
<td>mov_indirect[Format in {M42}]</td>
</tr>
<tr>
<td>mov-to-IND-CPUID</td>
<td>mov-to-IND[Field(reg) == cpuid]</td>
</tr>
<tr>
<td>mov-to-IND-DBR</td>
<td>mov-to-IND[Field(reg) == dbr]</td>
</tr>
<tr>
<td>mov-to-IND-IBR</td>
<td>mov-to-IND[Field(reg) == ibr]</td>
</tr>
<tr>
<td>mov-to-IND-PKR</td>
<td>mov-to-IND[Field(reg) == pkr]</td>
</tr>
<tr>
<td>mov-to-IND-PMC</td>
<td>mov-to-IND[Field(reg) == pmc]</td>
</tr>
<tr>
<td>mov-to-IND-PMD</td>
<td>mov-to-IND[Field(reg) == pmd]</td>
</tr>
<tr>
<td>mov-to-IND-priv</td>
<td>mov-to-IND</td>
</tr>
<tr>
<td>mov-to-IND-RR</td>
<td>mov-to-IND[Field(reg) == rr]</td>
</tr>
<tr>
<td>mov-to-PR</td>
<td>mov-to-PR-allreg, mov-to-PR-rotreg</td>
</tr>
<tr>
<td>mov-to-PR-allreg</td>
<td>mov_pr[Format in {I23}]</td>
</tr>
<tr>
<td>mov-to-PR-rotreg</td>
<td>mov_pr[Format in {I24}]</td>
</tr>
<tr>
<td>mov-to-PSR-I</td>
<td>mov_psr[Format in {M35}]</td>
</tr>
<tr>
<td>mov-to-PSR-um</td>
<td>mov_um[Format in {M35}]</td>
</tr>
<tr>
<td>mux</td>
<td>mux1, mux2</td>
</tr>
</tbody>
</table>
Table 5-5. Instruction Classes (Continued)

<table>
<thead>
<tr>
<th>Class</th>
<th>Events/Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>none</td>
<td>-</td>
</tr>
<tr>
<td>pack</td>
<td>pack2, pack4</td>
</tr>
<tr>
<td>padd</td>
<td>padd1, padd2, padd4</td>
</tr>
<tr>
<td>pavg</td>
<td>pavg1, pavg2</td>
</tr>
<tr>
<td>pavgsub</td>
<td>pavgsub1, pavgsub2</td>
</tr>
<tr>
<td>pcmp</td>
<td>pcmp1, pcmp2, pcmp4</td>
</tr>
<tr>
<td>pmax</td>
<td>pmax1, pmax2</td>
</tr>
<tr>
<td>pmin</td>
<td>pmin1, pmin2</td>
</tr>
<tr>
<td>pmpy</td>
<td>pmpy2</td>
</tr>
<tr>
<td>pmpyshr</td>
<td>pmpyshr2</td>
</tr>
<tr>
<td>pr-and-writers</td>
<td>pr-gen-writers-int[Field(ctype) in {and andcm}]. pr-gen-writers-int[Field(ctype) in {or andcm or orc}].</td>
</tr>
<tr>
<td>pr-gen-writers-fp</td>
<td>tclass, fcmp</td>
</tr>
<tr>
<td>pr-gen-writers-int</td>
<td>cmp, cmp4, tbit, tnat</td>
</tr>
<tr>
<td>pr-norm-writers-fp</td>
<td>pr-gen-writers-fp[Field(ctype)==]</td>
</tr>
<tr>
<td>pr-norm-writers-int</td>
<td>pr-gen-writers-int[Field(ctype)==]</td>
</tr>
<tr>
<td>pr-or-writers</td>
<td>pr-gen-writers-int[Field(ctype) in {or orcm}]. pr-gen-writers-int[Field(ctype) in {or andcm or orc}].</td>
</tr>
<tr>
<td>pr-readers-br</td>
<td>br.call, br.cond, br.call, bcall, bcall, br.cond, br.ret, br.wexit, br.wtop, break.b, nop.b, ReservedBQP</td>
</tr>
<tr>
<td>pr-readers-nobr-nomovpr</td>
<td>add, add1, addp4, adds, and, andcm, break.l, break.i, break.m, break.x, chk.s, chk.a, cmp, cmp4, cmpxchg, cxz, dep, extr, fp-arith, fp-non-arith, fc, fchlkf, fclrkf, fcmp, fetchadd, fpcmp, fsetc, fwb, getf, invalid-all, itc.i, itc.d, itrl, id, ld, ldtp, ldtpclt-all, mf, mix, mov-from-AR-M, mov-from-AR-IM, mov-from-AR-I, mov-to-AR-M, mov-to-AR-IM, mov-to-AR-I, mov-to-AR, mov-to-BR, mov-from-BR, mov-to-CR, mov-from-CR, mov-to-IND, mov-from-IND, mov-ip, mov-to-PSR-I, mov-to-PSR-Um, mov-from-PSR, mov-from-PSR-Um, movi, mux, nop.f, nop.i, nop.m, nop.x, or, pack, padd, pavg, pavgsub, pcmp, pmax, pmin, pmpy, pmpyshr, popcnt, probe-all, psad, pshl, pshladd, pshr, pshradd, psub, ptc.e, ptc.g, ptc.ga, plc.l, ptr.d, ptr.i, ptrn, ReservedQP, rsm, setf, shl, shlaadd, shlaaddp4, shr, shra, srzf.i, srzf.d, ssr, st, stf, sub, sum, sax, sync, tak, tbit, thash, tnat, tpa, itag, unpack, xchg, xma, xmpy, xor, zxt</td>
</tr>
<tr>
<td>pr-unc-writers-fp</td>
<td>pr-gen-writers-fp[Field(ctype)==unc]11, fprcpa11, fprsqrta11, frfcpa11, frsqrta11</td>
</tr>
<tr>
<td>pr-unc-writers-int</td>
<td>pr-gen-writers-int[Field(ctype)==unc]11</td>
</tr>
<tr>
<td>pr-writers</td>
<td>pr-writers-int, pr-writers-fp</td>
</tr>
<tr>
<td>pr-writers-fp</td>
<td>pr-norm-writers-fp, pr-unc-writers-fp</td>
</tr>
<tr>
<td>pr-writers-int</td>
<td>pr-norm-writers-int, pr-unc-writers-int, pr-and-writers, pr-or-writers</td>
</tr>
<tr>
<td>predicatable-instructions</td>
<td>mov-from-PR, mov-to-PR, pr-readers-br, pr-readers-nobr-nomovpr</td>
</tr>
<tr>
<td>priv-ops</td>
<td>mov-to-IND-priv, bsw, itc.i, itc.d, itrl, itrd, mov-to-CR, mov-from-CR, mov-to-PSR-I, mov-from-PSR, mov-from-IND-priv, plc.e, plc.g, plc.ga, plc.l, ptr.i, ptr.d, rfi, rsm, ssm, tak, tpa</td>
</tr>
<tr>
<td>probe-all</td>
<td>probe-fault, probe-nofault</td>
</tr>
<tr>
<td>probe-fault</td>
<td>probe[Format in {M40}]</td>
</tr>
<tr>
<td>probe-nofault</td>
<td>probe[Format in {M38 M39}]</td>
</tr>
<tr>
<td>psad</td>
<td>psad1</td>
</tr>
<tr>
<td>pshl</td>
<td>pshl2, pshl4</td>
</tr>
<tr>
<td>pshladd</td>
<td>pshladd2</td>
</tr>
<tr>
<td>pshr</td>
<td>pshr2, pshr4</td>
</tr>
<tr>
<td>pshradd</td>
<td>pshradd2</td>
</tr>
<tr>
<td>psb</td>
<td>psb1, psb2, psb4</td>
</tr>
</tbody>
</table>
## Table 5-5. Instruction Classes (Continued)

<table>
<thead>
<tr>
<th>Class</th>
<th>Events/Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>ReservedBQP</td>
<td>.15</td>
</tr>
<tr>
<td>ReservedQP</td>
<td>.16</td>
</tr>
<tr>
<td>rse-readers</td>
<td>alloc, br.call, br.ia, br.ret, brl.call, cover, flushrs, loadrs, mov-from-AR-BSP, mov-from-AR-BSPSTORE, mov-to-AR-BSPSTORE, mov-from-AR-RNAT, mov-to-AR-RNAT, rfi</td>
</tr>
<tr>
<td>rse-writers</td>
<td>alloc, br.call, br.ia, br.ret, brl.call, cover, flushrs, loadrs, mov-to-AR-BSPSTORE, rfi</td>
</tr>
<tr>
<td>st</td>
<td>st1, st2, st4, st8, st8.spill</td>
</tr>
<tr>
<td>st-postinc</td>
<td>stf[Format in (M10)], st[Format in (M5)]</td>
</tr>
<tr>
<td>stf</td>
<td>stfs, stfd, stfe, stf8, stf8.spill</td>
</tr>
<tr>
<td>sxt</td>
<td>sxt1, sxt2, sxt4</td>
</tr>
<tr>
<td>sys-mask-writers-partial</td>
<td>rsm, ssm</td>
</tr>
<tr>
<td>unpack</td>
<td>unpack1, unpack2, unpack4</td>
</tr>
<tr>
<td>unpredicatable-instructions</td>
<td>alloc, br.cloop, br.clop, br.exit, br.ia, brp, bsw, clrrrb, cover, epc, flushrs, loadrs, rfi</td>
</tr>
<tr>
<td>user-mask-writers-partial</td>
<td>rum, sum</td>
</tr>
<tr>
<td>xchg</td>
<td>xchg1, xchg2, xchg4, xchg8</td>
</tr>
<tr>
<td>zxt</td>
<td>zxt1, zxt2, zxt4</td>
</tr>
</tbody>
</table>
Part II: IA-32 Instruction Set Descriptions
This section lists all IA-32 instructions and their behavior in the Itanium System Environment and IA-32 System Environments on an processor based on the Itanium architecture. Unless noted otherwise all IA-32 and MMX and Streaming SIMD Extension instructions operate as defined in the Intel Architecture Software Developer’s Manual.

This volume describes the complete IA-32 Architecture instruction set, including the integer, floating-point, MMX technology and Streaming SIMD Extension technology, and system instructions. The instruction descriptions are arranged in alphabetical order. For each instruction, the forms are given for each operand combination, including the opcode, operands required, and a description. Also given for each instruction are a description of the instruction and its operands, an operational description, a description of the effect of the instructions on flags in the EFLAGS register, and a summary of the exceptions that can be generated.

For all IA-32 the following relationships hold:

- **Writes** – Writes of any IA-32 general purpose, floating-point or Streaming SIMD Extension, MMX technology registers by IA-32 instructions are reflected in the Itanium registers defined to hold that IA-32 state when IA-32 instruction set completes execution.
- **Reads** – Reads of any IA-32 general purpose, floating-point or Streaming SIMD Extension, MMX technology registers by IA-32 instructions see the state of the Itanium registers defined to hold the IA-32 state after entering the IA-32 instruction set.
- **State mappings** – IA-32 numeric instructions are controlled by and reflect their status in FCW, FSW, FTW, FCS, FIP, FOP, FDS and FEA. On exit from the IA-32 instruction set, Itanium numeric status and control resources defined to hold IA-32 state reflect the results of all IA-32 prior numeric instructions in FCR, FSR, FIR and FDR. Itanium numeric status and control resources defined to hold IA-32 state are honored by IA-32 numeric instructions when entering the IA-32 instruction set.

### 1.1 Additional Intel® Itanium™ Faults

The following fault behavior is defined for all IA-32 instructions in the Itanium System Environment:

- **IA-32 Faults** – All IA-32 faults are performed as defined in the Intel Architecture Software Developer’s Manual, unless otherwise noted. IA-32 faults are delivered on the IA-32 Exception interruption vector.
- **IA-32 GPFault** – Null segments are signified by the segment descriptor register’s P-bit being set to zero. IA-32 memory references through DSD, ESD, FSD, and GSD with the P-bit set to zero result in an IA-32 GPFault.
- **Itanium Low FP Reg Fault** – If PSR.dfl is 1, execution of any IA-32 MMX technology, Streaming SIMD Extension or floating-point instructions results in a Disabled FP Register fault (regardless of whether FR2-31 is referenced).
• **Itanium High FP Reg Fault** – If PSR.dfh is 1, execution of the first target IA-32 instruction following an `br.ia` or `rfi` results in a Disabled FP Register fault (regardless of whether FR32-127 is referenced).

• **Itanium Instruction Mem Faults** – The following additional Itanium memory faults can be generated on each virtual page referenced when fetching IA-32 or MMX or Streaming SIMD Extension instructions for execution:
  - Alternative instruction TLB fault
  - VHPT instruction fault
  - Instruction TLB fault
  - Instruction Page Not Present fault
  - Instruction NaT Page Consumption Abort
  - Instruction Key Miss fault
  - Instruction Key Permission fault
  - Instruction Access Rights fault
  - Instruction Access Bit fault

• **Itanium Data Mem Faults** – The following additional Itanium memory faults can be generated on each virtual page touched when reading or writing memory operands from the IA-32 instruction set including MMX and Streaming SIMD Extension instructions:
  - Nested TLB fault
  - Alternative data TLB fault
  - VHPT data fault
  - Data TLB fault
  - Data Page Not Present fault
  - Data NaT Page Consumption Abort
  - Data Key Miss fault
  - Data Key Permission fault
  - Data Access Rights fault
  - Data Dirty bit fault
  - Data Access bit fault

1.2 **Interpreting the IA-32 Instruction Reference Pages**

This section describes the information contained in the various sections of the instruction reference pages that make up the majority of this chapter. It also explains the notational conventions and abbreviations used in these sections.

1.2.1 **IA-32 Instruction Format**

The following is an example of the format used for each Intel Architecture instruction description in this chapter.
CMC—Complement Carry Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F5</td>
<td>CMC</td>
<td>Complement carry flag</td>
</tr>
</tbody>
</table>

1.2.1.1 Opcode Column

The “Opcode” column gives the complete object code produced for each form of the instruction. When possible, the codes are given as hexadecimal bytes, in the same order in which they appear in memory. Definitions of entries other than hexadecimal bytes are as follows:

- \(/\text{digit}\) – A digit between 0 and 7 indicates that the ModR/M byte of the instruction uses only the r/m (register or memory) operand. The reg field contains the digit that provides an extension to the instruction’s opcode.
- \(/r\) – Indicates that the ModR/M byte of the instruction contains both a register operand and an r/m operand.
- \(\text{cb, cw, cd, cp}\) – A 1-byte (cb), 2-byte (cw), 4-byte (cd), or 6-byte (cp) value following the opcode that is used to specify a code offset and possibly a new value for the code segment register.
- \(\text{ib, iw, id}\) – A 1-byte (ib), 2-byte (iw), or 4-byte (id) immediate operand to the instruction that follows the opcode, ModR/M bytes or scale-indexing bytes. The opcode determines if the operand is a signed value. All words and doublewords are given with the low-order byte first.
- \(+\text{rb}, +\text{rw}, +\text{rd}\) – A register code, from 0 through 7, added to the hexadecimal byte given at the left of the plus sign to form a single opcode byte. The register codes are given in Table 1-1.
- \(+\text{i}\) – A number used in floating-point instructions when one of the operands is ST(i) from the FPU register stack. The number i (which can range from 0 to 7) is added to the hexadecimal byte given at the left of the plus sign to form a single opcode byte.

<table>
<thead>
<tr>
<th>rb</th>
<th>rw</th>
<th>rd</th>
</tr>
</thead>
<tbody>
<tr>
<td>AL = 0</td>
<td>AX = 0</td>
<td>EAX = 0</td>
</tr>
<tr>
<td>CL = 1</td>
<td>CX = 1</td>
<td>ECX = 1</td>
</tr>
<tr>
<td>DL = 2</td>
<td>DX = 2</td>
<td>EDX = 2</td>
</tr>
<tr>
<td>BL = 3</td>
<td>BX = 3</td>
<td>EBX = 3</td>
</tr>
<tr>
<td>AH = 4</td>
<td>SP = 4</td>
<td>ESP = 4</td>
</tr>
<tr>
<td>CH = 5</td>
<td>BP = 5</td>
<td>EBP = 5</td>
</tr>
<tr>
<td>DH = 6</td>
<td>SI = 6</td>
<td>ESI = 6</td>
</tr>
<tr>
<td>BH = 7</td>
<td>DI = 7</td>
<td>EDI = 7</td>
</tr>
</tbody>
</table>

1.2.1.2 Instruction Column

The “Instruction” column gives the syntax of the instruction statement as it would appear in an ASM386 program. The following is a list of the symbols used to represent operands in the instruction statements:

- \(\text{rel8}\) – A relative address in the range from 128 bytes before the end of the instruction to 127 bytes after the end of the instruction.
• **rel16 and rel32** – A relative address within the same code segment as the instruction assembled. The rel16 symbol applies to instructions with an operand-size attribute of 16 bits; the rel32 symbol applies to instructions with an operand-size attribute of 32 bits.

• **ptr16:16 and ptr16:32** – A far pointer, typically in a code segment different from that of the instruction. The notation 16:16 indicates that the value of the pointer has two parts. The value to the left of the colon is a 16-bit selector or value destined for the code segment register. The value to the right corresponds to the offset within the destination segment. The ptr16:16 symbol is used when the instruction's operand-size attribute is 16 bits; the ptr16:32 symbol is used when the operand-size attribute is 32 bits.

• **r8** – One of the byte general-purpose registers AL, CL, DL, BL, AH, CH, DH, or BH.

• **r16** – One of the word general-purpose registers AX, CX, DX, BX, SP, BP, SI, or DI.

• **r32** – One of the doubleword general-purpose registers EAX, ECX, EDX, EBX, ESP, EBP, ESI, or EDI.

• **imm8** – An immediate byte value. The imm8 symbol is a signed number between −128 and +127 inclusive. For instructions in which imm8 is combined with a word or doubleword operand, the immediate value is sign-extended to form a word or doubleword. The upper byte of the word is filled with the topmost bit of the immediate value.

• **imm16** – An immediate word value used for instructions whose operand-size attribute is 16 bits. This is a number between −32,768 and +32,767 inclusive.

• **imm32** – An immediate doubleword value used for instructions whose operand-size attribute is 32 bits. It allows the use of a number between +2,147,483,647 and –2,147,483,648 inclusive.

• **r/m8** – A byte operand that is either the contents of a byte general-purpose register (AL, BL, CL, DL, AH, BH, CH, and DH), or a byte from memory.

• **r/m16** – A word general-purpose register or memory operand used for instructions whose operand-size attribute is 16 bits. The word general-purpose registers are: AX, BX, CX, DX, SP, BP, SI, and DI. The contents of memory are found at the address provided by the effective address computation.

• **r/m32** – A doubleword general-purpose register or memory operand used for instructions whose operand-size attribute is 32 bits. The doubleword general-purpose registers are: EAX, EBX, ECX, EDX, ESP, EBP, ESI, and EDI. The contents of memory are found at the address provided by the effective address computation.

• **m** – A 16- or 32-bit operand in memory.

• **m8** – A byte operand in memory, usually expressed as a variable or array name, but pointed to by the DS:(E)SI or ES:(E)DI registers. This nomenclature is used only with the string instructions and the XLAT instruction.

• **m16** – A word operand in memory, usually expressed as a variable or array name, but pointed to by the DS:(E)SI or ES:(E)DI registers. This nomenclature is used only with the string instructions.

• **m32** – A doubleword operand in memory, usually expressed as a variable or array name, but pointed to by the DS:(E)SI or ES:(E)DI registers. This nomenclature is used only with the string instructions.

• **m64** – A memory quadword operand in memory. This nomenclature is used only with the CMPXCHG8B instruction.

• **m16:16, m16:32** – A memory operand containing a far pointer composed of two numbers. The number to the left of the colon corresponds to the pointer's segment selector. The number to the right corresponds to its offset.
- **m16&32, m16&16, m32&32** – A memory operand consisting of data item pairs whose sizes are indicated on the left and the right side of the ampersand. All memory addressing modes are allowed. The m16&16 and m32&32 operands are used by the BOUND instruction to provide an operand containing an upper and lower bounds for array indices. The m16&32 operand is used by LIDT and LGDT to provide a word with which to load the limit field, and a doubleword with which to load the base field of the corresponding GDTR and IDTR registers.

- **moffs8, moffs16, moffs32** – A simple memory variable (memory offset) of type byte, word, or doubleword used by some variants of the MOV instruction. The actual address is given by a simple offset relative to the segment base. No ModR/M byte is used in the instruction. The number shown with moffs indicates its size, which is determined by the address-size attribute of the instruction.

- **Sreg** – A segment register. The segment register bit assignments are ES=0, CS=1, SS=2, DS=3, FS=4, and GS=5.

- **m32real, m64real, m80real** – A single-, double-, and extended-real (respectively) floating-point operand in memory.

- **m16int, m32int, m64int** – A word-, short-, and long-integer (respectively) floating-point operand in memory.

- **ST or ST(0)** – The top element of the FPU register stack.

- **ST(i)** – The i\(^{th}\) element from the top of the FPU register stack. (i = 0 through 7).

- **mm** – An MMX technology register. The 64-bit MMX technology registers are: MM0 through MM7.

- **mm/m32** – The low order 32 bits of an MMX technology register or a 32-bit memory operand. The 64-bit MMX technology registers are: MM0 through MM7. The contents of memory are found at the address provided by the effective address computation.

- **mm/m64** – An MMX technology register or a 64-bit memory operand. The 64-bit MMX technology registers are: MM0 through MM7. The contents of memory are found at the address provided by the effective address computation.

### 1.2.1.3 Description Column

The “Description” column following the “Instruction” column briefly explains the various forms of the instruction. The following “Description” and “Operation” sections contain more details of the instruction's operation.

### 1.2.1.4 Description

The “Description” section describes the purpose of the instructions and the required operands. It also discusses the effect of the instruction on flags.

### 1.2.2 Operation

The “Operation” section contains an algorithmic description (written in pseudo-code) of the instruction. The pseudo-code uses a notation similar to the Algol or Pascal language. The algorithms are composed of the following elements:

- Comments are enclosed within the symbol pairs “(*” and “*)”. 
• Compound statements are enclosed in keywords, such as IF, THEN, ELSE, and FI for an if statement, DO and OD for a do statement, or CASE... OF and ESAC for a case statement.

• A register name implies the contents of the register. A register name enclosed in brackets implies the contents of the location whose address is contained in that register. For example, ES:[DI] indicates the contents of the location whose ES segment relative address is in register DL. [SI] indicates the contents of the address contained in register SI relative to SI's default segment (DS) or overridden segment.

• Parentheses around the “E” in a general-purpose register name, such as (E)SI, indicates that an offset is read from the SI register if the current address-size attribute is 16 or is read from the ESI register if the address-size attribute is 32.

• Brackets are also used for memory operands, where they mean that the contents of the memory location is a segment-relative offset. For example, [SRC] indicates that the contents of the source operand is a segment-relative offset.

• A ← B; indicates that the value of B is assigned to A.

• The symbols =, ≠, ≥, and ≤ are relational operators used to compare two values, meaning equal, not equal, greater or equal, less or equal, respectively. A relational expression such as A = B is TRUE if the value of A is equal to B; otherwise it is FALSE.

• The expression “<< COUNT” and “>> COUNT” indicates that the destination operand should be shifted left or right, respectively, by the number of bits indicated by the count operand.

The following identifiers are used in the algorithmic descriptions:

• **OperandSize** and **AddressSize** – The OperandSize identifier represents the operand-size attribute of the instruction, which is either 16 or 32 bits. The AddressSize identifier represents the address-size attribute, which is either 16 or 32 bits. For example, the following pseudo-code indicates that the operand-size attribute depends on the form of the CMPS instruction used.

```
IF instruction = CMPSW
    THEN OperandSize ← 16;
ELSE
    IF instruction = CMPSD
        THEN OperandSize ← 32;
        FI;
FI;
```

See “Operand-Size and Address-Size Attributes” in Chapter 3 of the *Intel Architecture Software Developer’s Manual, Volume 1*, for general guidelines on how these attributes are determined.

• **StackAddrSize** – Represents the stack address-size attribute associated with the instruction, which has a value of 16 or 32 bits (see “Address-Size Attribute for Stack” in Chapter 4 of the *Intel Architecture Software Developer’s Manual, Volume 1*).

• **SRC** – Represents the source operand.

• **DEST** – Represents the destination operand.

The following functions are used in the algorithmic descriptions:

• **ZeroExtend(value)** – Returns a value zero-extended to the operand-size attribute of the instruction. For example, if the operand-size attribute is 32, zero extending a byte value of –10 converts the byte from F6H to a doubleword value of 000000F6H. If the value passed to the ZeroExtend function and the operand-size attribute are the same size, ZeroExtend returns the value unaltered.
• **SignExtend(value)** – Returns a value sign-extended to the operand-size attribute of the instruction. For example, if the operand-size attribute is 32, sign extending a byte containing the value –10 converts the byte from F6H to a doubleword value of FFFFFFF6H. If the value passed to the SignExtend function and the operand-size attribute are the same size, SignExtend returns the value unaltered.

• **SaturateSignedWordToSignedByte** – Converts a signed 16-bit value to a signed 8-bit value. If the signed 16-bit value is less than –128, it is represented by the saturated value –128 (80H); if it is greater than 127, it is represented by the saturated value 127 (7FH).

• **SaturateSignedDwordToSignedWord** – Converts a signed 32-bit value to a signed 16-bit value. If the signed 32-bit value is less than –32768, it is represented by the saturated value –32768 (8000H); if it is greater than 32767, it is represented by the saturated value 32767 (7FFFH).

• **SaturateSignedWordToUnsignedByte** – Converts a signed 16-bit value to an unsigned 8-bit value. If the signed 16-bit value is less than zero, it is represented by the saturated value zero (00H); if it is greater than 255, it is represented by the saturated value 255 (FFH).

• **SaturateToSignedByte** – Represents the result of an operation as a signed 8-bit value. If the result is less than –128, it is represented by the saturated value –128 (80H); if it is greater than 127, it is represented by the saturated value 127 (7FH).

• **SaturateToSignedWord** – Represents the result of an operation as a signed 16-bit value. If the result is less than –32768, it is represented by the saturated value –32768 (8000H); if it is greater than 32767, it is represented by the saturated value 32767 (7FFFH).

• **SaturateToUnsignedByte** – Represents the result of an operation as a signed 8-bit value. If the result is less than zero it is represented by the saturated value zero (00H); if it is greater than 255, it is represented by the saturated value 255 (FFH).

• **SaturateToUnsignedWord** – Represents the result of an operation as a signed 16-bit value. If the result is less than zero it is represented by the saturated value zero (00H); if it is greater than 65535, it is represented by the saturated value 65535 (FFFFH).

• **LowOrderWord(Dest * Src)** – Multiplies a word operand by a word operand and stores the least significant word of the doubleword result in the destination operand.

• **HighOrderWord(Dest * Src)** – Multiplies a word operand by a word operand and stores the most significant word of the doubleword result in the destination operand.

• **Push(value)** – Pushes a value onto the stack. The number of bytes pushed is determined by the operand-size attribute of the instruction.

• **Pop()** – Removes the value from the top of the stack and returns it. The statement EAX ← Pop(); assigns to EAX the 32-bit value from the top of the stack. Pop will return either a word or a doubleword depending on the operand-size attribute.

• **PopRegisterStack** – Marks the FPU ST(0) register as empty and increments the FPU register stack pointer (TOP) by 1.

• **Switch-Tasks** – Performs a task switch.

• **Bit(BitBase, BitOffset)** – Returns the value of a bit within a bit string, which is a sequence of bits in memory or a register. Bits are numbered from low-order to high-order within registers and within memory bytes. If the base operand is a register, the offset can be in the range 0..31. This offset addresses a bit within the indicated register. An example, the function Bit[EAX, 21] is illustrated in Figure 1-1.
If BitBase is a memory address, BitOffset can range from –2 GBits to 2 GBits. The addressed bit is numbered (Offset MOD 8) within the byte at address (BitBase + (BitOffset DIV 8)), where DIV is signed division with rounding towards negative infinity, and MOD returns a positive number. This operation is illustrated in Figure 1-2.

Figure 1-2. Memory Bit Indexing

1.2.3 Flags Affected

The “Flags Affected” section lists the flags in the EFLAGS register that are affected by the instruction. When a flag is cleared, it is equal to 0; when it is set, it is equal to 1. The arithmetic and logical instructions usually assign values to the status flags in a uniform manner (see Appendix A, EFLAGS Cross-Reference, in the Intel Architecture Software Developer’s Manual, Volume 1). Non-conventional assignments are described in the “Operation” section. The values of flags listed as undefined may be changed by the instruction in an indeterminate manner. Flags that are not listed are unchanged by the instruction.

1.2.4 FPU Flags Affected

The floating-point instructions have an “FPU Flags Affected” section that describes how each instruction can affect the four condition code flags of the FPU status word.
1.2.5 Protected Mode Exceptions

The “Protected Mode Exceptions” section lists the exceptions that can occur when the instruction is executed in protected mode and the reasons for the exceptions. Each exception is given a mnemonic that consists of a pound sign (#) followed by two letters and an optional error code in parentheses. For example, #GP(0) denotes a general protection exception with an error code of 0. Table 1-2 associates each two-letter mnemonic with the corresponding interrupt vector number and exception name. See Chapter 5, *Interrupt and Exception Handling*, in the *Intel Architecture Software Developer’s Manual, Volume 3*, for a detailed description of the exceptions.

Application programmers should consult the documentation provided with their operating systems to determine the actions taken when exceptions occur.

1.2.6 Real-address Mode Exceptions

The “Real-Address Mode Exceptions” section lists the exceptions that can occur when the instruction is executed in real-address mode.

### Table 1-2. Exception Mnemonics, Names, and Vector Numbers

<table>
<thead>
<tr>
<th>Vector No.</th>
<th>Mnemonic</th>
<th>Name</th>
<th>Source</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>#DE</td>
<td>Divide Error</td>
<td>DIV and IDIV instructions.</td>
</tr>
<tr>
<td>1</td>
<td>#DB</td>
<td>Debug</td>
<td>Any code or data reference.</td>
</tr>
<tr>
<td>3</td>
<td>#BP</td>
<td>Breakpoint</td>
<td>INT 3 instruction.</td>
</tr>
<tr>
<td>4</td>
<td>#OF</td>
<td>Overflow</td>
<td>INTO instruction.</td>
</tr>
<tr>
<td>5</td>
<td>#BR</td>
<td>BOUND Range Exceeded</td>
<td>BOUND instruction.</td>
</tr>
<tr>
<td>6</td>
<td>#UD</td>
<td>Invalid Opcode (Undefined Opcode)</td>
<td>UD2 instruction or reserved opcode.</td>
</tr>
<tr>
<td>7</td>
<td>#NM</td>
<td>Device Not Available (No Math Coprocessor)</td>
<td>Floating-point or WAIT/FWAIT instruction.</td>
</tr>
<tr>
<td>8</td>
<td>#DF</td>
<td>Double Fault</td>
<td>Any instruction that can generate an exception, an NMI, or an INTR.</td>
</tr>
<tr>
<td>10</td>
<td>#TS</td>
<td>Invalid TSS</td>
<td>Task switch or TSS access.</td>
</tr>
<tr>
<td>11</td>
<td>#NP</td>
<td>Segment Not Present</td>
<td>Loading segment registers or accessing system segments.</td>
</tr>
<tr>
<td>12</td>
<td>#SS</td>
<td>Stack Segment Fault</td>
<td>Stack operations and SS register loads.</td>
</tr>
<tr>
<td>13</td>
<td>#GP</td>
<td>General Protection</td>
<td>Any memory reference and other protection checks.</td>
</tr>
<tr>
<td>14</td>
<td>#PF</td>
<td>Page Fault</td>
<td>Any memory reference.</td>
</tr>
<tr>
<td>16</td>
<td>#MF</td>
<td>Floating-point Error (Math Fault)</td>
<td>Floating-point or WAIT/FWAIT instruction.</td>
</tr>
<tr>
<td>17</td>
<td>#AC</td>
<td>Alignment Check</td>
<td>Any data reference in memory.</td>
</tr>
<tr>
<td>18</td>
<td>#MC</td>
<td>Machine Check</td>
<td>Model dependent.</td>
</tr>
</tbody>
</table>

a. The UD2 instruction was introduced in the Pentium Pro processor.
b. This exception was introduced in the Intel® 486™ processor.
c. This exception was introduced in the Pentium processor and enhanced in the Pentium Pro processor.
1.2.7 Virtual-8086 Mode Exceptions

The “Virtual-8086 Mode Exceptions” section lists the exceptions that can occur when the instruction is executed in virtual-8086 mode.

1.2.8 Floating-point Exceptions

The “Floating-point Exceptions” section lists additional exceptions that can occur when a floating-point instruction is executed in any mode. All of these exception conditions result in a floating-point error exception (#MF, vector number 16) being generated. Table 1-3 associates each one- or two-letter mnemonic with the corresponding exception name. See “Floating-Point Exception Conditions” in Chapter 7 of the Intel Architecture Software Developer’s Manual, Volume 1, for a detailed description of these exceptions.

Table 1-3. Floating-point Exception Mnemonics and Names

<table>
<thead>
<tr>
<th>Vector No.</th>
<th>Mnemonic</th>
<th>Name</th>
<th>Source</th>
</tr>
</thead>
<tbody>
<tr>
<td>16</td>
<td>#IS</td>
<td>Floating-point invalid operation:</td>
<td>- FPU stack overflow or underflow</td>
</tr>
<tr>
<td>16</td>
<td>#IA</td>
<td>- Stack overflow or underflow</td>
<td>- Invalid FPU arithmetic operation</td>
</tr>
<tr>
<td>16</td>
<td>#Z</td>
<td>- Invalid arithmetic operation</td>
<td>FPU divide-by-zero</td>
</tr>
<tr>
<td>16</td>
<td>#D</td>
<td>Floating-point divide-by-zero</td>
<td>Attempting to operate on a denormal number</td>
</tr>
<tr>
<td>16</td>
<td>#O</td>
<td>Floating-point denormalized operation</td>
<td>FPU numeric overflow</td>
</tr>
<tr>
<td>16</td>
<td>#U</td>
<td>Floating-point numeric overflow</td>
<td>FPU numeric underflow</td>
</tr>
<tr>
<td>16</td>
<td>#P</td>
<td>Floating-point numeric underflow</td>
<td>Inexact result (precision)</td>
</tr>
</tbody>
</table>

1.3 IA-32 Base Instruction Reference

The remainder of this chapter provides detailed descriptions of each of the Intel Architecture instructions.
AAA—ASCII Adjust After Addition

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>37</td>
<td>AAA</td>
<td>ASCII adjust AL after addition</td>
</tr>
</tbody>
</table>

**Description**

Adjusts the sum of two unpacked BCD values to create an unpacked BCD result. The AL register is the implied source and destination operand for this instruction. The AAA instruction is only useful when it follows an ADD instruction that adds (binary addition) two unpacked BCD values and stores a byte result in the AL register. The AAA instruction then adjusts the contents of the AL register to contain the correct 1-digit unpacked BCD result.

If the addition produces a decimal carry, the AH register is incremented by 1, and the CF and AF flags are set. If there was no decimal carry, the CF and AF flags are cleared and the AH register is unchanged. In either case, bits 4 through 7 of the AL register are cleared to 0.

**Operation**

\[
\text{IF } ((AL \text{ AND FH}) > 9) \text{ OR } (AF = 1) \text{ THEN}
\]
\[
\begin{align*}
\text{AL} & \leftarrow (AL + 6); \\
\text{AH} & \leftarrow AH + 1; \\
\text{AF} & \leftarrow 1; \\
\text{CF} & \leftarrow 1;
\end{align*}
\]

\[
\text{ELSE}
\]
\[
\begin{align*}
\text{AF} & \leftarrow 0; \\
\text{CF} & \leftarrow 0;
\end{align*}
\]

\[
\text{FI;}
\]

\[
AL \leftarrow AL \text{ AND FH;}
\]

**Flags Affected**

The AF and CF flags are set to 1 if the adjustment results in a decimal carry; otherwise they are cleared to 0. The OF, SF, ZF, and PF flags are undefined.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.
AAD—ASCII Adjust AX Before Division

### Description

Adjusts two unpacked BCD digits (the least-significant digit in the AL register and the most-significant digit in the AH register) so that a division operation performed on the result will yield a correct unpacked BCD value. The AAD instruction is only useful when it precedes a DIV instruction that divides (binary division) the adjusted value in the AL register by an unpacked BCD value.

The AAD instruction sets the value in the AL register to \((AL + (10 \times AH))\), and then clears the AH register to 00H. The value in the AX register is then equal to the binary equivalent of the original unpacked two-digit number in registers AH and AL.

### Operation

\[
\begin{align*}
\text{tempAL} & \leftarrow AL; \\
\text{tempAH} & \leftarrow AH; \\
\text{AL} & \leftarrow (\text{tempAL} + (\text{tempAH} + \text{imm8})) \text{ AND FFH}; \\
\text{AH} & \leftarrow 0
\end{align*}
\]

The immediate value \((\text{imm8})\) is taken from the second byte of the instruction, which under normal assembly is 0AH (10 decimal). However, this immediate value can be changed to produce a different result.

### Flags Affected

The SF, ZF, and PF flags are set according to the result; the OF, AF, and CF flags are undefined.

### Additional Itanium System Environment Exceptions

Itanium Reg Faults NaT Register Consumption Abort.

### Exceptions (All Operating Modes)

None.
AAM—ASCII Adjust AX After Multiply

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D4 0A</td>
<td>AAM</td>
<td>ASCII adjust AX after multiply</td>
</tr>
</tbody>
</table>

**Description**

Adjusts the result of the multiplication of two unpacked BCD values to create a pair of unpacked BCD values. The AX register is the implied source and destination operand for this instruction. The AAM instruction is only useful when it follows an MUL instruction that multiplies (binary multiplication) two unpacked BCD values and stores a word result in the AX register. The AAM instruction then adjusts the contents of the AX register to contain the correct 2-digit unpacked BCD result.

**Operation**

\[
\begin{align*}
tempAL & \leftarrow AL; \\
AH & \leftarrow tempAL / imm8; \\
AL & \leftarrow tempAL \mod imm8;
\end{align*}
\]

The immediate value \((imm8)\) is taken from the second byte of the instruction, which under normal assembly is 0AH (10 decimal). However, this immediate value can be changed to produce a different result.

**Flags Affected**

The SF, ZF, and PF flags are set according to the result. The OF, AF, and CF flags are undefined.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.
AAS—ASCII Adjust AL After Subtraction

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>3F</td>
<td>AAS</td>
<td>ASCII adjust AL after subtraction</td>
</tr>
</tbody>
</table>

**Description**

Adjusts the result of the subtraction of two unpacked BCD values to create a unpacked BCD result. The AL register is the implied source and destination operand for this instruction. The AAS instruction is only useful when it follows a SUB instruction that subtracts (binary subtraction) one unpacked BCD value from another and stores a byte result in the AL register. The AAA instruction then adjusts the contents of the AL register to contain the correct 1-digit unpacked BCD result.

If the subtraction produced a decimal carry, the AH register is decremented by 1, and the CF and AF flags are set. If no decimal carry occurred, the CF and AF flags are cleared, and the AH register is unchanged. In either case, the AL register is left with its top nibble set to 0.

**Operation**

```plaintext
IF ((AL AND FH) > 9) OR (AF = 1)
THEN
    AL ← AL − 6;
    AH ← AH − 1;
    AF ← 1;
    CF ← 1;
ELSE
    CF ← 0;
    AF ← 0;
FI;
AL ← AL AND FH;
```

**Flags Affected**

The AF and CF flags are set to 1 if there is a decimal borrow; otherwise, they are cleared to 0. The OF, SF, ZF, and PF flags are undefined.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.
ADC—Add with Carry

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>14 ib</td>
<td>ADC AL,imm8</td>
<td>Add with carry imm8 to AL</td>
</tr>
<tr>
<td>15 iw</td>
<td>ADC AX,imm16</td>
<td>Add with carry imm16 to AX</td>
</tr>
<tr>
<td>15 id</td>
<td>ADC EAX,imm32</td>
<td>Add with carry imm32 to EAX</td>
</tr>
<tr>
<td>80 /2 ib</td>
<td>ADC r/m8,imm8</td>
<td>Add with carry imm8 to r/m8</td>
</tr>
<tr>
<td>81 /2 iw</td>
<td>ADC r/m16,imm16</td>
<td>Add with carry imm16 to r/m16</td>
</tr>
<tr>
<td>81 /2 id</td>
<td>ADC r/m32,imm32</td>
<td>Add with CF imm32 to r/m32</td>
</tr>
<tr>
<td>83 /2 ib</td>
<td>ADC r/m16,imm8</td>
<td>Add with CF sign-extended imm8 to r/m16</td>
</tr>
<tr>
<td>83 /2 ib</td>
<td>ADC r/m32,imm8</td>
<td>Add with CF sign-extended imm8 into r/m32</td>
</tr>
<tr>
<td>10 /r</td>
<td>ADC r/m8,r8</td>
<td>Add with carry byte register to r/m8</td>
</tr>
<tr>
<td>11 /r</td>
<td>ADC r/m16,r16</td>
<td>Add with carry r16 to r/m16</td>
</tr>
<tr>
<td>11 /r</td>
<td>ADC r/m32,r32</td>
<td>Add with CF r32 to r/m32</td>
</tr>
<tr>
<td>12 /r</td>
<td>ADC r8,r/m8</td>
<td>Add with carry r/m8 to byte register</td>
</tr>
<tr>
<td>13 /r</td>
<td>ADC r16,r/m16</td>
<td>Add with carry r/m16 to r16</td>
</tr>
<tr>
<td>13 /r</td>
<td>ADC r32,r/m32</td>
<td>Add with CF r/m32 to r32</td>
</tr>
</tbody>
</table>

Description

Adds the destination operand (first operand), the source operand (second operand), and the carry (CF) flag and stores the result in the destination operand. The destination operand can be a register or a memory location; the source operand can be an immediate, a register, or a memory location. The state of the CF flag represents a carry from a previous addition. When an immediate value is used as an operand, it is sign-extended to the length of the destination operand format.

The ADC instruction does not distinguish between signed or unsigned operands. Instead, the processor evaluates the result for both data types and sets the OF and CF flags to indicate a carry in the signed or unsigned result, respectively. The SF flag indicates the sign of the signed result.

The ADC instruction is usually executed as part of a multibyte or multiword addition in which an ADD instruction is followed by an ADC instruction.

Operation

\[
\text{DEST} \leftarrow \text{DEST} + \text{SRC} + \text{CF};
\]

Flags Affected

The OF, SF, ZF, AF, CF, and PF flags are set according to the result.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
ADC—Add with Carry (continued)

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
ADD—Add

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>04 ib</td>
<td>ADD AL,imm8</td>
<td>Add imm8 to AL</td>
</tr>
<tr>
<td>05 iw</td>
<td>ADD AX,imm16</td>
<td>Add imm16 to AX</td>
</tr>
<tr>
<td>05 id</td>
<td>ADD EAX,imm32</td>
<td>Add imm32 to EAX</td>
</tr>
<tr>
<td>80 /0 ib</td>
<td>ADD r/m8,imm8</td>
<td>Add imm8 to r/m8</td>
</tr>
<tr>
<td>81 /0 iw</td>
<td>ADD r/m16,imm16</td>
<td>Add imm16 to r/m16</td>
</tr>
<tr>
<td>81 /0 id</td>
<td>ADD r/m32,imm32</td>
<td>Add imm32 to r/m32</td>
</tr>
<tr>
<td>83 /0 ib</td>
<td>ADD r/m16,imm8</td>
<td>Add sign-extended imm8 to r/m16</td>
</tr>
<tr>
<td>83 /0 ib</td>
<td>ADD r/m32,imm8</td>
<td>Add sign-extended imm8 to r/m32</td>
</tr>
<tr>
<td>00 /r</td>
<td>ADD r/m8,r8</td>
<td>Add r8 to r/m8</td>
</tr>
<tr>
<td>01 /r</td>
<td>ADD r/m16,r16</td>
<td>Add r16 to r/m16</td>
</tr>
<tr>
<td>01 /r</td>
<td>ADD r/m32,r32</td>
<td>Add r32 to r/m32</td>
</tr>
<tr>
<td>02 /r</td>
<td>ADD r8,r/m8</td>
<td>Add r/m8 to r8</td>
</tr>
<tr>
<td>03 /r</td>
<td>ADD r16,r/m16</td>
<td>Add r/m16 to r16</td>
</tr>
<tr>
<td>03 /r</td>
<td>ADD r32,r/m32</td>
<td>Add r/m32 to r32</td>
</tr>
</tbody>
</table>

Description

Adds the first operand (destination operand) and the second operand (source operand) and stores the result in the destination operand. The destination operand can be a register or a memory location; the source operand can be an immediate, a register, or a memory location. When an immediate value is used as an operand, it is sign-extended to the length of the destination operand format.

The ADD instruction does not distinguish between signed or unsigned operands. Instead, the processor evaluates the result for both data types and sets the OF and CF flags to indicate a carry in the signed or unsigned result, respectively. The SF flag indicates the sign of the signed result.

Operation

\[
\text{DEST} \leftarrow \text{DEST} + \text{SRC};
\]

Flags Affected

The OF, SF, ZF, AF, CF, and PF flags are set according to the result.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
ADD—Add (continued)

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
AND—Logical AND

Description
Performs a bitwise AND operation on the destination (first) and source (second) operands and stores the result in the destination operand location. The source operand can be an immediate, a register, or a memory location; the destination operand can be a register or a memory location.

Operation
\[
\text{DEST} \leftarrow \text{DEST AND SRC};
\]

Flags Affected
The OF and CF flags are cleared; the SF, ZF, and PF flags are set according to the result. The state of the AF flag is undefined.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

Protected Mode Exceptions

- **#GP(0)**
  - If the destination operand points to a nonwritable segment.
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.

- **#SS(0)**
  - If a memory operand effective address is outside the SS segment limit.
AND—Logical AND (continued)

#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
ARPL—Adjust RPL Field of Segment Selector

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>63 /r</td>
<td>ARPL r/m16,r16</td>
<td>Adjust RPL of r/m16 to not less than RPL of r16</td>
</tr>
</tbody>
</table>

Description

Compares the RPL fields of two segment selectors. The first operand (the destination operand) contains one segment selector and the second operand (source operand) contains the other. (The RPL field is located in bits 0 and 1 of each operand.) If the RPL field of the destination operand is less than the RPL field of the source operand, the ZF flag is set and the RPL field of the destination operand is increased to match that of the source operand. Otherwise, the ZF flag is cleared and no change is made to the destination operand. (The destination operand can be a word register or a memory location; the source operand must be a word register.)

The ARPL instruction is provided for use by operating-system procedures (however, it can also be used by applications). It is generally used to adjust the RPL of a segment selector that has been passed to the operating system by an application program to match the privilege level of the application program. Here the segment selector passed to the operating system is placed in the destination operand and segment selector for the application program’s code segment is placed in the source operand. (The RPL field in the source operand represents the privilege level of the application program.) Execution of the ARPL instruction then insures that the RPL of the segment selector received by the operating system is no lower (does not have a higher privilege) than the privilege level of the application program. (The segment selector for the application program’s code segment can be read from the procedure stack following a procedure call.)

See the Intel Architecture Software Developer’s Manual, Volume 3 for more information about the use of this instruction.

Operation

IF DEST(RPL) < SRC(RPL)
THEN
   ZF ← 1;
   DEST(RPL) ← SRC(RPL);
ELSE
   ZF ← 0;
FI;

Flags Affected

The ZF flag is set to 1 if the RPL field of the destination operand is less than that of the source operand; otherwise, is cleared to 0.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
ARPL—Adjust RPL Field of Segment Selector (continued)

**Protected Mode Exceptions**

#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

#UD The ARPL instruction is not recognized in real address mode.

**Virtual 8086 Mode Exceptions**

#UD The ARPL instruction is not recognized in virtual 8086 mode.
BOUND—Check Array Index Against Bounds

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>62 /r</td>
<td>BOUND r16,m16&amp;16</td>
<td>Check if r16 (array index) is within bounds specified by m16&amp;16</td>
</tr>
<tr>
<td>62 /r</td>
<td>BOUND r32,m32&amp;32</td>
<td>Check if r32 (array index) is within bounds specified by m16&amp;16</td>
</tr>
</tbody>
</table>

**Description**

Determines if the first operand (array index) is within the bounds of an array specified the second operand (bounds operand). The array index is a signed integer located in a register. The bounds operand is a memory location that points to a pair of signed doubleword-integers (when the operand-size attribute is 32) or a pair of signed word-integers (when the operand-size attribute is 16). The first doubleword (or word) is the lower bound of the array and the second doubleword (or word) is the upper bound of the array. The array index must be greater than or equal to the lower bound and less than or equal to the upper bound plus the operand size in bytes. If the index is not within bounds, a BOUND range exceeded exception (#BR) is signaled. (When a this exception is generated, the saved return instruction pointer points to the BOUND instruction.)

The bounds limit data structure (two words or doublewords containing the lower and upper limits of the array) is usually placed just before the array itself, making the limits addressable via a constant offset from the beginning of the array. Because the address of the array already will be present in a register, this practice avoids extra bus cycles to obtain the effective address of the array bounds.

**Operation**

\[
\text{IF (ArrayIndex < LowerBound OR ArrayIndex > (UpperBound + OperandSize/8))} \\
\text{ (* Below lower bound or above upper bound *)} \\
\text{THEN} \\
\text{ #BR;} \\
\text{FI;}
\]

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
BOUND—Check Array Index Against Bounds (continued)

Protected Mode Exceptions

#BR If the bounds test fails.
#UD If second operand is not a memory location.
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#BR If the bounds test fails.
#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#BR If the bounds test fails.
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
BSF—Bit Scan Forward

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F BC</td>
<td>BSF r16,r/m16</td>
<td>Bit scan forward on r/m16</td>
</tr>
<tr>
<td>0F BC</td>
<td>BSF r32,r/m32</td>
<td>Bit scan forward on r/m32</td>
</tr>
</tbody>
</table>

**Description**

Searches the source operand (second operand) for the least significant set bit (1 bit). If a least significant 1 bit is found, its bit index is stored in the destination operand (first operand). The source operand can be a register or a memory location; the destination operand is a register. The bit index is an unsigned offset from bit 0 of the source operand. If the contents source operand are 0, the contents of the destination operand is undefined.

**Operation**

\[
\text{IF SRC = 0} \\
\quad \text{THEN} \\
\quad \quad \text{ZF } \leftarrow 1; \\
\quad \quad \text{DEST is undefined;} \\
\quad \text{ELSE} \\
\quad \quad \text{ZF } \leftarrow 0; \\
\quad \quad \text{temp } \leftarrow 0; \\
\quad \quad \text{WHILE Bit(SRC, temp) = 0} \\
\quad \quad \quad \text{DO} \\
\quad \quad \quad \quad \text{temp } \leftarrow \text{temp} + 1; \\
\quad \quad \quad \quad \text{DEST } \leftarrow \text{temp}; \\
\quad \quad \text{OD}; \\
\quad \text{FI;}
\]

**Flags Affected**

The ZF flag is set to 1 if all the source operand is 0; otherwise, the ZF flag is cleared. The CF, OF, SF, AF, and PF, flags are undefined.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
BSF—Bit Scan Forward (continued)

**Protected Mode Exceptions**

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

- **#GP** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS** If a memory operand effective address is outside the SS segment limit.

**Virtual 8086 Mode Exceptions**

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made.
BSR—Bit Scan Reverse

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F BD</td>
<td>BSR r16,r/m16</td>
<td>Bit scan reverse on r/m16</td>
</tr>
<tr>
<td>0F BD</td>
<td>BSR r32,r/m32</td>
<td>Bit scan reverse on r/m32</td>
</tr>
</tbody>
</table>

**Description**

Searches the source operand (second operand) for the most significant set bit (1 bit). If a most significant 1 bit is found, its bit index is stored in the destination operand (first operand). The source operand can be a register or a memory location; the destination operand is a register. The bit index is an unsigned offset from bit 0 of the source operand. If the contents source operand are 0, the contents of the destination operand is undefined.

**Operation**

\[
\text{IF SRC = 0} \\
\text{THEN} \\
\quad ZF \leftarrow 1; \\
\quad \text{DEST is undefined;} \\
\text{ELSE} \\
\quad ZF \leftarrow 0; \\
\quad \text{temp} \leftarrow \text{OperandSize} - 1; \\
\quad \text{WHILE Bit(SRC, temp) = 0} \\
\quad \DO \\
\quad \quad \text{temp} \leftarrow \text{temp} - 1; \\
\quad \quad \text{DEST} \leftarrow \text{temp}; \\
\quad \OD; \\
\text{FI;}
\]

**Flags Affected**

The ZF flag is set to 1 if all the source operand is 0; otherwise, the ZF flag is cleared. The CF, OF, SF, AF, and PF flags are undefined.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**: NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
BSR—Bit Scan Reverse (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
   If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
BSWAP—Byte Swap

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F C8+rd</td>
<td>BSWAP r32</td>
<td>Reverses the byte order of a 32-bit register.</td>
</tr>
</tbody>
</table>

**Description**

Reverses the byte order of a 32-bit (destination) register: bits 0 through 7 are swapped with bits 24 through 31, and bits 8 through 15 are swapped with bits 16 through 23. This instruction is provided for converting little-endian values to big-endian format and vice versa.

To swap bytes in a word value (16-bit register), use the XCHG instruction. When the BSWAP instruction references a 16-bit register, the result is undefined.

**Operation**

\[
\begin{align*}
\text{TEMP} & \leftarrow \text{DEST} \\
\text{DEST}(7..0) & \leftarrow \text{TEMP}(31..24) \\
\text{DEST}(15..8) & \leftarrow \text{TEMP}(23..16) \\
\text{DEST}(23..16) & \leftarrow \text{TEMP}(15..8) \\
\text{DEST}(31..24) & \leftarrow \text{TEMP}(7..0)
\end{align*}
\]

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.

**Intel Architecture Compatibility Information**

The BSWAP instruction is not supported on Intel Architecture processors earlier than the Intel486™ processor family. For compatibility with this instruction, include functionally-equivalent code for execution on Intel processors earlier than the Intel486 processor family.
BT—Bit Test

Description

Selects the bit in a bit string (specified with the first operand, called the bit base) at the bit-position designated by the bit offset operand (second operand) and stores the value of the bit in the CF flag. The bit base operand can be a register or a memory location; the bit offset operand can be a register or an immediate value. If the bit base operand specifies a register, the instruction takes the modulo 16 or 32 (depending on the register size) of the bit offset operand, allowing any bit position to be selected in a 16- or 32-bit register, respectively. If the bit base operand specifies a memory location, it represents the address of the byte in memory that contains the bit base (bit 0 of the specified byte) of the bit string. The offset operand then selects a bit position within the range $-2^{31}$ to $2^{31} - 1$ for a register offset and 0 to 31 for an immediate offset.

Some assemblers support immediate bit offsets larger than 31 by using the immediate bit offset field in combination with the displacement field of the memory operand. In this case, the low-order 3 or 5 bits (3 for 16-bit operands, 5 for 32-bit operands) of the immediate bit offset are stored in the immediate bit offset field, and the high-order bits are shifted and combined with the byte displacement in the addressing mode by the assembler. The processor will ignore the high order bits if they are not zero.

When accessing a bit in memory, the processor may access 4 bytes starting from the memory address for a 32-bit operand size, using by the following relationship:

$\text{Effective Address} + (4 \times (\text{BitOffset} \div 32))$

Or, it may access 2 bytes starting from the memory address for a 16-bit operand, using this relationship:

$\text{Effective Address} + (2 \times (\text{BitOffset} \div 16))$

It may do so even when only a single byte needs to be accessed to reach the given bit. When using this bit addressing mechanism, software should avoid referencing areas of memory close to address space holes. In particular, it should avoid references to memory-mapped I/O registers. Instead, software should use the MOV instructions to load from or store to these addresses, and use the register form of these instructions to manipulate the data.

Operation

$CF \leftarrow \text{Bit(BitBase, BitOffset)}$

Flags Affected

The CF flag contains the value of the selected bit. The OF, SF, ZF, AF, and PF flags are undefined.
**BT—Bit Test (continued)**

### Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Exception Type</th>
<th>Exceptions</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Reg Faults</td>
<td>NaT Register Consumption Abort.</td>
</tr>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

### Protected Mode Exceptions

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

### Real Address Mode Exceptions

- **#GP**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS**: If a memory operand effective address is outside the SS segment limit.

### Virtual 8086 Mode Exceptions

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made.
BTC—Bit Test and Complement

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F BB</td>
<td>BTC r/m16,r16</td>
<td>Store selected bit in CF flag and complement</td>
</tr>
<tr>
<td>0F BB</td>
<td>BTC r/m32,r32</td>
<td>Store selected bit in CF flag and complement</td>
</tr>
<tr>
<td>0F BA /7 ib</td>
<td>BTC r/m16,imm8</td>
<td>Store selected bit in CF flag and complement</td>
</tr>
<tr>
<td>0F BA /7 ib</td>
<td>BTC r/m32,imm8</td>
<td>Store selected bit in CF flag and complement</td>
</tr>
</tbody>
</table>

Description

Selects the bit in a bit string (specified with the first operand, called the bit base) at the bit-position designated by the bit offset operand (second operand), stores the value of the bit in the CF flag, and complements the selected bit in the bit string. The bit base operand can be a register or a memory location; the bit offset operand can be a register or an immediate value. If the bit base operand specifies a register, the instruction takes the modulo 16 or 32 (depending on the register size) of the bit offset operand, allowing any bit position to be selected in a 16- or 32-bit register, respectively. If the bit base operand specifies a memory location, it represents the address of the byte in memory that contains the bit base (bit 0 of the specified byte) of the bit string. The offset operand then selects a bit position within the range $-2^{31}$ to $2^{31} - 1$ for a register offset and 0 to 31 for an immediate offset.

Some assemblers support immediate bit offsets larger than 31 by using the immediate bit offset field in combination with the displacement field of the memory operand. See “BT—Bit Test” on page 3:388 for more information on this addressing mechanism.

Operation

\[
\text{CF} \leftarrow \text{Bit(BitBase, BitOffset)} \\
\text{Bit(BitBase, BitOffset)} \leftarrow \text{NOT Bit(BitBase, BitOffset)};
\]

Flags Affected

The CF flag contains the value of the selected bit before it is complemented. The OF, SF, ZF, AF, and PF flags are undefined.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
BTC—Bit Test and Complement (continued)

**Protected Mode Exceptions**

- **#GP(0)** If the destination operand points to a non-writable segment.
  
  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  
  If the DS, ES, FS, or GS register contains a null segment selector.

- **#SS(0)** If a memory operand effective address is outside the SS segment limit.

- **#PF(fault-code)** If a page fault occurs.

- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

- **#GP** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

- **#SS** If a memory operand effective address is outside the SS segment limit.

**Virtual 8086 Mode Exceptions**

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

- **#SS(0)** If a memory operand effective address is outside the SS segment limit.

- **#PF(fault-code)** If a page fault occurs.

- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made.
BTR—Bit Test and Reset

**Description**

Selects the bit in a bit string (specified with the first operand, called the bit base) at the bit-position designated by the bit offset operand (second operand), stores the value of the bit in the CF flag, and clears the selected bit in the bit string to 0. The bit base operand can be a register or a memory location; the bit offset operand can be a register or an immediate value. If the bit base operand specifies a register, the instruction takes the modulo 16 or 32 (depending on the register size) of the bit offset operand, allowing any bit position to be selected in a 16- or 32-bit register, respectively. If the bit base operand specifies a memory location, it represents the address of the byte in memory that contains the bit base (bit 0 of the specified byte) of the bit string. The offset operand then selects a bit position within the range $-2^{31}$ to $2^{31} - 1$ for a register offset and 0 to 31 for an immediate offset.

Some assemblers support immediate bit offsets larger than 31 by using the immediate bit offset field in combination with the displacement field of the memory operand. See “BT—Bit Test” on page 3:388 for more information on this addressing mechanism.

**Operation**

\[
\begin{align*}
CF & \leftarrow \text{Bit(BitBase, BitOffset)} \\
\text{Bit(BitBase, BitOffset)} & \leftarrow 0;
\end{align*}
\]

**Flags Affected**

The CF flag contains the value of the selected bit before it is cleared. The OF, SF, ZF, AF, and PF flags are undefined.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
BTR—Bit Test and Reset (continued)

Protected Mode Exceptions

#GP(0) If the destination operand points to a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
BTS—Bit Test and Set

### Description

Selects the bit in a bit string (specified with the first operand, called the bit base) at the bit-position designated by the bit offset operand (second operand), stores the value of the bit in the CF flag, and sets the selected bit in the bit string to 1. The bit base operand can be a register or a memory location; the bit offset operand can be a register or an immediate value. If the bit base operand specifies a register, the instruction takes the modulo 16 or 32 (depending on the register size) of the bit offset operand, allowing any bit position to be selected in a 16- or 32-bit register, respectively. If the bit base operand specifies a memory location, it represents the address of the byte in memory that contains the bit base (bit 0 of the specified byte) of the bit string. The offset operand then selects a bit position within the range \(-2^{31}\) to \(2^{31} - 1\) for a register offset and 0 to 31 for an immediate offset.

Some assemblers support immediate bit offsets larger than 31 by using the immediate bit offset field in combination with the displacement field of the memory operand. See “BT—Bit Test” on page 3:388 for more information on this addressing mechanism.

### Operation

\[
\begin{align*}
CF & \leftarrow \text{Bit(BitBase, BitOffset)} \\
\text{Bit(BitBase, BitOffset)} & \leftarrow 1;
\end{align*}
\]

### Flags Affected

The CF flag contains the value of the selected bit before it is set. The OF, SF, ZF, AF, and PF flags are undefined.

### Additional Itanium System Environment Exceptions

- **Itanium Reg Faults**: NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
BTS—Bit Test and Set (continued)

Protected Mode Exceptions

#GP(0) If the destination operand points to a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
CALL—Call Procedure

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>E8 cw</td>
<td>CALL rel16</td>
<td>Call near, displacement relative to next instruction</td>
</tr>
<tr>
<td>E8 cd</td>
<td>CALL rel32</td>
<td>Call near, displacement relative to next instruction</td>
</tr>
<tr>
<td>FF /2</td>
<td>CALL r/m16</td>
<td>Call near, r/m16 indirect</td>
</tr>
<tr>
<td>FF /2</td>
<td>CALL r/m32</td>
<td>Call near, r/m32 indirect</td>
</tr>
<tr>
<td>9A cd</td>
<td>CALL ptr16:16</td>
<td>Call far, to full pointer given</td>
</tr>
<tr>
<td>9A cp</td>
<td>CALL ptr16:32</td>
<td>Call far, to full pointer given</td>
</tr>
<tr>
<td>FF /3</td>
<td>CALL m16:16</td>
<td>Call far, address at r/m16</td>
</tr>
<tr>
<td>FF /3</td>
<td>CALL m16:32</td>
<td>Call far, address at r/m32</td>
</tr>
</tbody>
</table>

Description

Saves procedure linking information on the procedure stack and jumps to the procedure (called procedure) specified with the destination (target) operand. The target operand specifies the address of the first instruction in the called procedure. This operand can be an immediate value, a general-purpose register, or a memory location.

This instruction can be used to execute four different types of calls:

- Near call – A call to a procedure within the current code segment (the segment currently pointed to by the CS register), sometimes referred to as an intrasegment call.
- Far call – A call to a procedure located in a different segment than the current code segment, sometimes referred to as an intersegment call.
- Inter-privilege-level far call – A far call to a procedure in a segment at a different privilege level than that of the currently executing program or procedure. Results in an IA-32_Interrupt(Gate) in Itanium System Environment.
- Task switch – A call to a procedure located in a different task. Results in an IA-32_Interrupt(Gate) in Itanium System Environment.

The latter two call types (inter-privilege-level call and task switch) can only be executed in protected mode. See Chapter 6 in the Intel Architecture Software Developer’s Manual, Volume 3 for information on task switching with the CALL instruction.

When executing a near call, the processor pushes the value of the EIP register (which contains the address of the instruction following the CALL instruction) onto the procedure stack (for use later as a return-instruction pointer. The processor then jumps to the address specified with the target operand for the called procedure. The target operand specifies either an absolute address in the code segment (that is an offset from the base of the code segment) or a relative offset (a signed offset relative to the current value of the instruction pointer in the EIP register, which points to the instruction following the call). An absolute address is specified directly in a register or indirectly in a memory location (r/m16 or r/m32 target-operand form). (When accessing an absolute address indirectly using the stack pointer (ESP) as a base register, the base value used is the value of the ESP before the instruction executes.) A relative offset (rel16 or rel32) is generally specified as a label in assembly code, but at the machine code level, it is encoded as a signed, 16- or 32-bit immediate value, which is added to the instruction pointer.
CALL—Call Procedure (continued)

When executing a near call, the operand-size attribute determines the size of the target operand (16 or 32 bits) for absolute addresses. Absolute addresses are loaded directly into the EIP register. When a relative offset is specified, it is added to the value of the EIP register. If the operand-size attribute is 16, the upper two bytes of the EIP register are cleared to 0s, resulting in a maximum instruction pointer size of 16 bits. The CS register is not changed on near calls.

When executing a far call, the processor pushes the current value of both the CS and EIP registers onto the procedure stack for use as a return-instruction pointer. The processor then performs a far jump to the code segment and address specified with the target operand for the called procedure. Here the target operand specifies an absolute far address either directly with a pointer (ptr16:16 or ptr16:32) or indirectly with a memory location (m16:16 or m16:32). With the pointer method, the segment and address of the called procedure is encoded in the instruction using a 4-byte (16-bit operand size) or 6-byte (32-bit operand size) far address immediate. With the indirect method, the target operand specifies a memory location that contains a 4-byte (16-bit operand size) or 6-byte (32-bit operand size) far address. The operand-size attribute determines the size of the offset (16 or 32 bits) in the far address. The far address is loaded directly into the CS and EIP registers. If the operand-size attribute is 16, the upper two bytes of the EIP register are cleared to 0s.

Any far call from a 32-bit code segment to a 16-bit code segment should be made from the first 64 Kbytes of the 32-bit code segment, because the operand-size attribute of the instruction is set to 16, allowing only a 16-bit return address offset to be saved. Also, the call should be made using a 16-bit call gate so that 16-bit values will be pushed on the stack.

When the processor is operating in protected mode, a far call can also be used to access a code segment at a different privilege level or to switch tasks. Here, the processor uses the segment selector part of the far address to access the segment descriptor for the segment being jumped to. Depending on the value of the type and access rights information in the segment selector, the CALL instruction can perform:

- A far call to the same privilege level (described in the previous paragraph).
- An far call to a different privilege level. Results in an IA-32 Intercept(Gate) in Itanium System Environment.
- A task switch. Results in an IA-32 Intercept(Gate) in Itanium System Environment.

When executing an inter-privilege-level far call, the code segment for the procedure being called is accessed through a call gate. The segment selector specified by the target operand identifies the call gate. In executing a call through a call gate where a change of privilege level occurs, the processor switches to the stack for the privilege level of the called procedure, pushes the current values of the CS and EIP registers and the SS and ESP values for the old stack onto the new stack, then performs a far jump to the new code segment. The new code segment is specified in the call gate descriptor; the new stack segment is specified in the TSS for the currently running task. The jump to the new code segment occurs after the stack switch. On the new stack, the processor pushes the segment selector and stack pointer for the calling procedure’s stack, a set of parameters from the calling procedures stack, and the segment selector and instruction pointer for the calling procedure’s code segment. (A value in the call gate descriptor determines how many parameters to copy to the new stack.)
Finally, the processor jumps to the address of the procedure being called within the new code segment. The procedure address is the offset specified by the target operand. Here again, the target operand can specify the far address of the call gate and procedure either directly with a pointer (ptr16:16 or ptr16:32) or indirectly with a memory location (m16:16 or m16:32).
CALL—Call Procedure (continued)

Executing a task switch with the CALL instruction, is similar to executing a call through a call gate. Here the target operand specifies the segment selector of the task gate for the task being switched to and the address of the procedure being called in the task. The task gate in turn points to the TSS for the task, which contains the segment selectors for the task’s code and stack segments. The CALL instruction can also specify the segment selector of the TSS directly. See the Intel Architecture Software Developer’s Manual, Volume 3 for detailed information on the mechanics of a task switch.

Operation

IF near call
THEN IF near relative call
    IF the instruction pointer is not within code segment limit THEN #GP(0); FI;
    THEN IF OperandSize = 32
    THEN
        IF stack not large enough for a 4-byte return address THEN #SS(0); FI;
        Push(EIP);
        EIP ← EIP + DEST; (* DEST is rel32 *)
    ELSE (* OperandSize = 16 *)
        IF stack not large enough for a 2-byte return address THEN #SS(0); FI;
        Push(IP);
        EIP ← (EIP + DEST) AND 0000FFFFH; (* DEST is rel16 *)
    FI;
    FI;
ELSE (* near absolute call *)
    IF the instruction pointer is not within code segment limit THEN #GP(0); FI;
    IF OperandSize = 32
    THEN
        IF stack not large enough for a 4-byte return address THEN #SS(0); FI;
        Push(EIP);
        EIP ← DEST; (* DEST is r/m32 *)
    ELSE (* OperandSize = 16 *)
        IF stack not large enough for a 2-byte return address THEN #SS(0); FI;
        Push(IP);
        EIP ← DEST AND 0000FFFFH; (* DEST is r/m16 *)
    FI;
    FI;
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
FI;
IF far call AND (PE = 0 OR (PE = 1 AND VM = 1)) (* real address or virtual 8086 mode *)
THEN
    IF OperandSize = 32
    THEN
        IF stack not large enough for a 6-byte return address THEN #SS(0); FI;
        IF the instruction pointer is not within code segment limit THEN #GP(0); FI;
        Push(CS); (* padded with 16 high-order bits *)
        Push(EIP);
        CS ← DEST[47:32]; (* DEST is ptr16:32 or [m16:32] *)
        EIP ← DEST[31:0]; (* DEST is ptr16:32 or [m16:32] *)
    ELSE (* OperandSize = 16 *)
        IF stack not large enough for a 4-byte return address THEN #SS(0); FI;
        IF the instruction pointer is not within code segment limit THEN #GP(0); FI;
        Push(CS);
        Push(IP);
CALL—Call Procedure (continued)

\[
\begin{align*}
\text{CS} & \leftarrow \text{DEST}[31:16]; (* \text{DEST is ptr16:16 or [m16:16]*)} \\
\text{EIP} & \leftarrow \text{DEST}[15:0]; (* \text{DEST is ptr16:16 or [m16:16]*)} \\
\text{EIP} & \leftarrow \text{EIP AND 0000FFFFH}; (* \text{clear upper 16 bits]*) \\
\end{align*}
\]

FI;

IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);

FI;

IF far call AND (PE = 1 AND VM = 0) (* Protected mode, not virtual 8086 mode *)

THEN

IF segment selector in target operand null THEN #GP(0); FI;
IF segment selector index not within descriptor table limits

THEN #GP(new code selector);
FI;
Read type and access rights of selected segment descriptor;
IF segment type is not a conforming or nonconforming code segment, call gate,
task gate, or TSS THEN #GP(segment selector); FI;
Depending on type and access rights
GO TO CONFORMING-CODE-SEGMENT;
GO TO NONCONFORMING-CODE-SEGMENT;
GO TO CALL-GATE;
GO TO TASK-GATE;
GO TO TASK-STATE-SEGMENT;
FI;

CONFORMING-CODE-SEGMENT:
IF DPL > CPL THEN #GP(new code segment selector); FI;
IF not present THEN #NP(selector); FI;
IF OperandSize = 32

THEN
IF stack not large enough for a 6-byte return address THEN #SS(0); FI;
IF the instruction pointer is not within code segment limit THEN #GP(0); FI;
Push(CS); (* padded with 16 high-order bits *)
Push(EIP);
CS ← DEST(NewCodeSegmentSelector);
(* segment descriptor information also loaded *)
CS(RPL) ← CPL
EIP ← DEST(offset);
ELSE (* OperandSize = 16 *)
IF stack not large enough for a 4-byte return address THEN #SS(0); FI;
IF the instruction pointer is not within code segment limit THEN #GP(0); FI;
Push(CS);
Push(IP);
CS ← DEST(NewCodeSegmentSelector);
(* segment descriptor information also loaded *)
CS(RPL) ← CPL
EIP ← DEST(offset) AND 0000FFFFH; (* clear upper 16 bits *)
FI;
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
END;

NONCONFORMING-CODE-SEGMENT:
IF (RPL > CPL) OR (DPL ≠ CPL) THEN #GP(new code segment selector); FI;
IF stack not large enough for return address THEN #SS(0); FI;
tempEIP ← DEST(offset)
CALL—Call Procedure (continued)

IF OperandSize=16
    THEN
        tempEIP ← tempEIP AND 0000FFFFH; (* clear upper 16 bits *)
    FI;
IF tempEIP outside code segment limit THEN #GP(0); FI;
IF OperandSize = 32
    THEN
        Push(CS); (* padded with 16 high-order bits *)
        Push(EIP);
        CS ← DEST(NewCodeSegmentSelector);
        (* segment descriptor information also loaded *)
        CS(RPL) ← CPL;
        EIP ← tempEIP;
    ELSE (* OperandSize = 16 *)
        Push(CS);
        Push(IP);
        CS ← DEST(NewCodeSegmentSelector);
        (* segment descriptor information also loaded *)
        CS(RPL) ← CPL;
        EIP ← tempEIP;
    FI;
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
END;

CALL-GATE:
    IF call gate DPL < CPL or RPL THEN #GP(call gate selector); FI;
    IF not present THEN #NP(call gate selector); FI;
    IF Itanium System Environment THEN IA-32_Intercept(Gate,CALL);
    IF call gate code-segment selector is null THEN #GP(0); FI;
    IF call gate code-segment selector index is outside descriptor table limits
        THEN #GP(code segment selector); FI;
    Read code segment descriptor;
    IF code-segment segment descriptor does not indicate a code segment
    OR code-segment segment descriptor DPL > CPL
        THEN #GP(code segment selector); FI;
    IF code segment not present THEN #NP(new code segment selector); FI;
    IF code segment is non-conforming AND DPL < CPL
        THEN go to MORE-PRIVILEGE;
    ELSE go to SAME-PRIVILEGE;
    FI;
END;

MORE-PRIVILEGE:
    IF current TSS is 32-bit TSS
        THEN
            TSSstackAddress ← new code segment (DPL * 8) + 4
            IF (TSSstackAddress + 7) > TSS limit
                THEN #TS(current TSS selector); FI;
            newSS ← TSSstackAddress + 4;
            newESP ← stack address;
            ELSE (* TSS is 16-bit *)
                TSSstackAddress ← new code segment (DPL * 4) + 2
                IF (TSSstackAddress + 4) > TSS limit
                    THEN #TS(current TSS selector); FI;
CALL—Call Procedure (continued)

newESP ← TSSstackAddress;
newSS ← TSSstackAddress + 2;
FI;
IF stack segment selector is null THEN #TS(stack segment selector); FI;
IF stack segment selector index is not within its descriptor table limits
THEN #TS(SS selector); FI
Read code segment descriptor;
IF stack segment selector's RPL ≠ DPL of code segment
   OR stack segment DPL ≠ DPL of code segment
   OR stack segment is not a writable data segment
THEN #TS(SS selector); FI
IF stack segment not present THEN #SS(SS selector); FI;
IF CallGateSize = 32
   THEN
      IF stack does not have room for parameters plus 16 bytes
      THEN #SS(SS selector); FI;
      IF CallGate(InstructionPointer) not within code segment limit THEN #GP(0); FI;
      SS ← newSS;
      (* segment descriptor information also loaded *)
      ESP ← newESP;
      CS:EIP ← CallGate(CS:InstructionPointer);
      (* segment descriptor information also loaded *)
      Push(oldSS:oldESP); (* from calling procedure *)
      temp ← parameter count from call gate, masked to 5 bits;
      Push(parameters from calling procedure’s stack, temp)
      Push(oldCS:oldEIP); (* return address to calling procedure *)
   ELSE (* CallGateSize = 16 *)
      IF stack does not have room for parameters plus 8 bytes
      THEN #SS(0); FI;
      IF EIP not within code segment limit then #GP(0); FI;
      CS:EIP ← CallGate(CS:EIP) (* segment descriptor information also loaded *)
      Push(oldSS:oldESP); (* from calling procedure *)
      temp ← parameter count from call gate, masked to 5 bits;
      Push(parameters from calling procedure’s stack, temp)
      Push(oldCS:oldEIP); (* return address to calling procedure *)
   FI;
CPL ← CodeSegment(DPL)
CS(RPL) ← CPL
END;
SAME-PRIVILEGE:
IF CallGateSize = 32
   THEN
      IF stack does not have room for 8 bytes
      THEN #SS(0); FI;
      IF EIP not within code segment limit then #GP(0); FI;
      CS:EIP ← CallGate(CS:EIP) (* segment descriptor information also loaded *)
      Push(oldCS:oldEIP); (* return address to calling procedure *)
   ELSE (* CallGateSize = 16 *)
CALL—Call Procedure (continued)

IF stack does not have room for parameters plus 4 bytes
    THEN #SS(0); FI;
IF IP not within code segment limit THEN #GP(0); FI;
CS:IP ← CallGate(CS:instruction pointer)
(* segment descriptor information also loaded *)
Push(oldCS:oldIP); (* return address to calling procedure *)
FI;
CS(RPL) ← CPL
END;

TASK-GATE:
IF task gate DPL < CPL or RPL
    THEN #GP(task gate selector); FI;
IF task gate not present
    THEN #NP(task gate selector); FI;
IF Itanium System Environment THEN IA-32_Intercept(Gate,CALL);
Read the TSS segment selector in the task-gate descriptor;
IF TSS segment selector local/global bit is set to local
    OR index not within GDT limits
        THEN #GP(TSS selector); FI;
Access TSS descriptor in GDT;
IF TSS descriptor specifies that the TSS is busy (low-order 5 bits set to 00001)
    THEN #GP(TSS selector); FI;
IF TSS not present
    THEN #NP(TSS selector); FI;
SWITCH-TASKS (with nesting) to TSS;
IF EIP not within code segment limit
    THEN #GP(0); FI;
END;

TASK-STATE-SEGMENT:
IF TSS DPL < CPL or RPL
    OR TSS segment selector local/global bit is set to local
    OR TSS descriptor indicates TSS not available
        THEN #GP(TSS selector); FI;
IF TSS is not present
    THEN #NP(TSS selector); FI;
SWITCH-TASKS (with nesting) to TSS;
IF EIP not within code segment limit
    THEN #GP(0); FI;
END;
CALL—Call Procedure (continued)

Flags Affected

All flags are affected if a task switch occurs; no flags are affected if a task switch does not occur.

Additional Itanium System Environment Exceptions

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NuT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
IA-32_Intercept  Gate Intercept for CALLs through CALL Gates, Task Gates and Task Segments
IA-32_Exception  Taken Branch Debug Exception if PSR.tb is 1

Protected Mode Exceptions

#GP(0)  If target offset in destination operand is beyond the new code segment limit.
         If the segment selector in the destination operand is null.
         If the code segment selector in the gate is null.
         If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
         If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#GP(selector)  If code segment or gate or TSS selector index is outside descriptor table limits.
         If the segment descriptor pointed to by the segment selector in the destination operand is not for a conforming-code segment, nonconforming-code segment, call gate, task gate, or task state segment.
         If the DPL for a nonconforming-code segment is not equal to the CPL or the RPL for the segment’s segment selector is greater than the CPL.
         If the DPL for a conforming-code segment is greater than the CPL.
         If the DPL from a call-gate, task-gate, or TSS segment descriptor is less than the CPL or than the RPL of the call-gate, task-gate, or TSS’s segment selector.
         If the segment descriptor for a segment selector from a call gate does not indicate it is a code segment.
         If the segment selector from a call gate is beyond the descriptor table limits.
         If the DPL for a code-segment obtained from a call gate is greater than the CPL.
         If the segment selector for a TSS has its local/global bit set for local.
         If a TSS segment descriptor specifies that the TSS is busy or not available.

#SS(0)  If pushing the return address, parameters, or stack segment pointer onto the stack exceeds the bounds of the stack segment, when no stack switch occurs.
         If a memory operand effective address is outside the SS segment limit.
Warning: CALL—Call Procedure (continued)

#SS(selector) If pushing the return address, parameters, or stack segment pointer onto the stack exceeds the bounds of the stack segment, when a stack switch occurs.

If the SS register is being loaded as part of a stack switch and the segment pointed to is marked not present.

If stack segment does not have room for the return address, parameters, or stack segment pointer, when stack switch occurs.

#NP(selector) If a code segment, data segment, stack segment, call gate, task gate, or TSS is not present.

#TS(selector) If the new stack segment selector and ESP are beyond the end of the TSS.

If the new stack segment selector is null.

If the RPL of the new stack segment selector in the TSS is not equal to the DPL of the code segment being accessed.

If DPL of the stack segment descriptor for the new stack segment is not equal to the DPL of the code segment descriptor.

If the new stack segment is not a writable data segment.

If segment-selector index for stack segment is outside descriptor table limits.

#PF(fault-code) If a page fault occurs.

#AC(0) If an unaligned memory access occurs when the CPL is 3 and alignment checking is enabled.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the target offset is beyond the code segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the target offset is beyond the code segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If an unaligned memory access occurs when alignment checking is enabled.
CBW/CWDE—Convert Byte to Word/Convert Word to Doubleword

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>98</td>
<td>CBW</td>
<td>(AX \leftarrow \text{sign-extend of AL})</td>
</tr>
<tr>
<td>98</td>
<td>CWDE</td>
<td>(EAX \leftarrow \text{sign-extend of AX})</td>
</tr>
</tbody>
</table>

**Description**

Double the size of the source operand by means of sign extension. The CBW (convert byte to word) instruction copies the sign (bit 7) in the source operand into every bit in the AH register. The CWDE (convert word to doubleword) instruction copies the sign (bit 15) of the word in the AX register into the higher 16 bits of the EAX register.

The CBW and CWDE mnemonics reference the same opcode. The CBW instruction is intended for use when the operand-size attribute is 16 and the CWDE instruction for when the operand-size attribute is 32. Some assemblers may force the operand size to 16 when CBW is used and to 32 when CWDE is used. Others may treat these mnemonics as synonyms (CBW/CWDE) and use the current setting of the operand-size attribute to determine the size of values to be converted, regardless of the mnemonic used.

The CWDE instruction is different from the CWD (convert word to double) instruction. The CWD instruction uses the DX:AX register pair as a destination operand; whereas, the CWDE instruction uses the EAX register as a destination.

**Operation**

\[
\begin{align*}
&\text{IF } \text{OperandSize} = 16 (* \text{instruction} = \text{CBW} *) \\
&\quad \text{THEN } AX \leftarrow \text{SignExtend(AL)}; \\
&\quad \text{ELSE } (* \text{OperandSize} = 32, \text{instruction} = \text{CWDE} *) \\
&\quad\quad EAX \leftarrow \text{SignExtend(AX)}; \\
&\FI;
\]

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.
CDQ—Convert Double to Quad

See entry for CWD/CDQ — Convert Word to Double/Convert Double to Quad.
CLC—Clear Carry Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F8</td>
<td>CLC</td>
<td>Clear CF flag</td>
</tr>
</tbody>
</table>

**Description**

Clears the CF flag in the EFLAGS register.

**Operation**

\[ CF \leftarrow 0; \]

**Flags Affected**

The CF flag is cleared to 0. The OF, ZF, SF, AF, and PF flags are unaffected.

**Exceptions (All Operating Modes)**

None.
CLD—Clear Direction Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FC</td>
<td>CLD</td>
<td>Clear DF flag</td>
</tr>
</tbody>
</table>

**Description**

Clears the DF flag in the EFLAGS register. When the DF flag is set to 0, string operations increment the index registers (ESI and/or EDI).

**Operation**

\[
DF \leftarrow 0;
\]

**Flags Affected**

The DF flag is cleared to 0. The CF, OF, ZF, SF, AF, and PF flags are unaffected.

**Exceptions (All Operating Modes)**

None.
CLI—Clear Interrupt Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FA</td>
<td>CLI</td>
<td>Clear interrupt flag; interrupts disabled when interrupt flag cleared</td>
</tr>
</tbody>
</table>

**Description**

Clears the IF flag in the EFLAGS register. No other flags are affected. Clearing the IF flag causes the processor to ignore maskable external interrupts. The IF flag and the CLI and STI instruction have no affect on the generation of exceptions and NMI interrupts. In the Itanium System Environment, external interrupts are enabled for IA-32 instructions if PSR.i and (~CFLG.if or EFLAG.if) is 1 and for Itanium instructions if PSR.i is 1.

The following decision table indicates the action of the CLI instruction (bottom of the table) depending on the processor’s mode of operating and the CPL and IOPL of the currently running program or procedure (top of the table).

<table>
<thead>
<tr>
<th>PE =</th>
<th>0</th>
<th>1</th>
<th>1</th>
<th>1</th>
<th>1</th>
</tr>
</thead>
<tbody>
<tr>
<td>VM =</td>
<td>X</td>
<td>0</td>
<td>X</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>CPL</td>
<td>X</td>
<td>≤ IOPL</td>
<td>X</td>
<td>&gt; IOPL</td>
<td>X</td>
</tr>
<tr>
<td>IOPL</td>
<td>X</td>
<td>X</td>
<td>= 3</td>
<td>X</td>
<td>&lt; 3</td>
</tr>
<tr>
<td>IF ← 0</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>N</td>
<td>N</td>
</tr>
<tr>
<td>#GP(0)</td>
<td>N</td>
<td>N</td>
<td>N</td>
<td>Y</td>
<td>Y</td>
</tr>
</tbody>
</table>

Notes:
X Don’t care.
N Action in column 1 not taken.
Y Action in column 1 taken.

**Operation**

**OLD_IF ← IF;**

IF PE = 0 (* Executing in real-address mode *)
    THEN
        IF ← 0;
    ELSE
        IF VM = 0 (* Executing in protected mode *)
            THEN
                IF CR4.PVI = 1
                    THEN
                        IF CPL = 3
                            THEN
                                IF IOPL<3
                                    THEN VIF ← 0;
                                    ELSE IF ← 0;
                                    Fi;
                                ELSE (*CPL < 3*)
                                    IF IOPL < CPL
                                        THEN #GP(0);
                                    ELSE IF ← 0;
                        Fi;
                Fi;
            Fi;
CLI—Clear Interrupt Flag (continued)

\[
\begin{align*}
&\text{FI;} \\
&\text{FI;} \\
&\text{ELSE (*CR4.PVI=0 *)} \\
&\quad \text{IF IOPL < CPL} \\
&\quad \text{THEN} \\
&\quad \text{IF IOPL < CPL} \\
&\quad \text{THEN} \\
&\quad \text{ELSE IF < 0;} \\
&\quad \text{FI;} \\
&\quad \text{FI;} \\
&\quad \text{ELSE (* Executing in Virtual-8086 mode *)} \\
&\quad \text{IF IOPL = 3} \\
&\quad \text{THEN} \\
&\quad \text{IF IOPL = 3} \\
&\quad \text{THEN} \\
&\quad \text{ELSE} \\
&\quad \text{IF CR4.VME= 0} \\
&\quad \text{THEN #GP(0);} \\
&\quad \text{ELSE VIIF <- 0;} \\
&\quad \text{FI;} \\
&\quad \text{FI;} \\
&\quad \text{FI;} \\
&\quad \text{IF Itanium System Environment AND CFLG.ii AND IF != OLD_IF} \\
&\quad \text{THEN IA-32_Intercept(System_Flag,CLI);} \\
\end{align*}
\]

Flags Affected

The IF is cleared to 0 if the CPL is equal to or less than the IOPL; otherwise, the it is not affected. The other flags in the EFLAGS register are unaffected.

Additional Itanium System Environment Exceptions

IA-32_Interceptor System Flag Interception Trap if CFLG.ii is 1 and the IF flag changes state.

Protected Mode Exceptions

#GP(0) If the CPL is greater (has less privilege) than the IOPL of the current program or procedure.

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

#GP(0) If the CPL is greater (has less privilege) than the IOPL of the current program or procedure.
## CLTS—Clear Task-Switched Flag in CR0

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 06</td>
<td>CLTS</td>
<td>Clears TS flag in CR0</td>
</tr>
</tbody>
</table>

### Description

Clears the task-switched (TS) flag in the CR0 register. This instruction is intended for use in operating-system procedures. It is a privileged instruction that can only be executed at a CPL of 0. It is allowed to be executed in real-address mode to allow initialization for protected mode.

The processor sets the TS flag every time a task switch occurs. The flag is used to synchronize the saving of FPU context in multitasking applications. See the description of the TS flag in the *Intel Architecture Software Developer's Manual, Volume 3* for more information about this flag.

### Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,CLTS);

CR0(TS) ← 0;

### Flags Affected

The TS flag in CR0 register is cleared.

### Additional Itanium System Environment Exceptions

IA-32_Intercept Mandatory Instruction Intercept fault.

### Protected Mode Exceptions

#GP(0) If the CPL is greater than 0.

### Real Address Mode Exceptions

None.

### Virtual 8086 Mode Exceptions

#GP(0) If the CPL is greater than 0.
CMC—Complement Carry Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F5</td>
<td>CMC</td>
<td>Complement CF flag</td>
</tr>
</tbody>
</table>

Description

Complements the CF flag in the EFLAGS register.

Operation

CF ← NOT CF;

Flags Affected

The CF flag contains the complement of its original value. The OF, ZF, SF, AF, and PF flags are unaffected.

Exceptions (All Operating Modes)

None.
# CMOVcc—Conditional Move

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 47 cw/cd</td>
<td>CMOVA r16, r/m16</td>
<td>Move if above (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>0F 47 cw/cd</td>
<td>CMOVA r32, r/m32</td>
<td>Move if above (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>0F 43 cw/cd</td>
<td>CMOVAE r16, r/m16</td>
<td>Move if above or equal (CF=0)</td>
</tr>
<tr>
<td>0F 43 cw/cd</td>
<td>CMOVAE r32, r/m32</td>
<td>Move if above or equal (CF=0)</td>
</tr>
<tr>
<td>0F 42 cw/cd</td>
<td>CMOVB r16, r/m16</td>
<td>Move if below (CF=1)</td>
</tr>
<tr>
<td>0F 42 cw/cd</td>
<td>CMOVB r32, r/m32</td>
<td>Move if below (CF=1)</td>
</tr>
<tr>
<td>0F 46 cw/cd</td>
<td>COMVBE r16, r/m16</td>
<td>Move if below or equal (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 46 cw/cd</td>
<td>COMVBE r32, r/m32</td>
<td>Move if below or equal (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 42 cw/cd</td>
<td>COMVC r16, r/m16</td>
<td>Move if carry (CF=1)</td>
</tr>
<tr>
<td>0F 42 cw/cd</td>
<td>COMVC r32, r/m32</td>
<td>Move if carry (CF=1)</td>
</tr>
<tr>
<td>0F 44 cw/cd</td>
<td>CMOVE r16, r/m16</td>
<td>Move if equal (ZF=1)</td>
</tr>
<tr>
<td>0F 44 cw/cd</td>
<td>CMOVE r32, r/m32</td>
<td>Move if equal (ZF=1)</td>
</tr>
<tr>
<td>0F 4F cw/cd</td>
<td>CMOVG r16, r/m16</td>
<td>Move if greater (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>0F 4F cw/cd</td>
<td>CMOVG r32, r/m32</td>
<td>Move if greater (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>0F 4D cw/cd</td>
<td>CMOVGE r16, r/m16</td>
<td>Move if greater or equal (SF=OF)</td>
</tr>
<tr>
<td>0F 4D cw/cd</td>
<td>CMOVGE r32, r/m32</td>
<td>Move if greater or equal (SF=OF)</td>
</tr>
<tr>
<td>0F 4C cw/cd</td>
<td>CMOVL r16, r/m16</td>
<td>Move if less (SF&lt;OF)</td>
</tr>
<tr>
<td>0F 4C cw/cd</td>
<td>CMOVL r32, r/m32</td>
<td>Move if less (SF&lt;OF)</td>
</tr>
<tr>
<td>0F 4E cw/cd</td>
<td>CMOVLE r16, r/m16</td>
<td>Move if less or equal (ZF=1 or SF&lt;OF)</td>
</tr>
<tr>
<td>0F 4E cw/cd</td>
<td>CMOVLE r32, r/m32</td>
<td>Move if less or equal (ZF=1 or SF&lt;OF)</td>
</tr>
<tr>
<td>0F 46 cw/cd</td>
<td>CMOVN r16, r/m16</td>
<td>Move if not above (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 46 cw/cd</td>
<td>CMOVN r32, r/m32</td>
<td>Move if not above (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 42 cw/cd</td>
<td>CMOVNAE r16, r/m16</td>
<td>Move if not above or equal (CF=1)</td>
</tr>
<tr>
<td>0F 42 cw/cd</td>
<td>CMOVNAE r32, r/m32</td>
<td>Move if not above or equal (CF=1)</td>
</tr>
<tr>
<td>0F 43 cw/cd</td>
<td>CMOVNC r16, r/m16</td>
<td>Move if not carry (CF=0)</td>
</tr>
<tr>
<td>0F 43 cw/cd</td>
<td>CMOVNC r32, r/m32</td>
<td>Move if not carry (CF=0)</td>
</tr>
<tr>
<td>0F 45 cw/cd</td>
<td>CMOVNE r16, r/m16</td>
<td>Move if not equal (ZF=0)</td>
</tr>
<tr>
<td>0F 45 cw/cd</td>
<td>CMOVNE r32, r/m32</td>
<td>Move if not equal (ZF=0)</td>
</tr>
<tr>
<td>0F 4E cw/cd</td>
<td>CMOVNG r16, r/m16</td>
<td>Move if not greater (ZF=1 or SF&lt;OF)</td>
</tr>
<tr>
<td>0F 4E cw/cd</td>
<td>CMOVNG r32, r/m32</td>
<td>Move if not greater (ZF=1 or SF&lt;OF)</td>
</tr>
<tr>
<td>0F 4C cw/cd</td>
<td>CMOVNGE r16, r/m16</td>
<td>Move if not greater or equal (SF&lt;OF)</td>
</tr>
<tr>
<td>0F 4C cw/cd</td>
<td>CMOVNGE r32, r/m32</td>
<td>Move if not greater or equal (SF&lt;OF)</td>
</tr>
<tr>
<td>0F 4D cw/cd</td>
<td>CMOVNL r16, r/m16</td>
<td>Move if not less (SF=OF)</td>
</tr>
<tr>
<td>0F 4D cw/cd</td>
<td>CMOVNL r32, r/m32</td>
<td>Move if not less (SF=OF)</td>
</tr>
<tr>
<td>0F 4F cw/cd</td>
<td>CMOVNLE r16, r/m16</td>
<td>Move if not less or equal (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>0F 4F cw/cd</td>
<td>CMOVNLE r32, r/m32</td>
<td>Move if not less or equal (ZF=0 and SF=OF)</td>
</tr>
</tbody>
</table>
CMOV<sub>cc</sub>—Conditional Move (continued)

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 41 cw/cd</td>
<td>CMOVNO r16, r/m16</td>
<td>Move if not overflow (OF=0)</td>
</tr>
<tr>
<td>0F 41 cw/cd</td>
<td>CMOVNO r32, r/m32</td>
<td>Move if not overflow (OF=0)</td>
</tr>
<tr>
<td>0F 4B cw/cd</td>
<td>CMOVNP r16, r/m16</td>
<td>Move if not parity (PF=0)</td>
</tr>
<tr>
<td>0F 4B cw/cd</td>
<td>CMOVNP r32, r/m32</td>
<td>Move if not parity (PF=0)</td>
</tr>
<tr>
<td>0F 49 cw/cd</td>
<td>CMOVNS r16, r/m16</td>
<td>Move if not sign (SF=0)</td>
</tr>
<tr>
<td>0F 49 cw/cd</td>
<td>CMOVNS r32, r/m32</td>
<td>Move if not sign (SF=0)</td>
</tr>
<tr>
<td>0F 45 cw/cd</td>
<td>CMOVNZ r16, r/m16</td>
<td>Move if not zero (ZF=0)</td>
</tr>
<tr>
<td>0F 45 cw/cd</td>
<td>CMOVNZ r32, r/m32</td>
<td>Move if not zero (ZF=0)</td>
</tr>
<tr>
<td>0F 40 cw/cd</td>
<td>CMOVO r16, r/m16</td>
<td>Move if overflow (OF=0)</td>
</tr>
<tr>
<td>0F 40 cw/cd</td>
<td>CMOVO r32, r/m32</td>
<td>Move if overflow (OF=0)</td>
</tr>
<tr>
<td>0F 4A cw/cd</td>
<td>CMOVP r16, r/m16</td>
<td>Move if parity (PF=1)</td>
</tr>
<tr>
<td>0F 4A cw/cd</td>
<td>CMOVP r32, r/m32</td>
<td>Move if parity (PF=1)</td>
</tr>
<tr>
<td>0F 4A cw/cd</td>
<td>CMOVPE r16, r/m16</td>
<td>Move if parity even (PF=1)</td>
</tr>
<tr>
<td>0F 4A cw/cd</td>
<td>CMOVPE r32, r/m32</td>
<td>Move if parity even (PF=1)</td>
</tr>
<tr>
<td>0F 4B cw/cd</td>
<td>CMOVPO r16, r/m16</td>
<td>Move if parity odd (PF=0)</td>
</tr>
<tr>
<td>0F 4B cw/cd</td>
<td>CMOVPO r32, r/m32</td>
<td>Move if parity odd (PF=0)</td>
</tr>
<tr>
<td>0F 48 cw/cd</td>
<td>CMOVS r16, r/m16</td>
<td>Move if sign (SF=1)</td>
</tr>
<tr>
<td>0F 48 cw/cd</td>
<td>CMOVS r32, r/m32</td>
<td>Move if sign (SF=1)</td>
</tr>
<tr>
<td>0F 44 cw/cd</td>
<td>CMOVZ r16, r/m16</td>
<td>Move if zero (ZF=1)</td>
</tr>
<tr>
<td>0F 44 cw/cd</td>
<td>CMOVZ r32, r/m32</td>
<td>Move if zero (ZF=1)</td>
</tr>
</tbody>
</table>

Description

The CMOV<sub>cc</sub> instructions check the state of one or more of the status flags in the EFLAGS register (CF, OF, PF, SF, and ZF) and perform a move operation if the flags are in a specified state (or condition). A condition code (<i>cc</i>) is associated with each instruction to indicate the condition being tested for. If the condition is not satisfied, a move is not performed and execution continues with the instruction following the CMOV<sub>cc</sub> instruction.

If the condition is false for the memory form, some processor implementations will initiate the load (and discard the loaded data), possible memory faults can be generated. Other processor models will not initiate the load and not generate any faults if the condition is false.

These instructions can move a 16- or 32-bit value from memory to a general-purpose register or from one general-purpose register to another. Conditional moves of 8-bit register operands are not supported.

The conditions for each CMOV<sub>cc</sub> mnemonic is given in the description column of the above table. The terms “less” and “greater” are used for comparisons of signed integers and the terms “above” and “below” are used for unsigned integers.

Because a particular state of the status flags can sometimes be interpreted in two ways, two mnemonics are defined for some opcodes. For example, the CMOVA (conditional move if above) instruction and the CMOVNBE (conditional move if not below or equal) instruction are alternate mnemonics for the opcode 0F 47H.
CMOVcc—Conditional Move (continued)

The CMOVcc instructions are new for the Pentium Pro processor family; however, they may not be supported by all the processors in the family. Software can determine if the CMOVcc instructions are supported by checking the processor’s feature information with the CPUID instruction (see “CPUID—CPU Identification” on page 3:427).

Operation

\[
\text{temp} \leftarrow \text{DEST} \\
\text{IF condition TRUE} \\
\quad \text{THEN} \\
\quad \quad \text{DEST} \leftarrow \text{SRC} \\
\quad \text{ELSE} \\
\quad \quad \text{DEST} \leftarrow \text{temp} \\
\quad \text{FI};
\]

Flags Affected

None.

If the condition is false for the memory form, some processor implementations will initiate the load (and discard the loaded data), possible memory faults can be generated. Other processor models will not initiate the load and not generate any faults if the condition is false.

Additional Itanium System Environment Exceptions

Itanium Reg Faults   NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)   If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
          If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0)   If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)   If a page fault occurs.
#AC(0)   If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP   If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS   If a memory operand effective address is outside the SS segment limit.
CMOVcc—Conditional Move (continued)

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
CMP—Compare Two Operands

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>3C ib</td>
<td>CMP AL, imm8</td>
<td>Compare imm8 with AL</td>
</tr>
<tr>
<td>3D iw</td>
<td>CMP AX, imm16</td>
<td>Compare imm16 with AX</td>
</tr>
<tr>
<td>3D id</td>
<td>CMP EAX, imm32</td>
<td>Compare imm32 with EAX</td>
</tr>
<tr>
<td>80 /7 ib</td>
<td>CMP r/m8, imm8</td>
<td>Compare imm8 with r/m8</td>
</tr>
<tr>
<td>81 /7 iw</td>
<td>CMP r/m16, imm16</td>
<td>Compare imm16 with r/m16</td>
</tr>
<tr>
<td>81 /7 id</td>
<td>CMP r/m32,imm32</td>
<td>Compare imm32 with r/m32</td>
</tr>
<tr>
<td>83 /7 ib</td>
<td>CMP r/m32,imm8</td>
<td>Compare imm8 with r/m32</td>
</tr>
<tr>
<td>38 /r</td>
<td>CMP r/m8,r8</td>
<td>Compare r8 with r/m8</td>
</tr>
<tr>
<td>39 /r</td>
<td>CMP r/m16,r16</td>
<td>Compare r16 with r/m16</td>
</tr>
<tr>
<td>39 /r</td>
<td>CMP r/m32,r32</td>
<td>Compare r32 with r/m32</td>
</tr>
<tr>
<td>3A /r</td>
<td>CMP r8,r/m8</td>
<td>Compare r/m8 with r8</td>
</tr>
<tr>
<td>3B /r</td>
<td>CMP r16,r/m16</td>
<td>Compare r/m16 with r16</td>
</tr>
<tr>
<td>3B /r</td>
<td>CMP r32,r/m32</td>
<td>Compare r/m32 with r32</td>
</tr>
</tbody>
</table>

**Description**

Compares the first source operand with the second source operand and sets the status flags in the EFLAGS register according to the results. The comparison is performed by subtracting the second operand from the first operand and then setting the status flags in the same manner as the SUB instruction. When an immediate value is used as an operand, it is sign-extended to the length of the first operand.

The CMP instruction is typically used in conjunction with a conditional jump (Jcc), condition move (CMOVcc), or SETcc instruction. The condition codes used by the Jcc, CMOVcc, and SETcc instructions are based on the results of a CMP instruction.

**Operation**

\[
\text{temp} \leftarrow \text{SRC1} – \text{SignExtend} \text{(SRC2)}; \\
\text{ModifyStatusFlags; (* Modify status flags in the same manner as the SUB instruction*)}
\]

**Flags Affected**

The CF, OF, SF, ZF, AF, and PF flags are set according to the result.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
CMP—Compare Two Operands (continued)

**Protected Mode Exceptions**

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

**Virtual 8086 Mode Exceptions**

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
CMPS/CMPSB/CMPSW/CMPSD—Compare String Operands

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>A6</td>
<td>CMPS DS:(E)SI, ES:(E)DI</td>
<td>Compares byte at address DS:(E)SI with byte at address ES:(E)DI and sets the status flags accordingly</td>
</tr>
<tr>
<td>A7</td>
<td>CMPS DS:SI, ES:DI</td>
<td>Compares byte at address DS:SI with byte at address ES:DI and sets the status flags accordingly</td>
</tr>
<tr>
<td>A7</td>
<td>CMPS DS:ESI, ES:EDI</td>
<td>Compares byte at address DS:ESI with byte at address ES:EDI and sets the status flags accordingly</td>
</tr>
<tr>
<td>A6</td>
<td>CMPSB</td>
<td>Compares byte at address DS:(E)SI with byte at address ES:(E)DI and sets the status flags accordingly</td>
</tr>
<tr>
<td>A7</td>
<td>CMPSW</td>
<td>Compares byte at address DS:SI with byte at address ES:DI and sets the status flags accordingly</td>
</tr>
<tr>
<td>A7</td>
<td>CMPSD</td>
<td>Compares byte at address DS:ESI with byte at address ES:EDI and sets the status flags accordingly</td>
</tr>
</tbody>
</table>

Description

Compares the byte, word, or double word specified with the first source operand with the byte, word, or double word specified with the second source operand and sets the status flags in the EFLAGS register according to the results. The first source operand specifies the memory location at the address DS:ESI and the second source operand specifies the memory location at address ES:EDI. (When the operand-size attribute is 16, the SI and DI register are used as the source-index and destination-index registers, respectively.) The DS segment may be overridden with a segment override prefix, but the ES segment cannot be overridden.

The CMPSB, CMPSW, and CMPSD mnemonics are synonyms of the byte, word, and doubleword versions of the CMPS instructions. They are simpler to use, but provide no type or segment checking. (For the CMPS instruction, “DS:ESI” and “ES:EDI” must be explicitly specified in the instruction.)

After the comparison, the ESI and EDI registers are incremented or decremented automatically according to the setting of the DF flag in the EFLAGS register. (If the DF flag is 0, the ESI and EDI register are incremented; if the DF flag is 1, the ESI and EDI registers are decremented.) The registers are incremented or decremented by 1 for byte operations, by 2 for word operations, or by 4 for doubleword operations.

The CMPS, CMPSB, CMPSW, and CMPSD instructions can be preceded by the REP prefix for block comparisons of ECX bytes, words, or doublewords. More often, however, these instructions will be used in a LOOP construct that takes some action based on the setting of the status flags before the next comparison is made.
CMPS/CMPSB/CMPSW/CMPSD—Compare String Operands (continued)

**Operation**

\[
temp \leftarrow SRC1 - SRC2; \\
SetStatusFlags(temp); \\
IF (byte comparison) \\
\quad \text{THEN IF } DF = 0 \\
\quad \quad \text{THEN } (E)DI \leftarrow 1; (E)SI \leftarrow 1; \\
\quad \quad \text{ELSE } (E)DI \leftarrow -1; (E)SI \leftarrow -1; \\
\quad \FI; \\
\text{ELSE IF (word comparison) } \\
\quad \text{THEN IF } DF = 0 \\
\quad \quad \text{THEN } DI \leftarrow 2; (E)SI \leftarrow 2; \\
\quad \quad \text{ELSE } DI \leftarrow -2; (E)SI \leftarrow -2; \\
\quad \FI; \\
\quad \text{ELSE (* doubleword comparison *) } \\
\quad \quad \text{THEN IF } DF = 0 \\
\quad \quad \quad \text{THEN } EDI \leftarrow 4; (E)SI \leftarrow 4; \\
\quad \quad \quad \text{ELSE } EDI \leftarrow -4; (E)SI \leftarrow -4; \\
\quad \FI; \\
\FI; \\
\FI;
\]

**Flags Affected**

The CF, OF, SF, ZF, AF, and PF flags are set according to the temporary result of the comparison.

**Additional Itanium System Environment Exceptions**

**Itanium Reg Faults**  NaT Register Consumption Abort.

**Itanium Mem Faults**  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Protected Mode Exceptions**

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS  If a memory operand effective address is outside the SS segment limit.
CMPS/CMPSB/CMPSW/CMPSD—Compare String Operands (continued)

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
## CMPXCHG—Compare and Exchange

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F B0/r</td>
<td>CMPXCHG r/m8,r8</td>
<td>Compare AL with r/m8. If equal, ZF is set and r8 is loaded into r/m8. Else, clear ZF and load r/m8 into AL.</td>
</tr>
<tr>
<td>0F B1/r</td>
<td>CMPXCHG r/m16,r16</td>
<td>Compare AX with r/m16. If equal, ZF is set and r16 is loaded into r/m16. Else, clear ZF and load r/m16 into AL.</td>
</tr>
<tr>
<td>0F B1/r</td>
<td>CMPXCHG r/m32,r32</td>
<td>Compare EAX with r/m32. If equal, ZF is set and r32 is loaded into r/m32. Else, clear ZF and load r/m32 into AL.</td>
</tr>
</tbody>
</table>

### Description

Compares the value in the AL, AX, or EAX register (depending on the size of the operand) with the first operand (destination operand). If the two values are equal, the second operand (source operand) is loaded into the destination operand. Otherwise, the destination operand is loaded into the AL, AX, or EAX register.

This instruction can be used with a LOCK prefix to allow the instruction to be executed atomically. To simplify the interface to the processor’s bus, the destination operand receives a write cycle without regard to the result of the comparison. The destination operand is written back if the comparison fails; otherwise, the source operand is written into the destination. (The processor never produces a locked read without also producing a locked write.)

### Operation

\[ (*) \text{accumulator} = \text{AL, AX, or EAX, depending on whether *}) \]
\[ (*) \text{a byte, word, or doubleword comparison is being performed} (*) \]

\[
\text{IF Itanium System Environment AND External\_Atomic\_Lock\_Required AND DCR.Ic}
\text{THEN IA-32\_Intercept(LOCK,CMPXCHG);}\
\text{IF accumulator = DEST}
\text{THEN}
\text{ZF} \leftarrow 1$
\text{DEST} \leftarrow \text{SRC}
\text{ELSE}
\text{ZF} \leftarrow 0$
\text{accumulator} \leftarrow \text{DEST}
\text{FI;}
\]

### Flags Affected

The ZF flag is set if the values in the destination operand and register AL, AX, or EAX are; otherwise it is cleared. The CF, PF, AF, SF, and OF flags are set according to the results of the comparison operation.
CMPXCHG—Compare and Exchange (continued)

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
- IA-32_Intercept: Lock Intercept - If an external atomic bus lock is required to complete this operation and DCR.lc is 1, no atomic transaction occurs, this instruction is faulted and an IA-32_Intercept(Lock) fault is generated. The software lock handler is responsible for the emulation of this instruction.

Protected Mode Exceptions

- #GP(0): If the destination is located in a nonwritable segment.
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- #SS(0): If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code): If a page fault occurs.
- #AC(0): If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

- #GP: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- #SS: If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

- #GP(0): If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- #SS(0): If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code): If a page fault occurs.
- #AC(0): If alignment checking is enabled and an unaligned memory reference is made.

Intel Architecture Compatibility

This instruction is not supported on Intel processors earlier than the Intel486 processors.
CMPXCHG8B—Compare and Exchange 8 Bytes

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F C7 /1 m64</td>
<td>CMPXCHG8B m64</td>
<td>Compare EDX:EAX with m64. If equal, set ZF and load ECX:EBX into m64. Else, clear ZF and load m64 into EDX:EAX.</td>
</tr>
</tbody>
</table>

**Description**

Compares the 64-bit value in EDX:EAX with the operand (destination operand). If the values are equal, the 64-bit value in ECX:EBX is stored in the destination operand. Otherwise, the value in the destination operand is loaded into EDX:EAX. The destination operand is an 8-byte memory location. For the EDX:EAX and ECX:EBX register pairs, EDX and ECX contain the high-order 32 bits and EAX and EBX contain the low-order 32 bits of a 64-bit value.

This instruction can be used with a LOCK prefix to allow the instruction to be executed atomically. To simplify the interface to the processor’s bus, the destination operand receives a write cycle without regard to the result of the comparison. The destination operand is written back if the comparison fails; otherwise, the source operand is written into the destination. (The processor never produces a locked read without also producing a locked write.)

**Operation**

IF Itanium System Environment AND External_Atomic_Lock_Required AND DCR.lc THEN IA-32_Intercept(LOCK,CMPXCHG);

IF (EDX:EAX = DEST) THEN
   ZF ← 1
   DEST ← ECX:EBX
ELSE
   ZF ← 0
   EDX:EAX ← DEST

**Flags Affected**

The ZF flag is set if the destination operand and EDX:EAX are equal; otherwise it is cleared. The CF, PF, AF, SF, and OF flags are unaffected.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**
  - NaT Register Consumption Abort.
- **Itanium Mem Faults**
  - VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
- **IA-32_Intercept**
  - Lock Intercept - If an external atomic bus lock is required to complete this operation and DCR.lc is 1, no atomic transaction occurs, this instruction is faulted and an IA-32_Intercept(Lock) fault is generated. The software lock handler is responsible for the emulation of this instruction
### CMPXCHG8B—Compare and Exchange 8 Bytes (continued)

#### Protected Mode Exceptions

- **#UD** If the destination operand is not a memory location.
- **#GP(0)** If the destination is located in a nonwritable segment.
  
  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  
  If the DS, ES, FS, or GS register contains a null segment selector.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

#### Real Address Mode Exceptions

- **#GP** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS** If a memory operand effective address is outside the SS segment limit.

#### Virtual 8086 Mode Exceptions

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made.

#### Intel Architecture Compatibility

This instruction is not supported on Intel processors earlier than the Pentium processors.
CPUID—CPU Identification

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F A2</td>
<td>CPUID</td>
<td>EAX ← Processor identification information</td>
</tr>
</tbody>
</table>

**Description**

Provides processor identification information in registers EAX, EBX, ECX, and EDX. This information identifies Intel as the vendor, gives the family, model, and stepping of processor, feature information, and cache information. An input value loaded into the EAX register determines what information is returned, as shown in Table 1-4.

**Table 1-4. Information Returned by CPUID Instruction**

<table>
<thead>
<tr>
<th>Initial EAX Value</th>
<th>Information Provided about the Processor</th>
</tr>
</thead>
</table>
| 0                 | EAX  Maximum CPUID Input Value  
|                   | EBX  756E6547H “Genu” (G in BL)  
|                   | ECX  6C65746EH “ntel” (n in CL)  
|                   | EDX  49656E69H “inel” (i in DL)  |
| 1                 | EAX  Version Information (Family, Model, and Stepping ID)  
|                   | EBX  Reserved  
|                   | ECX  Reserved  
|                   | EDX  Feature Information  |
| 2                 | EAX  Cache Information  
|                   | EBX  Cache Information  
|                   | ECX  Cache Information  
|                   | EDX  Cache Information  |

The CPUID instruction can be executed at any privilege level to serialize instruction execution. Serializing instruction execution guarantees that any modifications to flags, registers, and memory for previous instructions are completed before the next instruction is fetched and executed.

When the input value in register EAX is 0, the processor returns the highest value the CPUID instruction recognizes in the EAX register. For the Itanium processor, the highest recognized value is 2. A vendor identification string is returned in the EBX, EDX, and ECX registers. For Intel processors, the vendor identification string is “GenuineIntel” as follows:

EBX ← 756e6547h (* “Genu”, with G in the low nibble of BL *)  
EDX ← 49656e69h (* “inel”, with i in the low nibble of DL *)  
ECX ← 6c65746eh (* “ntel”, with n in the low nibble of CL *)

When the input value is 1, the processor returns version information in the EAX register and feature information in the EDX register.
CPUID—CPU Identification (continued)

Figure 1-3. Version Information in Registers EAX

<table>
<thead>
<tr>
<th>Bit</th>
<th>IA-32 System Environment Feature</th>
<th>Intel® Itanium™ System Environment Feature</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>FPU—Floating-point Unit on Chip</td>
<td>Available</td>
<td>Processor contains an FPU and executes the Intel387 instruction set. Processor supports the following virtual 8086 mode enhancements: • CR4.VME bit enables virtual 8086 mode extensions. • CR4.PVI bit enables protected-mode virtual interrupts. • Expansion of the TSS with the software indirection bitmap. • EFLAGS.VIF bit enables the virtual interrupt flag. • EFLAGS.VIP bit enables the virtual interrupt pending flag.</td>
</tr>
<tr>
<td>1</td>
<td>VME—Virtual 8086 Mode Enhancements</td>
<td>Available</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>PSE—Page Size Extensions</td>
<td>Not Available</td>
<td>Processor supports 4-Mbyte pages, including the CR4.PSE bit for enabling page size extensions, the modified bit in page directory entries (PDEs), page directory entries, and page table entries (PTEs).</td>
</tr>
<tr>
<td>4</td>
<td>TSC—Time Stamp Counter</td>
<td>Available</td>
<td>Processor supports the RDTSC (read time stamp counter) instruction, including the CR4.TSD bit that, along with the CPL, controls whether the time stamp counter can be read.</td>
</tr>
<tr>
<td>5</td>
<td>MSR—Model Specific Registers</td>
<td>Not Available</td>
<td>Processor supports the RDMSR (read model-specific register) and WRMSR (write model-specific register) instructions.</td>
</tr>
</tbody>
</table>

The version information consists of an Intel Architecture extended family identifier, an extended model identifier, an Intel Architecture family identifier, a model identifier and a stepping ID.

See “Intel Application Note 485 — Intel Processor Identification with the CPUID Instruction” for more information on identifying earlier Intel Architecture processors. Intel releases information on stepping IDs as needed.

A feature flag set to 1 indicates the corresponding feature is supported. Software should identify Intel as the vendor to properly interpret the feature flags.

Note: IA-32 System Environment feature bits may be set in the Itanium System Environment even if that feature cannot be used within the Itanium System Environment.

Table 1-5. Feature Flags Returned in EDX Register

<table>
<thead>
<tr>
<th>Bit</th>
<th>IA-32 System Environment Feature</th>
<th>Intel® Itanium™ System Environment Feature</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>FPU—Floating-point Unit on Chip</td>
<td>Available</td>
<td>Processor contains an FPU and executes the Intel387 instruction set. Processor supports the following virtual 8086 mode enhancements: • CR4.VME bit enables virtual 8086 mode extensions. • CR4.PVI bit enables protected-mode virtual interrupts. • Expansion of the TSS with the software indirection bitmap. • EFLAGS.VIF bit enables the virtual interrupt flag. • EFLAGS.VIP bit enables the virtual interrupt pending flag.</td>
</tr>
<tr>
<td>1</td>
<td>VME—Virtual 8086 Mode Enhancements</td>
<td>Available</td>
<td></td>
</tr>
<tr>
<td>3</td>
<td>PSE—Page Size Extensions</td>
<td>Not Available</td>
<td>Processor supports 4-Mbyte pages, including the CR4.PSE bit for enabling page size extensions, the modified bit in page directory entries (PDEs), page directory entries, and page table entries (PTEs).</td>
</tr>
<tr>
<td>4</td>
<td>TSC—Time Stamp Counter</td>
<td>Available</td>
<td>Processor supports the RDTSC (read time stamp counter) instruction, including the CR4.TSD bit that, along with the CPL, controls whether the time stamp counter can be read.</td>
</tr>
<tr>
<td>5</td>
<td>MSR—Model Specific Registers</td>
<td>Not Available</td>
<td>Processor supports the RDMSR (read model-specific register) and WRMSR (write model-specific register) instructions.</td>
</tr>
</tbody>
</table>
### Table 1-5. Feature Flags Returned in EDX Register (Continued)

<table>
<thead>
<tr>
<th>Bit</th>
<th>IA-32 System Environment Feature</th>
<th>Intel® Itanium™ System Environment Feature</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>6</td>
<td>PAE—Physical Address Extension</td>
<td>Not Available. The Itanium architecture always supports more than 32-bits of physical addressing</td>
<td>Processor supports physical addresses greater than 32 bits, the extended page-table-entry format, an extra level in the page translation tables, and 2-MByte pages. The CR4.PAE bit enables this feature. The number of address bits is implementation specific.</td>
</tr>
<tr>
<td>7</td>
<td>MCE—Machine Check Exception</td>
<td>Not Available. Processor uses PAL defined MCHK architecture.</td>
<td>Processor supports the CR4.MCE bit, enabling machine check exceptions. However, this feature does not define the model-specific implementations of machine-check error logging, reporting, or processor shutdowns. Machine-check exception handlers might have to check the processor version to do model-specific processing of the exception or check for the presence of the machine-check feature.</td>
</tr>
<tr>
<td>8</td>
<td>CX8—CMPXCHG8B Instruction</td>
<td>Available</td>
<td>Processor supports the CMPXCHG8B (compare and exchange 8 bytes) instruction.</td>
</tr>
<tr>
<td>9</td>
<td>APIC</td>
<td>Not Available. Replaced by the Itanium interrupt mechanism.</td>
<td>Processor contains an on-chip Advanced Programmable Interrupt Controller (APIC) and it has been enabled and is available for use.</td>
</tr>
<tr>
<td>10</td>
<td>Reserved</td>
<td></td>
<td>returns zero</td>
</tr>
<tr>
<td>12</td>
<td>MTRR—Memory Type Range Registers</td>
<td>Not Available. Processor utilizes memory attributes from the TLB defined by the Itanium architecture</td>
<td>Processor supports machine-specific memory-type range registers (MTRRs). The MTRRs contains bit fields that indicate the processor’s MTRR capabilities, including which memory types the processor supports, the number of variable MTRRs the processor supports, and whether the processor supports fixed MTRRs.</td>
</tr>
<tr>
<td>13</td>
<td>PGE—PTE Global Flag</td>
<td>Not available. Superceded by virtual regions defined in the Itanium architecture.</td>
<td>Processor supports the CR4.PGE flag enabling the global bit in both PTDEs and PTEs. These bits are used to indicate translation lookaside buffer (TLB) entries that are common to different tasks and need not be flushed when control register CR3 is written.</td>
</tr>
<tr>
<td>14</td>
<td>MCA—Machine Check Architecture</td>
<td>Not Available. Processor uses PAL defined MCHK architecture.</td>
<td>Processor supports the MCG_CAP (machine check global capability) MSR. The MCG_CAP register indicates how many banks of error reporting MSRs the processor supports.</td>
</tr>
<tr>
<td>15</td>
<td>CMOV—Conditional Move and Compare Instructions</td>
<td>Available</td>
<td>Processor supports the CMOVcc instruction and, if the FPU feature flag (bit 0) is also set, supports the FCMOVcc and FCOMI instructions.</td>
</tr>
<tr>
<td>16</td>
<td>PSE-36 - 36-bit Page Size Extensions</td>
<td>Not Available. The Itanium architecture always supports more than 32-bits of physical addressing</td>
<td>Indicates whether the processor supports 4-MByte pages that are capable of addressing physical memory beyond 4GB. This feature indicates that the upper four bits of the physical address of the 4-Mbyte page is encoded by bits 13-16 of the page directory entry.</td>
</tr>
<tr>
<td>17</td>
<td>PAT - Memory Attribute Palette</td>
<td>Not available for paging. Superceded by virtual regions defined in the Intel® Itanium™ architecture.</td>
<td>Processors supports the IA-32 physical attribute table</td>
</tr>
<tr>
<td>18</td>
<td>PPN - Physical Processor Number</td>
<td>Not available. The Intel® Itanium™ architecture does not support this feature.</td>
<td>Processor supports a Physical Processor Number for each manufactured processor</td>
</tr>
<tr>
<td>23</td>
<td>MMX - Intel® MMX™ Technology</td>
<td>Available</td>
<td>Processor supports the Architecture Intel® MMX™ Technology</td>
</tr>
</tbody>
</table>
### Table 1-5. Feature Flags Returned in EDX Register (Continued)

<table>
<thead>
<tr>
<th>Bit</th>
<th>IA-32 System Environment Feature</th>
<th>Intel® Itanium™ System Environment Feature</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>24</td>
<td>FXSR</td>
<td>Available</td>
<td>Processor supports the Streaming SIMD Extension FXRSTOR and FXSAVE instruction</td>
</tr>
<tr>
<td>25</td>
<td>XMM - Streaming SIMD Extension Technology</td>
<td>Available</td>
<td>Processor supports the Intel Architecture Streaming SIMD Extension</td>
</tr>
<tr>
<td>30</td>
<td>Processor based on the Intel® Itanium™ architecture</td>
<td>Available</td>
<td>The processor is based on the Intel® Itanium™ architecture and is capable of executing the Intel® Itanium™ instruction set. IA-32 application level software MUST also check with the running operating system to see if the system can also support Itanium™-based code before switching to the Intel® Itanium™ instruction set.</td>
</tr>
</tbody>
</table>
CPUID—CPU Identification (continued)

When the input value is 2, the processor returns information about the processor’s internal caches and TLBs in the EAX, EBX, ECX, and EDX registers. The encoding of these registers is as follows:

- The least-significant byte in register EAX (register AL) indicates the number of times the CPUID instruction must be executed with an input value of 2 to get a complete description of the processor’s caches and TLBs.
- The most significant bit (bit 31) of each register indicates whether the register contains valid information (cleared to 0) or is reserved (set to 1).

Please see the processor specific supplement for further information on how to decode the return values for the processors internal caches and TLBs.

CPUID performs instruction serialization and a memory fence operation.

Operation

CASE (EAX) OF
EAX = 0:
   EAX ← highest input value understood by CPUID; (* 2 for Itanium processor *)
   EBX ← Vendor identification string;
   EDX ← Vendor identification string;
   ECX ← Vendor identification string;
   BREAK;
EAX = 1:
   EAX[3:0] ← Stepping ID;
   EAX[7:4] ← Model;
   EAX[11:8] ← Family;
   EAX[15:12] ← Reserved;
   EAX[19:16] ← Extended Model ID;
   EAX[27:20] ← Extended Family ID;
   EAX[31:28] ← Reserved;
   EBX ← Reserved;
   ECX ← Reserved;
   EDX ← Feature flags;
   BREAK;
EAX = 2:
   EAX[7:0] ← N_Param_Descrip_Blocks = 1;
   EAX[31:8], EBX, ECX, EDX = cache and TLB parameters
   BREAK;
DEFAULT: (* EAX > highest value recognized by CPUID *)
   EAX ← Reserved, Undefined;
   EBX ← Reserved, Undefined;
   ECX ← Reserved, Undefined;
   EDX ← Reserved, Undefined;
   BREAK;
ESAC;

memory_fence();
instruction_serialize();

Flags Affected
None.
CPUID—CPU Identification (continued)

**Additional Itanium System Environment Exceptions**
Itanium Reg Faults  NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**
None.

**Intel Architecture Compatibility**

The CPUID instruction is not supported in early models of the Intel486 processor or in any Intel Architecture processor earlier than the Intel486 processor. The ID flag in the EFLAGS register can be used to determine if this instruction is supported. If a procedure is able to set or clear this flag, the CPUID is supported by the processor running the procedure.
CWD/CDQ—Convert Word to Doubleword/Convert Doubleword to Quadword

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>99</td>
<td>CWD</td>
<td>DX:AX ← sign-extend of AX</td>
</tr>
<tr>
<td>99</td>
<td>CDQ</td>
<td>EDX:EAX ← sign-extend of EAX</td>
</tr>
</tbody>
</table>

Description

Doubles the size of the operand in register AX or EAX (depending on the operand size) by means of sign extension and stores the result in registers DX:AX or EDX:EAX, respectively. The CWD instruction copies the sign (bit 15) of the value in the AX register into every bit position in the DX register. The CDQ instruction copies the sign (bit 31) of the value in the EAX register into every bit position in the EDX register.

The CWD instruction can be used to produce a doubleword dividend from a word before a word division, and the CDQ instruction can be used to produce a quadword dividend from a doubleword before doubleword division.

The CWD and CDQ mnemonics reference the same opcode. The CWD instruction is intended for use when the operand-size attribute is 16 and the CDQ instruction for when the operand-size attribute is 32. Some assemblers may force the operand size to 16 when CWD is used and to 32 when CDQ is used. Others may treat these mnemonics as synonyms (CWD/CDQ) and use the current setting of the operand-size attribute to determine the size of values to be converted, regardless of the mnemonic used.

Operation

IF OperandSize = 16 (* CWD instruction *)
   THEN DX ← SignExtend(AX);
ELSE (* OperandSize = 32, CDQ instruction *)
   EDX ← SignExtend(EAX);
FI;

Additional Itanium System Environment Exceptions

Itanium Reg Faults    NaT Register Consumption Abort.

Flags Affected

None.

Exceptions (All Operating Modes)

None.
CWDE—Convert Word to Doubleword

See entry for CBW/CWDE—Convert Byte to Word/Convert Word to Doubleword.
DAA—Decimal Adjust AL after Addition

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>27</td>
<td>DAA</td>
<td>Decimal adjust AL after addition</td>
</tr>
</tbody>
</table>

**Description**

Adjusts the sum of two packed BCD values to create a packed BCD result. The AL register is the implied source and destination operand. The DAA instruction is only useful when it follows an ADD instruction that adds (binary addition) two 2-digit, packed BCD values and stores a byte result in the AL register. The DAA instruction then adjusts the contents of the AL register to contain the correct 2-digit, packed BCD result. If a decimal carry is detected, the CF and AF flags are set accordingly.

**Operation**

```plaintext
IF (((AL AND 0FH) > 9) or AF = 1)
    THEN
        AL ← AL + 6;
        CF ← CF OR CarryFromLastAddition; (* CF OR carry from AL ← AL + 6 *)
        AF ← 1;
    ELSE
        AF ← 0;
    FI;
IF ((AL AND F0H) > 90H) or CF = 1)
    THEN
        AL ← AL + 60H;
        CF ← 1;
    ELSE
        CF ← 0;
    FI;
```

**Example**

ADD AL, BL  
Before: AL=79H BL=35H EFLAGS(OSZAPC)=XXXXXX  
After: AL=AEH BL=35H EFLAGS(OSZAPC)=110000

DAA  
Before: AL=79H BL=35H EFLAGS(OSZAPC)=110000  
After: AL=AEH BL=35H EFLAGS(OSZAPC)=X00111

**Flags Affected**

The CF and AF flags are set if the adjustment of the value results in a decimal carry in either digit of the result (see “Operation” above). The SF, ZF, and PF flags are set according to the result. The OF flag is undefined.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.
DAS—Decimal Adjust AL after Subtraction

Description

Adjusts the result of the subtraction of two packed BCD values to create a packed BCD result. The AL register is the implied source and destination operand. The DAS instruction is only useful when it follows a SUB instruction that subtracts (binary subtraction) one 2-digit, packed BCD value from another and stores a byte result in the AL register. The DAS instruction then adjusts the contents of the AL register to contain the correct 2-digit, packed BCD result. If a decimal borrow is detected, the CF and AF flags are set accordingly.

Operation

\[
\begin{align*}
\text{IF } (\text{AL AND 0FH}) > 9 \text{ OR AF } = 1 & \quad \text{THEN} \\
\text{AL} & \leftarrow \text{AL} - 6; \\
\text{CF} & \leftarrow \text{CF OR BorrowFromLastSubtraction}; \\
\text{AF} & \leftarrow 1; \\
\text{ELSE} & \text{AF } \leftarrow 0; \\
\text{FI}; \\
\text{IF } ((\text{AL } > 9FH) \text{ OR CF } = 1) & \quad \text{THEN} \\
\text{AL} & \leftarrow \text{AL} - 60H; \\
\text{CF} & \leftarrow 1; \\
\text{ELSE} & \text{CF } \leftarrow 0; \\
\text{FI};
\end{align*}
\]

Example

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Before:</th>
<th>After:</th>
</tr>
</thead>
<tbody>
<tr>
<td>SUB AL, BL</td>
<td>AL=35H BL=47H EFLAGS(OSZAPC)=XXXXXX</td>
<td>AL=EEH BL=47H EFLAGS(OSZAPC)=010111</td>
</tr>
<tr>
<td>DAA</td>
<td>AL=EEH BL=47H EFLAGS(OSZAPC)=010111</td>
<td>AL=88H BL=47H EFLAGS(OSZAPC)=X10111</td>
</tr>
</tbody>
</table>

Flags Affected

The CF and AF flags are set if the adjustment of the value results in a decimal borrow in either digit of the result (see “Operation” above). The SF, ZF, and PF flags are set according to the result. The OF flag is undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.

Exceptions (All Operating Modes)

None.
DEC—Decrement by 1

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FE /1</td>
<td>DEC r/m8</td>
<td>Decrement r/m8 by 1</td>
</tr>
<tr>
<td>FF /1</td>
<td>DEC r/m16</td>
<td>Decrement r/m16 by 1</td>
</tr>
<tr>
<td>FF /1</td>
<td>DEC r/m32</td>
<td>Decrement r/m32 by 1</td>
</tr>
<tr>
<td>48+rw</td>
<td>DEC r16</td>
<td>Decrement r16 by 1</td>
</tr>
<tr>
<td>48+rd</td>
<td>DEC r32</td>
<td>Decrement r32 by 1</td>
</tr>
</tbody>
</table>

**Description**

Subtracts 1 from the operand, while preserving the state of the CF flag. The source operand can be a register or a memory location. This instruction allows a loop counter to be updated without disturbing the CF flag. (Use a SUB instruction with an immediate operand of 1 to perform a decrement operation that does updates the CF flag.)

**Operation**

DEST ← DEST – 1;

**Flags Affected**

The CF flag is not affected. The OF, SF, ZF, AF, and PF flags are set according to the result.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**: NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Protected Mode Exceptions**

- #GP(0) If the destination is located in a nonwritable segment.
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- #SS(0) If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code) If a page fault occurs.
- #AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

- #GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- #SS If a memory operand effective address is outside the SS segment limit.
DEC—Decrement by 1 (continued)

Virtual 8086 Mode Exceptions

#GP(0)    If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)    If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0)    If alignment checking is enabled and an unaligned memory reference is made.
DIV—Unsigned Divide

Description

Divides (unsigned) the value in the AL, AX, or EAX register (dividend) by the source operand (divisor) and stores the result in the AX, DX:AX, or EDX:EAX registers. The source operand can be a general-purpose register or a memory location. The action of this instruction depends on the operand size, as shown in the following table:

<table>
<thead>
<tr>
<th>Operand Size</th>
<th>Dividend</th>
<th>Divisor</th>
<th>Quotient</th>
<th>Remainder</th>
<th>Maximum Quotient</th>
</tr>
</thead>
<tbody>
<tr>
<td>Word/byte</td>
<td>AX</td>
<td>r/m8</td>
<td>AL</td>
<td>AH</td>
<td>255</td>
</tr>
<tr>
<td>Doubleword/word</td>
<td>DX:AX</td>
<td>r/m16</td>
<td>AX</td>
<td>DX</td>
<td>65,535</td>
</tr>
<tr>
<td>Quadword/doubleword</td>
<td>EDX:EAX</td>
<td>r/m32</td>
<td>EAX</td>
<td>EDX</td>
<td>$2^{32} - 1$</td>
</tr>
</tbody>
</table>

Non-integral results are truncated (chopped) towards 0. The remainder is always less than the divisor in magnitude. Overflow is indicated with the #DE (divide error) exception rather than with the CF flag.

Operation

IF SRC = 0
    THEN #DE; (* divide error *)
FI;
IF OpernadSize = 8 (* word/byte operation *)
    THEN
        temp ← AX / SRC;
        IF temp > FFH
            THEN #DE; (* divide error *)
            ELSE
                AL ← temp;
                AH ← AX MOD SRC;
        FI;
    ELSE
        IF OpernadSize = 16 (* doubleword/word operation *)
            THEN
                temp ← DX:AX / SRC;
                IF temp > FFFFH
                    THEN #DE; (* divide error *)
                    ELSE
                        AX ← temp;
                        DX ← DX:AX MOD SRC;
                FI;
DIV—Unsigned Divide (continued)

ELSE (* quadword/doubleword operation *)
  temp ← EDX:EAX / SRC;
  IF temp > FFFFFFFFH
    THEN #DE; (* divide error *) ;
    ELSE
      EAX ← temp;
      EDX ← EDX:EAX MOD SRC;
    FI;
  FI;
FI;

Flags Affected

The CF, OF, SF, ZF, AF, and PF flags are undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults       NaT Register Consumption Abort.
Itanium Mem Faults      VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB
                        Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data
                        Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data
                        Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#DE        If the source operand (divisor) is 0
          If the quotient is too large for the designated register.
#GP(0)     If a memory operand effective address is outside the CS, DS, ES, FS, or GS
          segment limit.
          If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0)     If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0)     If alignment checking is enabled and an unaligned memory reference is made
          while the current privilege level is 3.

Real Address Mode Exceptions

#DE        If the source operand (divisor) is 0.
          If the quotient is too large for the designated register.
#GP        If a memory operand effective address is outside the CS, DS, ES, FS, or GS
          segment limit.
          If the DS, ES, FS, or GS register contains a null segment selector.
DIV—Unsigned Divide (continued)

Virtual 8086 Mode Exceptions

#DE If the source operand (divisor) is 0.

If the quotient is too large for the designated register.

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
ENTER—Make Stack Frame for Procedure Parameters

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>C8 iw 00</td>
<td>ENTER imm16,0</td>
<td>Create a stack frame for a procedure</td>
</tr>
<tr>
<td>C8 iw 01</td>
<td>ENTER imm16,1</td>
<td>Create a nested stack frame for a procedure</td>
</tr>
<tr>
<td>C8 iw 0b</td>
<td>ENTER imm16,imm8</td>
<td>Create a nested stack frame for a procedure</td>
</tr>
</tbody>
</table>

**Description**

Creates a stack frame for a procedure. The first operand (size operand) specifies the size of the stack frame (that is, the number of bytes of dynamic storage allocated on the stack for the procedure). The second operand (nesting level operand) gives the lexical nesting level (0 to 31) of the procedure. The nesting level determines the number of stack frame pointers that are copied into the “display area” of the new stack frame from the preceding frame. Both of these operands are immediate values.

The stack-size attribute determines whether the BP (16 bits) or EBP (32 bits) register specifies the current frame pointer and whether SP (16 bits) or ESP (32 bits) specifies the stack pointer.

The ENTER and companion LEAVE instructions are provided to support block structured languages. They do not provide a jump or call to another procedure; they merely set up a new stack frame for an already called procedure. An ENTER instruction is commonly followed by a CALL, JMP, or Jcc instruction to transfer program control to the procedure being called.

If the nesting level is 0, the processor pushes the frame pointer from the EBP register onto the stack, copies the current stack pointer from the ESP register into the EBP register, and loads the ESP register with the current stack-pointer value minus the value in the size operand. For nesting levels of 1 or greater, the processor pushes additional frame pointers on the stack before adjusting the stack pointer. These additional frame pointers provide the called procedure with access points to other nested frames on the stack.

**Operation**

```plaintext
NestingLevel ← NestingLevel MOD 32
IF StackSize = 32
    THEN
        Push(EBP);
        FrameTemp ← ESP;
    ELSE (* StackSize = 16*)
        Push(BP);
        FrameTemp ← SP;
    FI;
IF NestingLevel = 0
    THEN GOTO CONTINUE;
FI;
IF (NestingLevel > 0)
    FOR i ← 1 TO (NestingLevel − 1)
        DO
            IF OperandSize = 32
                THEN
                    IF StackSize = 32
                        EBP ← EBP − 4;
                    FI;
                FI;
        FI;
```

**Opcode Instruction Description**

- C8 iw 00 ENTER imm16,0: Create a stack frame for a procedure
- C8 iw 01 ENTER imm16,1: Create a nested stack frame for a procedure
- C8 iw 0b ENTER imm16,imm8: Create a nested stack frame for a procedure
ENTER—Make Stack Frame for Procedure Parameters (continued)

Push([EBP]); (* doubleword push *)

ELSE (* StackSize = 16*)
    BP ← BP − 4;
    Push([BP]); (* doubleword push *)
FI;
ELSE (* OperandSize = 16 *)
    IF StackSize = 32
        THEN
            EBP ← EBP − 2;
            Push([EBP]); (* word push *)
        ELSE (* StackSize = 16*)
            BP ← BP − 2;
            Push([BP]); (* word push *)
        FI;
    FI;
OD;
IF OperandSize = 32
    THEN
        Push(FrameTemp); (* doubleword push *)
    ELSE (* OperandSize = 16 *)
        Push(FrameTemp); (* word push *)
    FI;
GOTO CONTINUE;
FI;
CONTINUE:
IF StackSize = 32
    THEN
        EBP ← FrameTemp
        ESP ← EBP − Size;
    ELSE (* StackSize = 16*)
        BP ← FrameTemp
        SP ← BP − Size;
    FI;
END;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults NaT Register Consumption Abort.

Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
ENTER—Make Stack Frame for Procedure Parameters (continued)

Protected Mode Exceptions

#SS(0) If the new value of the SP or ESP register is outside the stack segment limit.
#PF(fault-code) If a page fault occurs.

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

None.
F2XM1—Compute $2^x - 1$

**Description**

Calculates the exponential value of 2 to the power of the source operand minus 1. The source operand is located in register ST(0) and the result is also stored in ST(0). The value of the source operand must lie in the range $-1.0$ to $+1.0$. If the source value is outside this range, the result is undefined.

The following table shows the results obtained when computing the exponential value of various classes of numbers, assuming that neither overflow nor underflow occurs:

<table>
<thead>
<tr>
<th>ST(0) SRC</th>
<th>ST(0) DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>-1.0 to 0</td>
<td>0.5 to 0</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>+0</td>
<td>+0</td>
</tr>
<tr>
<td>+0 to 1</td>
<td>0 to 1</td>
</tr>
</tbody>
</table>

Values other than 2 can be exponentiated using the following formula:

$$x^y = 2^{(y \times \log_2 x)}$$

**Operation**

$$ST(0) \leftarrow (2^{ST(0)} - 1);$$

**FPU Flags Affected**

- **C1**: Set to 0 if stack underflow occurred.
- **C0, C2, C3**: Undefined.

Indicates rounding direction if the inexact-result exception (#P) is generated:

0 = not roundup; 1 = roundup.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

**Floating-point Exceptions**

- **#IS**: Stack underflow occurred.
- **#IA**: Source operand is an SNaN value or unsupported format.
- **#D**: Result is a denormal value.
- **#U**: Result is too small for destination format.
- **#P**: Value cannot be represented exactly in destination format.
F2XM1—Compute $2^x - 1$ (continued)

**Protected Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Real Address Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

#NM  EM or TS in CR0 is set.
FABS—Absolute Value

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 E1</td>
<td>FABS</td>
<td>Replace ST with its absolute value.</td>
</tr>
</tbody>
</table>

Description

Clears the sign bit of ST(0) to create the absolute value of the operand. The following table shows the results obtained when creating the absolute value of various classes of numbers.

<table>
<thead>
<tr>
<th>ST(0) SRC</th>
<th>ST(0) DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>←</td>
<td>+∞</td>
</tr>
<tr>
<td>−F</td>
<td>+F</td>
</tr>
<tr>
<td>−0</td>
<td>+0</td>
</tr>
<tr>
<td>+0</td>
<td>+0</td>
</tr>
<tr>
<td>+F</td>
<td>+F</td>
</tr>
<tr>
<td>+∞</td>
<td>+∞</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Note:
F means finite-real number.

Operation

ST(0) ← |ST(0)|

FPU Flags Affected

C1          Set to 0 if stack underflow occurred; otherwise, cleared to 0.
C0, C2, C3  Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Floating-point Exceptions

#IS          Stack underflow occurred.

Protected Mode Exceptions

#NM          EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM          EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM          EM or TS in CR0 is set.
FADD/FADDP/FIADD—Add

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D8 /0</td>
<td>FADD m32 real</td>
<td>Add m32real to ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DC /0</td>
<td>FADD m64real</td>
<td>Add m64real to ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>D8 C0+i</td>
<td>FADD ST(0), ST(i)</td>
<td>Add ST(0) to ST(i) and store result in ST(i)</td>
</tr>
<tr>
<td>DC C0+i</td>
<td>FADD ST(i), ST(0)</td>
<td>Add ST(i) to ST(0) and store result in ST(i)</td>
</tr>
<tr>
<td>DE C0+i</td>
<td>FADDP ST(i), ST(0)</td>
<td>Add ST(0) to ST(i), store result in ST(i), and pop the register stack</td>
</tr>
<tr>
<td>DE C1</td>
<td>FADDP</td>
<td>Add ST(0) to ST(1), store result in ST(1), and pop the register stack</td>
</tr>
<tr>
<td>DA /0</td>
<td>FIADD m32int</td>
<td>Add m32int to ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DE /0</td>
<td>FIADD m16int</td>
<td>Add m16int to ST(0) and store result in ST(0)</td>
</tr>
</tbody>
</table>

Description

Adds the destination and source operands and stores the sum in the destination location. The destination operand is always an FPU register; the source operand can be a register or a memory location. Source operands in memory can be in single-real, double-real, word-integer, or short-integer formats.

The no-operand version of the instruction adds the contents of the ST(0) register to the ST(1) register. The one-operand version adds the contents of a memory location (either a real or an integer value) to the contents of the ST(0) register. The two-operand version adds the contents of the ST(0) register to the ST(i) register or vice versa. The value in ST(0) can be doubled by coding:

FADD ST(0), ST(0);

The FADDP instructions perform the additional operation of popping the FPU register stack after storing the result. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. (The no-operand version of the floating-point add instructions always results in the register stack being popped. In some assemblers, the mnemonic for this instruction is FADD rather than FADDP.)

The FIADD instructions convert an integer source operand to extended-real format before performing the addition.

The table on the following page shows the results obtained when adding various classes of numbers, assuming that neither overflow nor underflow occurs.

When the sum of two operands with opposite signs is 0, the result is +0, except for the round toward −∞ mode, in which case the result is −0. When the source operand is an integer 0, it is treated as a +0.

When both operand are infinities of the same sign, the result is ±∞ of the expected sign. If both operands are infinities of opposite signs, an invalid-operation exception is generated.
FADD/FADDP/FIADD—Add (continued)

<table>
<thead>
<tr>
<th>SRC</th>
<th>DEST</th>
<th>+F</th>
<th>+0</th>
<th>-0</th>
<th>*</th>
<th>-∞</th>
<th>-F</th>
<th>-∞</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>-∞</td>
<td>-∞</td>
<td>-∞</td>
<td>-∞</td>
<td>-∞</td>
<td>-∞</td>
<td>NaN</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>-F or -I</td>
<td>-F</td>
<td>SRC</td>
<td>SRC</td>
<td>±F or ±0</td>
<td>+∞</td>
<td>NaN</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>-0</td>
<td>DEST</td>
<td>-0</td>
<td>±0</td>
<td>DEST</td>
<td>+∞</td>
<td>NaN</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>+0</td>
<td>DEST</td>
<td>±0</td>
<td>+0</td>
<td>DEST</td>
<td>+∞</td>
<td>NaN</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>+For +I</td>
<td>±F or ±0</td>
<td>SRC</td>
<td>SRC</td>
<td>+F</td>
<td>+∞</td>
<td>NaN</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>NaN</td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Notes:
- F means finite-real number.
- L means integer.
- * indicates floating-point invalid-arithmetic-operand (#IA) exception.

Operation
IF instruction is FIADD
THEN
   DEST ← DEST + ConvertExtendedReal(SRC);
ELSE (* source operand is real number *)
   DEST ← DEST + SRC;
FI;

IF instruction = FADDP
THEN
   PopRegisterStack;
FI;

FPU Flags Affected
C1 Set to 0 if stack underflow occurred.
   Indicates rounding direction if the inexact-result exception (#P) is generated:
   0 = not roundup; 1 = roundup.

C0, C2, C3 Undefined.

Additional Itanium System Environment Exceptions
Itanium Reg Faults Disabled FP Register Fault if PSR.df1 is 1, NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
FADD/FADDP/FIADD—Add (continued)

Floating-point Exceptions

#IS Stack underflow occurred.
#IA Operand is an SNaN value or unsupported format.
   Operands are infinities of unlike sign.
#D Result is a denormal value.
#U Result is too small for destination format.
#O Result is too large for destination format.
#P Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
   If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
FBLD—Load Binary Coded Decimal

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DF /4</td>
<td>FBLD m80 dec</td>
<td>Convert BCD value to real and push onto the FPU stack.</td>
</tr>
</tbody>
</table>

**Description**

Converts the BCD source operand into extended-real format and pushes the value onto the FPU stack. The source operand is loaded without rounding errors. The sign of the source operand is preserved, including that of −0.

The packed BCD digits are assumed to be in the range 0 through 9; the instruction does not check for invalid digits (AH through FH). Attempting to load an invalid encoding produces an undefined result.

**Operation**

\[
\text{TOP} \leftarrow \text{TOP} - 1; \\
\text{ST}(0) \leftarrow \text{ExtendedReal}(\text{SRC}); \\
\]

**FPU Flags Affected**

- **C1** Set to 1 if stack overflow occurred; otherwise, cleared to 0.
- **C0, C2, C3** Undefined.

**Floating-point Exceptions**

- **#IS** Stack overflow occurred.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults** Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- **Itanium Mem Faults** VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Protected Mode Exceptions**

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#NM** EM or TS in CR0 is set.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
FBLD—Load Binary Coded Decimal (continued)

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
FBSTP—Store BCD Integer and Pop

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DF /6</td>
<td>FBSTP m80bcd</td>
<td>Store ST(0) in m80bcd and pop ST(0).</td>
</tr>
</tbody>
</table>

**Description**

Converts the value in the ST(0) register to an 18-digit packed BCD integer, stores the result in the destination operand, and pops the register stack. If the source value is a non-integral value, it is rounded to an integer value, according to rounding mode specified by the RC field of the FPU control word. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1.

The destination operand specifies the address where the first byte destination value is to be stored. The BCD value (including its sign bit) requires 10 bytes of space in memory.

The following table shows the results obtained when storing various classes of numbers in packed BCD format.

<table>
<thead>
<tr>
<th>ST(0)</th>
<th>DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>-*</td>
<td>-*</td>
</tr>
<tr>
<td>-F &lt; -1</td>
<td>-D</td>
</tr>
<tr>
<td>-1 &lt; -F &lt; -0</td>
<td>**</td>
</tr>
<tr>
<td>-0</td>
<td>-0</td>
</tr>
<tr>
<td>+0</td>
<td>+0</td>
</tr>
<tr>
<td>+0 &lt; +F &lt; +1</td>
<td>**</td>
</tr>
<tr>
<td>+F &gt; +1</td>
<td>+D</td>
</tr>
<tr>
<td>+=</td>
<td>*=</td>
</tr>
<tr>
<td>NaN</td>
<td>*</td>
</tr>
</tbody>
</table>

Notes:
- F means finite-real number.
- D means packed-BCD number.
- * indicates floating-point invalid-operation (#IA) exception.
- ** ±0 or ±1, depending on the rounding mode.

If the source value is too large for the destination format and the invalid-operation exception is not masked, an invalid-operation exception is generated and no value is stored in the destination operand. If the invalid-operation exception is masked, the packed BCD indefinite value is stored in memory.

If the source value is a quiet NaN, an invalid-operation exception is generated. Quiet NaNs do not normally cause this exception to be generated.

**Operation**

DEST ← BCD(ST(0));
PpopRegisterStack;
FBSTP—Store BCD Integer and Pop (continued)

FPU Flags Affected

C1
Set to 0 if stack underflow occurred.
Indicates rounding direction if the inexact exception (#P) is generated: 0 = not roundup; 1 = roundup.

C0, C2, C3
Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults
Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults
VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Floating-point Exceptions

#IS
Stack underflow occurred.

#IA
Source operand is empty; contains a NaN, ±∞, or unsupported format; or contains value that exceeds 18 BCD digits in length.

#P
Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#GP(0)
If a segment register is being loaded with a segment selector that points to a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0)
If a memory operand effective address is outside the SS segment limit.

#NM
EM or TS in CR0 is set.

#PF(fault-code)
If a page fault occurs.

#AC(0)
If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS
If a memory operand effective address is outside the SS segment limit.

#NM
EM or TS in CR0 is set.
FBSTP—Store BCD Integer and Pop (continued)

**Virtual 8086 Mode Exceptions**

<table>
<thead>
<tr>
<th>Exception Code</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP(0)</td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.</td>
</tr>
<tr>
<td>#SS(0)</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#NM</td>
<td>EM or TS in CR0 is set.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made.</td>
</tr>
</tbody>
</table>
FCHS—Change Sign

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 E0</td>
<td>FCHS</td>
<td>Complements sign of ST(0)</td>
</tr>
</tbody>
</table>

Description

Complements the sign bit of ST(0). This operation changes a positive value into a negative value of equal magnitude or vice-versa. The following table shows the results obtained when creating the absolute value of various classes of numbers.

<table>
<thead>
<tr>
<th>ST(0) SRC</th>
<th>ST(0) DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>→</td>
<td>+∞</td>
</tr>
<tr>
<td>-F</td>
<td>+F</td>
</tr>
<tr>
<td>-0</td>
<td>+0</td>
</tr>
<tr>
<td>+0</td>
<td>-0</td>
</tr>
<tr>
<td>+F</td>
<td>-F</td>
</tr>
<tr>
<td>+∞</td>
<td>-∞</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Note:
F means finite-real number.

Operation

SignBit(ST(0)) ← NOT (SignBit(ST(0)))

FPU Flags Affected

C1 Set to 0 if stack underflow occurred; otherwise, cleared to 0.
C0, C2, C3 Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Floating-point Exceptions

#IS Stack underflow occurred.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FCLEX/FNCLEX—Clear Exceptions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9B DB E2</td>
<td>FCLEX</td>
<td>Clear floating-point exception flags after checking for pending unmasked floating-point exceptions.</td>
</tr>
<tr>
<td>DB E2</td>
<td>FNCLEX</td>
<td>Clear floating-point exception flags without checking for pending unmasked floating-point exceptions.</td>
</tr>
</tbody>
</table>

**Description**

Clears the floating-point exception flags (PE, UE, OE, ZE, DE, and IE), the exception summary status flag (ES), the stack fault flag (SF), and the busy flag (B) in the FPU status word. The FCLEX instruction checks for and handles any pending unmasked floating-point exceptions before clearing the exception flags; the FNCLEX instruction does not.

**Operation**

\[
\text{FPUStatusWord}[0..7] \leftarrow 0; \\
\text{FPUStatusWord}[15] \leftarrow 0;
\]

**FPU Flags Affected**

The PE, UE, OE, ZE, DE, IE, ES, SF, and B flags in the FPU status word are cleared. The C0, C1, C2, and C3 flags are undefined.

**Floating-point Exceptions**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

**Protected Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Real Address Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

#NM  EM or TS in CR0 is set.
FCMOVcc—Floating-point Conditional Move

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DA C0-i</td>
<td>FCMOVB ST(0), ST(i)</td>
<td>Move if below (CF=1)</td>
</tr>
<tr>
<td>DA C8-i</td>
<td>FCMOVE ST(0), ST(i)</td>
<td>Move if equal (ZF=1)</td>
</tr>
<tr>
<td>DA D0-i</td>
<td>FCMOVBE ST(0), ST(i)</td>
<td>Move if below or equal (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>DA D8-i</td>
<td>FCMOVU ST(0), ST(i)</td>
<td>Move if unordered (PF=1)</td>
</tr>
<tr>
<td>DB C0-i</td>
<td>FCMOVNB ST(0), ST(i)</td>
<td>Move if not below (CF=0)</td>
</tr>
<tr>
<td>DB C8-i</td>
<td>FCMOVNE ST(0), ST(i)</td>
<td>Move if not equal (ZF=0)</td>
</tr>
<tr>
<td>DB D0-i</td>
<td>FCMOVNBE ST(0), ST(i)</td>
<td>Move if not below or equal (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>DB D8-i</td>
<td>FCMOVNU ST(0), ST(i)</td>
<td>Move if not unordered (PF=0)</td>
</tr>
</tbody>
</table>

Description

Tests the status flags in the EFLAGS register and moves the source operand (second operand) to the destination operand (first operand) if the given test condition is true. The source operand is always in the ST(i) register and the destination operand is always ST(0).

The FCMOVcc instructions are useful for optimizing small IF constructions. They also help eliminate branching overhead for IF operations and the possibility of branch mispredictions by the processor.

A processor in the Pentium Pro processor family may not support the FCMOVcc instructions. Software can check if the FCMOVcc instructions are supported by checking the processor’s feature information with the CPUID instruction (see “CPUID—CPU Identification” on page 3:427). If both the CMOV and FPU feature bits are set, the FCMOVcc instructions are supported.

Operation

IF condition TRUE

ST(0) ← ST(i)
FI;

FPU Flags Affected

C1          Set to 0 if stack underflow occurred.
C0, C2, C3  Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Floating-point Exceptions

#IS          Stack underflow occurred.

Integer Flags Affected

None.
FCMOVcc—Floating-point Conditional Move (continued)

Protected Mode Exceptions
#NM EM or TS in CR0 is set.

Real Address Mode Exceptions
#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions
#NM EM or TS in CR0 is set.
**FCOM/FCOMP/FCOMPP—Compare Real**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D8 /2</td>
<td>FCOM m32real</td>
<td>Compare ST(0) with m32real.</td>
</tr>
<tr>
<td>DC /2</td>
<td>FCOM m64real</td>
<td>Compare ST(0) with m64real.</td>
</tr>
<tr>
<td>D8 D0+i</td>
<td>FCOM ST(i)</td>
<td>Compare ST(0) with ST(i).</td>
</tr>
<tr>
<td>D8 D1</td>
<td>FCOM</td>
<td>Compare ST(0) with ST(1).</td>
</tr>
<tr>
<td>D8 /3</td>
<td>FCOMP m32real</td>
<td>Compare ST(0) with m32real and pop register stack.</td>
</tr>
<tr>
<td>DC /3</td>
<td>FCOMP m64real</td>
<td>Compare ST(0) with m64real and pop register stack.</td>
</tr>
<tr>
<td>D8 D8+i</td>
<td>FCOMP ST(i)</td>
<td>Compare ST(0) with ST(i) and pop register stack.</td>
</tr>
<tr>
<td>D8 D9</td>
<td>FCOMP</td>
<td>Compare ST(0) with ST(1) and pop register stack.</td>
</tr>
<tr>
<td>DE D9</td>
<td>FCOMPP</td>
<td>Compare ST(0) with ST(1) and pop register stack twice.</td>
</tr>
</tbody>
</table>

**Description**

Compares the contents of register ST(0) and source value and sets condition code flags C0, C2, and C3 in the FPU status word according to the results (see the table below). The source operand can be a data register or a memory location. If no source operand is given, the value in ST(0) is compared with the value in ST(1). The sign of zero is ignored, so that \(-0.0 = +0.0\).

<table>
<thead>
<tr>
<th>Condition</th>
<th>C3</th>
<th>C2</th>
<th>C0</th>
</tr>
</thead>
<tbody>
<tr>
<td>ST(0) &gt; SRC</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>ST(0) &lt; SRC</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>ST(0) = SRC</td>
<td>1</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>Unordered(^a)</td>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
</tbody>
</table>

\(^a\) Flags not set if unmasked invalid-arithmetic-operand (#IA) exception is generated.

This instruction checks the class of the numbers being compared. If either operand is a NaN or is in an unsupported format, an invalid-arithmetic-operand exception (#IA) is raised and, if the exception is masked, the condition flags are set to “unordered.” If the invalid-arithmetic-operand exception is unmasked, the condition code flags are not set.

The FCOMP instruction pops the register stack following the comparison operation and the FCOMPP instruction pops the register stack twice following the comparison operation. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1.

The FCOM instructions perform the same operation as the FUCOM instructions. The only difference is how they handle QNaN operands. The FCOM instructions raise an invalid-arithmetic-operand exception (#IA) when either or both of the operands is a NaN value or is in an unsupported format. The FUCOM instructions perform the same operation as the FCOM instructions, except that they do not generate an invalid-arithmetic-operand exception for QNaNs.
FOM/FOMP/FOMEPP—Compare Real (continued)

Operation

CASE (relation of operands) OF
ST > SRC: C3, C2, C0 ← 000;
ST < SRC: C3, C2, C0 ← 001;
ST = SRC: C3, C2, C0 ← 100;
ESAC;
IF ST(0) or SRC = NaN or unsupported format
THEN
  #IA
  IF FPUControlWord.IM = 1
      THEN
          C3, C2, C0 ← 111;
  FI;
FI;
IF instruction = FOMP
THEN
  PopRegisterStack;
FI;
IF instruction = FOMEPP
THEN
  PopRegisterStack;
  PopRegisterStack;
FI;

FPU Flags Affected
C1 Set to 0 if stack underflow occurred; otherwise, cleared to 0.
C0, C2, C3 See table on previous page.

Additional Itanium System Environment Exceptions
Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Floating-point Exceptions
#IS Stack underflow occurred.
#IA One or both operands are NaN values or have unsupported formats.
  Register is marked empty.
#D One or both operands are denormal values.
Protected Mode Exceptions

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#NM**: EM or TS in CR0 is set.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

- **#GP**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS**: If a memory operand effective address is outside the SS segment limit.
- **#NM**: EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#NM**: EM or TS in CR0 is set.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made.
FCOMI/FCOMIP/FUCOMI/FUCOMIP—Compare Real and Set EFLAGS

**Description**

Compares the contents of register ST(0) and ST(i) and sets the status flags ZF, PF, and CF in the EFLAGS register according to the results (see the table below). The sign of zero is ignored for comparisons, so that −0.0 = +0.0.

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DB F0+i</td>
<td>FCOMI ST, ST(i)</td>
<td>Compare ST(0) with ST(i) and set status flags accordingly</td>
</tr>
<tr>
<td>DF F0+i</td>
<td>FCOMIP ST, ST(i)</td>
<td>Compare ST(0) with ST(i), set status flags accordingly, and pop register stack</td>
</tr>
<tr>
<td>DB E8+i</td>
<td>FUCOMI ST, ST(i)</td>
<td>Compare ST(0) with ST(i), check for ordered values, and set status flags accordingly</td>
</tr>
<tr>
<td>DF E8+i</td>
<td>FUCOMIP ST, ST(i)</td>
<td>Compare ST(0) with ST(i), check for ordered values, set status flags accordingly, and pop register stack</td>
</tr>
</tbody>
</table>

**Comparison Results**

<table>
<thead>
<tr>
<th>Comparison Results</th>
<th>ZF</th>
<th>PF</th>
<th>CF</th>
</tr>
</thead>
<tbody>
<tr>
<td>ST0 &gt; ST(i)</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>ST0 &lt; ST(i)</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>ST0 = ST(i)</td>
<td>1</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>Unordered a</td>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
</tbody>
</table>

a. Flags not set if unmasked invalid-arithmetic-operand (#IA) exception is generated.

The FCOMI/FCOMIP instructions perform the same operation as the FUCOMI/FUCOMIP instructions. The only difference is how they handle QNaN operands. The FCOMI/FCOMIP instructions set the status flags to “unordered” and generate an invalid-arithmetic-operand exception (#IA) when either or both of the operands is a NaN value (SNaN or QNaN) or is in an unsupported format.

The FUCOMI/FUCOMIP instructions perform the same operation as the FCOMI/FCOMIP instructions, except that they do not generate an invalid-arithmetic-operand exception for QNaNs.

If invalid-operation exception is unmasked, the status flags are not set if the invalid-arithmetic-operand exception is generated.

The FCOMIP and FUCOMIP instructions also pop the register stack following the comparison operation. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1.
FCOMI/FCOMIP/FUCOMI/FUCOMIP—Compare Real and Set EFLAGS
(continued)

Operation

CASE (relation of operands) OF
ST(0) > ST(i):    ZF, PF, CF ← 000;
ST(0) < ST(i):    ZF, PF, CF ← 001;
ST(0) = ST(i):    ZF, PF, CF ← 100;
ESAC;
IF instruction is FCOMI or FCOMIP
  THEN
    IF ST(0) or ST(i) = NaN or unsupported format
    THEN
      #IA
      IF FPUControlWord.IM = 1
        THEN
          ZF, PF, CF ← 111;
        FI;
      FI;
    IF instruction is FUCOMI or FUCOMIP
    THEN
      IF ST(0) or ST(i) = QNaN, but not SNaN or unsupported format
      THEN
        ZF, PF, CF ← 111;
      ELSE (* ST(0) or ST(i) is SNaN or unsupported format *)
        #IA;
        IF FPUControlWord.IM = 1
          THEN
            ZF, PF, CF ← 111;
          FI;
        FI;
    IF instruction is FCOMIP or FUCOMIP
    THEN
      PopRegisterStack;
  FI;

FPU Flags Affected
C1    Set to 0 if stack underflow occurred; otherwise, cleared to 0.
C0, C2, C3    Not affected.

Additional Itanium System Environment Exceptions
Itanium Reg Faults    Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
FCOMI/FCOMIP/FUCOMI/FUCOMIP—Compare Real and Set EFLAGS
(continued)

Floating-point Exceptions

#IS Stack underflow occurred.

#IA (FCOMI or FCOMIP instruction) One or both operands are NaN values or have unsupported formats.

(FUCOMI or FUCOMIP instruction) One or both operands are SNaN values (but not QNaNs) or have undefined formats. Detection of a QNaN value does not raise an invalid-operand exception.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FCOS—Cosine

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 FF</td>
<td>FCOS</td>
<td>Replace ST(0) with its cosine</td>
</tr>
</tbody>
</table>

### Description

Calculates the cosine of the source operand in register ST(0) and stores the result in ST(0). The source operand must be given in radians and must be within the range $-2^{63}$ to $+2^{63}$. The following table shows the results obtained when taking the cosine of various classes of numbers, assuming that neither overflow nor underflow occurs.

<table>
<thead>
<tr>
<th>ST(0) SRC</th>
<th>ST(0) DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>$-\infty$</td>
<td>*</td>
</tr>
<tr>
<td>$-F$</td>
<td>$-1$ to $+1$</td>
</tr>
<tr>
<td>$-0$</td>
<td>$+1$</td>
</tr>
<tr>
<td>$+0$</td>
<td>$+1$</td>
</tr>
<tr>
<td>$+F$</td>
<td>$-1$ to $+1$</td>
</tr>
<tr>
<td>$+\infty$</td>
<td>*</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
- $F$ means finite-real number.
- * indicates floating-point invalid-arithmetic-operand (#IA) exception.

If the source operand is outside the acceptable range, the C2 flag in the FPU status word is set, and the value in register ST(0) remains unchanged. The instruction does not raise an exception when the source operand is out of range. It is up to the program to check the C2 flag for out-of-range conditions. Source values outside the range $-2^{63}$ to $+2^{63}$ can be reduced to the range of the instruction by subtracting an appropriate integer multiple of $2\pi$ or by using the FPREM instruction with a divisor of $2\pi$.

### Operation

If $|ST(0)| < 2^{63}$
Then
- $C2 \leftarrow 0$;
- $ST(0) \leftarrow \cosine(ST(0))$;
Else (*source operand is out-of-range *)
- $C2 \leftarrow 1$;
Fi;
FCOS—Cosine (continued)

FPU Flags Affected

C1 Set to 0 if stack underflow occurred.
Indicates rounding direction if the inexact-result exception (#P) is generated:
0 = not roundup; 1 = roundup.
Undefined if C2 is 1.

C2 Set to 1 if source operand is outside the range \(-2^{63}\) to \(+2^{63}\); otherwise, cleared to 0.

C0, C3 Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Floating-point Exceptions

#IS Stack underflow occurred.
#IA Source operand is an SNaN value, \(\infty\), or unsupported format.
#D Result is a denormal value.
#U Result is too small for destination format.
#P Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FDECSTP—Decrement Stack-Top Pointer

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 F6</td>
<td>FDECSTP</td>
<td>Decrement TOP field in FPU status word.</td>
</tr>
</tbody>
</table>

Description

Subtracts one from the TOP field of the FPU status word (decrements the top-of-stack pointer). The contents of the FPU data registers and tag register are not affected.

Operation

\[
\text{IF TOP} = 0 \\
\quad \text{THEN TOP} \leftarrow 7; \\
\quad \text{ELSE TOP} \leftarrow \text{TOP} - 1; \\
\text{FI};
\]

FPU Flags Affected

The C1 flag is set to 0; otherwise, cleared to 0. The C0, C2, and C3 flags are undefined.

Floating-point Exceptions

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.df is 1, NaT Register Consumption Abort.

Protected Mode Exceptions

#NM  EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM  EM or TS in CR0 is set.
FDIV/FDIVP/FIDIV—Divide

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D8 /6</td>
<td>FDIV m32real</td>
<td>Divide ST(0) by m32real and store result in ST(0)</td>
</tr>
<tr>
<td>DC /6</td>
<td>FDIV m64real</td>
<td>Divide ST(0) by m64real and store result in ST(0)</td>
</tr>
<tr>
<td>D8 F0+i</td>
<td>FDIV ST(0), ST(i)</td>
<td>Divide ST(0) by ST(i) and store result in ST(0)</td>
</tr>
<tr>
<td>DC F8+i</td>
<td>FDIV ST(i), ST(0)</td>
<td>Divide ST(i) by ST(0) and store result in ST(i)</td>
</tr>
<tr>
<td>DE F8+i</td>
<td>FDIVP ST(i), ST(0)</td>
<td>Divide ST(i) by ST(0), store result in ST(i), and pop the register stack</td>
</tr>
<tr>
<td>DE F9</td>
<td>FDIVP</td>
<td>Divide ST(1) by ST(0), store result in ST(1), and pop the register stack</td>
</tr>
<tr>
<td>DA /6</td>
<td>FIDIV m32int</td>
<td>Divide ST(0) by m32int and store result in ST(0)</td>
</tr>
<tr>
<td>DE /6</td>
<td>FIDIV m16int</td>
<td>Divide ST(0) by m64int and store result in ST(0)</td>
</tr>
</tbody>
</table>

Description

Divides the destination operand by the source operand and stores the result in the destination location. The destination operand (dividend) is always in an FPU register; the source operand (divisor) can be a register or a memory location. Source operands in memory can be in single-real, double-real, word-integer, or short-integer formats.

The no-operand version of the instruction divides the contents of the ST(1) register by the contents of the ST(0) register. The one-operand version divides the contents of the ST(0) register by the contents of a memory location (either a real or an integer value). The two-operand version, divides the contents of the ST(0) register by the contents of the ST(i) register or vice versa.

The FDIVP instructions perform the additional operation of popping the FPU register stack after storing the result. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. The no-operand version of the floating-point divide instructions always results in the register stack being popped. In some assemblers, the mnemonic for this instruction is FDIV rather than FDIVP.

The FIDIV instructions convert an integer source operand to extended-real format before performing the division. When the source operand is an integer 0, it is treated as a +0.

If an unmasked divide by zero exception (#Z) is generated, no result is stored; if the exception is masked, an ∞ of the appropriate sign is stored in the destination operand.

The following table shows the results obtained when dividing various classes of numbers, assuming that neither overflow nor underflow occurs.
FDIV/FDIVP/FIDIV—Divide (continued)

### Notes:
- **F** means finite-real number.
- **I** means integer.
- * indicates floating-point invalid-arithmetic-operand (#IA) exception.
- ** indicates floating-point zero-divide (#Z) exception.

### Operation

\[
\text{IF } \text{SRC} = 0 \\
\text{THEN} \\
\text{#Z} \\
\text{ELSE} \\
\text{IF instruction is FIDIV} \\
\text{THEN} \\
\text{DEST} \leftarrow \text{DEST} / \text{ConvertExtendedReal}(\text{SRC}); \\
\text{ELSE (* source operand is real number *)} \\
\text{DEST} \leftarrow \text{DEST} / \text{SRC}; \\
\text{FI;} \\
\text{FI;} \\
\text{IF instruction = FDIVP} \\
\text{THEN} \\
\text{PopRegisterStack} \\
\text{FI;} \\
\]

### FPU Flags Affected

- **C1**
  - Set to 0 if stack underflow occurred.
  - Indicates rounding direction if the inexact-result exception (#P) is generated:
    - 0 = not roundup; 1 = roundup.

- **C0, C2, C3**
  - Undefined.
**FDIV/FDIVP/FIDIV—Divide** (continued)

### Additional Itanium System Environment Exceptions

- **Itanium Reg Faults**: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault.

### Floating-point Exceptions

- **#IS**: Stack underflow occurred.
- **#IA**: Operand is an SNaN value or unsupported format.
  \[-\infty / \pm\infty; \pm0 / \pm0\]
- **#D**: Result is a denormal value.
- **#Z**: DEST / ±0, where DEST is not equal to ±0.
- **#U**: Result is too small for destination format.
- **#O**: Result is too large for destination format.
- **#P**: Value cannot be represented exactly in destination format.

### Protected Mode Exceptions

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#NM**: EM or TS in CR0 is set.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

### Real Address Mode Exceptions

- **#GP**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS**: If a memory operand effective address is outside the SS segment limit.
- **#NM**: EM or TS in CR0 is set.

### Virtual 8086 Mode Exceptions

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#NM**: EM or TS in CR0 is set.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made.
FDIVR/FDIVRP/FIDIVR—Reverse Divide

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D8 /7</td>
<td>FDIVR m32real</td>
<td>Divide m32real by ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DC /7</td>
<td>FDIVR m64real</td>
<td>Divide m64real by ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>D8 F8+i</td>
<td>FDIVR ST(0), ST(i)</td>
<td>Divide ST(i) by ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DC F0+i</td>
<td>FDIVR ST(i), ST(0)</td>
<td>Divide ST(0) by ST(i) and store result in ST(i)</td>
</tr>
<tr>
<td>DE F0+i</td>
<td>FDIVRP ST(i), ST(0)</td>
<td>Divide ST(0) by ST(i), store result in ST(i), and pop the register stack</td>
</tr>
<tr>
<td>DE F1</td>
<td>FDIVRP</td>
<td>Divide ST(0) by ST(1), store result in ST(1), and pop the register stack</td>
</tr>
<tr>
<td>DA /7</td>
<td>FIDIVR m32int</td>
<td>Divide m32int by ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DE /7</td>
<td>FIDIVR m16int</td>
<td>Divide m64int by ST(0) and store result in ST(0)</td>
</tr>
</tbody>
</table>

Description

Divides the source operand by the destination operand and stores the result in the destination location. The destination operand (divisor) is always in an FPU register; the source operand (dividend) can be a register or a memory location. Source operands in memory can be in single-real, double-real, word-integer, or short-integer formats.

These instructions perform the reverse operations of the FDIV, FDIVP, and FIDIV instructions. They are provided to support more efficient coding.

The no-operand version of the instruction divides the contents of the ST(0) register by the contents of the ST(1) register. The one-operand version divides the contents of a memory location (either a real or an integer value) by the contents of the ST(0) register. The two-operand version, divides the contents of the ST(i) register by the contents of the ST(0) register or vice versa.

The FDIVRP instructions perform the additional operation of popping the FPU register stack after storing the result. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. The no-operand version of the floating-point divide instructions always results in the register stack being popped. In some assemblers, the mnemonic for this instruction is FDIVR rather than FDIVRP.

The FIDIVR instructions convert an integer source operand to extended-real format before performing the division.

If an unmasked divide by zero exception (#Z) is generated, no result is stored; if the exception is masked, an \( \pm \infty \) of the appropriate sign is stored in the destination operand.

The following table shows the results obtained when dividing various classes of numbers, assuming that neither overflow nor underflow occurs.
FDIVR/FDIVRP/FIDIVR—Reverse Divide (continued)

<table>
<thead>
<tr>
<th>SRC</th>
<th>DEST</th>
<th>−∞</th>
<th>−F</th>
<th>−0</th>
<th>+0</th>
<th>+F</th>
<th>+∞</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>*</td>
<td>+∞</td>
<td>−F</td>
<td>+F</td>
<td>+0</td>
<td>+F</td>
<td>+∞</td>
<td>NaN</td>
</tr>
<tr>
<td>−F</td>
<td>+0</td>
<td>+F</td>
<td>−0</td>
<td>−0</td>
<td>−0</td>
<td>−0</td>
<td>−0</td>
<td>NaN</td>
</tr>
<tr>
<td>−I</td>
<td>+0</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>NaN</td>
</tr>
<tr>
<td>−0</td>
<td>+0</td>
<td>*</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>NaN</td>
</tr>
<tr>
<td>+0</td>
<td>−0</td>
<td>*</td>
<td>−F</td>
<td>−F</td>
<td>−F</td>
<td>−F</td>
<td>−F</td>
<td>NaN</td>
</tr>
<tr>
<td>+F</td>
<td>+0</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>NaN</td>
</tr>
<tr>
<td>+∞</td>
<td>*</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>NaN</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
F  means finite-real number.
I  means integer.
* indicates floating-point invalid-arithmetic-operand (#IA) exception.
** indicates floating-point zero-divide (#Z) exception.

When the source operand is an integer 0, it is treated as a +0.

Operation

IF DEST = 0
THEN
   #Z
ELSE
   IF instruction is FIDIVR
   THEN
      DEST ← ConvertExtendedReal(SRC) / DEST;
   ELSE (* source operand is real number *)
      DEST ← SRC / DEST;
   FI;
   FI;
IF instruction = FDIVRP
THEN
   PopRegisterStack
FI;

FPU Flags Affected

C1  Set to 0 if stack underflow occurred.
    Indicates rounding direction if the inexact-result exception (#P) is generated:
    0 = not roundup; 1 = roundup.
C0, C2, C3  Undefined.
FDIVR/FDIVRP/FIDIVR—Reverse Divide (continued)

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Floating-point Exceptions

#IS  Stack underflow occurred.
#IA  Operand is an SNaN value or unsupported format.
    \( \pm\infty / \pm\infty ; \pm0 / \pm0 \)
#D   Result is a denormal value.
#Z   SRC / \pm0, where SRC is not equal to \pm0.
#U   Result is too small for destination format.
#O   Result is too large for destination format.
#P   Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
**FFREE—Free Floating-point Register**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DD C0+i</td>
<td>FFREE ST(i)</td>
<td>Sets tag for ST(i) to empty</td>
</tr>
</tbody>
</table>

**Description**

Sets the tag in the FPU tag register associated with register ST(i) to empty (11B). The contents of ST(i) and the FPU stack-top pointer (TOP) are not affected.

**Operation**

TAG(i) ← 11B;

**FPU Flags Affected**

C0, C1, C2, C3 undefined.

**Floating-point Exceptions**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1.

**Protected Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Real Address Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

#NM  EM or TS in CR0 is set.
FICOM/FICOMP—Compare Integer

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DE /2</td>
<td>FICOM m16int</td>
<td>Compare ST(0) with m16int</td>
</tr>
<tr>
<td>DA /2</td>
<td>FICOM m32int</td>
<td>Compare ST(0) with m32int</td>
</tr>
<tr>
<td>DE /3</td>
<td>FICOMP m16int</td>
<td>Compare ST(0) with m16int and pop stack register</td>
</tr>
<tr>
<td>DA /3</td>
<td>FICOMP m32int</td>
<td>Compare ST(0) with m32int and pop stack register</td>
</tr>
</tbody>
</table>

**Description**

Compares the value in ST(0) with an integer source operand and sets the condition code flags C0, C2, and C3 in the FPU status word according to the results (see table below). The integer value is converted to extended-real format before the comparison is made.

<table>
<thead>
<tr>
<th>Condition</th>
<th>C3</th>
<th>C2</th>
<th>C0</th>
</tr>
</thead>
<tbody>
<tr>
<td>ST(0) &gt; SRC</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>ST(0) &lt; SRC</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>ST(0) = SRC</td>
<td>1</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>Unordered</td>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
</tbody>
</table>

These instructions perform an “unordered comparison.” An unordered comparison also checks the class of the numbers being compared. If either operand is a NaN or is in an undefined format, the condition flags are set to “unordered.”

The sign of zero is ignored, so that –0.0 = +0.0.

The FICOMP instructions pop the register stack following the comparison. To pop the register stack, the processor marks the ST(0) register empty and increments the stack pointer (TOP) by 1.

**Operation**

```plaintext
CASE (relation of operands) OF
   ST(0) > SRC:  C3, C2, C0 ← 000;
   ST(0) < SRC:  C3, C2, C0 ← 001;
   ST(0) = SRC:  C3, C2, C0 ← 100;
   Unordered:    C3, C2, C0 ← 111;
ESAC;
IF instruction = FICOMP
   THEN
      PopRegisterStack;
   FI;
```

**FPU Flags Affected**

- C1: Set to 0 if stack underflow occurred; otherwise, set to 0.
- C0, C2, C3: See table on previous page.
Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Floating-point Exceptions

#IS  Stack underflow occurred.
#IA  One or both operands are NaN values or have unsupported formats.
#D   One or both operands are denormal values.

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
        If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#NM    EM or TS in CR0 is set.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS  If a memory operand effective address is outside the SS segment limit.
#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#NM    EM or TS in CR0 is set.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
FILD—Load Integer

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DF /0</td>
<td>FILD m16int</td>
<td>Push m16int onto the FPU register stack.</td>
</tr>
<tr>
<td>DB /0</td>
<td>FILD m32int</td>
<td>Push m32int onto the FPU register stack.</td>
</tr>
<tr>
<td>DF /5</td>
<td>FILD m64int</td>
<td>Push m64int onto the FPU register stack.</td>
</tr>
</tbody>
</table>

**Description**

Converts the signed-integer source operand into extended-real format and pushes the value onto the FPU register stack. The source operand can be a word, short, or long integer value. It is loaded without rounding errors. The sign of the source operand is preserved.

**Operation**

\[
\text{TOP} \leftarrow \text{TOP} - 1; \\
\text{ST}(0) \leftarrow \text{ExtendedReal(SRC)};
\]

**FPU Flags Affected**

- **C1**: Set to 1 if stack overflow occurred; cleared to 0 otherwise.
- **C0, C2, C3**: Undefined.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Floating-point Exceptions**

- #IS: Stack overflow occurred.

**Protected Mode Exceptions**

- #GP(0): If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  
  If the DS, ES, FS, or GS register contains a null segment selector.

- #SS(0): If a memory operand effective address is outside the SS segment limit.

- #NM: EM or TS in CR0 is set.

- #PF(fault-code): If a page fault occurs.

- #AC(0): If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
FILD—Load Integer (continued)

**Real Address Mode Exceptions**

- **#GP** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS** If a memory operand effective address is outside the SS segment limit.
- **#NM** EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#NM** EM or TS in CR0 is set.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made.
FINCSTP—Increment Stack-Top Pointer

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 F7</td>
<td>FINCSTP</td>
<td>Increment the TOP field in the FPU status register</td>
</tr>
</tbody>
</table>

**Description**

Adds one to the TOP field of the FPU status word (increments the top-of-stack pointer). The contents of the FPU data registers and tag register are not affected. This operation is not equivalent to popping the stack, because the tag for the previous top-of-stack register is not marked empty.

**Operation**

```
IF TOP = 7
    THEN TOP ← 0;
    ELSE TOP ← TOP + 1;
FI;
```

**FPU Flags Affected**

The C1 flag is set to 0; otherwise, generates an #IS fault. The C0, C2, and C3 flags are undefined.

**Floating-point Exceptions**

#IS

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1.

**Protected Mode Exceptions**

#NM EM or TS in CR0 is set.

**Real Address Mode Exceptions**

#NM EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

#NM EM or TS in CR0 is set.
FINIT/FNINIT—Initialize Floating-point Unit

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9B DB E3</td>
<td>FINIT</td>
<td>Initialize FPU after checking for pending unmasked floating-point exceptions.</td>
</tr>
<tr>
<td>DB E3</td>
<td>FNINIT</td>
<td>Initialize FPU without checking for pending unmasked floating-point exceptions.</td>
</tr>
</tbody>
</table>

Description

Sets the FPU control, status, tag, instruction pointer, and data pointer registers to their default states. The FPU control word is set to 037FH (round to nearest, all exceptions masked, 64-bit precision). The status word is cleared (no exception flags set, TOP is set to 0). The data registers in the register stack are left unchanged, but they are all tagged as empty (11B). Both the instruction and data pointers are cleared.

The FINIT instruction checks for and handles any pending unmasked floating-point exceptions before performing the initialization; the FNINIT instruction does not.

Operation

FPUCtrlWord ← 037FH;
FPUStrtWord ← 0;
FPUTagWord ← FFFFH;
FPUDtaPointer ← 0;
FPUIrstInstrOp ← 0;
FPULstInstrOp ← 0;

FPU Flags Affected

C0, C1, C2, C3 cleared to 0.

Floating-point Exceptions

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disable FP Register Fault if PSR.dfr is 1.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FIST/FISTP—Store Integer

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>DF /2</td>
<td>FIST m16int</td>
<td>Store ST(0) in m16int</td>
</tr>
<tr>
<td>DB /2</td>
<td>FIST m32int</td>
<td>Store ST(0) in m32int</td>
</tr>
<tr>
<td>DF /3</td>
<td>FISTP m16int</td>
<td>Store ST(0) in m16int and pop register stack</td>
</tr>
<tr>
<td>DB /3</td>
<td>FISTP m32int</td>
<td>Store ST(0) in m32int and pop register stack</td>
</tr>
<tr>
<td>DF /7</td>
<td>FISTP m64int</td>
<td>Store ST(0) in m64int and pop register stack</td>
</tr>
</tbody>
</table>

Description

The FIST instruction converts the value in the ST(0) register to a signed integer and stores the result in the destination operand. Values can be stored in word- or short-integer format. The destination operand specifies the address where the first byte of the destination value is to be stored.

The FISTP instruction performs the same operation as the FIST instruction and then pops the register stack. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. The FISTP instruction can also stores values in long-integer format.

The following table shows the results obtained when storing various classes of numbers in integer format.

<table>
<thead>
<tr>
<th>ST(0)</th>
<th>DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>*</td>
</tr>
<tr>
<td>−F &lt; −1</td>
<td>−1</td>
</tr>
<tr>
<td>−1 &lt; −F &lt; −0</td>
<td>**</td>
</tr>
<tr>
<td>−0</td>
<td>0</td>
</tr>
<tr>
<td>+0</td>
<td>0</td>
</tr>
<tr>
<td>+0 &lt; +F &lt; +1</td>
<td>**</td>
</tr>
<tr>
<td>+F &gt; +1</td>
<td>+1</td>
</tr>
<tr>
<td>+∞</td>
<td>*</td>
</tr>
<tr>
<td>NaN</td>
<td>*</td>
</tr>
</tbody>
</table>

Notes:
Fmeans finite-real number.
Imeans integer.
*indicates floating-point invalid-operation (#IA) exception.
**±0 or ±1, depending on the rounding mode.

If the source value is a non-integral value, it is rounded to an integer value, according to the rounding mode specified by the RC field of the FPU control word.

If the value being stored is too large for the destination format, is an ∞, is a NaN, or is in an unsupported format and if the invalid-arithmetic-operand exception (#IA) is unmasked, an invalid-operation exception is generated and no value is stored in the destination operand. If the invalid-operation exception is masked, the integer indefinite value is stored in the destination operand.
FIST/FISTP—Store Integer (continued)

Operation

\[
\text{DEST} \leftarrow \text{Integer(ST(0))};
\]

IF instruction = FISTP

THEN

\[
\text{PopRegisterStack};
\]

FI;

FPU Flags Affected

C1 Set to 0 if stack underflow occurred.

Indicates rounding direction of if the inexact exception (#P) is generated:

0 = not roundup; 1 = roundup.

Cleared to 0 otherwise.

C0, C2, C3 Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT register Consumption Abort.

Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Floating-point Exceptions

#IS Stack underflow occurred.

#IA Source operand is too large for the destination format

Source operand is a NaN value or unsupported format.

#P Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
FIST/FISTP—Store Integer (continued)

**Real Address Mode Exceptions**

- **#GP**
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS**
  - If a memory operand effective address is outside the SS segment limit.
- **#NM**
  - EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

- **#GP(0)**
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)**
  - If a memory operand effective address is outside the SS segment limit.
- **#NM**
  - EM or TS in CR0 is set.
- **#PF(fault-code)**
  - If a page fault occurs.
- **#AC(0)**
  - If alignment checking is enabled and an unaligned memory reference is made.
FLD—Load Real

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 /0</td>
<td>FLD m32real</td>
<td>Push m32real onto the FPU register stack.</td>
</tr>
<tr>
<td>DD /0</td>
<td>FLD m64real</td>
<td>Push m64real onto the FPU register stack.</td>
</tr>
<tr>
<td>DB /5</td>
<td>FLD m80real</td>
<td>Push m80real onto the FPU register stack.</td>
</tr>
<tr>
<td>D9 C0+i</td>
<td>FLD ST(i)</td>
<td>Push ST(i) onto the FPU register stack.</td>
</tr>
</tbody>
</table>

Description

Pushes the source operand onto the FPU register stack. If the source operand is in single- or double-real format, it is automatically converted to the extended-real format before being pushed on the stack.

The FLD instruction can also push the value in a selected FPU register [ST(i)] onto the stack. Here, pushing register ST(0) duplicates the stack top.

Operation

IF SRC is ST(i)
    THEN
        temp ← ST(i)
        TOP ← TOP − 1;
    IF SRC is memory-operand
        THEN
            ST(0) ← ExtendedReal(SRC);
        ELSE (* SRC is ST(i) *)
            ST(0) ← temp;

FPU Flags Affected

C1        Set to 1 if stack overflow occurred; otherwise, cleared to 0.
C0, C2, C3 Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Floating-point Exceptions

#IS        Stack overflow occurred.
#IA        Source operand is an SNaN value or unsupported format.
#D         Source operand is a denormal value. Does not occur if the source operand is in extended-real format.

FLD—Load Real (continued)
**Protected Mode Exceptions**

#GP(0)  If destination is located in a nonwritable segment.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
FLD1/FLDL2T/FLDL2E/FLDPI/FLDLG2/FLDLN2/FLDZ—Load Constant

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 E8</td>
<td>FLD1</td>
<td>Push +1.0 onto the FPU register stack.</td>
</tr>
<tr>
<td>D9 E9</td>
<td>FLDL2T</td>
<td>Push log₂10 onto the FPU register stack.</td>
</tr>
<tr>
<td>D9 EA</td>
<td>FLDL2E</td>
<td>Push log₂e onto the FPU register stack.</td>
</tr>
<tr>
<td>D9 EB</td>
<td>FLDPI</td>
<td>Push π onto the FPU register stack.</td>
</tr>
<tr>
<td>D9 EC</td>
<td>FLDLG2</td>
<td>Push log₁₀2 onto the FPU register stack.</td>
</tr>
<tr>
<td>D9 ED</td>
<td>FLDLN2</td>
<td>Push log₂e₂ onto the FPU register stack.</td>
</tr>
<tr>
<td>D9 EE</td>
<td>FLDZ</td>
<td>Push +0.0 onto the FPU register stack.</td>
</tr>
</tbody>
</table>

**Description**

Push one of seven commonly-used constants (in extended-real format) onto the FPU register stack. The constants that can be loaded with these instructions include +1.0, +0.0, log₂10, log₂e, π, log₁₀2, and log₂e₂. For each constant, an internal 66-bit constant is rounded (as specified by the RC field in the FPU control word) to external-real format. The inexact-result exception (#P) is not generated as a result of the rounding.

**Operation**

TOP ← TOP – 1;
ST(0) ← CONSTANT;

**FPU Flags Affected**

C1 Set to 1 if stack overflow occurred; otherwise, cleared to 0.
C0, C2, C3 Undefined.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults Disabled FP Register Fault if PSR.df1 is 1.

**Floating-point Exceptions**

#IS Stack overflow occurred.

**Protected Mode Exceptions**

#NM EM or TS in CR0 is set.

**Real Address Mode Exceptions**

#NM EM or TS in CR0 is set.
Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.

Intel Architecture Compatibility Information

When the RC field is set to round-to-nearest, the FPU produces the same constants that is produced by the Intel 8087 and Intel287 math coprocessors.
FLDCW—Load Control Word

Description

Loads the 16-bit source operand into the FPU control word. The source operand is a memory location. This instruction is typically used to establish or change the FPU’s mode of operation.

If one or more exception flags are set in the FPU status word prior to loading a new FPU control word and the new control word unmasks one or more of those exceptions, a floating-point exception will be generated upon execution of the next floating-point instruction (except for the no-wait floating-point instructions). To avoid raising exceptions when changing FPU operating modes, clear any pending exceptions (using the FCLEX or FNCLEX instruction) before loading the new control word.

Operation

FPUCW ← SRC;

FPU Flags Affected

C0, C1, C2, C3 undefined.

Floating-point Exceptions

None; however, this operation might unmask a pending exception in the FPU status word. That exception is then generated upon execution of the next waiting floating-point instruction.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Exception Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Reg Faults</td>
<td>Disabled FP Register Fault if PSR.dfl is 1.</td>
</tr>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
FLDCW—Load Control Word (continued)

**Real Address Mode Exceptions**

- **#GP**
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

- **#SS**
  - If a memory operand effective address is outside the SS segment limit.

- **#NM**
  - EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

- **#GP(0)**
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

- **#SS(0)**
  - If a memory operand effective address is outside the SS segment limit.

- **#NM**
  - EM or TS in CR0 is set.

- **#PF(fault-code)**
  - If a page fault occurs.

- **#AC(0)**
  - If alignment checking is enabled and an unaligned memory reference is made.
Description

Loads the complete FPU operating environment from memory into the FPU registers. The source operand specifies the first byte of the operating-environment data in memory. This data is typically written to the specified memory location by a FSTENV or FNSTENV instruction.

The FPU operating environment consists of the FPU control word, status word, tag word, instruction pointer, data pointer, and last opcode. See the *Intel Architecture Software Developer’s Manual* for the layout in memory of the loaded environment, depending on the operating mode of the processor (protected or real) and the size of the current address attribute (16-bit or 32-bit). In virtual-8086 mode, the real mode layouts are used.

The FLDENV instruction should be executed in the same operating mode as the corresponding FSTENV/FNSTENV instruction.

If one or more unmasked exception flags are set in the new FPU status word, a floating-point exception will be generated upon execution of the next floating-point instruction (except for the no-wait floating-point instructions). To avoid generating exceptions when loading a new environment, clear all the exception flags in the FPU status word that is being loaded.

Operation

\[
\begin{align*}
\text{FPUCtrlWord} & \leftarrow \text{SRC(FPUCtlWord)}; \\
\text{FPUStrWord} & \leftarrow \text{SRC(FPUStrWord)}; \\
\text{FPUTagWord} & \leftarrow \text{SRC(FPUTagWord)}; \\
\text{FPUDataPtr} & \leftarrow \text{SRC(FPUDataPtr)}; \\
\text{FPUInstrPt} & \leftarrow \text{SRC(FPUInstrPt)}; \\
\text{FPULastInstrOp} & \leftarrow \text{SRC(FPULastInstrOp)};
\end{align*}
\]

FPU Flags Affected

The C0, C1, C2, C3 flags are loaded.

Additional Itanium System Environment Exceptions

- **Itanium Reg Faults**: Disabled FP Register Fault if PSR.dfl is 1.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Floating-point Exceptions

None; however, if an unmasked exception is loaded in the status word, it is generated upon execution of the next waiting floating-point instruction.

---

**FLDENV—Load FPU Environment**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 /4</td>
<td>FLDENV m14/28byte</td>
<td>Load FPU environment from m14byte or m28byte.</td>
</tr>
</tbody>
</table>
FLDENV—Load FPU Environment (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
**FMUL/FMULP/FIMUL—Multiply**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D8 /1</td>
<td>FMUL m32real</td>
<td>Multiply ST(0) by m32real and store result in ST(0)</td>
</tr>
<tr>
<td>DC /1</td>
<td>FMUL m64real</td>
<td>Multiply ST(0) by m64real and store result in ST(0)</td>
</tr>
<tr>
<td>D8 C8+i</td>
<td>FMUL ST(0), ST(i)</td>
<td>Multiply ST(0) by ST(i) and store result in ST(0)</td>
</tr>
<tr>
<td>DC C8+i</td>
<td>FMUL ST(i), ST(0)</td>
<td>Multiply ST(i) by ST(0) and store result in ST(i)</td>
</tr>
<tr>
<td>DE C8+i</td>
<td>FMULP ST(i), ST(0)</td>
<td>Multiply ST(i) by ST(0), store result in ST(i), and pop the register stack</td>
</tr>
<tr>
<td>DE C9</td>
<td>FMULP</td>
<td>Multiply ST(0) by ST(1), store result in ST(0), and pop the register stack</td>
</tr>
<tr>
<td>DA /1</td>
<td>FIMUL m32int</td>
<td>Multiply m32int by ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DE /1</td>
<td>FIMUL m16int</td>
<td>Multiply m16int by ST(0) and store result in ST(0)</td>
</tr>
</tbody>
</table>

**Description**

Multiplies the destination and source operands and stores the product in the destination location. The destination operand is always an FPU data register; the source operand can be a register or a memory location. Source operands in memory can be in single-real, double-real, word-integer, or short-integer formats.

The no-operand version of the instruction multiplies the contents of the ST(0) register by the contents of the ST(1) register. The one-operand version multiplies the contents of a memory location (either a real or an integer value) by the contents of the ST(0) register. The two-operand version multiplies the contents of the ST(0) register by the contents of the ST(i) register or vice versa.

The FMULP instructions perform the additional operation of popping the FPU register stack after storing the product. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. The no-operand version of the floating-point multiply instructions always results in the register stack being popped. In some assemblers, the mnemonic for this instruction is FMUL rather than FMULP.

The FIMUL instructions convert an integer source operand to extended-real format before performing the multiplication.

The sign of the result is always the exclusive-OR of the source signs, even if one or more of the values being multiplied is 0 or $\infty$. When the source operand is an integer 0, it is treated as a +0.

The following table shows the results obtained when multiplying various classes of numbers, assuming that neither overflow nor underflow occurs.
FMUL/FMULP/FIMUL—Multiply (continued)

### Notes:
- F means finite-real number.
- I means Integer.
- * indicates invalid-arithmetic-operand (#IA) exception.

### Operation

If instruction is FIMUL

\[
\text{THEN} \quad \text{DEST} \leftarrow \text{DEST} \times \text{ConvertExtendedReal}(\text{SRC})
\]

Else (* source operand is real number *)

\[
\text{DEST} \leftarrow \text{DEST} \times \text{SRC}
\]

FI;

If instruction = FMULP

Then

\[
\text{PopRegisterStack}
\]

FI;

### FPU Flags Affected
- C1: Set to 0 if stack underflow occurred.
- Indicates rounding direction if the inexact-result exception (#P) fault is generated: 0 = not roundup; 1 = roundup.
- C0, C2, C3: Undefined.

### Floating-point Exceptions
- #IS: Stack underflow occurred.
- #IA: Operand is an SNaN value or unsupported format.
  - One operand is ±0 and the other is ±∞.
- #D: Source operand is a denormal value.
- #U: Result is too small for destination format.
- #O: Result is too large for destination format.
- #P: Value cannot be represented exactly in destination format.
Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
        If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#NM  EM or TS in CR0 is set.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
FNOP—No Operation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 D0</td>
<td>FNOP</td>
<td>No operation is performed.</td>
</tr>
</tbody>
</table>

**Description**

Performs no FPU operation. This instruction takes up space in the instruction stream but does not affect the FPU or machine context, except the EIP register.

**FPU Flags Affected**

C0, C1, C2, C3 undefined.

**Floating-point Exceptions**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1.

**Protected Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Real Address Mode Exceptions**

#NM  EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

#NM  EM or TS in CR0 is set.
**FPATAN—Partial Arctangent**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 F3</td>
<td>FPATAN</td>
<td>Replace ST(1) with arctan(ST(1)/ST(0)) and pop the register stack</td>
</tr>
</tbody>
</table>

**Description**

Computes the arctangent of the source operand in register ST(1) divided by the source operand in register ST(0), stores the result in ST(1), and pops the FPU register stack. The result in register ST(0) has the same sign as the source operand ST(1) and a magnitude less than $+\pi$.

The following table shows the results obtained when computing the arctangent of various classes of numbers, assuming that underflow does not occur.

**Table 1-6. FPATAN Zeros and NaNs**

<table>
<thead>
<tr>
<th>ST(1)</th>
<th>ST(0)</th>
</tr>
</thead>
<tbody>
<tr>
<td>F</td>
<td>$-3\pi/4$</td>
</tr>
<tr>
<td>$-F$</td>
<td>$-p$</td>
</tr>
<tr>
<td>$-0$</td>
<td>$-p$</td>
</tr>
<tr>
<td>$+0$</td>
<td>$+\pi$</td>
</tr>
<tr>
<td>$+F$</td>
<td>$+\pi$</td>
</tr>
<tr>
<td>$+\infty$</td>
<td>$+3\pi/4$</td>
</tr>
</tbody>
</table>

Note:

*F* means finite-real number.

There is no restriction on the range of source operands that FPATAN can accept.

**Operation**

```
ST(1) ← arctan(ST(1) / ST(0));
PopRegisterStack;
```

**FPU Flags Affected**

<table>
<thead>
<tr>
<th>Flag</th>
<th>Behavior</th>
</tr>
</thead>
<tbody>
<tr>
<td>C1</td>
<td>Set to 0 if stack underflow occurred. Indicates rounding direction if the inexact-result exception (#P) is generated: 0 = not roundup; 1 = roundup.</td>
</tr>
<tr>
<td>C0, C2, C3</td>
<td>Undefined.</td>
</tr>
</tbody>
</table>

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
FPATAN—Partial Arctangent (continued)

Floating-point Exceptions

#IS  Stack underflow occurred.
#IA  Source operand is an SNaN value or unsupported format.
#D   Source operand is a denormal value.
#U   Result is too small for destination format.
#P   Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#NM   EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM   EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM   EM or TS in CR0 is set.

Intel Architecture Compatibility Information

The source operands for this instruction are restricted for the 80287 math coprocessor to the following range:

\[ 0 \leq |\text{ST}(1)| < |\text{ST}(0)| < +\infty \]
FPREM—Partial Remainder

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 F8</td>
<td>FPREM</td>
<td>Replace ST(0) with the remainder obtained on dividing ST(0) by ST(1)</td>
</tr>
</tbody>
</table>

Description

Computes the remainder obtained on dividing the value in the ST(0) register (the dividend) by the value in the ST(1) register (the divisor or modulus), and stores the result in ST(0). The remainder represents the following value:

Remainder = ST(0) - (N * ST(1))

Here, N is an integer value that is obtained by truncating the real-number quotient of [ST(0) / ST(1)] toward zero. The sign of the remainder is the same as the sign of the dividend. The magnitude of the remainder is less than that of the modulus, unless a partial remainder was computed (as described below).

This instruction produces an exact result; the precision (inexact) exception does not occur and the rounding control has no effect. The following table shows the results obtained when computing the remainder of various classes of numbers, assuming that underflow does not occur.

Table 1-7. FPREM Zeros and NaNs

<table>
<thead>
<tr>
<th>ST(0)</th>
<th>−∞</th>
<th>−F</th>
<th>−0</th>
<th>+0</th>
<th>+F</th>
<th>+∞</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>NaN</td>
</tr>
<tr>
<td>−F</td>
<td>ST(0)</td>
<td>−F or −0</td>
<td>**</td>
<td>**</td>
<td>−F or −0</td>
<td>ST(0)</td>
<td>NaN</td>
</tr>
<tr>
<td>−0</td>
<td>−0</td>
<td>−0</td>
<td>*</td>
<td>*</td>
<td>−0</td>
<td>−0</td>
<td>NaN</td>
</tr>
<tr>
<td>+0</td>
<td>+0</td>
<td>+0</td>
<td>*</td>
<td>*</td>
<td>+0</td>
<td>+0</td>
<td>NaN</td>
</tr>
<tr>
<td>+F</td>
<td>ST(0)</td>
<td>+F or +0</td>
<td>**</td>
<td>**</td>
<td>+F or +0</td>
<td>ST(0)</td>
<td>NaN</td>
</tr>
<tr>
<td>+∞</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>NaN</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
F means finite-real number.
* indicates floating-point invalid-arithmetic-operand (#IA) exception.
** indicates floating-point zero-divide (#Z) exception.

When the result is 0, its sign is the same as that of the dividend. When the modulus is ∞, the result is equal to the value in ST(0).

The FPREM instruction does not compute the remainder specified in IEEE Std. 754. The IEEE specified remainder can be computed with the FPREM1 instruction. The FPREM instruction is provided for compatibility with the Intel 8087 and Intel287 math coprocessors.
FPREM—Partial Remainder (continued)

The FPREM instruction gets its name “partial remainder” because of the way it computes the remainder. This instruction arrives at a remainder through iterative subtraction. It can, however, reduce the exponent of ST(0) by no more than 63 in one execution of the instruction. If the instruction succeeds in producing a remainder that is less than the modulus, the operation is complete and the C2 flag in the FPU status word is cleared. Otherwise, C2 is set, and the result in ST(0) is called the partial remainder. The exponent of the partial remainder will be less than the exponent of the original dividend by at least 32. Software can re-execute the instruction (using the partial remainder in ST(0) as the dividend) until C2 is cleared.

Note: While executing such a remainder-computation loop, a higher-priority interrupting routine that needs the FPU can force a context switch in-between the instructions in the loop.

An important use of the FPREM instruction is to reduce the arguments of periodic functions. When reduction is complete, the instruction stores the three least-significant bits of the quotient in the C3, C1, and C0 flags of the FPU status word. This information is important in argument reduction for the tangent function (using a modulus of \(\pi/4\)), because it locates the original angle in the correct one of eight sectors of the unit circle.

Operation

\[
D \leftarrow \text{exponent(ST(0))} - \text{exponent(ST(1))};
\]

IF \(D < 64\) THEN

\[
Q \leftarrow \text{Integer(TruncateTowardZero(ST(0) / ST(1)))};
\]

\[
\text{ST(0)} \leftarrow \text{ST(0)} - (\text{ST(1)} \times Q);
\]

\[
\text{C2} \leftarrow 0;
\]

\[
\text{C0, C3, C1} \leftarrow \text{LeastSignificantBits(Q)}; (* Q2, Q1, Q0 *)
\]

ELSE

\[
\text{C2} \leftarrow 1;
\]

\[
\text{N} \leftarrow \text{an implementation-dependent number between 32 and 63};
\]

\[
\text{QQ} \leftarrow \text{Integer(TruncateTowardZero((ST(0) / ST(1)) / 2^{(D-N)}))};
\]

\[
\text{ST(0)} \leftarrow \text{ST(0)} - (\text{ST(1)} \times \text{QQ} \times 2^{(D-N)});
\]

FI;

FPU Flags Affected

- **C0**: Set to bit 2 (Q2) of the quotient.
- **C1**: Set to 0 if stack underflow occurred; otherwise, set to least significant bit of quotient (Q0).
- **C2**: Set to 0 if reduction complete; set to 1 if incomplete.
- **C3**: Set to bit 1 (Q1) of the quotient.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
FPREM—Partial Remainder (continued)

Floating-point Exceptions

#IS Stack underflow occurred.
#IA Source operand is an SNaN value, modulus is 0, dividend is $\infty$, or unsupported format.
#D Source operand is a denormal value.
#U Result is too small for destination format.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FPREM1—Partial Remainder

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 F5</td>
<td>FPREM1</td>
<td>Replace ST(0) with the IEEE remainder obtained on dividing ST(0) by ST(1)</td>
</tr>
</tbody>
</table>

### Description

Computes the IEEE remainder obtained on dividing the value in the ST(0) register (the dividend) by the value in the ST(1) register (the divisor or modulus), and stores the result in ST(0). The remainder represents the following value:

\[
\text{Remainder} = ST(0) - (N \times ST(1))
\]

Here, \( N \) is an integer value that is obtained by rounding the real-number quotient of \([ST(0) / ST(1)]\) to the nearest integer value. The sign of the remainder is the same as the sign of the dividend. The magnitude of the remainder is less than half the magnitude of the modulus, unless a partial remainder was computed (as described below).

This instruction produces an exact result; the precision (inexact) exception does not occur and the rounding control has no effect. The following table shows the results obtained when computing the remainder of various classes of numbers, assuming that underflow does not occur.

### Table 1-8. FPREM1 Zeros and NaNs

<table>
<thead>
<tr>
<th>ST(0)</th>
<th>ST(1)</th>
<th>−∞</th>
<th>−F</th>
<th>−0</th>
<th>+0</th>
<th>+F</th>
<th>+∞</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>ST(0)</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>NaN</td>
</tr>
<tr>
<td>−F</td>
<td>−F or −0</td>
<td>*</td>
<td>*</td>
<td>**</td>
<td>**</td>
<td>−F or −0</td>
<td>ST(0)</td>
<td>NaN</td>
</tr>
<tr>
<td>−0</td>
<td>−0</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>0</td>
<td>0</td>
<td>NaN</td>
<td></td>
</tr>
<tr>
<td>+0</td>
<td>+0</td>
<td>+0</td>
<td>*</td>
<td>*</td>
<td>0</td>
<td>0</td>
<td>NaN</td>
<td></td>
</tr>
<tr>
<td>+F</td>
<td>ST(0)</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>**</td>
<td>+F or +0</td>
<td>ST(0)</td>
<td>NaN</td>
</tr>
<tr>
<td>+∞</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>NaN</td>
<td></td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td></td>
</tr>
</tbody>
</table>

Notes:

- \( F \) means finite-real number.
- * indicates floating-point invalid-arithmetic-operand (#IA) exception.
- ** indicates floating-point zero-divide (#Z) exception.

When the result is 0, its sign is the same as that of the dividend. When the modulus is \( \infty \), the result is equal to the value in ST(0).

The FPREM1 instruction computes the remainder specified in IEEE Std 754. This instruction operates differently from the FPREM instruction in the way that it rounds the quotient of ST(0) divided by ST(1) to an integer (see the “Operation” below).
FPREM1—Partial Remainder (continued)

Like the FPREM instruction, the FPREM1 computes the remainder through iterative subtraction, but can reduce the exponent of ST(0) by no more than 63 in one execution of the instruction. If the instruction succeeds in producing a remainder that is less than one half the modulus, the operation is complete and the C2 flag in the FPU status word is cleared. Otherwise, C2 is set, and the result in ST(0) is called the *partial remainder*. The exponent of the partial remainder will be less than the exponent of the original dividend by at least 32. Software can re-execute the instruction (using the partial remainder in ST(0) as the dividend) until C2 is cleared.

**Note:** While executing such a remainder-computation loop, a higher-priority interrupting routine that needs the FPU can force a context switch in-between the instructions in the loop.

An important use of the FPREM1 instruction is to reduce the arguments of periodic functions. When reduction is complete, the instruction stores the three least-significant bits of the quotient in the C3, C1, and C0 flags of the FPU status word. This information is important in argument reduction for the tangent function (using a modulus of $\pi/4$), because it locates the original angle in the correct one of eight sectors of the unit circle.

**Operation**

\[
D \leftarrow \text{exponent}(\text{ST}(0)) - \text{exponent}(\text{ST}(1));
\]

\[
\text{IF } D < 64 \text{ THEN}
\]

\[
Q \leftarrow \text{Integer(RoundTowardNearestInteger}(\text{ST}(0) / \text{ST}(1)));
\]

\[
\text{ST}(0) \leftarrow \text{ST}(0) - (\text{ST}(1) * Q);
\]

\[
C2 \leftarrow 0;
\]

\[
C0, C3, C1 \leftarrow \text{LeastSignificantBits}(Q); (* Q2, Q1, Q0 *)
\]

\[
\text{ELSE}
\]

\[
C2 \leftarrow 1;
\]

\[
N \leftarrow \text{an implementation-dependent number between 32 and 63};
\]

\[
QQ \leftarrow \text{Integer(TruncateTowardZero}((\text{ST}(0) / \text{ST}(1)) / 2^{(0 - N)}));
\]

\[
\text{ST}(0) \leftarrow \text{ST}(0) - (\text{ST}(1) * QQ * 2^{(0 - N)});
\]

\[
\text{FI};
\]

**FPU Flags Affected**

- **C0** Set to bit 2 (Q2) of the quotient.
- **C1** Set to 0 if stack underflow occurred; otherwise, set to least significant bit of quotient (Q0).
- **C2** Set to 0 if reduction complete; set to 1 if incomplete.
- **C3** Set to bit 1 (Q1) of the quotient.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults** Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
FPREM1—Partial Remainder (continued)

Floating-point Exceptions

#IS Stack underflow occurred.

#IA Source operand is an SNaN value, modulus (divisor) is 0, dividend is \( \infty \), or unsupported format.

#D Source operand is a denormal value.

#U Result is too small for destination format.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FPTAN—Partial Tangent

Description

Computes the tangent of the source operand in register ST(0), stores the result in ST(0), and pushes a 1.0 onto the FPU register stack. The source operand must be given in radians and must be less than ±2^{63}. The following table shows the unmasked results obtained when computing the partial tangent of various classes of numbers, assuming that underflow does not occur.

<table>
<thead>
<tr>
<th>ST(0) SRC</th>
<th>ST(0) DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>*</td>
</tr>
<tr>
<td>−F</td>
<td>−F to +F</td>
</tr>
<tr>
<td>−0</td>
<td>0</td>
</tr>
<tr>
<td>+0</td>
<td>+0</td>
</tr>
<tr>
<td>+F</td>
<td>−F to +F</td>
</tr>
<tr>
<td>+∞</td>
<td>*</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
- F means finite-real number.
- * indicates floating-point invalid-arithmetic-operand (#IA) exception.

If the source operand is outside the acceptable range, the C2 flag in the FPU status word is set, and the value in register ST(0) remains unchanged. The instruction does not raise an exception when the source operand is out of range. It is up to the program to check the C2 flag for out-of-range conditions. Source values outside the range −2^{63} to +2^{63} can be reduced to the range of the instruction by subtracting an appropriate integer multiple of 2π or by using the FPREM instruction with a divisor of 2π.

The value 1.0 is pushed onto the register stack after the tangent has been computed to maintain compatibility with the Intel 8087 and Intel287 math coprocessors. This operation also simplifies the calculation of other trigonometric functions. For instance, the cotangent (which is the reciprocal of the tangent) can be computed by executing a FDIVR instruction after the FPTAN instruction.

Operation

IF

IF ST(0) < 2^{63}
THEN
  C2 ← 0;
  ST(0) ← tan(ST(0));
  TOP ← TOP − 1;
  ST(0) ← 1.0;
ELSE (*source operand is out-of-range *)
  C2 ← 1;
FI;
FPTAN—Partial Tangent (continued)

FPU Flags Affected

C1 Set to 0 if stack underflow occurred; set to 1 if stack overflow occurred.
Indicates rounding direction if the inexact-result exception (#P) is generated:
0 = not roundup; 1 = roundup.

C2 Set to 1 if source operand is outside the range $-2^{63}$ to $+2^{63}$; otherwise, cleared to 0.

C0, C3 Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Floating-point Exceptions

#IS Stack underflow occurred.
#IA Source operand is an SNaN value, $\infty$, or unsupported format.
#D Source operand is a denormal value.
#U Result is too small for destination format.
#P Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FRNDINT—Round to Integer

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 FC</td>
<td>FRNDINT</td>
<td>Round ST(0) to an integer.</td>
</tr>
</tbody>
</table>

Description

Rounds the source value in the ST(0) register to the nearest integral value, depending on the current rounding mode (setting of the RC field of the FPU control word), and stores the result in ST(0).

If the source value is \( \infty \), the value is not changed. If the source value is not an integral value, the floating-point inexact-result exception (#P) is generated.

Operation

\[
\text{ST}(0) \leftarrow \text{RoundToIntegralValue}(\text{ST}(0));
\]

FPU Flags Affected

- **C1** Set to 0 if stack underflow occurred.
- Indicates rounding direction if the inexact-result exception (#P) is generated:
  - 0 = not roundup; 1 = roundup.
- **C0, C2, C3** Undefined.

Floating-point Exceptions

- **#IS** Stack underflow occurred.
- **#IA** Source operand is an SNaN value or unsupported format.
- **#D** Source operand is a denormal value.
- **#P** Source operand is not an integral value.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Protected Mode Exceptions

- **#NM** EM or TS in CR0 is set.

Real Address Mode Exceptions

- **#NM** EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

- **#NM** EM or TS in CR0 is set.
FRSTOR—Restore FPU State

Description

Loads the FPU state (operating environment and register stack) from the memory area specified with the source operand. This state data is typically written to the specified memory location by a previous FSAVE/FNSAVE instruction.

The FPU operating environment consists of the FPU control word, status word, tag word, instruction pointer, data pointer, and last opcode. See the Intel Architecture Software Developer’s Manual for the layout in memory of the stored environment, depending on the operating mode of the processor (protected or real) and the size of the current address attribute (16-bit or 32-bit). In virtual-8086 mode, the real mode layouts are used. The contents of the FPU register stack are stored in the 80 bytes immediately follow the operating environment image.

The FRSTOR instruction should be executed in the same operating mode as the corresponding FSAVE/FNSAVE instruction.

If one or more unmasked exception bits are set in the new FPU status word, a floating-point exception will be generated. To avoid raising exceptions when loading a new operating environment, clear all the exception flags in the FPU status word that is being loaded.

Operation

\[
\begin{align*}
\text{FPUControlWord} & \leftarrow \text{SRC(FPUControlWord)}; \\
\text{FPUStatusWord} & \leftarrow \text{SRC(FPUStatusWord)}; \\
\text{FPUTagWord} & \leftarrow \text{SRC(FPUTagWord)}; \\
\text{FPUDataPointer} & \leftarrow \text{SRC(FPUDataPointer)}; \\
\text{FPUInstructionPointer} & \leftarrow \text{SRC(FPUInstructionPointer)}; \\
\text{FPULastInstructionOpcode} & \leftarrow \text{SRC(FPULastInstructionOpcode)}; \\
\end{align*}
\]

ST(0) \leftarrow \text{SRC(ST(0))};
ST(1) \leftarrow \text{SRC(ST(1))};
ST(2) \leftarrow \text{SRC(ST(2))};
ST(3) \leftarrow \text{SRC(ST(3))};
ST(4) \leftarrow \text{SRC(ST(4))};
ST(5) \leftarrow \text{SRC(ST(5))};
ST(6) \leftarrow \text{SRC(ST(6))};
ST(7) \leftarrow \text{SRC(ST(7))};

FPU Flags Affected

The C0, C1, C2, C3 flags are loaded.

Floating-point Exceptions

None; however, this operation might unmask an existing exception that has been detected but not generated, because it was masked. Here, the exception is generated at the completion of the instruction.
FRSTOR—Restore FPU State (continued)

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Exception Type</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Reg Faults</td>
<td>Disabled FP Register Fault if PSR.dfl is 1.</td>
</tr>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

Protected Mode Exceptions

<table>
<thead>
<tr>
<th>Exception</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP(0)</td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit. If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.</td>
</tr>
<tr>
<td>#SS(0)</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#NM</td>
<td>EM or TS in CR0 is set.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.</td>
</tr>
</tbody>
</table>

Real Address Mode Exceptions

<table>
<thead>
<tr>
<th>Exception</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP</td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.</td>
</tr>
<tr>
<td>#SS</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#NM</td>
<td>EM or TS in CR0 is set.</td>
</tr>
</tbody>
</table>

Virtual 8086 Mode Exceptions

<table>
<thead>
<tr>
<th>Exception</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP(0)</td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.</td>
</tr>
<tr>
<td>#SS(0)</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#NM</td>
<td>EM or TS in CR0 is set.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made.</td>
</tr>
</tbody>
</table>
FSAVE/FNSAVE—Store FPU State

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9B DD /6</td>
<td>FSAVE m94/108byte</td>
<td>Store FPU state to m94byte or m108byte after checking for pending unmasked floating-point exceptions. Then re-initialize the FPU.</td>
</tr>
<tr>
<td>DD /6</td>
<td>FNSAVE m94/108byte</td>
<td>Store FPU environment to m94byte or m108byte without checking for pending unmasked floating-point exceptions. Then re-initialize the FPU.</td>
</tr>
</tbody>
</table>

Description

Stores the current FPU state (operating environment and register stack) at the specified destination in memory, and then re-initializes the FPU. The FSAVE instruction checks for and handles pending unmasked floating-point exceptions before storing the FPU state; the FNSAVE instruction does not.

The FPU operating environment consists of the FPU control word, status word, tag word, instruction pointer, data pointer, and last opcode. See the Intel Architecture Software Developer’s Manual for the layout in memory of the stored environment, depending on the operating mode of the processor (protected or real) and the size of the current address attribute (16-bit or 32-bit). In virtual-8086 mode, the real mode layouts are used. The contents of the FPU register stack are stored in the 80 bytes immediately follow the operating environment image.

The saved image reflects the state of the FPU after all floating-point instructions preceding the FSAVE/FNSAVE instruction in the instruction stream have been executed.

After the FPU state has been saved, the FPU is reset to the same default values it is set to with the FINIT/FNINIT instructions (see “FINIT/FNINIT—Initialize Floating-point Unit” on page 3:481).

The FSAVE/FNSAVE instructions are typically used when the operating system needs to perform a context switch, an exception handler needs to use the FPU, or an application program needs to pass a “clean” FPU to a procedure.

Operation

(* Save FPU State and Registers *)
DEST(FPUControlWord) ← FPUControlWord;
DEST(FPUStatusWord) ← FPUStatusWord;
DEST(FPUTagWord) ← FPUTagWord;
DEST(FPUDataPointer) ← FPUDataPointer;
DEST(FPUInstructionPointer) ← FPUInstructionPointer;
DEST(FPULastInstructionOpcode) ← FPULastInstructionOpcode;
DEST(ST(0)) ← ST(0);
DEST(ST(1)) ← ST(1);
DEST(ST(2)) ← ST(2);
DEST(ST(3)) ← ST(3);
DEST(ST(4)) ← ST(4);
DEST(ST(5)) ← ST(5);
DEST(ST(6)) ← ST(6);
DEST(ST(7)) ← ST(7);
(* Initialize FPU *)
FPUControlWord ← 037FH;
FSAVE/FNSAVE—Store FPU State (continued)

FPUSStatusWord ← 0;
FPUTagWord ← FFFFH;
FPUTDataPointer ← 0;
FPUIInstructionPointer ← 0;
FPULastInstructionOpcode ← 0;

FPU Flags Affected

The C0, C1, C2, and C3 flags are saved and then cleared.

Floating-point Exceptions

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If destination is located in a nonwritable segment.
        If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
        If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0)  If a memory operand effective address is outside the SS segment limit.
#NM     EM or TS in CR0 is set.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS  If a memory operand effective address is outside the SS segment limit.
#NM  EM or TS in CR0 is set.
FSAVE/FNSAVE—Store FPU State (continued)

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.

Intel Architecture Compatibility Information

For Intel math coprocessors and FPUs prior to the Pentium processor, an FWAIT instruction should be executed before attempting to read from the memory image stored with a prior FSAVE/FNSAVE instruction. This FWAIT instruction helps insure that the storage operation has been completed.
FSCALE—Scale

Description

Multiplies the destination operand by 2 to the power of the source operand and stores the result in the destination operand. This instruction provides rapid multiplication or division by integral powers of 2. The destination operand is a real value that is located in register ST(0). The source operand is the nearest integer value that is smaller than the value in the ST(1) register (that is, the value in register ST(1) is truncate toward 0 to its nearest integer value to form the source operand). The actual scaling operation is performed by adding the source operand (integer value) to the exponent of the value in register ST(0). The following table shows the results obtained when scaling various classes of numbers, assuming that neither overflow nor underflow occurs.

<table>
<thead>
<tr>
<th>ST(0)</th>
<th>-N</th>
<th>0</th>
<th>+N</th>
</tr>
</thead>
<tbody>
<tr>
<td>-∞</td>
<td>-∞</td>
<td>-∞</td>
<td>-∞</td>
</tr>
<tr>
<td>-F</td>
<td>-F</td>
<td>-F</td>
<td>-F</td>
</tr>
<tr>
<td>-0</td>
<td>-0</td>
<td>-0</td>
<td>-0</td>
</tr>
<tr>
<td>+0</td>
<td>+0</td>
<td>+0</td>
<td>+0</td>
</tr>
<tr>
<td>+F</td>
<td>+F</td>
<td>+F</td>
<td>+F</td>
</tr>
<tr>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
F means finite-real number.
N means integer.

In most cases, only the exponent is changed and the mantissa (significand) remains unchanged. However, when the value being scaled in ST(0) is a denormal value, the mantissa is also changed and the result may turn out to be a normalized number. Similarly, if overflow or underflow results from a scale operation, the resulting mantissa will differ from the source’s mantissa.

The FSCALE instruction can also be used to reverse the action of the FXTRACT instruction, as shown in the following example:

```plaintext
FXTRACT;
FSCALE;
FSTP ST(1);
```

In this example, the FXTRACT instruction extracts the significand and exponent from the value in ST(0) and stores them in ST(0) and ST(1) respectively. The FSCALE then scales the significand in ST(0) by the exponent in ST(1), recreating the original value before the FXTRACT operation was performed. The FSTP ST(1) instruction returns the recreated value to the FPU register where it originally resided.

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 FD</td>
<td>FSCALE</td>
<td>Scale ST(0) by ST(1).</td>
</tr>
</tbody>
</table>
FScale—Scale (continued)

Operation
\[ ST(0) \leftarrow ST(0) \times 2^{ST(1)}; \]

FPU Flags Affected

- **C1** Set to 0 if stack underflow occurred.
  Indicates rounding direction if the inexact-result exception (#P) is generated:
  \(0 = \text{not roundup}; \ 1 = \text{roundup}.

- **C0, C2, C3** Undefined.

Floating-point Exceptions

- **#IS** Stack underflow occurred.
- **#IA** Source operand is an SNaN value or unsupported format.
- **#D** Source operand is a denormal value.
- **#U** Result is too small for destination format.
- **#O** Result is too large for destination format.
- **#P** Value cannot be represented exactly in destination format.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Protected Mode Exceptions

- **#NM** EM or TS in CR0 is set.

Real Address Mode Exceptions

- **#NM** EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

- **#NM** EM or TS in CR0 is set.
**FSIN—Sine**

**Description**

 Calculates the sine of the source operand in register ST(0) and stores the result in ST(0). The source operand must be given in radians and must be within the range $-2^{63}$ to $+2^{63}$. The following table shows the results obtained when taking the sine of various classes of numbers, assuming that underflow does not occur.

<table>
<thead>
<tr>
<th>SRC (ST(0))</th>
<th>DEST (ST(0))</th>
</tr>
</thead>
<tbody>
<tr>
<td>$-\infty$</td>
<td>*</td>
</tr>
<tr>
<td>$-F$</td>
<td>$-1$ to $+1$</td>
</tr>
<tr>
<td>$-0$</td>
<td>$-0$</td>
</tr>
<tr>
<td>$+0$</td>
<td>$+0$</td>
</tr>
<tr>
<td>$+F$</td>
<td>$-1$ to $+1$</td>
</tr>
<tr>
<td>$+\infty$</td>
<td>*</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

**Notes:**

F means finite-real number.

* indicates floating-point invalid-arithmetic-operand (#IA) exception.

If the source operand is outside the acceptable range, the C2 flag in the FPU status word is set, and the value in register ST(0) remains unchanged. The instruction does not raise an exception when the source operand is out of range. It is up to the program to check the C2 flag for out-of-range conditions. Source values outside the range $-2^{63}$ to $+2^{63}$ can be reduced to the range of the instruction by subtracting an appropriate integer multiple of $2\pi$ or by using the FPREM instruction with a divisor of $2\pi$.

**Operation**

**IF I**

**IF** ST(0) < $2^{63}$

**THEN**

\[
\begin{align*}
& \text{C2} \leftarrow 0; \\
& \text{ST}(0) \leftarrow \sin(\text{ST}(0)); \\
& \text{ELSE} (* \text{ source operand out of range } *) \\
& \text{C2} \leftarrow 1; \\
& \text{FI};
\end{align*}
\]
FSIN—Sine (continued)

FPU Flags Affected
C1 Set to 0 if stack underflow occurred.
Indicates rounding direction if the inexact-result exception (#P) is generated:
0 = not round; 1 = round.
C2 Set to 1 if source operand is outside the range $-2^{63}$ to $+2^{63}$; otherwise, cleared
to 0.
C0, C3 Undefined.

Additional Itanium System Environment Exceptions
Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Floating-point Exceptions
#IS Stack underflow occurred.
#IA Source operand is an SNaN value, $\infty$, or unsupported format.
#D Source operand is a denormal value.
#P Value cannot be represented exactly in destination format.

Protected Mode Exceptions
#NM EM or TS in CR0 is set.

Real Address Mode Exceptions
#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions
#NM EM or TS in CR0 is set.
FSINCOS—Sine and Cosine

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 FB</td>
<td>FSINCOS</td>
<td>Compute the sine and cosine of ST(0); replace ST(0) with the sine, and push the cosine onto the register stack.</td>
</tr>
</tbody>
</table>

**Description**

Computes both the sine and the cosine of the source operand in register ST(0), stores the sine in ST(0), and pushes the cosine onto the top of the FPU register stack. (This instruction is faster than executing the FSIN and FCOS instructions in succession.)

The source operand must be given in radians and must be within the range $-2^{63}$ to $+2^{63}$. The following table shows the results obtained when taking the sine and cosine of various classes of numbers, assuming that underflow does not occur.

<table>
<thead>
<tr>
<th>SRC</th>
<th>DEST</th>
</tr>
</thead>
<tbody>
<tr>
<td>ST(0)</td>
<td>ST(0) Cosine</td>
</tr>
<tr>
<td>$-\infty$</td>
<td>$\ast$</td>
</tr>
<tr>
<td>$-F$</td>
<td>$-1$ to $+1$</td>
</tr>
<tr>
<td>$-0$</td>
<td>$+1$</td>
</tr>
<tr>
<td>$+0$</td>
<td>$+1$</td>
</tr>
<tr>
<td>$+F$</td>
<td>$-1$ to $+1$</td>
</tr>
<tr>
<td>$+\infty$</td>
<td>$\ast$</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:

F means finite-real number.

* indicates floating-point invalid-arithmetic-operand (#IA) exception.

If the source operand is outside the acceptable range, the C2 flag in the FPU status word is set, and the value in register ST(0) remains unchanged. The instruction does not raise an exception when the source operand is out of range. It is up to the program to check the C2 flag for out-of-range conditions. Source values outside the range $-2^{63}$ to $+2^{63}$ can be reduced to the range of the instruction by subtracting an appropriate integer multiple of $2\pi$ or by using the FPREM instruction with a divisor of $2\pi$.

**Operation**

```
IF ST(0) < 2^{63}
THEN
    C2 ← 0;
    TEMP ← cosine(ST(0));
    ST(0) ← sine(ST(0));
    TOP ← TOP − 1;
    ST(0) ← TEMP;
ELSE (* source operand out of range *)
    C2 ← 1;
FI:
```
FSINCOS—Sine and Cosine (continued)

FPU Flags Affected

C1  Set to 0 if stack underflow occurred; set to 1 if stack overflow occurs.
Indicates rounding direction if the inexact-result exception (#P) is generated:
0 = not roundup; 1 = roundup.

C2  Set to 1 if source operand is outside the range $-2^{63}$ to $+2^{63}$; otherwise, cleared
to 0.

C0, C3  Undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Floating-point Exceptions

#IS  Stack underflow occurred.

#IA  Source operand is an SNaN value, $\infty$, or unsupported format.

#D  Source operand is a denormal value.

#U  Result is too small for destination format.

#P  Value cannot be represented exactly in destination format.

Protected Mode Exceptions

#NM  EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM  EM or TS in CR0 is set.
FSQRT—Square Root

Description

Calculates the square root of the source value in the ST(0) register and stores the result in ST(0).

The following table shows the results obtained when taking the square root of various classes of numbers, assuming that neither overflow nor underflow occurs.

<table>
<thead>
<tr>
<th>SRC (ST(0))</th>
<th>DEST (ST(0))</th>
</tr>
</thead>
<tbody>
<tr>
<td>$\infty$</td>
<td>*</td>
</tr>
<tr>
<td>$-\infty$</td>
<td>*</td>
</tr>
<tr>
<td>$-F$</td>
<td>$-0$</td>
</tr>
<tr>
<td>$-0$</td>
<td>$+0$</td>
</tr>
<tr>
<td>$+0$</td>
<td>$+F$</td>
</tr>
<tr>
<td>$+\infty$</td>
<td>$+\infty$</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
- $F$ means finite-real number.
- * indicates floating-point invalid-arithmetic-operand (#IA) exception.

Operation

$ST(0) \leftarrow \text{SquareRoot}(ST(0))$;

FPU Flags Affected

- $C1$: Set to 0 if stack underflow occurred.
  Indicates rounding direction if inexact-result exception (#P) is generated: $0 = \text{not roundup}; 1 = \text{roundup}$.
- $C0, C2, C3$: Undefined.

Floating-point Exceptions

- #IS: Stack underflow occurred.
- #IA: Source operand is an SNaN value or unsupported format.
- #D: Source operand is a negative value (except for $-0$).
- #P: Value cannot be represented exactly in destination format.
Additional Itanium System Environment Exceptions
Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Protected Mode Exceptions
#NM  EM or TS in CR0 is set.

Real Address Mode Exceptions
#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions
#NM  EM or TS in CR0 is set.
FST/FSTP—Store Real

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 /2</td>
<td>FST m32real</td>
<td>Copy ST(0) to m32real</td>
</tr>
<tr>
<td>DD /2</td>
<td>FST m64real</td>
<td>Copy ST(0) to m64real</td>
</tr>
<tr>
<td>DD D0+i</td>
<td>FST ST(i)</td>
<td>Copy ST(0) to ST(i)</td>
</tr>
<tr>
<td>D9 /3</td>
<td>FSTP m32real</td>
<td>Copy ST(0) to m32real and pop register stack</td>
</tr>
<tr>
<td>DD /3</td>
<td>FSTP m64real</td>
<td>Copy ST(0) to m64real and pop register stack</td>
</tr>
<tr>
<td>DB /7</td>
<td>FSTP m80real</td>
<td>Copy ST(0) to m80real and pop register stack</td>
</tr>
<tr>
<td>DD D8+i</td>
<td>FSTP ST(i)</td>
<td>Copy ST(0) to ST(i) and pop register stack</td>
</tr>
</tbody>
</table>

Description

The FST instruction copies the value in the ST(0) register to the destination operand, which can be a memory location or another register in the FPU registers stack. When storing the value in memory, the value is converted to single- or double-real format.

The FSTP instruction performs the same operation as the FST instruction and then pops the register stack. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. The FSTP instruction can also stores values in memory in extended-real format.

If the destination operand is a memory location, the operand specifies the address where the first byte of the destination value is to be stored. If the destination operand is a register, the operand specifies a register in the register stack relative to the top of the stack.

If the destination size is single- or double-real, the significand of the value being stored is rounded to the width of the destination (according to rounding mode specified by the RC field of the FPU control word), and the exponent is converted to the width and bias of the destination format. If the value being stored is too large for the destination format, a numeric overflow exception (#O) is generated and, if the exception is unmasked, no value is stored in the destination operand. If the value being stored is a denormal value, the denormal exception (#D) is not generated. This condition is simply signaled as a numeric underflow exception (#U) condition.

If the value being stored is ±0, ±∞, or a NaN, the least-significant bits of the significand and the exponent are truncated to fit the destination format. This operation preserves the value’s identity as a 0, ∞, or NaN.

If the destination operand is a non-empty register, the invalid-operation exception is not generated.

Operation

DEST ← ST(0);
IF instruction = FSTP
    THEN
        PopRegisterStack;
FI;
FST/FSTP—Store Real (continued)

**FPU Flags Affected**

C1 Set to 0 if stack underflow occurred.

Indicates rounding direction of if the floating-point inexact exception (#P) is generated: 0 = not roundup; 1 = roundup.

C0, C2, C3 Undefined.

**Floating-point Exceptions**

#IS Stack underflow occurred.

#IA Source operand is an SNaN value or unsupported format.

#U Result is too small for the destination format.

#O Result is too large for the destination format.

#P Value cannot be represented exactly in destination format.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults

Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults

VTID Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Protected Mode Exceptions**

#GP(0) If the destination is located in a nonwritable segment.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.
FST/FSTP—Store Real (continued)

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
FSTCW/FNSTCW—Store Control Word

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9B D9 /7</td>
<td>FSTCW m2byte</td>
<td>Store FPU control word to m2byte after checking for pending unmasked floating-point exceptions.</td>
</tr>
<tr>
<td>D9 /7</td>
<td>FNSTCW m2byte</td>
<td>Store FPU control word to m2byte without checking for pending unmasked floating-point exceptions.</td>
</tr>
</tbody>
</table>

Description

Stores the current value of the FPU control word at the specified destination in memory. The FSTCW instruction checks for and handles pending unmasked floating-point exceptions before storing the control word; the FNSTCW instruction does not.

Operation

DEST ← FPUControlWord;

FPU Flags Affected

The C0, C1, C2, and C3 flags are undefined.

Floating-point Exceptions

None.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

- #GP(0): If the destination is located in a nonwritable segment.
- #SS(0): If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code): If a page fault occurs.
- #AC(0): If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
FSTCW/FNSTCW—Store Control Word (continued)

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
FSTENV/FNSTENV—Store FPU Environment

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9B D9 /6</td>
<td>FSTENV m14/28byte</td>
<td>Store FPU environment to m14byte or m28byte after checking for pending unmasked floating-point exceptions. Then mask all floating-point exceptions.</td>
</tr>
<tr>
<td>D9 /6</td>
<td>FNSTENV m14/28byte</td>
<td>Store FPU environment to m14byte or m28byte without checking for pending unmasked floating-point exceptions. Then mask all floating-point exceptions.</td>
</tr>
</tbody>
</table>

**Description**

Saves the current FPU operating environment at the memory location specified with the destination operand, and then masks all floating-point exceptions. The FPU operating environment consists of the FPU control word, status word, tag word, instruction pointer, data pointer, and last opcode. See the Intel Architecture Software Developer’s Manual for the layout in memory of the stored environment, depending on the operating mode of the processor (protected or real) and the size of the current address attribute (16-bit or 32-bit). (In virtual-8086 mode, the real mode layouts are used.)

The FSTENV instruction checks for and handles any pending unmasked floating-point exceptions before storing the FPU environment; the FNSTENV instruction does not. The saved image reflects the state of the FPU after all floating-point instructions preceding the FSTENV/FNSTENV instruction in the instruction stream have been executed.

These instructions are often used by exception handlers because they provide access to the FPU instruction and data pointers. The environment is typically saved in the procedure stack. Masking all exceptions after saving the environment prevents floating-point exceptions from interrupting the exception handler.

**Operation**

DEST(FPUControlWord) ← FPUControlWord;
DEST(FPUStatusWord) ← FPUStatusWord;
DEST(FPUTagWord) ← FPUTagWord;
DEST(FPUDataPointer) ← FPUDataPointer;
DEST(FPUInstructionPointer) ← FPUInstructionPointer;
DEST(FPULastInstructionOpcode) ← FPULastInstructionOpcode;

**FPU Flags Affected**

The C0, C1, C2, and C3 are undefined.

**Floating-point Exceptions**

None.
Additional Itanium System Environment Exceptions

Itanium Reg Faults
- Disabled FP Register Fault if PSR.dfl is 1

Itanium Mem Faults
- VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NM EM or TS in CR0 is set.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
FSTSW/FNSTSW—Store Status Word

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9B DD /7</td>
<td>FSTSW m2byte</td>
<td>Store FPU status word at m2byte after checking for pending unmasked floating-point exceptions.</td>
</tr>
<tr>
<td>9B DF E0</td>
<td>FSTSW AX</td>
<td>Store FPU status word in AX register after checking for pending unmasked floating-point exceptions.</td>
</tr>
<tr>
<td>DD /7</td>
<td>FNSTSW m2byte</td>
<td>Store FPU status word at m2byte without checking for pending unmasked floating-point exceptions.</td>
</tr>
<tr>
<td>DF E0</td>
<td>FNSTSW AX</td>
<td>Store FPU status word in AX register without checking for pending unmasked floating-point exceptions.</td>
</tr>
</tbody>
</table>

Description

Stores the current value of the FPU status word in the destination location. The destination operand can be either a two-byte memory location or the AX register. The FSTSW instruction checks for and handles pending unmasked floating-point exceptions before storing the status word; the FNSTSW instruction does not.

The FNSTSW AX form of the instruction is used primarily in conditional branching (for instance, after an FPU comparison instruction or an FPREM, FPREM1, or FXAM instruction), where the direction of the branch depends on the state of the FPU condition code flags. This instruction can also be used to invoke exception handlers (by examining the exception flags) in environments that do not use interrupts. When the FNSTSW AX instruction is executed, the AX register is updated before the processor executes any further instructions. The status stored in the AX register is thus guaranteed to be from the completion of the prior FPU instruction.

Operation

DEST ← FPUStatusWord;

FPU Flags Affected

The C0, C1, C2, and C3 are undefined.

Floating-point Exceptions

None.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NuT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
FSTSW/FNSTSW—Store Status Word (continued)

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a
null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made
while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
FSUB/FSUBP/FISUB—Subtract

Description

Subtracts the source operand from the destination operand and stores the difference in the destination location. The destination operand is always an FPU data register; the source operand can be a register or a memory location. Source operands in memory can be in single-real, double-real, word-integer, or short-integer formats.

The no-operand version of the instruction subtracts the contents of the ST(0) register from the ST(1) register and stores the result in ST(1). The one-operand version subtracts the contents of a memory location (either a real or an integer value) from the contents of the ST(0) register and stores the result in ST(0). The two-operand version subtracts the contents of the ST(0) register from the ST(i) register or vice versa.

The FSUBP instructions perform the additional operation of popping the FPU register stack following the subtraction. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. The no-operand version of the floating-point subtract instructions always results in the register stack being popped. In some assemblers, the mnemonic for this instruction is FSUB rather than FSUBP.

The FISUB instructions convert an integer source operand to extended-real format before performing the subtraction.

The following table shows the results obtained when subtracting various classes of numbers from one another, assuming that neither overflow nor underflow occurs. Here, the SRC value is subtracted from the DEST value (DEST − SRC = result).

When the difference between two operands of like sign is 0, the result is +0, except for the round toward −∞ mode, in which case the result is −0. This instruction also guarantees that +0 − (−0) = +0, and that −0 − (+0) = −0. When the source operand is an integer 0, it is treated as a +0.

When one operand is ∞, the result is ∞ of the expected sign. If both operands are ∞ of the same sign, an invalid-operation exception is generated.

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D8 /4</td>
<td>FSUB m32real</td>
<td>Subtract m32real from ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DC /4</td>
<td>FSUB m64real</td>
<td>Subtract m64real from ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>D8 E0+i</td>
<td>FSUB ST(0), ST(i)</td>
<td>Subtract ST(i) from ST(0) and store result in ST(i)</td>
</tr>
<tr>
<td>DC E8+i</td>
<td>FSUB ST(i), ST(0)</td>
<td>Subtract ST(0) from ST(i) and store result in ST(i)</td>
</tr>
<tr>
<td>DE E8+i</td>
<td>FSUBP ST(i), ST(0)</td>
<td>Subtract ST(0) from ST(i), store result in ST(i), and pop register stack</td>
</tr>
<tr>
<td>DE E9</td>
<td>FSUBP</td>
<td>Subtract ST(0) from ST(1), store result in ST(1), and pop register stack</td>
</tr>
<tr>
<td>DA /4</td>
<td>FISUB m32int</td>
<td>Subtract m32int from ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DE /4</td>
<td>FISUB m16int</td>
<td>Subtract m16int from ST(0) and store result in ST(0)</td>
</tr>
</tbody>
</table>
FSUB/FSUBP/FISUB—Subtract (continued)

Table 1-9. FSUB Zeros and NaNs

<table>
<thead>
<tr>
<th>SRC</th>
<th>−∞</th>
<th>−F or −I</th>
<th>−0</th>
<th>+0</th>
<th>+F or +I</th>
<th>+∞</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>NaN</td>
</tr>
<tr>
<td>−F</td>
<td>+∞</td>
<td>±F or ±0</td>
<td>DEST</td>
<td>DEST</td>
<td>−F</td>
<td>−∞</td>
<td>NaN</td>
</tr>
<tr>
<td>−0</td>
<td>+∞</td>
<td>−SRC</td>
<td>±0</td>
<td>−0</td>
<td>−SRC</td>
<td>−∞</td>
<td>NaN</td>
</tr>
<tr>
<td>+0</td>
<td>+∞</td>
<td>−SRC</td>
<td>+0</td>
<td>±0</td>
<td>−SRC</td>
<td>−∞</td>
<td>NaN</td>
</tr>
<tr>
<td>+F</td>
<td>+∞</td>
<td>+F</td>
<td>DEST</td>
<td>DEST</td>
<td>±F or ±0</td>
<td>−∞</td>
<td>NaN</td>
</tr>
<tr>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>+∞</td>
<td>*</td>
<td>NaN</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
F means finite-real number.
I means integer.
* indicates floating-point invalid-arithmetic-operand (#IA) exception.

Operation

IF instruction is FISUB
THEN
    DEST ← DEST − ConvertExtendedReal(SRC);
ELSE (* source operand is real number *)
    DEST ← DEST − SRC;
FI;

IF instruction = FSUBP
THEN
    PopRegisterStack
FI;

FPU Flags Affected

C1 Set to 0 if stack underflow occurred.
Indicates rounding direction if the inexact-result exception (#P) fault is generated: 0 = not roundup; 1 = roundup.

C0, C2, C3 Undefined.

Floating-point Exceptions

#IS Stack underflow occurred.

#IA Operand is an SNaN value or unsupported format.
Operands are infinities of like sign.

#D Source operand is a denormal value.

#U Result is too small for destination format.

#O Result is too large for destination format.

#P Value cannot be represented exactly in destination format.
FSUB/FSUBP/FISUB—Subtract (continued)

Additional Itanium System Environment Exceptions
Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions
#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
        If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#NM    EM or TS in CR0 is set.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions
#GP    If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS    If a memory operand effective address is outside the SS segment limit.
#NM    EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions
#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#NM    EM or TS in CR0 is set.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
FSUBR/FSUBRP/FISUBR—Reverse Subtract

Description

Subtracts the destination operand from the source operand and stores the difference in the destination location. The destination operand is always an FPU register; the source operand can be a register or a memory location. Source operands in memory can be in single-real, double-real, word-integer, or short-integer formats.

These instructions perform the reverse operations of the FSUB, FSUBP, and FISUB instructions. They are provided to support more efficient coding.

The no-operand version of the instruction subtracts the contents of the ST(1) register from the ST(0) register and stores the result in ST(1). The one-operand version subtracts the contents of the ST(0) register from the contents of a memory location (either a real or an integer value) and stores the result in ST(0). The two-operand version, subtracts the contents of the ST(i) register from the ST(0) register or vice versa.

The FSUBRP instructions perform the additional operation of popping the FPU register stack following the subtraction. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1. The no-operand version of the floating-point reverse subtract instructions always results in the register stack being popped. In some assemblers, the mnemonic for this instruction is FSUBR rather than FSUBRP.

The FISUBR instructions convert an integer source operand to extended-real format before performing the subtraction.

The following table shows the results obtained when subtracting various classes of numbers from one another, assuming that neither overflow nor underflow occurs. Here, the DEST value is subtracted from the SRC value (SRC – DEST = result).

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D8 /5</td>
<td>FSUBR m32real</td>
<td>Subtract ST(0) from m32real and store result in ST(0)</td>
</tr>
<tr>
<td>DC /5</td>
<td>FSUBR m64real</td>
<td>Subtract ST(0) from m64real and store result in ST(0)</td>
</tr>
<tr>
<td>D8 E8+i</td>
<td>FSUBR ST(0), ST(i)</td>
<td>Subtract ST(0) from ST(i) and store result in ST(0)</td>
</tr>
<tr>
<td>DC E0+i</td>
<td>FSUBR ST(i), ST(0)</td>
<td>Subtract ST(i) from ST(0) and store result in ST(0)</td>
</tr>
<tr>
<td>DE E0+i</td>
<td>FSUBRP ST(i), ST(0)</td>
<td>Subtract ST(0) from ST(i), store result in ST(i), and pop register stack</td>
</tr>
<tr>
<td>DE E1</td>
<td>FSUBRP</td>
<td>Subtract ST(1) from ST(0), store result in ST(1), and pop register stack</td>
</tr>
<tr>
<td>DA /5</td>
<td>FISUBR m32int</td>
<td>Subtract ST(0) from m32int and store result in ST(0)</td>
</tr>
<tr>
<td>DE /5</td>
<td>FISUBR m16int</td>
<td>Subtract ST(0) from m16int and store result in ST(0)</td>
</tr>
</tbody>
</table>
FSUBR/FSUBRP/FISUBR—Reverse Subtract (continued)

When the difference between two operands of like sign is 0, the result is +0, except for the round
toward −∞ mode, in which case the result is −0. This instruction also guarantees that +0 − (−0) = +0,
and that −0 − (+0) = −0. When the source operand is an integer 0, it is treated as a +0.

When one operand is ∞, the result is ∞ of the expected sign. If both operands are ∞ of the same
sign, an invalid-operation exception is generated.

Table 1-10. FSUBR Zeros and NaNs

<table>
<thead>
<tr>
<th>DEST</th>
<th>−∞</th>
<th>−F</th>
<th>−0</th>
<th>+0</th>
<th>+F</th>
<th>+∞</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td></td>
<td>+∞</td>
<td>−0</td>
<td>+0</td>
<td>+∞</td>
<td>+∞</td>
<td>NaN</td>
</tr>
<tr>
<td>−F or −I</td>
<td>−∞</td>
<td>±F or ±0</td>
<td>−DEST</td>
<td>−DEST</td>
<td>+F</td>
<td>+∞</td>
<td>NaN</td>
</tr>
<tr>
<td>−0</td>
<td>−∞</td>
<td>SRC</td>
<td>±0</td>
<td>+0</td>
<td>SRC</td>
<td>+∞</td>
<td>NaN</td>
</tr>
<tr>
<td>+0</td>
<td>−∞</td>
<td>SRC</td>
<td>−0</td>
<td>±0</td>
<td>SRC</td>
<td>+∞</td>
<td>NaN</td>
</tr>
<tr>
<td>+F or +I</td>
<td>−∞</td>
<td>−F</td>
<td>−DEST</td>
<td>−DEST</td>
<td>±F or ±0</td>
<td>+∞</td>
<td>NaN</td>
</tr>
<tr>
<td>+∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>−∞</td>
<td>*</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

Notes:
F means finite-real number.
I means integer.
* indicates floating-point invalid-arithmetic-operand (#IA) exception.

Operation

IF instruction is FISUBR
THEN
   DEST ← ConvertExtendedReal(SRC) − DEST;
ELSE (* source operand is real number *)
   DEST ← SRC − DEST;
FI;
IF instruction = FSUBRP
THEN
   PopRegisterStack
FI;

FPU Flags Affected

C1 Set to 0 if stack underflow occurred.
   Indicates rounding direction if the inexact-result exception (#P) fault is
generated: 0 = not roundup; 1 = roundup.

C0, C2, C3 Undefined.
FSUBR/FSUBRP/FISUBR—Reverse Subtract (continued)

Floating-point Exceptions

#IS Stack underflow occurred.
#IA Operand is an SNaN value or unsupported format.
Operands are infinities of like sign.
#D Source operand is a denormal value.
#U Result is too small for destination format.
#O Result is too large for destination format.
#P Value cannot be represented exactly in destination format.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault,
Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault,
Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault,
Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a
null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made
while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#NM EM or TS in CR0 is set.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
FTST—TEST

Description

Comparedes the value in the ST(0) register with 0.0 and sets the condition code flags C0, C2, and C3 in the FPU status word according to the results (see table below).

<table>
<thead>
<tr>
<th>Condition</th>
<th>C3</th>
<th>C2</th>
<th>C0</th>
</tr>
</thead>
<tbody>
<tr>
<td>ST(0) &gt; 0.0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>ST(0) &lt; 0.0</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>ST(0) = 0.0</td>
<td>1</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>Unordered</td>
<td>1</td>
<td>1</td>
<td>1</td>
</tr>
</tbody>
</table>

This instruction performs an “unordered comparison.” An unordered comparison also checks the class of the numbers being compared (see “FXAM—Examine” on page 3:541). If the value in register ST(0) is a NaN or is in an undefined format, the condition flags are set to “unordered.”

The sign of zero is ignored, so that –0.0 = +0.0.

Operation

CASE (relation of operands) OF
   Not comparable: C3, C2, C0 ← 111;
   ST(0) > 0.0: C3, C2, C0 ← 000;
   ST(0) < 0.0: C3, C2, C0 ← 001;
   ST(0) = 0.0: C3, C2, C0 ← 100;
   ESAC;

FPU Flags Affected

C1 Set to 0 if stack underflow occurred; otherwise, cleared to 0.
C0, C2, C3 See above table.

Floating-point Exceptions

#IS Stack underflow occurred.
#IA One or both operands are NaN values or have unsupported formats.
#D One or both operands are denormal values.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.
FTST—TEST (continued)

Real Address Mode Exceptions
#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions
#NM EM or TS in CR0 is set.
FUCOM/FUCOMP/FUCOMPP—Unordered Compare Real

**Description**

Performs an unordered comparison of the contents of register ST(0) and ST(i) and sets condition code flags C0, C2, and C3 in the FPU status word according to the results (see the table below). If no operand is specified, the contents of registers ST(0) and ST(1) are compared. The sign of zero is ignored, so that \(-0.0 = +0.0\).

An unordered comparison checks the class of the numbers being compared (see “FXAM—Examine” on page 3:541). The FUCOM instructions perform the same operation as the FCOM instructions. The only difference is that the FUCOM instruction raises the invalid-arithmetic-operand exception (#IA) only when either or both operands is an SNaN or is in an unsupported format; QNaNs cause the condition code flags to be set to unordered, but do not cause an exception to be generated. The FCOM instruction raises an invalid-operation exception when either or both of the operands is a NaN value of any kind or is in an unsupported format.

As with the FCOM instructions, if the operation results in an invalid-arithmetic-operand exception being raised, the condition code flags are set only if the exception is masked.

The FUCOMP instructions pop the register stack following the comparison operation and the FUCOMPP instructions pops the register stack twice following the comparison operation. To pop the register stack, the processor marks the ST(0) register as empty and increments the stack pointer (TOP) by 1.

**Operation**

\[
\text{CASE (relation of operands) OF} \\
\text{ST > SRC: } C3, C2, C0 \leftarrow 000;  \\
\text{ST < SRC: } C3, C2, C0 \leftarrow 001;  \\
\text{ST = SRC: } C3, C2, C0 \leftarrow 100;  \\
\text{ESAC;} \\
\text{IF ST(0) or SRC = QNaN, but not SNaN or unsupported format}
\]
FUCOM/FUCOMP/FUCOMPP—Unordered Compare Real (continued)

THEN
   C3, C2, C0 ← 111;
ELSE (* ST(0) or SRC is SNaN or unsupported format *)
   IF FPUControlWord.IM = 1
      THEN
         C3, C2, C0 ← 111;
   FI;
FI;
IF instruction = FUCOMP
   THEN
      PopRegisterStack;
FI;
IF instruction = FUCOMPP
   THEN
      PopRegisterStack;
PpopRegisterStack;
FI;

FPU Flags Affected

C1 Set to 0 if stack underflow occurred.
C0, C2, C3 See table on previous page.

Floating-point Exceptions

#IS Stack underflow occurred.
#IA One or both operands are SNaN values or have unsupported formats. Detection of a QNaN value in and of itself does not raise an invalid-operand exception.
#D One or both operands are denormal values.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FWAIT—Wait

See entry for WAIT.
FXAM—Examine

**Description**

Examines the contents of the ST(0) register and sets the condition code flags C0, C2, and C3 in the FPU status word to indicate the class of value or number in the register (see the table below).

<table>
<thead>
<tr>
<th>Class</th>
<th>C3</th>
<th>C2</th>
<th>C0</th>
</tr>
</thead>
<tbody>
<tr>
<td>Unsupported</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>NaN</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>Normal finite number</td>
<td>0</td>
<td>1</td>
<td>0</td>
</tr>
<tr>
<td>Infinity</td>
<td>0</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>Zero</td>
<td>1</td>
<td>0</td>
<td>0</td>
</tr>
<tr>
<td>Empty</td>
<td>1</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>Denormal number</td>
<td>1</td>
<td>1</td>
<td>0</td>
</tr>
</tbody>
</table>

The C1 flag is set to the sign of the value in ST(0), regardless of whether the register is empty or full.

**Operation**

\[ C1 \leftarrow \text{sign bit of ST}; \quad (\ast 0 \text{ for positive, } 1 \text{ for negative } \ast) \]

CASE (class of value or number in ST(0)) OF

- Unsupported: \( C3, C2, C0 \leftarrow 000; \)
- NaN: \( C3, C2, C0 \leftarrow 001; \)
- Normal: \( C3, C2, C0 \leftarrow 010; \)
- Infinity: \( C3, C2, C0 \leftarrow 011; \)
- Zero: \( C3, C2, C0 \leftarrow 100; \)
- Empty: \( C3, C2, C0 \leftarrow 101; \)
- Denormal: \( C3, C2, C0 \leftarrow 110; \)

**FPU Flags Affected**

C1: Sign of value in ST(0).
C0, C2, C3: See table above.

**Floating-point Exceptions**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
FXAM—Examine (continued)

Protected Mode Exceptions

#NM EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM EM or TS in CR0 is set.
FXCH—Exchange Register Contents

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 C8+i</td>
<td>FXCH ST(i)</td>
<td>Exchange the contents of ST(0) and ST(i)</td>
</tr>
<tr>
<td>D9 C9</td>
<td>FXCH</td>
<td>Exchange the contents of ST(0) and ST(1)</td>
</tr>
</tbody>
</table>

**Description**

Exchanges the contents of registers ST(0) and ST(i). If no source operand is specified, the contents of ST(0) and ST(1) are exchanged.

This instruction provides a simple means of moving values in the FPU register stack to the top of the stack [ST(0)], so that they can be operated on by those floating-point instructions that can only operate on values in ST(0). For example, the following instruction sequence takes the square root of the third register from the top of the register stack:

```
FXCH ST(3);
FSQRT;
FXCH ST(3);
```

**Operation**

IF number-of-operands is 1
THEN

```
temp ← ST(0);
ST(0) ← SRC;
SRC ← temp;
```
ELSE

```
temp ← ST(0);
ST(0) ← ST(1);
ST(1) ← temp;
```

**FPU Flags Affected**

C1 Set to 0 if stack underflow occurred; otherwise, cleared to 0.

C0, C2, C3 Undefined.

**Floating-point Exceptions**

#IS Stack underflow occurred.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

**Protected Mode Exceptions**

#NM EM or TS in CR0 is set.
FXCH—Exchange Register Contents (continued)

Real Address Mode Exceptions
#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions
#NM  EM or TS in CR0 is set.
FXTRACT—Extract Exponent and Significand

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 F4</td>
<td>FXTRACT</td>
<td>Separate value in ST(0) into exponent and significand, store exponent in ST(0), and push the significand onto the register stack.</td>
</tr>
</tbody>
</table>

**Description**

Separates the source value in the ST(0) register into its exponent and significand, stores the exponent in ST(0), and pushes the significand onto the register stack. Following this operation, the new top-of-stack register ST(0) contains the value of the original significand expressed as a real number. The sign and significand of this value are the same as those found in the source operand, and the exponent is 3FFFH (biased value for a true exponent of zero). The ST(1) register contains the value of the original operand’s true (unbiased) exponent expressed as a real number. (The operation performed by this instruction is a superset of the IEEE-recommended logb(x) function.)

This instruction and the F2XM1 instruction are useful for performing power and range scaling operations. The FXTRACT instruction is also useful for converting numbers in extended-real format to decimal representations (e.g. for printing or displaying).

If the floating-point zero-divide exception (#Z) is masked and the source operand is zero, an exponent value of $-\infty$ is stored in register ST(1) and 0 with the sign of the source operand is stored in register ST(0).

**Operation**

```
TEMP ← Significand(ST(0));
ST(0) ← Exponent(ST(0));
TOP← TOP − 1;
ST(0) ← TEMP;
```

**FPU Flags Affected**

- $C1$ Set to 0 if stack underflow occurred; set to 1 if stack overflow occurred.
- $C0, C2, C3$ Undefined.

**Floating-point Exceptions**

- **#IS** Stack underflow occurred.
- **#OA** Stack overflow occurred.
- **#IA** Source operand is an SNaN value or unsupported format.
- **#Z** ST(0) operand is ±0.
- **#D** Source operand is a denormal value.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
FXTRACT—Extract Exponent and Significand (continued)

Protected Mode Exceptions

#NM  EM or TS in CR0 is set.

Real Address Mode Exceptions

#NM  EM or TS in CR0 is set.

Virtual 8086 Mode Exceptions

#NM  EM or TS in CR0 is set.
FYL2X—Compute $y \times \log_2 x$

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D9 F1</td>
<td>FYL2X</td>
<td>Replace ST(1) with $(ST(1) \times \log_2 ST(0))$ and pop the register stack</td>
</tr>
</tbody>
</table>

**Description**

Calculates $(ST(1) \times \log_2 (ST(0)))$, stores the result in register ST(1), and pops the FPU register stack. The source operand in ST(0) must be a non-zero positive number.

The following table shows the results obtained when taking the log of various classes of numbers, assuming that neither overflow nor underflow occurs.

**Table 1-11. FYL2X Zeros and NaNs**

<table>
<thead>
<tr>
<th>ST(0)</th>
<th>$-\infty$</th>
<th>$-F$</th>
<th>$+0$</th>
<th>$+0$</th>
<th>$+F$</th>
<th>$+\infty$</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>ST(1)</td>
<td>$-\infty$</td>
<td>*</td>
<td>*</td>
<td>$+\infty$</td>
<td>$+\infty$</td>
<td>$-\infty$</td>
<td>NaN</td>
</tr>
<tr>
<td>$-F$</td>
<td>*</td>
<td>*</td>
<td>**</td>
<td>**</td>
<td>$\pm F$</td>
<td>$-\infty$</td>
<td>NaN</td>
</tr>
<tr>
<td>$-0$</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>$+0$</td>
<td>*</td>
<td>NaN</td>
<td></td>
</tr>
<tr>
<td>$+0$</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>*</td>
<td>$+0$</td>
<td>*</td>
<td>NaN</td>
</tr>
<tr>
<td>$+F$</td>
<td>*</td>
<td>*</td>
<td>**</td>
<td>**</td>
<td>$\pm F$</td>
<td>$+\infty$</td>
<td>NaN</td>
</tr>
<tr>
<td>$+\infty$</td>
<td>*</td>
<td>*</td>
<td>$-\infty$</td>
<td>$-\infty$</td>
<td>$-\infty$</td>
<td>$+\infty$</td>
<td>NaN</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

**Notes:**

- $F$ means finite-real number.
- * indicates floating-point invalid-operation (#IA) exception.
- ** indicates floating-point zero-divide (#Z) exception.

If the divide-by-zero exception is masked and register ST(0) contains $\pm 0$, the instruction returns $\infty$ with a sign that is the opposite of the sign of the source operand in register ST(1).

The FYL2X instruction is designed with a built-in multiplication to optimize the calculation of logarithms with an arbitrary positive base ($b$):

$$\log_b x = (\log_2 b)^{-1} \times \log_2 x$$

**Operation**

**IF**

ST(1) ← ST(1) * log₂ST(0);
PPopRegisterStack;

**FPU Flags Affected**

- C1 Set to 0 if stack underflow occurred.
  Indicates rounding direction if the inexact-result exception (#P) is generated:
  0 = not roundup; 1 = roundup.
- C0, C2, C3 Undefined.
FYL2X—Compute $y \times \log_2 x$ (continued)

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

**Floating-point Exceptions**

- **#IS**  Stack underflow occurred.
- **#IA**  Either operand is an SNaN or unsupported format.
  - Source operand in register ST(0) is a negative finite value (not $-0$).
- **#Z**  Source operand in register ST(0) is $\pm 0$.
- **#D**  Source operand is a denormal value.
- **#U**  Result is too small for destination format.
- **#O**  Result is too large for destination format.
- **#P**  Value cannot be represented exactly in destination format.

**Protected Mode Exceptions**

- **#NM**  EM or TS in CR0 is set.

**Real Address Mode Exceptions**

- **#NM**  EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

- **#NM**  EM or TS in CR0 is set.
FYL2XP1—Compute $y \cdot \log_2(x+1)$

**Opcode** | **Instruction** | **Description**
---|---|---
D9 F9 | FYL2XP1 | Replace ST(1) with $ST(1) \cdot \log_2(ST(0) + 1.0)$ and pop the register stack

**Description**

Calculates the log epsilon ($ST(1) \cdot \log_2(ST(0) + 1.0)$), stores the result in register ST(1), and pops the FPU register stack. The source operand in ST(0) must be in the range:

$-(1 - \sqrt{2}/2)$ to $(1 - \sqrt{2}/2)$

The source operand in ST(1) can range from $-\infty$ to $+\infty$. If either of the source operands is outside its acceptable range, the result is undefined and no exception is generated.

The following table shows the results obtained when taking the log epsilon of various classes of numbers, assuming that underflow does not occur:

**Table 1-12. FYL2XP1 Zeros and NaNs**

<table>
<thead>
<tr>
<th>ST(1)</th>
<th>−∞</th>
<th>−(1 − (\sqrt{2}/2))</th>
<th>0</th>
<th>+0 to +(1 − (\sqrt{2}/2))</th>
<th>+∞</th>
<th>NaN</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>*</td>
<td>*</td>
<td>−∞</td>
<td>−∞</td>
<td>NaN</td>
<td></td>
</tr>
<tr>
<td>−F</td>
<td>*</td>
<td>+F</td>
<td>+0</td>
<td>−0</td>
<td>−F</td>
<td>−∞</td>
</tr>
<tr>
<td>−0</td>
<td>*</td>
<td>+0</td>
<td>+0</td>
<td>−0</td>
<td>−0</td>
<td>*</td>
</tr>
<tr>
<td>+0</td>
<td>*</td>
<td>−0</td>
<td>−0</td>
<td>+0</td>
<td>+0</td>
<td>*</td>
</tr>
<tr>
<td>+F</td>
<td>*</td>
<td>−F</td>
<td>−0</td>
<td>+0</td>
<td>+F</td>
<td>+∞</td>
</tr>
<tr>
<td>+∞</td>
<td>*</td>
<td>−∞</td>
<td>−∞</td>
<td>*</td>
<td>*</td>
<td>+∞</td>
</tr>
<tr>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
<td>NaN</td>
</tr>
</tbody>
</table>

**Notes:**

- F means finite-real number.
- * indicates floating-point invalid-operation (#IA) exception.

This instruction provides optimal accuracy for values of epsilon [the value in register ST(0)] that are close to 0. When the epsilon value ($\varepsilon$) is small, more significant digits can be retained by using the FYL2XP1 instruction than by using ($\varepsilon+1$) as an argument to the FYL2X instruction. The ($\varepsilon+1$) expression is commonly found in compound interest and annuity calculations. The result can be simply converted into a value in another logarithm base by including a scale factor in the ST(1) source operand. The following equation is used to calculate the scale factor for a particular logarithm base, where $n$ is the logarithm base desired for the result of the FYL2XP1 instruction:

$$
\text{scale factor} = \log_n 2
$$

**Operation**

I

$ST(1) \leftarrow ST(1) \cdot \log_2(ST(0) + 1.0)$;

PopRegisterStack;
FYL2XP1—Compute \( y \times \log_2(x + 1) \) (continued)

**FPU Flags Affected**

- **C1** Set to 0 if stack underflow occurred.
  - Indicates rounding direction if the inexact-result exception (#P) is generated:
    \( 0 = \text{not roundup}; \ 1 = \text{roundup} \).
- **C0, C2, C3** Undefined.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

**Floating-point Exceptions**

- **#IS** Stack underflow occurred.
- **#IA** Either operand is an SNaN value or unsupported format.
- **#D** Source operand is a denormal value.
- **#U** Result is too small for destination format.
- **#O** Result is too large for destination format.
- **#P** Value cannot be represented exactly in destination format.

**Protected Mode Exceptions**

- **#NM** EM or TS in CR0 is set.

**Real Address Mode Exceptions**

- **#NM** EM or TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

- **#NM** EM or TS in CR0 is set.
HLT—Halt

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F4</td>
<td>HLT</td>
<td>Halt</td>
</tr>
</tbody>
</table>

**Description**

Stops instruction execution and places the processor in a HALT state. An enabled interrupt, NMI, or a reset will resume execution. If an interrupt (including NMI) is used to resume execution after a HLT instruction, the saved instruction pointer (CS:EIP) points to the instruction following the HLT instruction.

The HLT instruction is a privileged instruction. When the processor is running in protected or virtual 8086 mode, the privilege level of a program or procedure must to 0 to execute the HLT instruction.

**Operation**

IF Itanium System Environment THEN IA-32_Intercept(INST,HALT);
Enter Halt state;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

IA-32_Intercept Mandatory Instruction Intercept.

**Protected Mode Exceptions**

#GP(0) If the current privilege level is not 0.

**Real Address Mode Exceptions**

None.

**Virtual 8086 Mode Exceptions**

#GP(0) If the current privilege level is not 0.
**IDIV—Signed Divide**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F6 /7</td>
<td>IDIV r/m8</td>
<td>Signed divide AX (where AH must contain sign-extension of AL) by r/m byte. (Results: AL=Quotient, AH=Remainder)</td>
</tr>
<tr>
<td>F7 /7</td>
<td>IDIV r/m16</td>
<td>Signed divide DX:AX (where DX must contain sign-extension of AX) by r/m word. (Results: AX=Quotient, DX=Remainder)</td>
</tr>
<tr>
<td>F7 /7</td>
<td>IDIV r/m32</td>
<td>Signed divide EDX:EAX (where EDX must contain sign-extension of EAX) by r/m doubleword. (Results: EAX=Quotient, EDX=Remainder)</td>
</tr>
</tbody>
</table>

**Description**

Divides (signed) the value in the AL, AX, or EAX register by the source operand and stores the result in the AX, DX:AX, or EDX:EAX registers. The source operand can be a general-purpose register or a memory location. The action of this instruction depends on the operand size, as shown in the following table:

**Table 1-13. IDIV Operands**

<table>
<thead>
<tr>
<th>Operand Size</th>
<th>Dividend</th>
<th>Divisor</th>
<th>Quotient</th>
<th>Remainder</th>
<th>Quotient Range</th>
</tr>
</thead>
<tbody>
<tr>
<td>Word/byte</td>
<td>AX</td>
<td>r/m8</td>
<td>AL</td>
<td>AH</td>
<td>−128 to +127</td>
</tr>
<tr>
<td>Doubleword/word</td>
<td>DX:AX</td>
<td>r/m16</td>
<td>AX</td>
<td>DX</td>
<td>−32,768 to +32,767</td>
</tr>
<tr>
<td>Quadword/doubleword</td>
<td>EDX:EAX</td>
<td>r/m32</td>
<td>EAX</td>
<td>EDX</td>
<td>$-2^{31} \text{ to } 2^{32} - 1$</td>
</tr>
</tbody>
</table>

Non-integral results are truncated (chopped) towards 0. The sign of the remainder is always the same as the sign of the dividend. The absolute value of the remainder is always less than the absolute value of the divisor. Overflow is indicated with the #DE (divide error) exception rather than with the OF flag.

**Operation**

IF SRC = 0
THEN #DE; (* divide error *)
FI;
IF OpernadSize = 8 (* word/byte operation *)
THEN
   temp ← AX / SRC; (* signed division *)
   IF (temp > 7FH) OR (temp < 80H)
      (* if a positive result is greater than 7FH or a negative result is less than 80H *)
      THEN #DE; (* divide error *)
      ELSE
         AL ← temp;
         AH ← AX SignedModulus SRC;
      FI;
   ELSE
      IF OpernadSize = 16 (* doubleword/word operation *)
      THEN
IDIV—Signed Divide (continued)

```assembly
temp ← DX:AX / SRC; (* signed division *)
IF (temp > 7FFFH) OR (temp < 8000H)
(* if a positive result is greater than 7FFFH *)
(* or a negative result is less than 8000H *)
THEN #DE; (* divide error *)
ELSE
    AX ← temp;
    DX ← DX:AX SignedModulus SRC;
FI;
ELSE (* quadword/doubleword operation *)
    temp ← EDX:EAX / SRC; (* signed division *)
    IF (temp > 7FFFFFFFFH) OR (temp < 80000000H)
    (* if a positive result is greater than 7FFFFFFFFH *)
    (* or a negative result is less than 80000000H *)
    THEN #DE; (* divide error *)
    ELSE
        EAX ← temp;
        EDX ← EDXE:AX SignedModulus SRC;
    FI;
FI;
```

Flags Affected

The CF, OF, SF, ZF, AF, and PF flags are undefined.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

Protected Mode Exceptions

#DE If the source operand (divisor) is 0.

The signed result (quotient) is too large for the destination.

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
IDIV—Signed Divide (continued)

Real Address Mode Exceptions

#DE If the source operand (divisor) is 0.
The signed result (quotient) is too large for the destination.

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#DE If the source operand (divisor) is 0.
The signed result (quotient) is too large for the destination.

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
IMUL—Signed Multiply

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F6 /5</td>
<td>IMUL r/m8</td>
<td>AX ← AL * r/m byte</td>
</tr>
<tr>
<td>F7 /5</td>
<td>IMUL r/m16</td>
<td>DX:AX ← AX * r/m word</td>
</tr>
<tr>
<td>F7 /5</td>
<td>IMUL r/m32</td>
<td>EDX:EAX ← EAX * r/m doubleword</td>
</tr>
<tr>
<td>0F AF /r</td>
<td>IMUL r16,r/m16</td>
<td>word register ← word register * r/m word</td>
</tr>
<tr>
<td>0F AF /r</td>
<td>IMUL r32,r/m32</td>
<td>doubleword register ← doubleword register * r/m doubleword</td>
</tr>
<tr>
<td>6B /r ib</td>
<td>IMUL r16,r16,imm8</td>
<td>word register ← r16:imm8 * sign-extended immediate byte</td>
</tr>
<tr>
<td>6B /r ib</td>
<td>IMUL r32,r32,imm8</td>
<td>doubleword register ← r32:imm8 * sign-extended immediate byte</td>
</tr>
<tr>
<td>6B /r ib</td>
<td>IMUL r16,imm8</td>
<td>word register ← word register * sign-extended immediate byte</td>
</tr>
<tr>
<td>6B /r ib</td>
<td>IMUL r32,imm8</td>
<td>doubleword register ← doubleword register * sign-extended immediate byte</td>
</tr>
<tr>
<td>69 /r iw</td>
<td>IMUL r16,imm16</td>
<td>word register ← r16:imm16 * immediate word</td>
</tr>
<tr>
<td>69 /r id</td>
<td>IMUL r32,imm32</td>
<td>doubleword register ← r32:imm32 * immediate doubleword</td>
</tr>
<tr>
<td>69 /r iw</td>
<td>IMUL r16,imm16</td>
<td>word register ← r16:imm16 * immediate word</td>
</tr>
<tr>
<td>69 /r id</td>
<td>IMUL r32,imm32</td>
<td>doubleword register ← r32:imm32 * immediate doubleword</td>
</tr>
</tbody>
</table>

Description

Performs a signed multiplication of two operands. This instruction has three forms, depending on the number of operands.

- **One-operand form.** This form is identical to that used by the MUL instruction. Here, the source operand (in a general-purpose register or memory location) is multiplied by the value in the AL, AX, or EAX register (depending on the operand size) and the product is stored in the AX, DX:AX, or EDX:EAX registers, respectively.

- **Two-operand form.** With this form the destination operand (the first operand) is multiplied by the source operand (second operand). The destination operand is a general-purpose register and the source operand is an immediate value, a general-purpose register, or a memory location. The product is then stored in the destination operand location.

- **Three-operand form.** This form requires a destination operand (the first operand) and two source operands (the second and the third operands). Here, the first source operand (which can be a general-purpose register or a memory location) is multiplied by the second source operand (an immediate value). The product is then stored in the destination operand (a general-purpose register).

When an immediate value is used as an operand, it is sign-extended to the length of the destination operand format.

The CF and OF flags are set when significant bits are carried into the upper half of the result. The CF and OF flags are cleared when the result fits exactly in the lower half of the result.
IMUL—Signed Multiply (continued)

The three forms of the IMUL instruction are similar in that the length of the product is calculated to twice the length of the operands. With the one-operand form, the product is stored exactly in the destination. With the two- and three-operand forms, however, result is truncated to the length of the destination before it is stored in the destination register. Because of this truncation, the CF or OF flag should be tested to ensure that no significant bits are lost.

The two- and three-operand forms may also be used with unsigned operands because the lower half of the product is the same regardless if the operands are signed or unsigned. The CF and OF flags, however, cannot be used to determine if the upper half of the result is non-zero.

Operation

IF (NumberOfOperands = 1)
  THEN IF (OperandSize = 8)
    THEN
      AX ← AL * SRC (* signed multiplication *)
      IF ((AH = 00H) OR (AH = FFH))
        THEN CF = 0; OF = 0;
        ELSE CF = 1; OF = 1;
      FI;
    ELSE IF (OperandSize = 16)
      THEN
        DX:AX ← AX * SRC (* signed multiplication *)
        IF ((DX = 0000H) OR (DX = FFFFH))
          THEN CF = 0; OF = 0;
          ELSE CF = 1; OF = 1;
        FI;
      ELSE (*OperandSize = 32 *)
        EDX:EAX ← EAX * SRC (* signed multiplication *)
        IF ((EDX = 00000000H) OR (EDX = FFFFFFFFH))
          THEN CF = 0; OF = 0;
          ELSE CF = 1; OF = 1;
        FI;
      FI;
    ELSE IF (NumberOfOperands = 2)
      THEN
        temp ← DEST * SRC (* signed multiplication; temp is double DEST size *)
        DEST ← DEST * SRC (* signed multiplication *)
        IF temp ≠ DEST
          THEN CF = 1; OF = 1;
          ELSE CF = 0; OF = 0;
        FI;
      ELSE (*NumberOfOperands = 3 *)
        DEST ← SRC1 * SRC2 (* signed multiplication *)
        temp ← SRC1 * SRC2 (* signed multiplication; temp is double SRC1 size *)
        IF temp ≠ DEST
          THEN CF = 1; OF = 1;
          ELSE CF = 0; OF = 0;
        FI;
      FI;
    FI;
  FI;
FI;
IMUL—Signed Multiply (continued)

Flags Affected

For the one operand form of the instruction, the CF and OF flags are set when significant bits are carried into the upper half of the result and cleared when the result fits exactly in the lower half of the result. For the two- and three-operand forms of the instruction, the CF and OF flags are set when the result must be truncated to fit in the destination operand size and cleared when the result fits exactly in the destination operand size. The SF, ZF, AF, and PF flags are undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
        If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS  If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
IN—Input from Port

### Description

Copies the value from the I/O port specified with the second operand (source operand) to the destination operand (first operand). The source operand can be a byte-immediate or the DX register; the destination operand can be register AL, AX, or EAX, depending on the size of the port being accessed (8, 16, or 32 bits, respectively). Using the DX register as a source operand allows I/O port addresses from 0 to 65,535 to be accessed; using a byte immediate allows I/O port addresses 0 to 255 to be accessed.

When accessing an 8-bit I/O port, the opcode determines the port size; when accessing a 16- and 32-bit I/O port, the operand-size attribute determines the port size.

At the machine code level, I/O instructions are shorter when accessing 8-bit I/O ports. Here, the upper eight bits of the port address will be 0.

This instruction is only useful for accessing I/O ports located in the processor’s I/O address space.

I/O transactions are performed after all prior data memory operations. No subsequent data memory operations can pass an I/O transaction.

In the Itanium System Environment, I/O port references are mapped into the 64-bit virtual address pointed to by the IOBase register, with four ports per 4K-byte virtual page. Operating systems can utilize the TLB in the Itanium architecture to grant or deny permission to any four I/O ports. The I/O port space can be mapped into any arbitrary 64-bit physical memory location by operating system code. If CFLG.io is 1 and CPL>IOPL, the TSS is consulted for I/O permission. If CFLG.io is 0 or CPL<=$IOPL, permission is granted regardless of the state of the TSS I/O permission bitmap (the bitmap is not referenced).

If the referenced I/O port is mapped to an unimplemented virtual address (via the I/O Base register) or if data translations are disabled (PSR.dt is 0) a GPFault is generated on the referencing IN instruction.

### Operation

\[
\text{IF } ((\text{PE} = 1) \text{ AND } ((\text{VM} = 1) \text{ OR } (\text{CPL} > \text{IOPL})))
\]

\[
\text{THEN } (* \text{ Protected mode or virtual-8086 mode with CPL > IOPL } *)
\]

\[
\text{IF } (\text{CFLG.io AND Any I/O Permission Bit for I/O port being accessed} = 1)
\]

\[
\text{THEN } \#\text{GP}(0);
\]

\[
\text{FI};
\]

\[
\text{ELSE } (* \text{ Real-address mode or protected mode with CPL} \leq \text{IOPL } *)
\]

\[
(* \text{ or virtual-8086 mode with all I/O permission bits for I/O port cleared } *)
\]

---

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>E4 ib</td>
<td>IN AL,imm8</td>
<td>Input byte from imm8 I/O port address into AL</td>
</tr>
<tr>
<td>E5 ib</td>
<td>IN AX,imm8</td>
<td>Input byte from imm8 I/O port address into AX</td>
</tr>
<tr>
<td>E5 ib</td>
<td>IN EAX,imm8</td>
<td>Input byte from imm8 I/O port address into EAX</td>
</tr>
<tr>
<td>EC</td>
<td>IN AL,DX</td>
<td>Input byte from I/O port in DX into AL</td>
</tr>
<tr>
<td>ED</td>
<td>IN AX,DX</td>
<td>Input word from I/O port in DX into AX</td>
</tr>
<tr>
<td>ED</td>
<td>IN EAX,DX</td>
<td>Input doubleword from I/O port in DX into EAX</td>
</tr>
</tbody>
</table>
IN—Input from Port (continued)

FI;
IF (Itanium_System_Environment THEN
   SRC_VA = IOBase | (Port{15:2}<<12) | Port{11:0};
   SRC_PA = translate(SRC_VA);
   DEST ← [SRC_PA]; (* Reads from I/O port *)
FI;

memory_fence();
DEST <-SRC;
memory-fence();

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults   NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
IA-32_Exception     Debug traps for data breakpoints and single step
IA-32_Exception     Alignment faults
#GP(0)              Referenced Port is to an unimplemented virtual address or PSR.dt is zero.

Protected Mode Exceptions

#GP(0)               If the CPL is greater than (has less privilege) the I/O privilege level (IOPL) and any of the corresponding I/O permission bits in TSS for the I/O port being accessed is 1 when CFLG.io is 1.

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

#GP(0)               If any of the I/O permission bits in the TSS for the I/O port being accessed is 1.
INC—Increment by 1

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FE /0</td>
<td>INC r/m8</td>
<td>Increment r/m byte by 1</td>
</tr>
<tr>
<td>FF /0</td>
<td>INC r/m16</td>
<td>Increment r/m word by 1</td>
</tr>
<tr>
<td>FF /0</td>
<td>INC r/m32</td>
<td>Increment r/m doubleword by 1</td>
</tr>
<tr>
<td>40+ rw</td>
<td>INC r16</td>
<td>Increment word register by 1</td>
</tr>
<tr>
<td>40+ rd</td>
<td>INC r32</td>
<td>Increment doubleword register by 1</td>
</tr>
</tbody>
</table>

Description

Adds 1 to the operand, while preserving the state of the CF flag. The source operand can be a register or a memory location. This instruction allows a loop counter to be updated without disturbing the CF flag. (Use a ADD instruction with an immediate operand of 1 to perform a increment operation that does updates the CF flag.)

Operation

\[
\text{DEST} \leftarrow \text{DEST} - 1;
\]

Flags Affected

The CF flag is not affected. The OF, SF, ZF, AF, and PF flags are set according to the result.

Additional Itanium System Environment Exceptions

Itanium Reg Faults: NaT Register Consumption Abort.
Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
INC—Increment by 1 (continued)

**Virtual 8086 Mode Exceptions**

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made.
INS/INSB/INSW/INSD—Input from Port to String

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>6C</td>
<td>INS ES:(E)DI, DX</td>
<td>Input byte from port DX into ES:(E)DI</td>
</tr>
<tr>
<td>6D</td>
<td>INS ES:DI, DX</td>
<td>Input word from port DX into ES:DI</td>
</tr>
<tr>
<td>6D</td>
<td>INS ES:EDI, DX</td>
<td>Input doubleword from port DX into ES:EDI</td>
</tr>
<tr>
<td>6C</td>
<td>INSB</td>
<td>Input byte from port DX into ES:(E)DI</td>
</tr>
<tr>
<td>6D</td>
<td>INSW</td>
<td>Input word from port DX into ES:DI</td>
</tr>
<tr>
<td>6D</td>
<td>INSD</td>
<td>Input doubleword from port DX into ES:EDI</td>
</tr>
</tbody>
</table>

Description

Copies the data from the I/O port specified with the second operand (source operand) to the destination operand (first operand). The source operand must be the DX register, allowing I/O port addresses from 0 to 65,535 to be accessed. When accessing an 8-bit I/O port, the opcode determines the port size; when accessing a 16- and 32-bit I/O port, the operand-size attribute determines the port size.

The destination operand is a memory location at the address ES:EDI. (When the operand-size attribute is 16, the DI register is used as the destination-index register.) The ES segment cannot be overridden with a segment override prefix.

The INSB, INSW, and INSD mnemonics are synonyms of the byte, word, and doubleword versions of the INS instructions. (For the INS instruction, “ES:EDI” must be explicitly specified in the instruction.)

After the byte, word, or doubleword is transferred from the I/O port to the memory location, the EDI register is incremented or decremented automatically according to the setting of the DF flag in the EFLAGS register. (If the DF flag is 0, the EDI register is incremented; if the DF flag is 1, the EDI register is decremented.) The EDI register is incremented or decremented by 1 for byte operations, by 2 for word operations, or by 4 for doubleword operations.

The INS, INSB, INSW, and INSD instructions can be preceded by the REP prefix for block input of ECX bytes, words, or doublewords.

This instruction is only useful for accessing I/O ports located in the processor’s I/O address space.

I/O transactions are performed after all prior data memory operations. No subsequent data memory operations can pass an I/O transaction.

In the Itanium System Environment, I/O port references are mapped into the 64-bit virtual address pointed to by the IOBase register, with four ports per 4K-byte virtual page. Operating systems can utilize the TLBs in the Itanium architecture to grant or deny permission to any four I/O ports. The I/O port space can be mapped into any arbitrary 64-bit physical memory location by operating system code. If CFLG.io is 1 and CPL>IOPL, the TSS is consulted for I/O permission. If CFLG.io is 0 or CPL<=IOPL, permission is granted regardless of the state of the TSS I/O permission bitmap (the bitmap is not referenced).

If the referenced I/O port is mapped to an unimplemented virtual address (via the IOBase register) or if data translations are disabled (PSR.dt is 0) a GPFault is generated on the referencing INS instruction.
INS/INSB/INSW/INSD—Input from Port to String (continued)

Operation

IF ((PE = 1) AND ((VM = 1) OR (CPL > IOPL)))
   THEN (* Protected mode or virtual-8086 mode with CPL > IOPL *)
      IF (CFLG.io AND Any I/O Permission Bit for I/O port being accessed = 1)
         THEN #GP(0);
      ELSE (* I/O operation is allowed *)
      FI;
   IF (Itanium_System_Environment) THEN
      SRC_VA = IOBase | (Port(15:2)<<12) | Port(11:0);
      SRC_PA = translate(SRC_VA);
      DEST ← [SRC_PA]; (* Reads from I/O port *)
   FI;
memory_fence();
DEST ← SRC;
memory_fence();
   IF (byte transfer)
      THEN IF DF = 0
         THEN (E)DI ← 1;
         ELSE (E)DI ← -1;
      FI;
   ELSE IF (word transfer)
      THEN IF DF = 0
         THEN DI ← 2;
         ELSE DI ← -2;
      FI;
   ELSE (* doubleword transfer *)
      THEN IF DF = 0
         THEN EDI ← 4;
         ELSE EDI ← -4;
      FI;
   FI;
FI;
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults   NaT Register Consumption Abort.
Itanium Mem Faults   VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
IA-32_Exception     Debug traps for data breakpoints and single step
IA-32_Exception     Alignment faults
#GP(0)              Referenced Port is to an unimplemented virtual address or PSR.dt is zero.
INS/INSB/INSW/INSD—Input from Port to String (continued)

Protected Mode Exceptions

#GP(0)  If the CPL is greater than (has less privilege) the I/O privilege level (IOPL) and any of the corresponding I/O permission bits in TSS for the I/O port being accessed is 1 and when CFLG.io is 1.

If the destination is located in a nonwritable segment.

If an illegal memory operand effective address in the ES segments is given.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS  If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0)  If any of the I/O permission bits in the TSS for the I/O port being accessed is 1.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
**INTₙ/INTO/INT3—Call to Interrupt Procedure**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>CC</td>
<td>INT3</td>
<td>Interrupt 3—trap to debugger</td>
</tr>
<tr>
<td>CD ib</td>
<td>INT imm8</td>
<td>Interrupt vector numbered by immediate byte</td>
</tr>
<tr>
<td>CE</td>
<td>INTO</td>
<td>Interrupt 4—if overflow flag is 1</td>
</tr>
</tbody>
</table>

**Description**

The INTₙ instruction generates a call to the interrupt or exception handler specified with the destination operand. The destination operand specifies an interrupt vector from 0 to 255, encoded as an 8-bit unsigned intermediate value. The first 32 interrupt vectors are reserved by Intel for system use. Some of these interrupts are used for internally generated exceptions.

The INTₙ instruction is the general mnemonic for executing a software-generated call to an interrupt handler. The INTO instruction is a special mnemonic for calling overflow exception (#OF), interrupt vector 4. The overflow interrupt checks the OF flag in the EFLAGS register and calls the overflow interrupt handler if the OF flag is set to 1.

The INT3 instruction is a special mnemonic for calling the debug exception handler. The action of the INT3 instruction (opcode CC) is slightly different from the operation of the INT 3 instruction (opcode CC03), as follows:

- Interrupt redirection does not happen when in VME mode; the interrupt is handled by a protected-mode handler.
- The virtual-8086 mode IOPL checks do not occur. The interrupt is taken without faulting at any IOPL level.

The action of the INTₙ instruction (including the INTO and INT3 instructions) is similar to that of a far call made with the CALL instruction. The primary difference is that with the INTₙ instruction, the EFLAGS register is pushed onto the stack before the return address. (The return address is a far address consisting of the current values of the CS and EIP registers.) Returns from interrupt procedures are handled with the IRET instruction, which pops the EFLAGS information and return address from the stack.

The interrupt vector specifies an interrupt descriptor in the interrupt descriptor table (IDT); that is, it provides index into the IDT. The selected interrupt descriptor in turn contains a pointer to an interrupt or exception handler procedure. In protected mode, the IDT contains an array of 8-byte descriptors, each of which points to an interrupt gate, trap gate, or task gate. In real-address mode, the IDT is an array of 4-byte far pointers (2-byte code segment selector and a 2-byte instruction pointer), each of which point directly to procedure in the selected segment.

The following decision table indicates which action in the lower portion of the table is taken given the conditions in the upper portion of the table. Each Y in the lower section of the decision table represents a procedure defined in the “Operation” section for this instruction (except #GP).
INT\(n\)/INTO/INT3—Call to Interrupt Procedure (continued)

Table 1-14. INT Cases

<table>
<thead>
<tr>
<th>PE</th>
<th>0</th>
<th>1</th>
<th>1</th>
<th>1</th>
<th>1</th>
<th>1</th>
<th>1</th>
<th>1</th>
</tr>
</thead>
<tbody>
<tr>
<td>VM</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>0</td>
<td>1</td>
<td>1</td>
</tr>
<tr>
<td>IOPL</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>&lt;3</td>
<td>=3</td>
<td>–</td>
</tr>
<tr>
<td>DPL/CPL RELATIONSHIP</td>
<td>–</td>
<td>DPL&lt; CPL</td>
<td>–</td>
<td>DPL&gt; CPL</td>
<td>DPL= CPL or C</td>
<td>DPL&lt; CPL &amp; NC</td>
<td>–</td>
<td>–</td>
</tr>
<tr>
<td>INTERRUPT TYPE</td>
<td>–</td>
<td>S/W</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>–</td>
<td>–</td>
</tr>
<tr>
<td>GATE TYPE</td>
<td>–</td>
<td>–</td>
<td>Task</td>
<td>Trap or Interrupt</td>
<td>Trap or Interrupt</td>
<td>Trap or Interrupt</td>
<td>Trap or Interrupt</td>
<td>Trap or Interrupt</td>
</tr>
<tr>
<td>REAL-ADDRESS-MODE</td>
<td>Y</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>PROTECTED-MODE</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
</tr>
<tr>
<td>TRAP-OR-INTERRUPT-GATE</td>
<td></td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td>Y</td>
<td></td>
</tr>
<tr>
<td>INTER-PRIVILEGE-LEVEL-INTERRUPT</td>
<td>Y</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>INTRA-PRIVILEGE-LEVEL-INTERRUPT</td>
<td>Y</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>INTERRUPT-FROM-VIRTUAL-8086-MODE</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>TASK-GATE</td>
<td>Y</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>#GP</td>
<td>Y</td>
<td>Y</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

Notes:
– Don’t Care
Y Yes, Action Taken
Blank Action Not Taken

When the processor is executing in virtual-8086 mode, the IOPL determines the action of the INT\(n\) instruction. If the IOPL is less than 3, the processor generates a general protection exception (#GP); if the IOPL is 3, the processor executes a protected mode interrupt to privilege level 0. The interrupt gate’s DPL must be set to three and the target CPL of the interrupt handler procedure must be 0 to execute the protected mode interrupt to privilege level 0.

The interrupt descriptor table register (IDTR) specifies the base linear address and limit of the IDT. The initial base address value of the IDTR after the processor is powered up or reset is 0.

**Operation**

The following operational description applies not only to the INT\(n\) and INTO instructions, but also to external interrupts and exceptions.

**IF Itanium System Environment THEN**

**IF INT3 Form THEN IA-32_Exception(3);**

**IF INTO Form THEN IA-32_Exception(4);**

**IF INT Form THEN IA-32_Interrupt(N);**

**FI;**
INTᵣ/INTO/INT3—Call to Interrupt Procedure (continued)

/*IN the Itanium System Environment all of the following operations are intercepted*/
IF PE=0
   THEN
       GOTO REAL-ADDRESS-MODE;
   ELSE (* PE=1 *)
       GOTO PROTECTED-MODE;
FI;

REAL-ADDRESS-MODE:
   IF ((DEST * 4) + 3) is not within IDT limit THEN #GP; FI;
   IF stack not large enough for a 6-byte return information THEN #SS; FI;
   Push (EFLAGS[15:0]);
   IF ← 0; (* Clear interrupt flag *)
   TF ← 0; (* Clear trap flag *)
   AC ← 0; (* Clear AC flag *)
   Push(CS);
   Push(IP);
   (* No error codes are pushed *)
   CS ← IDT(Descriptor (vector * 4), selector));
   EIP ← IDT(Descriptor (vector * 4), offset)); (* 16 bit offset AND 0000FFFFH *)
END;

PROTECTED-MODE:
   IF ((DEST * 8) + 7) is not within IDT limits
      OR selected IDT descriptor is not an interrupt-, trap-, or task-gate type
      THEN #GP((DEST * 8) + 2 + EXT);
      (* EXT is bit 0 in error code *)
   FI;
   IF software interrupt (* generated by INTᵣ, INT3, or INTO *)
      THEN
         IF gate descriptor DPL < CPL
            THEN #GP((vector number * 8) + 2 );
            (* PE=1, DPL< CPL, software interrupt *)
            FI;
         FI;
         IF gate not present THEN #NP((vector number * 8) + 2 + EXT); FI;
   IF task gate (* specified in the selected interrupt table descriptor *)
      THEN GOTO TASK-GATE;
      ELSE GOTO TRAP-OR-INTERRUPT-GATE; (* PE=1, trap/interrupt gate *)
      FI;
END;

TASK-GATE: (* PE=1, task gate *)
   Read segment selector in task gate (IDT descriptor);
   IF local/global bit is set to local
      OR index not within GDT limits
      THEN #GP(TSS selector);
   FI;
   Access TSS descriptor in GDT;
   IF TSS descriptor specifies that the TSS is busy (low-order 5 bits set to 00001)
      THEN #GP(TSS selector);
   FI;
   IF TSS not present
INTn/INTO/INT3—Call to Interrupt Procedure (continued)

THEN #NP(TSS selector);
FI;
SWITCH-TASKS (with nesting) to TSS;
IF interrupt caused by fault with error code
THEN
   IF stack limit does not allow push of two bytes
   THEN #SS(0);
   FI;
   Push(error code);
FI;
IF EIP not within code segment limit
THEN #GP(0);
FI;
END;
TRAP-OR-_interrupt-GATE
Read segment selector for trap or interrupt gate (IDT descriptor);
IF segment selector for code segment is null
   THEN #GP(0H + EXT); (* null selector with EXT flag set *)
FI;
IF segment selector is not within its descriptor table limits
   THEN #GP(selector + EXT);
FI;
Read trap or interrupt handler descriptor;
IF descriptor does not indicate a code segment
   OR code segment descriptor DPL > CPL
   THEN #GP(selector + EXT);
FI;
IF trap or interrupt gate segment is not present,
   THEN #NP(selector + EXT);
FI;
IF code segment is non-conforming AND DPL < CPL
   THEN IF VM=0
      THEN
         GOTO INTER-PRIVILEGE-LEVEL- INTERRUPT:
         (* PE=1, interrupt or trap gate, nonconforming *)
         (* code segment, DPL< CPL, VM=0 *)
      ELSE (* VM=1 *)
         IF code segment DPL ≠ 0 THEN #GP(new code segment selector); FI;
         GOTO INTERRUPT-FROM-VIRTUAL-8086-MODE;
         (* PE=1, interrupt or trap gate, DPL< CPL, VM=1 *)
      FI;
   ELSE (* PE=1, interrupt or trap gate, DPL ≥ CPL *)
      IF VM=1 THEN #GP(new code segment selector); FI;
      IF code segment is conforming OR code segment DPL = CPL
         THEN
            GOTO INTRA-PRIVILEGE-LEVEL- INTERRUPT;
         ELSE
            #GP(CodeSegmentSelector + EXT); (* PE=1, interrupt or trap gate, nonconforming *)
            (* code segment, DPL>CPL *)
         FI;
   FI;
END;
INTn/INTO/INT3—Call to Interrupt Procedure (continued)

INTER-PRIVILEGE-LEVEL-INTERRUPT
(* PE=1, interrupt or trap gate, non-conforming code segment, DPL<CPL *)
(* Check segment selector and descriptor for stack of new privilege level in current TSS *)
IF current TSS is 32-bit TSS
  THEN
    TSSstackAddress ← new code segment (DPL * 8) + 4
    IF (TSSstackAddress + 7) > TSS limit
      THEN #TS(current TSS selector); FI;
    NewSS ← TSSstackAddress + 4;
    NewESP ← stack address;
  ELSE (* TSS is 16-bit *)
    TSSstackAddress ← new code segment (DPL * 4) + 2
    IF (TSSstackAddress + 4) > TSS limit
      THEN #TS(current TSS selector); FI;
    NewESP ← TSSstackAddress;
    NewSS ← TSSstackAddress + 2;
  FI;
  IF segment selector is null THEN #TS(EXT); FI;
  IF segment selector index is not within its descriptor table limits
    OR segment selector’s RPL ≠ DPL of code segment,
      THEN #TS(SS selector + EXT);
  FI;
Read segment descriptor for stack segment in GDT or LDT;
  IF stack segment DPL ≠ DPL of code segment,
    OR stack segment does not indicate writable data segment,
      THEN #TS(SS selector + EXT);
  FI;
  IF stack segment not present THEN #SS(SS selector+EXT); FI;
  IF 32-bit gate
    THEN
      IF new stack does not have room for 24 bytes (error code pushed)
        OR 20 bytes (no error code pushed)
          THEN #SS(segment selector + EXT);
        FI;
      ELSE (* 16-bit gate *)
        IF new stack does not have room for 12 bytes (error code pushed)
          OR 10 bytes (no error code pushed);
            THEN #SS(segment selector + EXT);
        FI;
      FI;
    FI;
  IF instruction pointer is not within code segment limits THEN #GP(0); FI;
SS:ESP ← TSS(SS:ESP) (* segment descriptor information also loaded *)
  IF 32-bit gate
    THEN
      CS:EIP ← Gate(CS:EIP); (* segment descriptor information also loaded *)
    ELSE (* 16-bit gate *)
      CS:IP ← Gate(CS:IP); (* segment descriptor information also loaded *)
    FI;
  IF 32-bit gate
    THEN
      Push(far pointer to old stack); (* old SS and ESP, 3 words padded to 4 *)
      Push(EFLAGS);
      Push(far pointer to return instruction); (* old CS and EIP, 3 words padded to 4*);
      Push(ErrorCode); (* if needed, 4 bytes *)
INTn/INTO/INT3—Call to Interrupt Procedure (continued)

ELSE (* 16-bit gate *)
    Push(far pointer to old stack); (* old SS and SP, 2 words *); Push(EFLAGS);
    Push(far pointer to return instruction); (* old CS and IP, 2 words *);
    Push(ErrorCode); (* if needed, 2 bytes *)
FI;
CPL ← CodeSegmentDescriptor(DPL);
CS(RPL) ← CPL;
IF interrupt gate
    THEN IF ← 0 (* interrupt flag to 0 (disabled *) ; FI;
    TF ← 0;
    VM ← 0;
    RF ← 0;
    NT ← 0;
END;
INTERRUPT-FROM-VIRTUAL-8086-MODE:
    (* Check segment selector and descriptor for privilege level 0 stack in current TSS *)
    IF current TSS is 32-bit TSS
        THEN
            TSSstackAddress ← new code segment (DPL * 8) + 4
            IF (TSSstackAddress + 7) > TSS limit
                THEN #TS(current TSS selector); FI;
            NewSS ← TSSstackAddress + 4;
            NewESP ← stack address;
        ELSE (* TSS is 16-bit *)
            TSSstackAddress ← new code segment (DPL * 4) + 2
            IF (TSSstackAddress + 4) > TSS limit
                THEN #TS(current TSS selector); FI;
            NewESP ← TSSstackAddress;
            NewSS ← TSSstackAddress + 2;
        FI;
    IF segment selector is null THEN #TS(EXT); FI;
    IF segment selector index is not within its descriptor table limits
        OR segment selector’s RPL ≠ DPL of code segment,
        THEN #TS(SS selector + EXT);
    FI;
Access segment descriptor for stack segment in GDT or LDT;
    IF stack segment DPL ≠ DPL of code segment,
        OR stack segment does not indicate writable data segment,
        THEN #TS(SS selector + EXT);
    FI;
    IF stack segment not present THEN #SS(SS selector+EXT); FI;
    IF 32-bit gate
        THEN
            IF new stack does not have room for 40 bytes (error code pushed)
                OR 36 bytes (no error code pushed);
                THEN #SS(segment selector + EXT);
            FI;
        ELSE (* 16-bit gate *)
            IF new stack does not have room for 20 bytes (error code pushed)
                OR 18 bytes (no error code pushed);
                THEN #SS(segment selector + EXT);
            FI;
    FI;
INTn/INTO/INT3—Call to Interrupt Procedure (continued)

IF instruction pointer is not within code segment limits THEN #GP(0); FI;

IF CR4.VME = 0 THEN
  IF IOPL=3 THEN
    IF Gate DPL = 3 THEN (*CPL=3, VM=1, IOPL=3, VME=0, gate DPL=3)
      IF Target CPL != 0 THEN #GP(0); ELSE Goto VM86_INTERRUPT_TO_PRIV0; FI;
    ELSE (*Gate DPL < 3*) #GP(0); FI;
  ELSE (*IOPL < 3*) #GP(0); FI;
ELSE (*VME = 1*)
  (*Check whether interrupt is directed for INT n instruction only,
  (*executes virtual 8086 interrupt, protected mode interrupt or faults*)
  Ptr <- [TSS + 66]; (*Fetch IO permission bitmap pointer*)
  IF BIT[Ptr-32,N] = 0 (*software redirection bitmap is 32 bytes below IO Permission*)
    THEN (*Interrupt redirected*)
      Goto VM86_INTERRUPT_TO_VM86;
    ELSE
      IF IOPL = 3 THEN
        IF Gate DPL = 3 THEN
          IF Target CPL != 0 THEN #GP(0); ELSE Goto VM86_INTERRUPT_TO_PRIV0; FI;
        ELSE (*IOPL < 3*) #GP(0); FI;
      ELSE (*IOPL < 3*) #GP(0); FI;
    FI;
  END;

VM86_INTERRUPT_TO_PRIV0:

tempEFLAGS ← EFLAGS;
VM ← 0;
TF ← 0;
RF ← 0;
IF service through interrupt gate THEN IF ← 0; FI;
TempSS ← SS;
TempESP ← ESP;
INTn/INTO/INT3—Call to Interrupt Procedure (continued)

SS:ESP ← TSS(SS0:ESP0); (* Change to level 0 stack segment *)
(* Following pushes are 16 bits for 16-bit gate and 32 bits for 32-bit gates *)
(* Segment selector pushes in 32-bit mode are padded to two words *)
Push(GS);
Push(FS);
Push(DS);
Push(ES);
Push(TempSS);
Push(TempESP);
Push(TempEFlags);
Push(CS);
Push(EIP);
GS ← 0; (*segment registers nullified, invalid in protected mode *)
FS ← 0;
DS ← 0;
ES ← 0;
CS ← Gate(CS);
IF OperandSize=32
    THEN
        EIP ← Gate(instruction pointer);
    ELSE (* OperandSize is 16 *)
        EIP ← Gate(instruction pointer) AND 0000FFFFH;
    FI;
(* Starts execution of new routine in Protected Mode *)
END;

VM86_INTERRUPT_TO_VM86:
IF IOPL = 3
    THEN
        push(FLAGS OR 3000H); (*Push FLAGS w/ IOPL bits as 11B or IOPL 3*)
        push(CS);
        push(IP);
        CS ← [N*4 + 2]; (*N is vector num, read from interrupt table*)
        IP ← [N*4];
        FLAGS ← FLAGS AND 7CD5H; (*Clear TF and IF in EFLAGS like 8086*)
    ELSE
        TempFlags ← FLAGS OR 3000H; (*Set IOPL to 11B or IOPL 3*)
        TempFlags.IF ← EFLAGS.VIF;
        push(TempFlags);
        push(CS);
        push(IP);
        CS ← [N*4 + 2]; (*N is vector num, read from interrupt table*)
        IP ← [N*4];
        FLAGS ← FLAGS AND 77ED5H; (*Clear VIF and TF and IF in EFLAGS like 8086*)
    FI;
END;

INTRA-PRIVILEGE-LEVEL-INTERRUPT:
(* PE=1, DPL = CPL or conforming segment *)
IF 32-bit gate
    THEN
        IF current stack does not have room for 16 bytes (error code pushed)
            OR 12 bytes (no error code pushed); THEN #SS(0);
        FI;
INTₙ/INTO/INT3—Call to Interrupt Procedure (continued)

ELSE (* 16-bit gate *)
    IF current stack does not have room for 8 bytes (error code pushed)
        OR 6 bytes (no error code pushed); THEN #SS(0);
    FI;
    IF instruction pointer not within code segment limit THEN #GP(0); FI;
    IF 32-bit gate
        THEN
            Push (EFLAGS);
            Push (far pointer to return instruction); (* 3 words padded to 4 *)
            CS:EIP ← Gate(CS:EIP); (* segment descriptor information also loaded *)
            Push (ErrorCode); (* if any *)
        ELSE (* 16-bit gate *)
            Push (FLAGS);
            Push (far pointer to return location); (* 2 words *)
            CS:IP ← Gate(CS:IP); (* segment descriptor information also loaded *)
            Push (ErrorCode); (* if any *)
        FI;
    CS(RPL) ← CPL;
    IF interrupt gate
        THEN
            IF ← 0; FI;
            TF ← 0;
            NT ← 0;
            VM ← 0;
            RF ← 0;
        FI;
    END;

Flags Affected

The EFLAGS register is pushed onto stack. The IF, TF, NT, AC, RF, and VM flags may be cleared, depending on the mode of operation of the processor when the INT instruction is executed (see “Operation” section.)

Additional Itanium System Environment Exceptions

IA-32_Exception If INT3 or INTO form, vector numbers are 3 and 4 respectively.
IA-32_Interrupt If INTₙ form, vector number is N.

Protected Mode Exceptions

#GP(0) If the instruction pointer in the IDT or in the interrupt-, trap-, or task gate is beyond the code segment limits.

#GP(selector) If the segment selector in the interrupt-, trap-, or task gate is null.
If a interrupt-, trap-, or task gate, code segment, or TSS segment selector index is outside its descriptor table limits.
If the interrupt vector is outside the IDT limits.
If an IDT descriptor is not an interrupt-, trap-, or task-descriptor.
If an interrupt is generated by the INTₙ instruction and the DPL of an interrupt-, trap-, or task-descriptor is less than the CPL.
INTn/INTO/INT3—Call to Interrupt Procedure (continued)

If the segment selector in an interrupt- or trap-gate does not point to a segment descriptor for a code segment.

If the segment selector for a TSS has its local/global bit set for local.

If a TSS segment descriptor specifies that the TSS is busy or not available.

#SS(0)  If pushing the return address, flags, or error code onto the stack exceeds the bounds of the stack segment and no stack switch occurs.

#SS(selector)  If the SS register is being loaded and the segment pointed to is marked not present.

If pushing the return address, flags, error code, or stack segment pointer exceeds the bounds of the stack segment.

#NP(selector)  If code segment, interrupt-, trap-, or task gate, or TSS is not present.

#TS(selector)  If the RPL of the stack segment selector in the TSS is not equal to the DPL of the code segment being accessed by the interrupt or trap gate.

If DPL of the stack segment descriptor pointed to by the stack segment selector in the TSS is not equal to the DPL of the code segment descriptor for the interrupt or trap gate.

If the stack segment selector in the TSS is null.

If the stack segment for the TSS is not a writable data segment.

If segment-selector index for stack segment is outside descriptor table limits.

#PF(fault-code)  If a page fault occurs.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the interrupt vector is outside the IDT limits.

#SS  If stack limit violation on push.

If pushing the return address, flags, or error code onto the stack exceeds the bounds of the stack segment when a stack switch occurs.

Virtual 8086 Mode Exceptions

#GP(0)  (For INTn instruction) If the IOPL is less than 3 and the DPL of the interrupt-, trap-, or task-gate descriptor is not equal to 3.

If the instruction pointer in the IDT or in the interrupt-, trap-, or task gate is beyond the code segment limits.

#GP(selector)  If the segment selector in the interrupt-, trap-, or task gate is null.

If a interrupt-, trap-, or task gate, code segment, or TSS segment selector index is outside its descriptor table limits.

If the interrupt vector is outside the IDT limits.

If an IDT descriptor is not an interrupt-, trap-, or task-descriptor.
INT\textsubscript{n}/INTO/INT3—Call to Interrupt Procedure (continued)

If an interrupt is generated by the INT\textsubscript{n} instruction and the DPL of an interrupt-, trap-, or task-descriptor is less than the CPL.

If the segment selector in an interrupt- or trap-gate does not point to a segment descriptor for a code segment.

If the segment selector for a TSS has its local/global bit set for local.

\#SS(selector) If the SS register is being loaded and the segment pointed to is marked not present.

If pushing the return address, flags, error code, stack segment pointer, or data segments exceeds the bounds of the stack segment.

\#NP(selector) If code segment, interrupt-, trap-, or task gate, or TSS is not present.

\#TS(selector) If the RPL of the stack segment selector in the TSS is not equal to the DPL of the code segment being accessed by the interrupt or trap gate.

If DPL of the stack segment descriptor for the TSS’s stack segment is not equal to the DPL of the code segment descriptor for the interrupt or trap gate.

If the stack segment selector in the TSS is null.

If the stack segment for the TSS is not a writable data segment.

If segment-selector index for stack segment is outside descriptor table limits.

\#PF(fault-code) If a page fault occurs.

\#BP If the INT3 instruction is executed.

\#OF If the INTO instruction is executed and the OF flag is set.
INVD—Invalidate Internal Caches

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 08</td>
<td>INVD</td>
<td>Flush internal caches; initiate flushing of external caches.</td>
</tr>
</tbody>
</table>

**Description**

Invalidates (flushes) the processor’s internal caches and issues a special-function bus cycle that directs external caches to also flush themselves. Data held in internal caches is not written back to main memory.

After executing this instruction, the processor does not wait for the external caches to complete their flushing operation before proceeding with instruction execution. It is the responsibility of hardware to respond to the cache flush signal.

The INVD instruction is a privileged instruction. When the processor is running in protected mode, the CPL of a program or procedure must be 0 to execute this instruction. This instruction is also implementation-dependent; its function may be implemented differently on future Intel Architecture processors.

Use this instruction with care. Data cached internally and not written back to main memory will be lost. Unless there is a specific requirement or benefit to flushing caches without writing back modified cache lines (for example, testing or fault recovery where cache coherency with main memory is not a concern), software should use the WBINVD instruction.

**Operation**

IF Itanium System Environment THEN IA-32_Interceptor(INST,INVD);
Flush(InternalCaches);
SignalFlush(ExternalCaches);
Continue (* Continue execution);

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

IA-32_Interceptor Mandatory Instruction Intercept

**Protected Mode Exceptions**

#GP(0) If the current privilege level is not 0.

**Real Address Mode Exceptions**

None.

**Virtual 8086 Mode Exceptions**

#GP(0) The INVD instruction cannot be executed at the virtual 8086 mode.
INVD—Invalidate Internal Caches (continued)

Intel Architecture Compatibility

This instruction is not supported on Intel Architecture processors earlier than the Intel486 processor.
INVLPG—Invalidate TLB Entry

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 01/7</td>
<td>INVLPG m</td>
<td>Invalidate TLB Entry for page that contains m</td>
</tr>
</tbody>
</table>

Description

Invalidates (flushes) the translation lookaside buffer (TLB) entry specified with the source operand. The source operand is a memory address. The processor determines the page that contains that address and flushes the TLB entry for that page.

The INVLPG instruction is a privileged instruction. When the processor is running in protected mode, the CPL of a program or procedure must be 0 to execute this instruction. This instruction is also implementation-dependent; its function may be implemented differently on future Intel Architecture processors.

The INVLPG instruction normally flushes the TLB entry only for the specified page; however, in some cases, it flushes the entire TLB.

Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,INVLPG);
Flush(RelevantTLBEntries);
Continue (* Continue execution);

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Intercept Mandatory Instruction Intercept

Protected Mode Exceptions

#GP(0) If the current privilege level is not 0.
#UD Operand is a register.

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

#GP(0) The INVLPG instruction cannot be executed at the virtual 8086 mode.

Intel Architecture Compatibility

This instruction is not supported on Intel Architecture processors earlier than the Intel486 processor.
IRET/IRETD—Interrupt Return

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>CF</td>
<td>IRET</td>
<td>Interrupt return (16-bit operand size)</td>
</tr>
<tr>
<td>CF</td>
<td>IRETD</td>
<td>Interrupt return (32-bit operand size)</td>
</tr>
</tbody>
</table>

Description

Returns program control from an exception or interrupt handler to a program or procedure that was interrupted by an exception, an external interrupt or, a software-generated interrupt, or returns from a nested task. IRET and IRETD are mnemonics for the same opcode. The IRETD mnemonic (interrupt return double) is intended for use when returning from an interrupt when using the 32-bit operand size; however, most assemblers use the IRET mnemonic interchangeably for both operand sizes.

In Real Address Mode, the IRET instruction performs a far return to the interrupted program or procedure. During this operation, the processor pops the return instruction pointer, return code segment selector, and EFLAGS image from the stack to the EIP, CS, and EFLAGS registers, respectively, and then resumes execution of the interrupted program or procedure.

In Protected Mode, the action of the IRET instruction depends on the settings of the NT (nested task) and VM flags in the EFLAGS register and the VM flag in the EFLAGS image stored on the current stack. Depending on the setting of these flags, the processor performs the following types of interrupt returns:

- Real Mode.
- Return from virtual-8086 mode.
- Return to virtual-8086 mode.
- Intra-privilege level return.
- Inter-privilege level return.

Return from nested task (task switch)

All forms of IRET result in an IA-32_Intercept(Inst,IRET) in the Itanium System Environment.

If the NT flag (EFLAGS register) is cleared, the IRET instruction performs a far return from the interrupt procedure, without a task switch. The code segment being returned to must be equally or less privileged than the interrupt handler routine (as indicated by the RPL field of the code segment selector popped from the stack). As with a real-address mode interrupt return, the IRET instruction pops the return instruction pointer, return code segment selector, and EFLAGS image from the stack to the EIP, CS, and EFLAGS registers, respectively, and then resumes execution of the interrupted program or procedure. If the return is to another privilege level, the IRET instruction also pops the stack pointer and SS from the stack, before resuming program execution. If the return is to virtual-8086 mode, the processor also pops the data segment registers from the stack.

If the NT flag is set, the IRET instruction performs a return from a nested task (switches from the called task back to the calling task) or reverses the operation of an interrupt or exception that caused a task switch. The updated state of the task executing the IRET instruction is saved in its TSS. If the task is reentered later, the code that follows the IRET instruction is executed.

IRET performs an instruction serialization and a memory fence operation.
IRET/IRETD—Interrupt Return (continued)

Operation

IF(Itanium System Environment)
THEN IA-32_Intercept(Inst.IRET);

IF PE = 0
THEN
   GOTO REAL-ADDRESS-MODE;
ELSE
   GOTO PROTECTED-MODE;
FI;

REAL-ADDRESS-MODE;
IF OperandSize = 32
THEN
   IF top 12 bytes of stack not within stack limits THEN #SS; FI;
   IF instruction pointer not within code segment limits THEN #GP(0); FI;
   EIP ← Pop();
   CS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
   tempEFLAGS ← Pop();
   EFLAGS ← (tempEFLAGS AND 257FD5H) OR (EFLAGS AND 1A0000H);
ELSE (* OperandSize = 16 *)
   IF top 6 bytes of stack are not within stack limits THEN #SS; FI;
   IF instruction pointer not within code segment limits THEN #GP(0); FI;
   EIP ← Pop();
   EIP ← EIP AND 0000FFFFH;
   CS ← Pop(); (* 16-bit pop *)
   EFLAGS[15:0] ← Pop();
FI;

END;

PROTECTED-MODE:
IF VM = 1 (* Virtual-8086 mode: PE=1, VM=1 *)
THEN
   GOTO RETURN-FROM-VIRTUAL-8086-MODE; (* PE=1, VM=1 *)
FI;
IF NT = 1
THEN
   GOTO TASK-RETURN;(*PE=1, VM=0, NT=1 *)
FI;
IF OperandSize=32
THEN
   IF top 12 bytes of stack not within stack limits
      THEN #SS(0)
      FI;
   tempEIP ← Pop();
   tempCS ← Pop();
   tempEFLAGS ← Pop();
   ELSE (* OperandSize = 16 *)
      IF top 6 bytes of stack are not within stack limits
         THEN #SS(0);
      FI;

IRET/IRETD—Interrupt Return (continued)

FI;
    tempEIP ← Pop();
    tempCS ← Pop();
    tempEFLAGS ← Pop();
    tempEIP ← tempEIP AND FFFFH;
    tempEFLAGS ← tempEFLAGS AND FFFFH;
FI;
    IF tempEFLAGS(VM) = 1 AND CPL=0
        THEN
            GOTO RETURN-TO-VIRTUAL-8086-MODE;
            (* PE=1, VM=1 in EFLAGS image *)
        ELSE
            GOTO PROTECTED-MODE-RETURN;
            (* PE=1, VM=0 in EFLAGS image *)
        FI;
RETURN-FROM-VIRTUAL-8086-MODE:
    (* Processor is in virtual-8086 mode when IRET is executed and stays in virtual-8086 mode *)
    IF CR4.VME = 0
        THEN
            IF IOPL=3 (* Virtual mode: PE=1, VM=1, IOPL=3 *)
                THEN
                    IF OperandSize = 32
                        THEN
                            IF top 12 bytes of stack not within stack limits THEN #SS(0); FI;
                            IF instruction pointer not within code segment limits THEN #GP(0); FI;
                            EIP ← Pop();
                            CS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
                            EFLAGS ← Pop();
                            (*VM,IOPL,VIP,and VIF EFLAGS bits are not modified by pop *)
                        ELSE (* OperandSize = 16 *)
                            IF top 6 bytes of stack are not within stack limits THEN #SS(0); FI;
                            IF instruction pointer not within code segment limits THEN #GP(0); FI;
                            EIP ← Pop();
                            EIP ← EIP AND 0000FFFFH;
                            CS ← Pop(); (* 16-bit pop *)
                            EFLAGS[15:0] ← Pop(); (* IOPL in EFLAGS is not modified by pop *)
                        FI;
                    ELSE #GP(0); (* trap to virtual-8086 monitor: PE=1, VM=1, IOPL<3 *)
                FI;
            ELSE (*VME is 1*)
                IF IOPL = 3
                    THEN
                        IF OperandSize = 32
                            THEN
                                EIP ← Pop();
                                CS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
                                TempEFlags ← Pop();
                                FLAGS = (EFLAGS AND 1B3000H) OR (TempEFlags AND 244FD7H)
                                (*VM,IOPL,RF,VIP,and VIF EFLAGS bits are not modified by pop *)
                            ELSE (* OperandSize = 16 *)
                                EIP ← Pop();
                                EIP ← EIP AND 0000FFFFH;
                            FI;
                        ELSE #GP(0); (* trap to virtual-8086 monitor: PE=1, VM=1, IOPL<3 *)
                    FI;
                ELSE (*IOPL<3*)
                    IF OperandSize = 32
                        THEN
                            EIP ← Pop();
                            CS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
                            TempEFlags ← Pop();
                            FLAGS = (EFLAGS AND 1B3000H) OR (TempEFlags AND 244FD7H)
                            (*VM,IOPL,RF,VIP,and VIF EFLAGS bits are not modified by pop *)
                        ELSE (* OperandSize = 16 *)
                            EIP ← Pop();
                            EIP ← EIP AND 0000FFFFH;
IRET/IRETD—Interrupt Return (continued)

```
CS ← Pop(); (* 16-bit pop *)
TempFlags ← Pop();
FLAGS = (FLAGS AND 3000H) OR (TempFlags AND 4FD5H)
(* IOPL unmodified*)
FI;
ELSE (* IOPL < 3 *)
IF OperandSize = 16
THEN
  IF ((STACK.TF != 0) OR (EFLAGS.VIP=1 AND STACK.IF=1))
    THEN #GP(0);
  ELSE
    IP ← Pop(); (*Word Pops*)
    CS ← Pop();
    TempFlags ← Pop();
    (*FLAGS IOPL, IF and TF are not modified*)
    FLAGS = (FLAGS AND 3302H) OR (TempFlags AND 4CD5H)
    EFLAGS.VIF ← TempFlags.IF;
    FI;
  ELSE (*OperandSize = 32 *)
    #GP(0);
  FI;
ELSE (*IOPL < 3*)
IF OperandSize = 16
THEN
  #GP(0);
END;

RETURN-TO-VIRTUAL-8086-MODE:

(* Interrupted procedure was in virtual-8086 mode: PE=1, VM=1 in flags image *)
IF top 24 bytes of stack are not within stack segment limits
  THEN #SS(0);
FI;
IF instruction pointer not within code segment limits
  THEN #GP(0);
FI;
CS ← tempCS;
EIP ← tempEIP;
EFLAGS ← tempEFLAGS
TempESP ← Pop();
TempSS ← Pop();
ES ← Pop(); (* pop 2 words; throw away high-order word *)
DS ← Pop(); (* pop 2 words; throw away high-order word *)
FS ← Pop(); (* pop 2 words; throw away high-order word *)
GS ← Pop(); (* pop 2 words; throw away high-order word *)
SS:ESP ← TempSS:TempESP;
(* Resume execution in Virtual 8086 mode *)
END;

TASK-RETURN: (* PE=1, VM=1, NT=1 *)
Read segment selector in link field of current TSS;
IF local/global bit is set to local
  OR index not within GDT limits
    THEN #GP(TSS selector);
FI;
Access TSS for task specified in link field of current TSS;
```
IRET/IRETD—Interrupt Return (continued)

IF TSS descriptor type is not TSS or if the TSS is marked not busy
THEN #GP(TSS selector);
FI;
IF TSS not present
THEN #NP(TSS selector);
FI;
SWITCH-TASKS (without nesting) to TSS specified in link field of current TSS;
Mark the task just abandoned as NOT BUSY;
IF EIP is not within code segment limit
THEN #GP(0);
FI;
END;

PROTECTED-MODE-RETURN: (* PE=1, VM=0 in flags image *)
IF return code segment selector is null THEN GP(0); FI;
IF return code segment selector addresses descriptor beyond descriptor table limit
THEN GP(selector); FI;
Read segment descriptor pointed to by the return code segment selector
IF return code segment descriptor is not a code segment THEN #GP(selector); FI;
IF return code segment selector RPL < CPL THEN #GP(selector); FI;
IF return code segment descriptor is conforming
    AND return code segment DPL > return code segment selector RPL
    THEN #GP(selector); FI;
IF return code segment descriptor is not present THEN #NP(selector); FI;
IF return code segment selector RPL > CPL
    THEN GOTO RETURN-OUTER-PRIVILEGE-LEVEL;
    ELSE GOTO RETURN-TO-SAME-PRIVILEGE-LEVEL
FI;
END;

RETURN-TO-SAME-PRIVILEGE-LEVEL: (* PE=1, VM=0 in flags image, RPL=CPL *)
IF EIP is not within code segment limits THEN #GP(0); FI;
EIP ← tempEIP;
CS ← tempCS; (* segment descriptor information also loaded *)
EFLAGS (CF, PF, AF, ZF, SF, TF, DF, OF, NT) ← tempEFLAGS;
IF OperandSize=32
    THEN
        EFLAGS(RF, AC, ID) ← tempEFLAGS;
FI;
IF CPL ≤ IOPL
    THEN
        EFLAGS(IF) ← tempEFLAGS;
FI;
IF CPL = 0
    THEN
        EFLAGS(IOPL) ← tempEFLAGS;
        IF OperandSize=32
            THEN EFLAGS(VM, VIF, VIP) ← tempEFLAGS;
        FI;
FI;
END;

RETURN-TO-OUTER-PRIVILEGE-LEVEL:
IRET/IRETD—Interrupt Return (continued)

IF OperandSize=32
THEN
  IF top 8 bytes on stack are not within limits THEN #SS(0); FI;
ELSE (* OperandSize=16 *)
  IF top 4 bytes on stack are not within limits THEN #SS(0); FI;
FI;
Read return segment selector;
IF stack segment selector is null THEN #GP(0); FI;
IF return stack segment selector index is not within its descriptor table limits
THEN #GP(SSselector); FI;
Read segment descriptor pointed to by return segment selector;
IF stack segment selector RPL ≠ RPL of the return code segment selector
  OR the stack segment descriptor does not indicate a a writable data segment;
  OR stack segment DPL ≠ RPL of the return code segment selector
THEN #GP(SS selector);
FI;
IF stack segment is not present THEN #NP(SS selector); FI;
EIP ← tempEIP;
CS ← tempCS;
EFLAGS (CF, PF, AF, ZF, SF, TF, DF, OF, NT) ← tempEFLAGS;
IF OperandSize=32
THEN
  EFLAGS(RF, AC, ID) ← tempEFLAGS;
FI;
IF CPO ≤ IOPL
THEN
  EFLAGS(IF) ← tempEFLAGS;
FI;
IF CPL = 0
THEN
  EFLAGS(IOPL) ← tempEFLAGS;
  IF OperandSize=32
    THEN EFLAGS(VM, VIF, VIP) ← tempEFLAGS;
  FI;
FI;
CPL ← RPL of the return code segment selector;
FOR each of segment register (ES, FS, GS, and DS) DO:
  IF segment register points to data or non-conforming code segment
    AND CPL > segment descriptor DPL (* stored in hidden part of segment register *)
    THEN (* segment register invalid *)
      SegmentSelector ← 0; (* null segment selector *)
  FI;
OD;
END:
IRET/IRETD—Interrupt Return (continued)

Flags Affected

All the flags and fields in the EFLAGS register are potentially modified, depending on the mode of operation of the processor.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
<tr>
<td>IA-32_Intercept</td>
<td>Instruction Intercept Trap for ALL forms of IRET.</td>
</tr>
</tbody>
</table>

Protected Mode Exceptions

#GP(0)  If the return code or stack segment selector is null.

If the return instruction pointer is not within the return code segment limit.

#GP(selector)  If a segment selector index is outside its descriptor table limits.

If the return code segment selector RPL is greater than the CPL.

If the DPL of a conforming-code segment is greater than the return code segment selector RPL.

If the DPL for a nonconforming-code segment is not equal to the RPL of the code segment selector.

If the stack segment descriptor DPL is not equal to the RPL of the return code segment selector.

If the stack segment is not a writable data segment.

If the stack segment selector RPL is not equal to the RPL of the return code segment selector.

If the segment descriptor for a code segment does not indicate it is a code segment.

If the segment selector for a TSS has its local/global bit set for local.

If a TSS segment descriptor specifies that the TSS is busy or not available.

#SS(0)  If the top bytes of stack are not within stack limits.

#NP(selector)  If the return code or stack segment is not present.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If an unaligned memory reference occurs when the CPL is 3 and alignment checking is enabled.

Real Address Mode Exceptions

#GP  If the return instruction pointer is not within the return code segment limit.

#SS  If the top bytes of stack are not within stack limits.
IRET/IRETD—Interrupt Return (continued)

**Virtual 8086 Mode Exceptions**

- **#GP(0)**: If the return instruction pointer is not within the return code segment limit.
- **#PF(fault-code)**: If a page fault occurs.
- **#SS(0)**: If the top bytes of stack are not within stack limits.
- **#AC(0)**: If an unaligned memory reference occurs and alignment checking is enabled.
## Jcc—Jump if Condition Is Met

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>77 cb</td>
<td>JA rel8</td>
<td>Jump short if above (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>73 cb</td>
<td>JAE rel8</td>
<td>Jump short if above or equal (CF=0)</td>
</tr>
<tr>
<td>72 cb</td>
<td>JB rel8</td>
<td>Jump short if below (CF=1)</td>
</tr>
<tr>
<td>76 cb</td>
<td>JBE rel8</td>
<td>Jump short if below or equal (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>72 cb</td>
<td>JC rel8</td>
<td>Jump short if carry (CF=1)</td>
</tr>
<tr>
<td>E3 cb</td>
<td>JCXZ rel8</td>
<td>Jump short if CX register is 0</td>
</tr>
<tr>
<td>E3 cb</td>
<td>JECXZ rel8</td>
<td>Jump short if ECX register is 0</td>
</tr>
<tr>
<td>74 cb</td>
<td>JE rel8</td>
<td>Jump short if equal (ZF=1)</td>
</tr>
<tr>
<td>7F cb</td>
<td>JG rel8</td>
<td>Jump short if greater (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>7D cb</td>
<td>JGE rel8</td>
<td>Jump short if greater or equal (SF=OF)</td>
</tr>
<tr>
<td>7C cb</td>
<td>JL rel8</td>
<td>Jump short if less (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>7E cb</td>
<td>JLE rel8</td>
<td>Jump short if less or equal (ZF=1 or SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>76 cb</td>
<td>JNA rel8</td>
<td>Jump short if not above (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>72 cb</td>
<td>JNAE rel8</td>
<td>Jump short if not above or equal (CF=1)</td>
</tr>
<tr>
<td>73 cb</td>
<td>JNB rel8</td>
<td>Jump short if not below (CF=0)</td>
</tr>
<tr>
<td>77 cb</td>
<td>JNBE rel8</td>
<td>Jump short if not below or equal (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>73 cb</td>
<td>JNC rel8</td>
<td>Jump short if not carry (CF=0)</td>
</tr>
<tr>
<td>75 cb</td>
<td>JNE rel8</td>
<td>Jump short if not equal (ZF=0)</td>
</tr>
<tr>
<td>7E cb</td>
<td>JNG rel8</td>
<td>Jump short if not greater (ZF=1 or SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>7C cb</td>
<td>JNGE rel8</td>
<td>Jump short if not greater or equal (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>7D cb</td>
<td>JNL rel8</td>
<td>Jump short if not less (SF=OF)</td>
</tr>
<tr>
<td>7F cb</td>
<td>JNLE rel8</td>
<td>Jump short if not less or equal (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>71 cb</td>
<td>JNO rel8</td>
<td>Jump short if not overflow (OF=0)</td>
</tr>
<tr>
<td>7B cb</td>
<td>JNP rel8</td>
<td>Jump short if not parity (PF=0)</td>
</tr>
<tr>
<td>79 cb</td>
<td>JNS rel8</td>
<td>Jump short if not sign (SF=0)</td>
</tr>
<tr>
<td>75 cb</td>
<td>JNZ rel8</td>
<td>Jump short if not zero (ZF=0)</td>
</tr>
<tr>
<td>70 cb</td>
<td>JO rel8</td>
<td>Jump short if overflow (OF=1)</td>
</tr>
<tr>
<td>7A cb</td>
<td>JP rel8</td>
<td>Jump short if parity (PF=1)</td>
</tr>
<tr>
<td>7A cb</td>
<td>JPE rel8</td>
<td>Jump short if parity even (PF=1)</td>
</tr>
<tr>
<td>7B cb</td>
<td>JPO rel8</td>
<td>Jump short if parity odd (PF=0)</td>
</tr>
<tr>
<td>78 cb</td>
<td>JS rel8</td>
<td>Jump short if sign (SF=1)</td>
</tr>
<tr>
<td>74 cb</td>
<td>JZ rel8</td>
<td>Jump short if zero (ZF = 1)</td>
</tr>
<tr>
<td>0F 87 cw/cd</td>
<td>JA rel16/32</td>
<td>Jump near if above (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>0F 83 cw/cd</td>
<td>JAE rel16/32</td>
<td>Jump near if above or equal (CF=0)</td>
</tr>
<tr>
<td>0F 82 cw/cd</td>
<td>JB rel16/32</td>
<td>Jump near if below (CF=1)</td>
</tr>
<tr>
<td>0F 86 cw/cd</td>
<td>JBE rel16/32</td>
<td>Jump near if below or equal (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 82 cw/cd</td>
<td>JC rel16/32</td>
<td>Jump near if carry (CF=1)</td>
</tr>
<tr>
<td>0F 84 cw/cd</td>
<td>JE rel16/32</td>
<td>Jump near if equal (ZF=1)</td>
</tr>
<tr>
<td>0F 84 cw/cd</td>
<td>JZ rel16/32</td>
<td>Jump near if 0 (ZF=1)</td>
</tr>
<tr>
<td>0F 8F cw/cd</td>
<td>JG rel16/32</td>
<td>Jump near if greater (ZF=0 and SF=OF)</td>
</tr>
</tbody>
</table>
Jcc—Jump if Condition Is Met (continued)

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 8D</td>
<td>JGE rel16/32</td>
<td>Jump near if greater or equal (SF=OF)</td>
</tr>
<tr>
<td>0F 8C</td>
<td>JL rel16/32</td>
<td>Jump near if less (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 8E</td>
<td>JLE rel16/32</td>
<td>Jump near if less or equal (ZF=1 or SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 86</td>
<td>JNA rel16/32</td>
<td>Jump near if not above (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 82</td>
<td>JNAE rel16/32</td>
<td>Jump near if not above or equal (CF=1)</td>
</tr>
<tr>
<td>0F 83</td>
<td>JNB rel16/32</td>
<td>Jump near if not below (CF=0)</td>
</tr>
<tr>
<td>0F 87</td>
<td>JNBE rel16/32</td>
<td>Jump near if not below or equal (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>0F 83</td>
<td>JNC rel16/32</td>
<td>Jump near if not carry (CF=0)</td>
</tr>
<tr>
<td>0F 85</td>
<td>JNAE rel16/32</td>
<td>Jump near if not equal (ZF=0)</td>
</tr>
<tr>
<td>0F 8E</td>
<td>JNG rel16/32</td>
<td>Jump near if not greater (ZF=1 or SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 8C</td>
<td>JNGE rel16/32</td>
<td>Jump near if not greater or equal (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 8D</td>
<td>JNL rel16/32</td>
<td>Jump near if not less (SF=OF)</td>
</tr>
<tr>
<td>0F 8F</td>
<td>JNLE rel16/32</td>
<td>Jump near if not less or equal (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>0F 81</td>
<td>JNO rel16/32</td>
<td>Jump near if not overflow (OF=0)</td>
</tr>
<tr>
<td>0F 8B</td>
<td>JNP rel16/32</td>
<td>Jump near if not parity (PF=0)</td>
</tr>
<tr>
<td>0F 89</td>
<td>JNS rel16/32</td>
<td>Jump near if not sign (SF=0)</td>
</tr>
<tr>
<td>0F 85</td>
<td>JNZ rel16/32</td>
<td>Jump near if not zero (ZF=0)</td>
</tr>
<tr>
<td>0F 80</td>
<td>JO rel16/32</td>
<td>Jump near if overflow (OF=1)</td>
</tr>
<tr>
<td>0F 8A</td>
<td>JP rel16/32</td>
<td>Jump near if parity (PF=1)</td>
</tr>
<tr>
<td>0F 8A</td>
<td>JPE rel16/32</td>
<td>Jump near if parity even (PF=1)</td>
</tr>
<tr>
<td>0F 8B</td>
<td>JPO rel16/32</td>
<td>Jump near if parity odd (PF=0)</td>
</tr>
<tr>
<td>0F 88</td>
<td>JS rel16/32</td>
<td>Jump near if sign (SF=1)</td>
</tr>
<tr>
<td>0F 84</td>
<td>JZ rel16/32</td>
<td>Jump near if 0 (ZF=1)</td>
</tr>
</tbody>
</table>

Description

Checks the state of one or more of the status flags in the EFLAGS register (CF, OF, PF, SF, and ZF) and, if the flags are in the specified state (condition), performs a jump to the target instruction specified by the destination operand. A condition code (cc) is associated with each instruction to indicate the condition being tested for. If the condition is not satisfied, the jump is not performed and execution continues with the instruction following the Jcc instruction.

The target instruction is specified with a relative offset (a signed offset relative to the current value of the instruction pointer in the EIP register). A relative offset (rel8, rel16, or rel32) is generally specified as a label in assembly code, but at the machine code level, it is encoded as a signed, 8-bit or 32-bit immediate value, which is added to the instruction pointer. Instruction coding is most efficient for offsets of −128 to +127. If the operand-size attribute is 16, the upper two bytes of the EIP register are cleared to 0s, resulting in a maximum instruction pointer size of 16 bits.

The conditions for each Jcc mnemonic are given in the “Description” column of the above table. The terms “less” and “greater” are used for comparisons of signed integers and the terms “above” and “below” are used for unsigned integers.
Jcc—Jump if Condition Is Met (continued)

Because a particular state of the status flags can sometimes be interpreted in two ways, two mnemonics are defined for some opcodes. For example, the JA (jump if above) instruction and the JNBE (jump if not below or equal) instruction are alternate mnemonics for the opcode 77H.

The Jcc instruction does not support far jumps (jumps to other code segments). When the target for the conditional jump is in a different segment, use the opposite condition from the condition being tested for the Jcc instruction, and then access the target with an unconditional far jump (JMP instruction) to the other segment. For example, the following conditional far jump is illegal:

```
JZ  FARLABEL;
```

To accomplish this far jump, use the following two instructions:

```
JNZ BEYOND;
JMP FARLABEL;
BEYOND:
```

The JECXZ and JCXZ instructions differs from the other Jcc instructions because they do not check the status flags. Instead they check the contents of the ECX and CX registers, respectively, for 0. These instructions are useful at the beginning of a conditional loop that terminates with a conditional loop instruction (such as LOOPNE). They prevent entering the loop when the ECX or CX register is equal to 0, which would cause the loop to execute $2^{32}$ or 64K times, respectively, instead of zero times.

All conditional jumps are converted to code fetches of one or two cache lines, regardless of jump address or cacheability.

**Operation**

If condition

```
IF condition
THEN
  EIP ← EIP + SignExtend(DEST);
  IF OperandSize = 16
  THEN
    EIP ← EIP AND 0000FFFFH;
  FI;
  IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
FI;
```

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

IA-32_Exception       Taken Branch Debug Exception if PSR.tb is 1

**Protected Mode Exceptions**

#GP(0)       If the offset being jumped to is beyond the limits of the CS segment.
Jcc—Jump if Condition Is Met (continued)

Real Address Mode Exceptions

#GP If the offset being jumped to is beyond the limits of the CS segment or is outside of the effective address space from 0 to FFFFH. This condition can occur if 32-address size override prefix is used.

Virtual 8086 Mode Exceptions

#GP(0) If the offset being jumped to is beyond the limits of the CS segment or is outside of the effective address space from 0 to FFFFH. This condition can occur if 32-address size override prefix is used.
JMP—Jump

### Description

Transfers program control to a different point in the instruction stream without recording return information. The destination (target) operand specifies the address of the instruction being jumped to. This operand can be an immediate value, a general-purpose register, or a memory location.

- **Near jump** — A jump to an instruction within the current code segment (the segment currently pointed to by the CS register), sometimes referred to as an intrasegment call.
- **Far jump** — A jump to an instruction located in a different segment than the current code segment, sometimes referred to as an intersegment call.
- **Task switch** — A jump to an instruction located in a different task. (This is a form of a far jump.) *Results in an IA-32 Intercept(Gate) in Itanium System Environment.*

A task switch can only be executed in protected mode (see Chapter 6 in the *Intel Architecture Software Developer’s Manual, Volume 3* for information on task switching with the JMP instruction).

When executing a near jump, the processor jumps to the address (within the current code segment) that is specified with the target operand. The target operand specifies either an absolute address (that is an offset from the base of the code segment) or a relative offset (a signed offset relative to the current value of the instruction pointer in the EIP register). An absolute address is specified directly in a register or indirectly in a memory location (*r/m16* or *r/m32* operand form). A relative offset (*rel8*, *rel16*, or *rel32*) is generally specified as a label in assembly code, but at the machine code level, it is encoded as a signed, 8-bit or 32-bit immediate value, which is added to the value in the EIP register (that is, to the instruction following the JMP instruction). The operand-size attribute determines the size of the target operand (16 or 32 bits) for absolute addresses. Absolute addresses are loaded directly into the EIP register. When a relative offset is specified, it is added to the value of the EIP register. If the operand-size attribute is 16, the upper two bytes of the EIP register are cleared to 0s, resulting in a maximum instruction pointer size of 16 bits. The CS register is not changed on near jumps.

### Opcode Instruction Description

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>EB cb</td>
<td>JMP rel8</td>
<td>Jump near, relative address</td>
</tr>
<tr>
<td>E9 cw</td>
<td>JMP rel16</td>
<td>Jump near, relative address</td>
</tr>
<tr>
<td>E9 cd</td>
<td>JMP rel32</td>
<td>Jump near, relative address</td>
</tr>
<tr>
<td>FF /4</td>
<td>JMP r/m16</td>
<td>Jump near, indirect address</td>
</tr>
<tr>
<td>FF /4</td>
<td>JMP r/m32</td>
<td>Jump near, indirect address</td>
</tr>
<tr>
<td>EA cd</td>
<td>JMP ptr16:16</td>
<td>Jump far, absolute address</td>
</tr>
<tr>
<td>EA cp</td>
<td>JMP ptr16:32</td>
<td>Jump far, absolute address</td>
</tr>
<tr>
<td>FF /5</td>
<td>JMP m16:16</td>
<td>Jump far, indirect address</td>
</tr>
<tr>
<td>FF /5</td>
<td>JMP m16:32</td>
<td>Jump far, indirect address</td>
</tr>
</tbody>
</table>
JMP—Jump (continued)

When executing a far jump, the processor jumps to the code segment and address specified with the target operand. Here the target operand specifies an absolute far address either directly with a pointer (ptr16:16 or ptr16:32) or indirectly with a memory location (m16:16 or m16:32). With the pointer method, the segment and address of the called procedure is encoded in the instruction using a 4-byte (16-bit operand size) or 6-byte (32-bit operand size) far address immediate. With the indirect method, the target operand specifies a memory location that contains a 4-byte (16-bit operand size) or 6-byte (32-bit operand size) far address. The operand-size attribute determines the size of the offset (16 or 32 bits) in the far address. The far address is loaded directly into the CS and EIP registers. If the operand-size attribute is 16, the upper two bytes of the EIP register are cleared to 0s.

When the processor is operating in protected mode, a far jump can also be used to access a code segment through a call gate or to switch tasks. Here, the processor uses the segment selector part of the far address to access the segment descriptor for the segment being jumped to. Depending on the value of the type and access rights information in the segment selector, the JMP instruction can perform:

- A far jump to a conforming or non-conforming code segment (same mechanism as the far jump described in the previous paragraph, except that the processor checks the access rights of the code segment being jumped to).
- An far jump through a call gate.
- A task switch. Results in an IA-32_Intercept(Gate) in Itanium System Environment.

The JMP instruction cannot be used to perform inter-privilege level jumps.

When executing an far jump through a call gate, the segment selector specified by the target operand identifies the call gate. (The offset part of the target operand is ignored.) The processor then jumps to the code segment specified in the call gate descriptor and begins executing the instruction at the offset specified in the gate. No stack switch occurs. Here again, the target operand can specify the far address of the call gate and instruction either directly with a pointer (ptr16:16 or ptr16:32) or indirectly with a memory location (m16:16 or m16:32).

Executing a task switch with the JMP instruction, is similar to executing a jump through a call gate. Here the target operand specifies the segment selector of the task gate for the task being switched to. (The offset part of the target operand is ignored). The task gate in turn points to the TSS for the task, which contains the segment selectors for the task’s code, data, and stack segments and the instruction pointer to the target instruction. One form of the JMP instruction allows the jump to be made directly to a TSS, without going through a task gate. See Chapter 13 in Intel Architecture Software Developer’s Manual, Volume 3 the for detailed information on the mechanics of a task switch.

All branches are converted to code fetches of one or two cache lines, regardless of jump address or cacheability.
JMP—Jump (continued)

**Operation**

IF near jump
THEN IF near relative jump
    THEN tempEIP ← EIP + DEST; (* EIP is instruction following JMP instruction*)
    ELSE (* near absolute jump *)
        tempEIP ← DEST;
    FI;
IF tempEIP is beyond code segment limit THEN #GP(0); FI;
IF OperandSize = 32
    THEN EIP ← tempEIP;
    ELSE (* OperandSize=16 *)
        EIP ← tempEIP AND 0000FFFFH;
    FI;
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
FI:

IF far jump AND (PE = 0 OR (PE = 1 AND VM = 1)) (* real address or virtual 8086 mode *)
THEN
tempEIP ← DEST(offset); (* DEST is ptr16:32 or [m16:32] *)
IF tempEIP is beyond code segment limit THEN #GP(0); FI;
CS ← DEST(segment selector); (* DEST is ptr16:32 or [m16:32] *)
IF OperandSize = 32
    THEN EIP ← tempEIP; (* DEST is ptr16:32 or [m16:32] *)
    ELSE (* OperandSize = 16 *)
        EIP ← tempEIP AND 0000FFFFH; (* clear upper 16 bits *)
    FI;
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
FI;

IF far call AND (PE = 1 AND VM = 0) (* Protected mode, not virtual 8086 mode *)
THEN
    IF effective address in the CS, DS, ES, FS, GS, or SS segment is illegal
    OR segment selector in target operand null
    THEN #GP(0);  
    FI;
    IF segment selector index not within descriptor table limits
    THEN #GP(new selector);
    FI;
    Read type and access rights of segment descriptor;
    IF segment type is not a conforming or nonconforming code segment, call gate,
    task gate, or TSS THEN #GP(segment selector); FI;
    Depending on type and access rights
    GO TO CONFORMING-CODE-SEGMENT;
    GO TO NONCONFORMING-CODE-SEGMENT;
    GO TO CALL-GATE;
    GO TO TASK-GATE;
    GO TO TASK-STATE-SEGMENT;
ELSE
    #GP(segment selector);
FI;
JMP—Jump (continued)

CONFORMING-CODE-SEGMENT:
IF DPL > CPL THEN #GP(segment selector); FI;
IF segment not present THEN #NP(segment selector); FI;
\[ \text{tempEIP } \leftarrow \text{DEST}(\text{offset}); \]
IF OperandSize=16
THEN tempEIP \leftarrow \text{tempEIP AND 0000FFFFH};
FI;
IF tempEIP not in code segment limit THEN #GP(0); FI;
CS \leftarrow \text{DEST}(\text{SegmentSelector}); (* segment descriptor information also loaded *)
CS(RPL) \leftarrow \text{CPL}
EIP \leftarrow \text{tempEIP};
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
END;

NONCONFORMING-CODE-SEGMENT:
IF (RPL > CPL) OR (DPL \neq \text{CPL}) THEN #GP(code segment selector); FI;
IF segment not present THEN #NP(segment selector); FI;
IF instruction pointer outside code segment limit THEN #GP(0); FI;
\[ \text{tempEIP } \leftarrow \text{DEST}(\text{offset}); \]
IF OperandSize=16
THEN tempEIP \leftarrow \text{tempEIP AND 0000FFFFH};
FI;
IF tempEIP not in code segment limit THEN #GP(0); FI;
CS \leftarrow \text{DEST}(\text{SegmentSelector}); (* segment descriptor information also loaded *)
CS(RPL) \leftarrow \text{CPL}
EIP \leftarrow \text{tempEIP};
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
END;

CALL-GATE:
IF call gate DPL < CPL
\[ \text{OR call gate DPL } < \text{call gate segment-selector RPL} \]
THEN #GP(call gate selector); FI;
IF call gate not present THEN #NP(call gate selector); FI;
IF Itanium System Environment THEN IA-32_Intercept(Gate,JMP);
IF call gate code-segment selector is null THEN #GP(0); FI;
IF call gate code-segment selector index is outside descriptor table limits
THEN #GP(code segment selector); FI;
Read code segment descriptor;
IF code-segment segment descriptor does not indicate a code segment
\[ \text{OR code-segment segment descriptor is conforming and DPL } > \text{CPL} \]
\[ \text{OR code-segment segment descriptor is non-conforming and DPL } \neq \text{CPL} \]
THEN #GP(code segment selector); FI;
IF code segment is not present THEN #NP(code-segment selector); FI;
IF instruction pointer is not within code-segment limit THEN #GP(0); FI;
\[ \text{tempEIP } \leftarrow \text{DEST}(\text{offset}); \]
IF GateSize=16
THEN tempEIP \leftarrow \text{tempEIP AND 0000FFFFH};
FI;
IF tempEIP not in code segment limit THEN #GP(0); FI;
CS \leftarrow \text{DEST}(\text{SegmentSelector}); (* segment descriptor information also loaded *)
CS(RPL) \leftarrow \text{CPL}
EIP \leftarrow \text{tempEIP};
JMP—Jump (continued)

END;

TASK-GATE:
  IF task gate DPL < CPL
    OR task gate DPL < task gate segment-selector RPL
    THEN #GP(task gate selector); FI;
  IF task gate not present THEN #NP(gate selector); FI;
  IF Itanium System Environment THEN IA-32_Intercept(Gate,JMP);
  Read the TSS segment selector in the task-gate descriptor;
  IF TSS segment selector local/global bit is set to local
    OR index not within GDT limits
    OR TSS descriptor specifies that the TSS is busy
    THEN #GP(TSS selector); FI;
  IF TSS not present THEN #NP(TSS selector); FI;
  IF Itanium System Environment THEN IA-32_Intercept(Gate,JMP);
  SWITCH-TASKS to TSS;
  IF EIP not within code segment limit THEN #GP(0); FI;
END;

TASK-STATE-SEGMENT:
  IF TSS DPL < CPL
    OR TSS DPL < TSS segment-selector RPL
    OR TSS descriptor indicates TSS not available
    THEN #GP(TSS selector); FI;
  IF TSS is not present THEN #NP(TSS selector); FI;
  IF Itanium System Environment THEN IA-32_Intercept(Gate,JMP);
  SWITCH-TASKS to TSS;
  IF EIP not within code segment limit THEN #GP(0); FI;
END;

Flags Affected

All flags are affected if a task switch occurs; no flags are affected if a task switch does not occur.

Additional Itanium System Environment Exceptions

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
IA-32_Intercept   Gate Intercept for JMP through CALL Gates, Task Gates and Task Segments
IA-32_Exception   Taken Branch Debug Exception if PSR.tb is 1

Protected Mode Exceptions

#GP(0)  If offset in target operand, call gate, or TSS is beyond the code segment limits.
       If the segment selector in the destination operand, call gate, task gate, or TSS is null.
       If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
JMP—Jump (continued)

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#GP(selector) If segment selector index is outside descriptor table limits.

If the segment descriptor pointed to by the segment selector in the destination operand is not for a conforming-code segment, nonconforming-code segment, call gate, task gate, or task state segment.

If the DPL for a nonconforming-code segment is not equal to the CPL.

(When not using a call gate.) If the RPL for the segment’s segment selector is greater than the CPL.

If the DPL for a conforming-code segment is greater than the CPL.

If the DPL from a call-gate, task-gate, or TSS segment descriptor is less than the CPL or than the RPL of the call-gate, task-gate, or TSS’s segment selector.

If the segment descriptor for selector in a call gate does not indicate it is a code segment.

If the segment descriptor for the segment selector in a task gate does not indicate available TSS.

If the segment selector for a TSS has its local/global bit set for local.

If a TSS segment descriptor specifies that the TSS is busy or not available.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#NP (selector) If the code segment being accessed is not present.

If call gate, task gate, or TSS not present.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3. (Only occurs when fetching target from memory.)

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If the target operand is beyond the code segment limits.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made. (Only occurs when fetching target from memory.)
### JMPE—Jump to Intel® Itanium™ Instruction Set

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 00 /6</td>
<td>JMPE r/m16</td>
<td>Jump to Intel® Itanium™ instruction set, indirect address specified by r/m16</td>
</tr>
<tr>
<td>0F 00 /6</td>
<td>JMPE r/m32</td>
<td>Jump to Intel® Itanium™ instruction set, indirect address specified by r/m32</td>
</tr>
<tr>
<td>0F B8</td>
<td>JMPE disp16</td>
<td>Jump to Intel® Itanium™ instruction set, absolute address specified by addr16</td>
</tr>
<tr>
<td>0F B8</td>
<td>JMPE disp32</td>
<td>Jump to Intel® Itanium™ instruction set, absolute address specified by addr32</td>
</tr>
</tbody>
</table>

#### Description

This instruction is available only on processors based on the Itanium architecture in the Itanium System Environment. Otherwise, execution of this instruction at privilege levels 1, 2, and 3 results in an Illegal Opcode fault, and at privilege level 0, termination of the IA-32 System Environment on a processor based on the Itanium architecture.

JMPE switches the processor to the Itanium instruction set and starts execution at the specified target address. There are two forms: an indirect form, r/m16/32, and an unsigned absolute form, disp16/32. Both 16 and 32-bit formats are supported.

The absolute form computes the 16-byte aligned 64-bit virtual target address in the Itanium instruction set by adding the unsigned 16 or 32-bit displacement to the current CS base (IP[31:0] = disp16/32 + CSD.base). The indirect form specifies the virtual target address by the contents of a register or memory location (IP[31:0] = [r/m16/32] + CSD.base). Target addresses are constrained to the lower 4G-bytes of the 64-bit virtual address space within virtual region 0.

GR[1] is loaded with the next sequential instruction address following JMPE.

If PSR.di is 1, the instruction is nullified and a Disabled Instruction Set Transition fault is generated. If Itanium branch debugging is enabled, an IA-32_Exception(Debug) trap is taken after JMPE completes execution.

JMPE can be performed at any privilege level and does not change the privilege level of the processor.

JMPE performs a FWAIT operation, any pending IA-32 unmasked floating-point exceptions are reported as faults on the JMPE instruction.

JMPE does not perform a memory fence or serialization operation.

Successful execution of JMPE clears EFLAG.rf and PSR.id to zero.

If the register stack engine is enabled for eager execution, the register stack engine may immediately start loading registers when the processor enters the Itanium instruction set.
JMPE—Jump to Intel® Itanium™ Instruction Set (continued)

Operation

if (NOT Itanium System Environment) {
    if (PSR.cpl==0) Terminate_IA-32_System_Env();
    else IA_32_Exception(IllegalOpcode);
} else if (PSR.di==1) {
    Disabled/Instruction_Set_Transition_Fault();
} else if (pending_numeric_exceptions()) {
    IA_32_exception(FPError);
} else {
    if (absolute_form) { //compute virtual target
        IP{31:0} = disp16/32 + AR[CSD].base;//disp is 16/32-bit unsigned value
    } else if (indirect_form) {
        IP{31:0} = [r/m16/32] + AR[CSD].base;
    }
    PSR.is = 0; //set Itanium Instruction Set bit
    IP{3:0}= 0; //Force 16-byte alignment
    IP{63:32} = 0; //zero extend from 32-bits to 64-bits
    GR[1]{31:0} = EIP + AR[CSD].base; //next sequential instruction address
    GR[1]{63:32} = 0;
    PSR.id = EFLAG.rf = 0;
    if (PSR.tb) //taken branch trap
        IA_32_Exception(Debug);
}

Flags Affected

None (other than EFLAG.rf)

Additional Itanium System Environment Exceptions

Itanium Reg Faults NaT Register Consumption Fault.
Disabled ISA Disabled Instruction Set Transition Fault, if PSR.di is 1
IA-32_Exception Floating-point Error, if any floating-point exceptions are pending
IA-32_Exception Taken Branch trap, if PSR.tb is 1.

IA-32 System Environment Exceptions (All Operating Modes)

#UD JMPE raises an invalid opcode exception at privilege levels 1, 2 and 3. Privilege level 0 results in termination of the IA-32 System Environment on a processor based on the Itanium architecture.
LAHF—Load Status Flags into AH Register

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9F</td>
<td>LAHF</td>
<td>Load: (AH = EFLAGS(SF;ZF:0;AF:0;PF:1;CF))</td>
</tr>
</tbody>
</table>

**Description**

Moves the low byte of the EFLAGS register (which includes status flags SF, ZF, AF, PF, and CF) to the AH register. Reserved bits 1, 3, and 5 of the EFLAGS register are set in the AH register as shown in the “Operation” below.

**Operation**

\[AH \leftarrow EFLAGS(SF;ZF:0;AF:0;PF:1;CF);\]

**Flags Affected**

None (that is, the state of the flags in the EFLAGS register are not affected).

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults    NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.
LAR—Load Access Rights Byte

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 02 /r</td>
<td>LAR r16,r/m16</td>
<td>$r16 \leftarrow r/m16$ masked by FF00H</td>
</tr>
<tr>
<td>0F 02 /r</td>
<td>LAR r32,r/m32</td>
<td>$r32 \leftarrow r/m32$ masked by 00FxFF00H</td>
</tr>
</tbody>
</table>

Description

Loads the access rights from the segment descriptor specified by the second operand (source operand) into the first operand (destination operand) and sets the ZF flag in the EFLAGS register. The source operand (which can be a register or a memory location) contains the segment selector for the segment descriptor being accessed. The destination operand is a general-purpose register.

The processor performs access checks as part of the loading process. Once loaded in the destination register, software can perform additional checks on the access rights information.

When the operand size is 32 bits, the access rights for a segment descriptor comprise the type and DPL fields and the S, P, AVL, D/B, and G flags, all of which are located in the second doubleword (bytes 4 through 7) of the segment descriptor. The doubleword is masked by 00FxFF00H before it is loaded into the destination operand. When the operand size is 16 bits, the access rights comprise the type and DPL fields. Here, the two lower-order bytes of the doubleword are masked by FF00H before being loaded into the destination operand.

This instruction performs the following checks before it loads the access rights in the destination register:

- Checks that the segment selector is not null.
- Checks that the segment selector points to a descriptor that is within the limits of the GDT or LDT being accessed.
- Checks that the descriptor type is valid for this instruction. All code and data segment descriptors are valid for (can be accessed with) the LAR instruction. The valid system segment and gate descriptor types are given in the following table.
- If the segment is not a conforming code segment, it checks that the specified segment descriptor is visible at the CPL (that is, if the CPL and the RPL of the segment selector are less than or equal to the DPL of the segment selector).

If the segment descriptor cannot be accessed or is an invalid type for the instruction, the ZF flag is cleared and no access rights are loaded in the destination operand.

The LAR instruction can only be executed in protected mode.
LAR—Load Access Rights Byte (continued)

Table 1-15. LAR Descriptor Validity

<table>
<thead>
<tr>
<th>Type</th>
<th>Name</th>
<th>Valid</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>1</td>
<td>Available 16-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>2</td>
<td>LDT</td>
<td>Yes</td>
</tr>
<tr>
<td>3</td>
<td>Busy 16-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>4</td>
<td>16-bit call gate</td>
<td>Yes</td>
</tr>
<tr>
<td>5</td>
<td>16-bit/32-bit task gate</td>
<td>Yes</td>
</tr>
<tr>
<td>6</td>
<td>16-bit trap gate</td>
<td>No</td>
</tr>
<tr>
<td>7</td>
<td>16-bit interrupt gate</td>
<td>No</td>
</tr>
<tr>
<td>8</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>9</td>
<td>Available 32-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>A</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>B</td>
<td>Busy 32-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>C</td>
<td>32-bit call gate</td>
<td>Yes</td>
</tr>
<tr>
<td>D</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>E</td>
<td>32-bit trap gate</td>
<td>No</td>
</tr>
<tr>
<td>F</td>
<td>32-bit interrupt gate</td>
<td>No</td>
</tr>
</tbody>
</table>

Operation

IF SRC(Offset) > descriptor table limit THEN ZF ← 0; FI;
Read segment descriptor;
IF SegmentDescriptor(Type) ≠ conforming code segment
AND (CPL > DPL) OR (RPL > DPL)
OR Segment type is not valid for instruction
THEN
    ZF ← 0
ELSE
    IF OperandSize = 32
        THEN
            DEST ← [SRC] AND 00FxFF00H;
        ELSE (*OperandSize = 16*)
            DEST ← [SRC] AND FF00H;
        FI;
    FI;

Flags Affected

The ZF flag is set to 1 if the access rights are loaded successfully; otherwise, it is cleared to 0.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
LAR—Load Access Rights Byte (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
   If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3. (Only occurs when fetching target from memory.)

Real Address Mode Exceptions

#UD The LAR instruction is not recognized in real address mode.

Virtual 8086 Mode Exceptions

#UD The LAR instruction cannot be executed in virtual 8086 mode.
LDS/LES/LFS/LGS/LSS—Load Far Pointer

### Description

Load a far pointer (segment selector and offset) from the second operand (source operand) into a segment register and the first operand (destination operand). The source operand specifies a 48-bit or a 32-bit pointer in memory depending on the current setting of the operand-size attribute (32 bits or 16 bits, respectively). The instruction opcode and the destination operand specify a segment register/general-purpose register pair. The 16-bit segment selector from the source operand is loaded into the segment register implied with the opcode (DS, SS, ES, FS, or GS). The 32-bit or 16-bit offset is loaded into the register specified with the destination operand.

If one of these instructions is executed in protected mode, additional information from the segment descriptor pointed to by the segment selector in the source operand is loaded in the hidden part of the selected segment register.

Also in protected mode, a null selector (values 0000 through 0003) can be loaded into DS, ES, FS, or GS registers without causing a protection exception. (Any subsequent reference to a segment whose corresponding segment register is loaded with a null selector, causes a general-protection exception (#GP) and no memory reference to the segment occurs.)

### Operation

IF ProtectedMode
THEN IF SS is loaded
   THEN IF SegmentSelector = null
      THEN #GP(0);
   FI;
ELSE IF Segment selector index is not within descriptor table limits
OR Segment selector RPL ≠ CPL
OR Access rights indicate nonwritable data segment
OR DPL ≠ CPL
   THEN #GP(selector);
   FI;
ELSE IF Segment marked not present
   THEN SS(selector);
   FI;
SS ← SegmentSelector(SRC);
SS ← SegmentDescriptor([SRC]);
ELSE IF DS, ES, FS, or GS is loaded with non-null segment selector
    THEN IF Segment selector index is not within descriptor table limits
        OR Access rights indicate segment neither data nor readable code segment
        OR (Segment is data or nonconforming-code segment
            AND both RPL and CPL > DPL)
            THEN #GP(selector);
        FI;
    ELSE IF Segment marked not present
        THEN #NP(selector);
    FI;
    SegmentRegister ← SegmentSelector(SRC) AND RPL;
    SegmentRegister ← SegmentDescriptor([SRC]);
ELSE IF DS, ES, FS or GS is loaded with a null selector:
    SegmentRegister ← NullSelector;
    SegmentRegister(DescriptorValidBit) ← 0; (*hidden flag; not accessible by software*)
FI;
FI;
IF (Real-Address or Virtual 8086 Mode)
    THEN
        SS ← SegmentSelector(SRC);
FI;
DEST ← Offset(SRC);

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults       NaT Register Consumption Abort.
Itanium Mem Faults      VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#UD        If source operand is not a memory location.
#GP(0)     If a null selector is loaded into the SS register.
            If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
            If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#GP(selector)  If the SS register is being loaded and any of the following is true: the segment selector index is not within the descriptor table limits, the segment selector RPL is not equal to CPL, the segment is a nonwritable data segment, or DPL is not equal to CPL.
LDS/LES/LFS/LGS/LSS—Load Far Pointer (continued)

If the DS, ES, FS, or GS register is being loaded with a non-null segment selector and any of the following is true: the segment selector index is not within descriptor table limits, the segment is neither a data nor a readable code segment, or the segment is a data or nonconforming-code segment and both RPL and CPL are greater than DPL.

#SS(0) If a memory operand effective address is outside the SS segment limit.
#SS(selector) If the SS register is being loaded and the segment is marked not present.
#NP(selector) If DS, ES, FS, or GS register is being loaded with a non-null segment selector and the segment is marked not present.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
#UD If source operand is not a memory location.

Virtual 8086 Mode Exceptions

#UD If source operand is not a memory location.
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
LEA—Load Effective Address

Description

Computes the effective address of the second operand (the source operand) and stores it in the first operand (destination operand). The source operand is a memory address (offset part) specified with one of the processors addressing modes; the destination operand is a general-purpose register. The address-size and operand-size attributes affect the action performed by this instruction, as shown in the following table. The operand-size attribute of the instruction is determined by the chosen register; the address-size attribute is determined by the attribute of the code segment.

Table 1-16. LEA Address and Operand Sizes

<table>
<thead>
<tr>
<th>Operand Size</th>
<th>Address Size</th>
<th>Action Performed</th>
</tr>
</thead>
<tbody>
<tr>
<td>16</td>
<td>16</td>
<td>16-bit effective address is calculated and stored in requested 16-bit register destination.</td>
</tr>
<tr>
<td>16</td>
<td>32</td>
<td>32-bit effective address is calculated. The lower 16 bits of the address are stored in the requested 16-bit register destination.</td>
</tr>
<tr>
<td>32</td>
<td>16</td>
<td>16-bit effective address is calculated. The 16-bit address is zero-extended and stored in the requested 32-bit register destination.</td>
</tr>
<tr>
<td>32</td>
<td>32</td>
<td>32-bit effective address is calculated and stored in the requested 32-bit register destination.</td>
</tr>
</tbody>
</table>

Different assemblers may use different algorithms based on the size attribute and symbolic reference of the source operand.

Operation

IF OperandSize = 16 AND AddressSize = 16
THEN
   DEST ← EffectiveAddress(SRC); (* 16-bit address *)
ELSE IF OperandSize = 16 AND AddressSize = 32
THEN
   temp ← EffectiveAddress(SRC); (* 32-bit address *)
   DEST ← temp[0..15]; (* 16-bit address *)
ELSE IF OperandSize = 32 AND AddressSize = 16
THEN
   temp ← EffectiveAddress(SRC); (* 16-bit address *)
   DEST ← ZeroExtend(temp); (* 32-bit address *)
ELSE IF OperandSize = 32 AND AddressSize = 32
THEN
   DEST ← EffectiveAddress(SRC); (* 32-bit address *)
FI;
FI;
LEA—Load Effective Address (continued)

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults   NaT Register Consumption Abort.

Protected Mode Exceptions

#UD       If source operand is not a memory location.

Real Address Mode Exceptions

#UD       If source operand is not a memory location.

Virtual 8086 Mode Exceptions

#UD       If source operand is not a memory location.
LEAVE—High Level Procedure Exit

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>C9</td>
<td>LEAVE</td>
<td>Set SP to BP, then pop BP</td>
</tr>
<tr>
<td>C9</td>
<td>LEAVE</td>
<td>Set ESP to EBP, then pop EBP</td>
</tr>
</tbody>
</table>

**Description**

Executes a return from a procedure or group of nested procedures established by an earlier ENTER instruction. The instruction copies the frame pointer (in the EBP register) into the stack pointer register (ESP), releasing the stack space used by a procedure for its local variables. The old frame pointer (the frame pointer for the calling procedure that issued the ENTER instruction) is then popped from the stack into the EBP register, restoring the calling procedure’s frame.

A RET instruction is commonly executed following a LEAVE instruction to return program control to the calling procedure and remove any arguments pushed onto the stack by the procedure being returned from.

**Operation**

IF StackAddressSize = 32
THEN
   ESP ← EBP;
ELSE (* StackAddressSize = 16*)
   SP ← BP;
FI;
IF OperandSize = 32
THEN
   EBP ← Pop();
ELSE (* OperandSize = 16*)
   BP ← Pop();
FI;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults       NaT Register Consumption Abort.
Itanium Mem Faults      VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Protected Mode Exceptions**

#SS(0)       If the EBP register points to a location that is not within the limits of the current stack segment.
LEAVE—High Level Procedure Exit (continued)

Real Address Mode Exceptions

#GP If the EBP register points to a location outside of the effective address space from 0 to 0FFFFH.

Virtual 8086 Mode Exceptions

#GP(0) If the EBP register points to a location outside of the effective address space from 0 to 0FFFFH.
LES—Load Full Pointer

See entry for LDS/LES/LFS/LGS/LSS.
LFS—Load Full Pointer

See entry for LDS/LES/LFS/LGS/LSS.
LGDT/LIDT—Load Global/Interrupt Descriptor Table Register

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 01 /2</td>
<td>LGDT m16&amp;32</td>
<td>Load m into GDTR</td>
</tr>
<tr>
<td>0F 01 /3</td>
<td>LIDT m16&amp;32</td>
<td>Load m into IDTR</td>
</tr>
</tbody>
</table>

Description

Loads the values in the source operand into the global descriptor table register (GDTR) or the interrupt descriptor table register (IDTR). The source operand is a pointer to 6 bytes of data in memory that contains the base address (a linear address) and the limit (size of table in bytes) of the global descriptor table (GDT) or the interrupt descriptor table (IDT). If operand-size attribute is 32 bits, a 16-bit limit (lower 2 bytes of the 6-byte data operand) and a 32-bit base address (upper 4 bytes of the data operand) are loaded into the register. If the operand-size attribute is 16 bits, a 16-bit limit (lower 2 bytes) and a 24-bit base address (third, fourth, and fifth byte) are loaded. Here, the high-order byte of the operand is not used and the high-order byte of the base address in the GDTR or IDTR is filled with zeros.

The LGDT and LIDT instructions are used only in operating-system software; they are not used in application programs. They are the only instructions that directly load a linear address (that is, not a segment-relative address) and a limit in protected mode. They are commonly executed in real-address mode to allow processor initialization prior to switching to protected mode.

Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,LGDT/LIDT);
IF instruction is LIDT
    THEN
        IF OperandSize = 16
            THEN
                IDTR(Limit) ← SRC[0:15];
                IDTR(Base) ← SRC[16:47] AND 00FFFFFFH;
            ELSE (* 32-bit Operand Size *)
                IDTR(Limit) ← SRC[0:15];
                IDTR(Base) ← SRC[16:47];
        FI;
        ELSE (* instruction is LGDT *)
            IF OperandSize = 16
                THEN
                    GDTR(Limit) ← SRC[0:15];
                    GDTR(Base) ← SRC[16:47] AND 00FFFFFFH;
                ELSE (* 32-bit Operand Size *)
                    GDTR(Limit) ← SRC[0:15];
                    GDTR(Base) ← SRC[16:47];
            FI;
        FI;

Flags Affected

None.
LGDT/LIDT—Load Global/Interrupt Descriptor Table Register (continued)

Additional Itanium System Environment Exceptions
IA-32_Intercept  Mandatory Instruction Intercept for LIDT and LGDT

Protected Mode Exceptions
#UD  If source operand is not a memory location.
#GP(0)  If the current privilege level is not 0.
        If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
        If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.

Real Address Mode Exceptions
#UD  If source operand is not a memory location.
#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS  If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions
#UD  If source operand is not a memory location.
#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
LGS—Load Full Pointer

See entry for LDS/LES/LFS/LGS/LSS.
LLDT—Load Local Descriptor Table Register

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 00 /2</td>
<td>LLDT r/m16</td>
<td>Load segment selector r/m16 into LDTR</td>
</tr>
</tbody>
</table>

### Description

Loads the source operand into the segment selector field of the local descriptor table register (LDTR). The source operand (a general-purpose register or a memory location) contains a segment selector that points to a local descriptor table (LDT). After the segment selector is loaded in the LDTR, the processor uses the segment selector to locate the segment descriptor for the LDT in the global descriptor table (GDT). It then loads the segment limit and base address for the LDT from the segment descriptor into the LDTR. The segment registers DS, ES, SS, FS, GS, and CS are not affected by this instruction, nor is the LDTR field in the task state segment (TSS) for the current task.

If the source operand is 0, the LDTR is marked invalid and all references to descriptors in the LDT (except by the LAR, VERR, VERW or LSL instructions) cause a general protection exception (#GP).

The operand-size attribute has no effect on this instruction.

The LLDT instruction is provided for use in operating-system software; it should not be used in application programs. Also, this instruction can only be executed in protected mode.

### Operation

**IF Itanium System Environment THEN IA-32_Interceptor(INST,LLDT);**

**IF SRC(Offset) > descriptor table limit THEN #GP(segment selector); FI;**

Read segment descriptor;

**IF SegmentDescriptor(Type) ≠ LDT THEN #GP(segment selector); FI;**

**IF segment descriptor is not present THEN #NP(segment selector);**

LDTR(SegmentSelector) ← SRC;

LDTR(SegmentDescriptor) ← GDTSegmentDescriptor;

### Flags Affected

None.

### Additional Itanium System Environment Exceptions

IA-32_Interceptor Instruction Intercept

### Protected Mode Exceptions

**#GP(0)** If the current privilege level is not 0.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register contains a null segment selector.
LLDT—Load Local Descriptor Table Register (continued)

#GP(selector)  If the selector operand does not point into the Global Descriptor Table or if the entry in the GDT is not a Local Descriptor Table.
   Segment selector is beyond GDT limit.
#SS(0)        If a memory operand effective address is outside the SS segment limit.
#NP(selector)  If the LDT descriptor is not present.
#PF(fault-code) If a page fault occurs.

Real Address Mode Exceptions
#UD            The LLDT instruction is not recognized in real address mode.

Virtual 8086 Mode Exceptions
#UD            The LLDT instruction is recognized in virtual 8086 mode.
**LIDT—Load Interrupt Descriptor Table Register**

See entry for LGDT/LIDT—Load Global Descriptor Table Register/Load Interrupt Descriptor Table Register.
LMSW—Load Machine Status Word

Description

Loads the source operand into the machine status word, bits 0 through 15 of register CR0. The source operand can be a 16-bit general-purpose register or a memory location. Only the low-order 4 bits of the source operand (which contains the PE, MP, EM, and TS flags) are loaded into CR0. The PG, CD, NW, AM, WP, NE, and ET flags of CR0 are not affected. The operand-size attribute has no effect on this instruction.

If the PE flag of the source operand (bit 0) is set to 1, the instruction causes the processor to switch to protected mode. The PE flag in the CR0 register is a sticky bit. Once set to 1, the LMSW instruction cannot be used clear this flag and force a switch back to real address mode.

The LMSW instruction is provided for use in operating-system software; it should not be used in application programs. In protected or virtual 8086 mode, it can only be executed at CPL 0.

This instruction is provided for compatibility with the Intel 286 processor; programs and procedures intended to run on processors more recent than the Intel 286 should use the MOV (control registers) instruction to load the machine status word.

This instruction is a serializing instruction.

Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,LMSW);

CR0[0:3] ← SRC[0:3];

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Intercept Mandatory Instruction Intercept

Protected Mode Exceptions

#GP(0) If the current privilege level is not 0.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 01 /6</td>
<td>LMSW r/m16</td>
<td>Loads r/m16 in machine status word of CR0</td>
</tr>
</tbody>
</table>
LMSW—Load Machine Status Word (continued)

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

Virtual 8086 Mode Exceptions

#GP(0)  If the current privilege level is not 0.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#PF(fault-code)  If a page fault occurs.
LOCK—Assert LOCK# Signal Prefix

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F0</td>
<td>LOCK</td>
<td>Asserts LOCK# signal for duration of the accompanying instruction</td>
</tr>
</tbody>
</table>

**Description**

Causes the processor’s LOCK# signal to be asserted during execution of the accompanying instruction (turns the instruction into an atomic instruction). In a multiprocessor environment, the LOCK# signal insures that the processor has exclusive use of any shared memory while the signal is asserted.

The LOCK prefix can be prepended only to the following instructions and to those forms of the instructions that use a memory operand: ADD, ADC, AND, BTC, BTR, BTS, CMPXCHG, DEC, INC, NEG, NOT, OR, SBB, SUB, XOR, XADD, and XCHG. An undefined opcode exception will be generated if the LOCK prefix is used with any other instruction. The XCHG instruction always asserts the LOCK# signal regardless of the presence or absence of the LOCK prefix.

The LOCK prefix is typically used with the BTS instruction to perform a read-modify-write operation on a memory location in shared memory environment.

The integrity of the LOCK prefix is not affected by the alignment of the memory field. Memory locking is observed for arbitrarily misaligned fields.

**Operation**

\[
\text{IF Itanium System Environment AND External_Bus_Lock_Required AND DCR.lc} \\
\text{THEN IA-32\_Intercept(LOCK);} \\
\text{AssertLOCK#(DurationOfAccompanyingInstruction)}
\]

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
- **IA-32\_Intercept**: Lock Intercept - If an external atomic bus lock is required to complete this operation and DCR.lc is 1, no atomic transaction occurs, the instruction is faulted and an IA-32\_Intercept(Lock) fault is generated. The software lock handler is responsible for the emulation of the instruction.

**Protected Mode Exceptions**

- **#UD**: If the LOCK prefix is used with an instruction not listed in the “Description” section above. Other exceptions can be generated by the instruction that the LOCK prefix is being applied to.
LOCK—Assert LOCK# Signal Prefix (continued)

Real Address Mode Exceptions

#UD If the LOCK prefix is used with an instruction not listed in the “Description” section above. Other exceptions can be generated by the instruction that the LOCK prefix is being applied to.

Virtual 8086 Mode Exceptions

#UD If the LOCK prefix is used with an instruction not listed in the “Description” section above. Other exceptions can be generated by the instruction that the LOCK prefix is being applied to.
LODS/LODSB/LODSW/LODSD—Load String Operand

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>AC</td>
<td>LODS DS:(E)SI</td>
<td>Load byte at address DS:(E)SI into AL</td>
</tr>
<tr>
<td>AD</td>
<td>LODS DS:SI</td>
<td>Load word at address DS:SI into AX</td>
</tr>
<tr>
<td>AD</td>
<td>LODS DS:ESI</td>
<td>Load doubleword at address DS:ESI into EAX</td>
</tr>
<tr>
<td>AC</td>
<td>LODSB</td>
<td>Load byte at address DS:(E)SI into AL</td>
</tr>
<tr>
<td>AD</td>
<td>LODSW</td>
<td>Load word at address DS:SI into AX</td>
</tr>
<tr>
<td>AD</td>
<td>LODSD</td>
<td>Load doubleword at address DS:ESI into EAX</td>
</tr>
</tbody>
</table>

**Description**

Load a byte, word, or doubleword from the source operand into the AL, AX, or EAX register, respectively. The source operand is a memory location at the address DS:ESI. (When the operand-size attribute is 16, the SI register is used as the source-index register.) The DS segment may be overridden with a segment override prefix.

The LODSB, LODSW, and LODSD mnemonics are synonyms of the byte, word, and doubleword versions of the LODS instructions. (For the LODS instruction, “DS:ESI” must be explicitly specified in the instruction.)

After the byte, word, or doubleword is transfer from the memory location into the AL, AX, or EAX register, the ESI register is incremented or decremented automatically according to the setting of the DF flag in the EFLAGS register. (If the DF flag is 0, the ESI register is incremented; if the DF flag is 1, the ESI register is decremented.) The ESI register is incremented or decremented by 1 for byte operations, by 2 for word operations, or by 4 for doubleword operations.

The LODS, LODSB, LODSW, and LODSD instructions can be preceded by the REP prefix for block loads of ECX bytes, words, or doublewords. More often, however, these instructions are used within a LOOP construct, because further processing of the data moved into the register is usually necessary before the next transfer can be made. See “REP/REPE/REPZ/REPNE/REPNZ—Repeat String Operation Prefix” on page 3:685 for a description of the REP prefix.

**Operation**

\[
\text{IF (byte load)} \\
\text{THEN} \\
\quad \text{AL} \leftarrow \text{SRC}; \text{(* byte load *)} \\
\quad \text{THEN IF DF = 0} \\
\quad \quad \text{THEN (E)SI} \leftarrow 1; \\
\quad \quad \text{ELSE (E)SI} \leftarrow -1; \\
\quad \text{FI}; \\
\text{ELSE IF (word load)} \\
\text{THEN} \\
\quad \text{AX} \leftarrow \text{SRC}; \text{(* word load *)} \\
\quad \text{THEN IF DF = 0} \\
\quad \quad \text{THEN SI} \leftarrow 2; \\
\quad \quad \text{ELSE SI} \leftarrow -2; \\
\quad \text{FI}; \\
\text{ELSE (* doubleword transfer *)} \\
\quad \text{EAX} \leftarrow \text{SRC}; \text{(* doubleword load *)} \\
\quad \text{THEN IF DF = 0} 
\]
LODS/LODSB/LODSW/LODSD—Load String Operand (continued)

    THEN ESI ← 4;
    ELSE ESI ← −4;
    FI;
    FI;
    FI;

Flags Affected
None.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

    If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
LOOP/LOOPcc—Loop According to ECX Counter

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>E2 cb</td>
<td>LOOP rel8</td>
<td>Decrement count; jump short if count ≠ 0</td>
</tr>
<tr>
<td>E1 cb</td>
<td>LOOPE rel8</td>
<td>Decrement count; jump short if count ≠ 0 and ZF=1</td>
</tr>
<tr>
<td>E1 cb</td>
<td>LOOPZ rel8</td>
<td>Decrement count; jump short if count ≠ 0 and ZF=1</td>
</tr>
<tr>
<td>E0 cb</td>
<td>LOOPNE rel8</td>
<td>Decrement count; jump short if count ≠ 0 and ZF=0</td>
</tr>
<tr>
<td>E0 cb</td>
<td>LOOPNZ rel8</td>
<td>Decrement count; jump short if count ≠ 0 and ZF=0</td>
</tr>
</tbody>
</table>

**Description**

Performs a loop operation using the ECX or CX register as a counter. Each time the LOOP instruction is executed, the count register is decremented, then checked for 0. If the count is 0, the loop is terminated and program execution continues with the instruction following the LOOP instruction. If the count is not zero, a near jump is performed to the destination (target) operand, which is presumably the instruction at the beginning of the loop. If the address-size attribute is 32 bits, the ECX register is used as the count register; otherwise the CX register is used.

The target instruction is specified with a relative offset (a signed offset relative to the current value of the instruction pointer in the EIP register). This offset is generally specified as a label in assembly code, but at the machine code level, it is encoded as a signed, 8-bit immediate value, which is added to the instruction pointer. Offsets of −128 to +127 are allowed with this instruction.

Some forms of the loop instruction (LOOPcc) also accept the ZF flag as a condition for terminating the loop before the count reaches zero. With these forms of the instruction, a condition code (cc) is associated with each instruction to indicate the condition being tested for. Here, the LOOPcc instruction itself does not affect the state of the ZF flag; the ZF flag is changed by other instructions in the loop.

All branches are converted to code fetches of one or two cache lines, regardless of jump address or cacheability.

**Operation**

```
IF AddressSize = 32
    THEN
        Count is ECX;
    ELSE (* AddressSize = 16 *)
        Count is CX;
    FI;
    Count ← Count – 1;

IF instruction in not LOOP
    THEN
        IF (instruction = LOOPE) OR (instruction = LOOPZ)
            THEN
                IF (ZF =1) AND (Count ≠ 0)
                    THEN BranchCond ← 1;
                        ELSE BranchCond ← 0;
                    FI;
                FI;
        IF (instruction = LOOPNE) OR (instruction = LOOPNZ)
```
LOOP/LOOPcc—Loop According to ECX Counter (continued)

THEN
   IF (ZF = 0) AND (Count ≠ 0)
      THEN BranchCond ← 1;
      ELSE BranchCond ← 0;
   FI;
ELSE (* instruction = LOOP *)
   IF (Count ≠ 0)
      THEN BranchCond ← 1;
      ELSE BranchCond ← 0;
   FI;
ELSE BranchCond = 1
   THEN
      EIP ← EIP + SignExtend(DEST);
      IF OperandSize = 16
         THEN
            EIP ← EIP AND 0000FFFFH;
         FI;
      IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
      ELSE
         Terminate loop and continue program execution at EIP;
      FI;
   FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.
IA-32_Exception  Taken Branch Debug Exception if PSR.tb is 1

Protected Mode Exceptions

#GP(0)  If the offset jumped to is beyond the limits of the code segment.

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

None.
LSL—Load Segment Limit

**Description**

Loads the unscrambled segment limit from the segment descriptor specified with the second operand (source operand) into the first operand (destination operand) and sets the ZF flag in the EFLAGS register. The source operand (which can be a register or a memory location) contains the segment selector for the segment descriptor being accessed. The destination operand is a general-purpose register.

The processor performs access checks as part of the loading process. Once loaded in the destination register, software can compare the segment limit with the offset of a pointer.

The segment limit is a 20-bit value contained in bytes 0 and 1 and in the first 4 bits of byte 6 of the segment descriptor. If the descriptor has a byte granular segment limit (the granularity flag is set to 0), the destination operand is loaded with a byte granular value (byte limit). If the descriptor has a page granular segment limit (the granularity flag is set to 1), the LSL instruction will translate the page granular limit (page limit) into a byte limit before loading it into the destination operand. The translation is performed by shifting the 20-bit “raw” limit left 12 bits and filling the low-order 12 bits with 1s.

When the operand size is 32 bits, the 32-bit byte limit is stored in the destination operand. When the operand size is 16 bits, a valid 32-bit limit is computed; however, the upper 16 bits are truncated and only the low-order 16 bits are loaded into the destination operand.

This instruction performs the following checks before it loads the segment limit into the destination register:

- Checks that the segment selector is not null.
- Checks that the segment selector points to a descriptor that is within the limits of the GDT or LDT being accessed.
- Checks that the descriptor type is valid for this instruction. All code and data segment descriptors are valid for (can be accessed with) the LSL instruction. The valid special segment and gate descriptor types are given in the following table.
- If the segment is not a conforming code segment, the instruction checks that the specified segment descriptor is visible at the CPL (that is, if the CPL and the RPL of the segment selector are less than or equal to the DPL of the segment selector).

If the segment descriptor cannot be accessed or is an invalid type for the instruction, the ZF flag is cleared and no value is loaded in the destination operand.

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 03 /r</td>
<td>LSL r16,r/m16</td>
<td>Load: r16 ← segment limit, selector r/m16</td>
</tr>
<tr>
<td>0F 03 /r</td>
<td>LSL r32,r/m32</td>
<td>Load: r32 ← segment limit, selector r/m32</td>
</tr>
</tbody>
</table>
LSL—Load Segment Limit (continued)

<table>
<thead>
<tr>
<th>Type</th>
<th>Name</th>
<th>Valid</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>1</td>
<td>Available 16-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>2</td>
<td>LDT</td>
<td>Yes</td>
</tr>
<tr>
<td>3</td>
<td>Busy 16-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>4</td>
<td>16-bit call gate</td>
<td>No</td>
</tr>
<tr>
<td>5</td>
<td>16-bit/32-bit task gate</td>
<td>No</td>
</tr>
<tr>
<td>6</td>
<td>16-bit trap gate</td>
<td>No</td>
</tr>
<tr>
<td>7</td>
<td>16-bit interrupt gate</td>
<td>No</td>
</tr>
<tr>
<td>8</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>9</td>
<td>Available 32-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>A</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>B</td>
<td>Busy 32-bit TSS</td>
<td>Yes</td>
</tr>
<tr>
<td>C</td>
<td>32-bit call gate</td>
<td>No</td>
</tr>
<tr>
<td>D</td>
<td>Reserved</td>
<td>No</td>
</tr>
<tr>
<td>E</td>
<td>32-bit trap gate</td>
<td>No</td>
</tr>
<tr>
<td>F</td>
<td>32-bit interrupt gate</td>
<td>No</td>
</tr>
</tbody>
</table>

Operation

- IF SRC(Offset) > descriptor table limit
  THEN ZF ← 0; FI;
- Read segment descriptor;
- IF SegmentDescriptor(Type) ≠ conforming code segment
  AND (CPL > DPL) OR (RPL > DPL)
  OR Segment type is not valid for instruction
  THEN
    ZF ← 0
  ELSE
    temp ← SegmentLimit([SRC]);
    IF (G = 1)
      THEN
        temp ← ShiftLeft(12, temp) OR 00000FFFH;
      FI;
    IF OperandSize = 32
      THEN
        DEST ← temp;
      ELSE (*OperandSize = 16*)
        DEST ← temp AND FFFFH;
      FI;
    FI;

Flags Affected

The ZF flag is set to 1 if the segment limit is loaded successfully; otherwise, it is cleared to 0.
**LSL—Load Segment Limit (continued)**

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

**Protected Mode Exceptions**

<table>
<thead>
<tr>
<th>Code</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP(0)</td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit. If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.</td>
</tr>
<tr>
<td>#SS(0)</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.</td>
</tr>
</tbody>
</table>

**Real Address Mode Exceptions**

<table>
<thead>
<tr>
<th>Code</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#UD</td>
<td>The LSL instruction is not recognized in real address mode.</td>
</tr>
</tbody>
</table>

**Virtual 8086 Mode Exceptions**

<table>
<thead>
<tr>
<th>Code</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#UD</td>
<td>The LSL instruction is not recognized in virtual 8086 mode.</td>
</tr>
</tbody>
</table>
LSS—Load Full Pointer

See entry for LDS/LES/LFS/LGS/LSS.
LTR—Load Task Register

### Description

Loads the source operand into the segment selector field of the task register. The source operand (a general-purpose register or a memory location) contains a segment selector that points to a task state segment (TSS). After the segment selector is loaded in the task register, the processor uses to segment selector to locate the segment descriptor for the TSS in the global descriptor table (GDT). It then loads the segment limit and base address for the TSS from the segment descriptor into the task register. The task pointed to by the task register is marked busy, but a switch to the task does not occur.

The LTR instruction is provided for use in operating-system software; it should not be used in application programs. It can only be executed in protected mode when the CPL is 0. It is commonly used in initialization code to establish the first task to be executed.

The operand-size attribute has no effect on this instruction.

### Operation

**IF Itanium System Environment THEN IA-32_Intercept(INST, LTR);**

IF SRC(Offset) > descriptor table limit OR IF SRC(type) ≠ global
   THEN #GP(segment selector);
FI;
Reat segment descriptor;
IF segment descriptor is not for an available TSS THEN #GP(segment selector); FI;
IF segment descriptor is not present THEN #NP(segment selector);
TSSsegmentDescriptor(busy) ← 1;
(* Locked read-modify-write operation on the entire descriptor when setting busy flag *)
TaskRegister(SegmentSelector) ← SRC;
TaskRegister(SegmentDescriptor) ← TSSSegmentDescriptor;

### Flags Affected

None.

### Additional Itanium System Environment Exceptions

IA-32_Intercept   Mandatory Instruction Intercept.

### Protected Mode Exceptions

#GP(0)       If the current privilege level is not 0.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
LTR—Load Task Register (continued)

#GP(selector) If the source selector points to a segment that is not a TSS or to one for a task that is already busy.

If the selector points to LDT or is beyond the GDT limit.

#NP(selector) If the TSS is marked not present.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

Real Address Mode Exceptions

#UD The LTR instruction is not recognized in real address mode.

Virtual 8086 Mode Exceptions

#UD The LTR instruction is not recognized in virtual 8086 mode.
## MOV—Move

### Description

Copies the second operand (source operand) to the first operand (destination operand). The source operand can be an immediate value, general-purpose register, segment register, or memory location; the destination register can be a general-purpose register, segment register, or memory location. Both operands must be the same size, which can be a byte, a word, or a doubleword.

The MOV instruction cannot be used to load the CS register. Attempting to do so results in an invalid opcode exception (#UD). To load the CS register, use the RET instruction.

### Opcode

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>88 / r</td>
<td>MOV r/m8,r8</td>
<td>Move r8 to r/m8</td>
</tr>
<tr>
<td>89 / r</td>
<td>MOV r/m16,r16</td>
<td>Move r16 to r/m16</td>
</tr>
<tr>
<td>89 / r</td>
<td>MOV r/m32,32</td>
<td>Move r32 to r/m32</td>
</tr>
<tr>
<td>8A / r</td>
<td>MOV r8,r/m8</td>
<td>Move r/m8 to r8</td>
</tr>
<tr>
<td>8B / r</td>
<td>MOV r16,r/m16</td>
<td>Move r/m16 to r16</td>
</tr>
<tr>
<td>8B / r</td>
<td>MOV r32,r/m32</td>
<td>Move r/m32 to r32</td>
</tr>
<tr>
<td>8C / r</td>
<td>MOV r/m16,Sreg**</td>
<td>Move segment register to r/m16</td>
</tr>
<tr>
<td>8E / r</td>
<td>MOV Sreg,r/m16</td>
<td>Move r/m16 to segment register</td>
</tr>
<tr>
<td>A0</td>
<td>MOV AL,moffs8*</td>
<td>Move byte at (seg:offset) to AL</td>
</tr>
<tr>
<td>A1</td>
<td>MOV AX,moffs16*</td>
<td>Move word at (seg:offset) to AX</td>
</tr>
<tr>
<td>A1</td>
<td>MOV EAX,moffs32*</td>
<td>Move doubleword at (seg:offset) to EAX</td>
</tr>
<tr>
<td>A2</td>
<td>MOV moffs8*,AL</td>
<td>Move AL to (seg:offset)</td>
</tr>
<tr>
<td>A3</td>
<td>MOV moffs16*,AX</td>
<td>Move AX to (seg:offset)</td>
</tr>
<tr>
<td>A3</td>
<td>MOV moffs32*,EAX</td>
<td>Move EAX to (seg:offset)</td>
</tr>
<tr>
<td>B0+ rb</td>
<td>MOV r8,imm8</td>
<td>Move imm8 to r8</td>
</tr>
<tr>
<td>B8+ rw</td>
<td>MOV r16,imm16</td>
<td>Move imm16 to r16</td>
</tr>
<tr>
<td>B8+ rd</td>
<td>MOV r32,imm32</td>
<td>Move imm32 to r32</td>
</tr>
<tr>
<td>C6 / 0</td>
<td>MOV r/m8,imm8</td>
<td>Move imm8 to r/m8</td>
</tr>
<tr>
<td>C7 / 0</td>
<td>MOV r/m16,imm16</td>
<td>Move imm16 to r/m16</td>
</tr>
<tr>
<td>C7 / 0</td>
<td>MOV r/m32,imm32</td>
<td>Move imm32 to r/m32</td>
</tr>
</tbody>
</table>

### Notes:

* The `moffs8`, `moffs16`, and `moffs32` operands specify a simple offset relative to the segment base, where 8, 16, and 32 refer to the size of the data. The address-size attribute of the instruction determines the size of the offset, either 16 or 32 bits.

** In 32-bit mode, the assembler may require the use of the 16-bit operand size prefix (a byte with the value 66H preceding the instruction).
MOV—Move (continued)

If the destination operand is a segment register (DS, ES, FS, GS, or SS), the source operand must be a valid segment selector. In protected mode, moving a segment selector into a segment register automatically causes the segment descriptor information associated with that segment selector to be loaded into the hidden (shadow) part of the segment register. While loading this information, the segment selector and segment descriptor information is validated (see the “Operation” algorithm below). The segment descriptor data is obtained from the GDT or LDT entry for the specified segment selector.

A null segment selector (values 0000-0003) can be loaded into the DS, ES, FS, and GS registers without causing a protection exception. However, any subsequent attempt to reference a segment whose corresponding segment register is loaded with a null value causes a general protection exception (#GP) and no memory reference occurs.

Loading the SS register with a MOV instruction inhibits all external interrupts and traps until after the execution of the next instruction in the IA-32 System Environment. For the Itanium System Environment, MOV to SS results in an IA-32_Intercept(SystemFlag) trap after the instruction completes. This operation allows a stack pointer to be loaded into the ESP register with the next instruction (MOV ESP, stack-pointer value) before an interrupt occurs. The LSS instruction offers a more efficient method of loading the SS and ESP registers.

When moving data in 32-bit mode between a segment register and a 32-bit general-purpose register, the Pentium Pro processor does not require the use of a 16-bit operand size prefix; however, some assemblers do require this prefix. The processor assumes that the sixteen least-significant bits of the general-purpose register are the destination or source operand. When moving a value from a segment selector to a 32-bit register, the processor fills the two high-order bytes of the register with zeros.

Operation

DEST ← SRC;

Loading a segment register while in protected mode results in special checks and actions, as described in the following listing. These checks are performed on the segment selector and the segment descriptor it points to.

IF SS is loaded;
    THEN
        IF segment selector is null
            THEN #GP(0);
        FI;
        IF segment selector index is outside descriptor table limits
            OR segment selector’s RPL ≠ CPL
            OR segment is not a writable data segment
            OR DPL ≠ CPL
            THEN #GP(selector);
        FI;
        IF segment not marked present
            THEN #SS(selector);
    ELSE
        SS ← segment selector;
        SS ← segment descriptor;
    FI;
MOV—Move (continued)

FI;
IF DS, ES, FS or GS is loaded with non-null selector;

THEN
 IF segment selector index is outside descriptor table limits
 OR segment is not a data or readable code segment
 OR ((segment is a data or nonconforming code segment)
    AND (both RPL and CPL > DPL))
 THEN #GP(selector);
 IF segment not marked present
 THEN #NP(selector);
 ELSE
  SegmentRegister ← segment selector;
  SegmentRegister ← segment descriptor;
 FI;
FI;
IF DS, ES, FS or GS is loaded with a null selector;
 THEN
  SegmentRegister ← null segment selector;
  SegmentRegister ← null segment descriptor;
 FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Intercept System Flag Intercept trap for Move to SS
Itanium Reg Faults NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If attempt is made to load SS register with null segment selector.
If the destination operand is in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.

#GP(selector) If segment selector index is outside descriptor table limits.
If the SS register is being loaded and the segment selector’s RPL and the segment descriptor’s DPL are not equal to the CPL.
If the SS register is being loaded and the segment pointed to is a nonwritable data segment.
If the DS, ES, FS, or GS register is being loaded and the segment pointed to is not a data or readable code segment.
MOV—Move (continued)

If the DS, ES, FS, or GS register is being loaded and the segment pointed to is a data or nonconforming code segment, but both the RPL and the CPL are greater than the DPL.

- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#SS(selector)** If the SS register is being loaded and the segment pointed to is marked not present.
- **#NP** If the DS, ES, FS, or GS register is being loaded and the segment pointed to is marked not present.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
- **#UD** If attempt is made to load the CS register.

### Real Address Mode Exceptions

- **#GP** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS** If a memory operand effective address is outside the SS segment limit.
- **#UD** If attempt is made to load the CS register.

### Virtual 8086 Mode Exceptions

- **#GP(0)** If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)** If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)** If a page fault occurs.
- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made.
- **#UD** If attempt is made to load the CS register.
MOV—Move to/from Control Registers

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 22 /r</td>
<td>MOV CR0, r32</td>
<td>Move r32 to CR0</td>
</tr>
<tr>
<td>0F 22 /r</td>
<td>MOV CR2, r32</td>
<td>Move r32 to CR2</td>
</tr>
<tr>
<td>0F 22 /r</td>
<td>MOV CR3, r32</td>
<td>Move r32 to CR3</td>
</tr>
<tr>
<td>0F 22 /r</td>
<td>MOV CR4, r32</td>
<td>Move r32 to CR4</td>
</tr>
<tr>
<td>0F 20 /r</td>
<td>MOV r32,CR0</td>
<td>Move CR0 to r32</td>
</tr>
<tr>
<td>0F 20 /r</td>
<td>MOV r32,CR2</td>
<td>Move CR2 to r32</td>
</tr>
<tr>
<td>0F 20 /r</td>
<td>MOV r32,CR3</td>
<td>Move CR3 to r32</td>
</tr>
<tr>
<td>0F 20 /r</td>
<td>MOV r32,CR4</td>
<td>Move CR4 to r32</td>
</tr>
</tbody>
</table>

Description

Moves the contents of a control register (CR0, CR2, CR3, or CR4) to a general-purpose register or vice versa. The operand size for these instructions is always 32 bits, regardless of the operand-size attribute. (See the Intel Architecture Software Developer’s Manual, Volume 3 for a detailed description of the flags and fields in the control registers.)

When loading a control register, a program should not attempt to change any of the reserved bits; that is, always set reserved bits to the value previously read.

At the opcode level, the reg field within the ModR/M byte specifies which of the control registers is loaded or read. The 2 bits in the mod field are always 11B. The r/m field specifies the general-purpose register loaded or read.

These instructions have the following side effects:

- When writing to control register CR3, all non-global TLB entries are flushed (see the Intel Architecture Software Developer’s Manual, Volume 3).
- When modifying any of the paging flags in the control registers (PE and PG in register CR0 and PGE, PSE, and PAE in register CR4), all TLB entries are flushed, including global entries. This operation is implementation specific for the Pentium Pro processor. Software should not depend on this functionality in future Intel Architecture processors.
- If the PG flag is set to 1 and control register CR4 is written to set the PAE flag to 1 (to enable the physical address extension mode), the pointers (PDPTRs) in the page-directory pointers table will be loaded into the processor (into internal, non-architectural registers).
- If the PAE flag is set to 1 and the PG flag set to 1, writing to control register CR3 will cause the PDPTRs to be reloaded into the processor.
- If the PAE flag is set to 1 and control register CR0 is written to set the PG flag, the PDPTRs are reloaded into the processor.

Operation

IF Itanium System Environment AND Move To CR Form THEN IA-32_Intercept(INST,MOVCR);
DEST ← SRC;

Flags Affected

The OF, SF, ZF, AF, PF, and CF flags are undefined.
MOV—Move to/from Control Registers (continued)

Additional Itanium System Environment Exceptions

IA-32_Interceptor  Move To CR#, Mandatory Instruction Intercept.

Move From CR#, read the virtualized control register values, CR0{15:6}
return zeros.

Protected Mode Exceptions

#GP(0)  If the current privilege level is not 0.

If an attempt is made to write a 1 to any reserved bit in CR4.

If an attempt is made to write reserved bits in the page-directory pointers table
(used in the extended physical addressing mode) when the PAE flag in control
register CR4 and the PG flag in control register CR0 are set to 1.

Real Address Mode Exceptions

#GP  If an attempt is made to write a 1 to any reserved bit in CR4.

Virtual 8086 Mode Exceptions

#GP(0)  These instructions cannot be executed in virtual 8086 mode.
MOV—Move to/from Debug Registers

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 21/r</td>
<td>MOV r32, DR0-DR3</td>
<td>Move debug registers to r32</td>
</tr>
<tr>
<td>0F 21/r</td>
<td>MOV r32, DR4-DR5</td>
<td>Move debug registers to r32</td>
</tr>
<tr>
<td>0F 21/r</td>
<td>MOV r32, DR6-DR7</td>
<td>Move debug registers to r32</td>
</tr>
<tr>
<td>0F 23/r</td>
<td>MOV DR0-DR3, r32</td>
<td>Move r32 to debug registers</td>
</tr>
<tr>
<td>0F 23/r</td>
<td>MOV DR4-DR5, r32</td>
<td>Move r32 to debug registers</td>
</tr>
<tr>
<td>0F 23/r</td>
<td>MOV DR6-DR7, r32</td>
<td>Move r32 to debug registers</td>
</tr>
</tbody>
</table>

Description

Moves the contents of two or more debug registers (DR0 through DR3, DR4 and DR5, or DR6 and DR7) to a general-purpose register or vice versa. The operand size for these instructions is always 32 bits, regardless of the operand-size attribute. (See the Intel Architecture Software Developer’s Manual, Volume 3 for a detailed description of the flags and fields in the debug registers.)

The instructions must be executed at privilege level 0 or in real-address mode.

When the debug extension (DE) flag in register CR4 is clear, these instructions operate on debug registers in a manner that is compatible with Intel386™ and Intel486 processors. In this mode, references to DR4 and DR5 refer to DR6 and DR7, respectively. When the DE set in CR4 is set, attempts to reference DR4 and DR5 result in an undefined opcode (#UD) exception.

At the opcode level, the reg field within the ModR/M byte specifies which of the debug registers is loaded or read. The two bits in the mod field are always 11. The r/m field specifies the general-purpose register loaded or read.

Operation

IF Itanium System Environment THEN IA-32_Interceptor(INST,MOVDR);

IF ((DE = 1) and (SRC or DEST = DR4 or DR5))
THEN
   #UD;
ELSE
   DEST ← SRC;

Flags Affected

The OF, SF, ZF, AF, PF, and CF flags are undefined.

Additional Itanium System Environment Exceptions

IA-32_Interceptor  Mandatory Instruction Intercept.

Protected Mode Exceptions

#GP(0)  If the current privilege level is not 0.
#UD    If the DE (debug extensions) bit of CR4 is set and a MOV instruction is executed involving DR4 or DR5.
MOV—Move to/from Debug Registers (continued)

#DB If any debug register is accessed while the GD flag in debug register DR7 is set.

Real Address Mode Exceptions

#UD If the DE (debug extensions) bit of CR4 is set and a MOV instruction is executed involving DR4 or DR5.

#DB If any debug register is accessed while the GD flag in debug register DR7 is set.

Virtual 8086 Mode Exceptions

#GP(0) The debug registers cannot be loaded or read when in virtual 8086 mode.
MOVS/MOVSB/MOVSW/MOVSD—Move Data from String to String

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>A4</td>
<td>MOVS ES:(E)DI, DS:(E)SI</td>
<td>Move byte at address DS:(E)SI to address ES:(E)DI</td>
</tr>
<tr>
<td>A5</td>
<td>MOVS ES:DI, DS:SI</td>
<td>Move word at address DS:SI to address ES:DI</td>
</tr>
<tr>
<td>A5</td>
<td>MOVS EDI, DS:ESI</td>
<td>Move doubleword at address DS:ESI to address ES:EDI</td>
</tr>
<tr>
<td>A4</td>
<td>MOVSB</td>
<td>Move byte at address DS:(E)SI to address ES:(E)DI</td>
</tr>
<tr>
<td>A5</td>
<td>MOVSW</td>
<td>Move word at address DS:SI to address ES:DI</td>
</tr>
<tr>
<td>A5</td>
<td>MOVSD</td>
<td>Move doubleword at address DS:ESI to address ES:EDI</td>
</tr>
</tbody>
</table>

Description

Moves the byte, word, or doubleword specified with the second operand (source operand) to the location specified with the first operand (destination operand). The source operand specifies the memory location at the address DS:ESI and the destination operand specifies the memory location at address ES:EDI. (When the operand-size attribute is 16, the SI and DI register are used as the source-index and destination-index registers, respectively.) The DS segment may be overridden with a segment override prefix, but the ES segment cannot be overridden.

The MOVSB, MOVSW, and MOVSD mnemonics are synonyms of the byte, word, and doubleword versions of the MOVS instructions. They are simpler to use, but provide no type or segment checking. (For the MOVS instruction, “DS:ESI” and “ES:EDI” must be explicitly specified in the instruction.)

After the transfer, the ESI and EDI registers are incremented or decremented automatically according to the setting of the DF flag in the EFLAGS register. (If the DF flag is 0, the ESI and EDI register are incremented; if the DF flag is 1, the ESI and EDI registers are decremented.) The registers are incremented or decremented by 1 for byte operations, by 2 for word operations, or by 4 for doubleword operations.

The MOVS, MOVSB, MOVSW, and MOVSD instructions can be preceded by the REP prefix (see “REP/REPE/REPZ/REPNZ/REPNE/REPNZ—Repeat Following String Operation” on page 3:685) for block moves of ECX bytes, words, or doublewords.

Operation

```
DEST ← SRC;
IF (byte move)
  THEN IF DF = 0
    THEN (E)DI ← 1;
    ELSE (E)DI ← −1;
  FI;
ELSE IF (word move)
  THEN IF DF = 0
    THEN DI ← 2;
    ELSE DI ← −2;
```
MOVS/MOVSB/MOVSW/MOVSD—Move Data from String to String
(continued)

FI;
ELSE (* doubleword move*)
    THEN IF DF = 0
        THEN EDI ← 4;
        ELSE EDI ← −4;
    FI;
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS  If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
MOVSX—Move with Sign-Extension

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F BE /r</td>
<td>MOVSX r16, /rm8</td>
<td>Move byte to word with sign-extension</td>
</tr>
<tr>
<td>0F BE /r</td>
<td>MOVSX r32, /rm8</td>
<td>Move byte to doubleword, sign-extension</td>
</tr>
<tr>
<td>0F BF /r</td>
<td>MOVSX r32, /rm16</td>
<td>Move word to doubleword, sign-extension</td>
</tr>
</tbody>
</table>

Description
Copies the contents of the source operand (register or memory location) to the destination operand (register) and sign extends the value to 16 or 32 bits. The size of the converted value depends on the operand-size attribute.

Operation
DEST ← SignExtend(SRC);

Flags Affected
None.

Additional Itanium System Environment Exceptions
Itanium Reg Faults NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions
#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
MOVZX—Move with Zero-Extend

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F B6 /r</td>
<td>MOVZX r16,r/m8</td>
<td>Move byte to word with zero-extension</td>
</tr>
<tr>
<td>0F B6 /r</td>
<td>MOVZX r32,r/m8</td>
<td>Move byte to doubleword, zero-extension</td>
</tr>
<tr>
<td>0F B7 /r</td>
<td>MOVZX r32,r/m16</td>
<td>Move word to doubleword, zero-extension</td>
</tr>
</tbody>
</table>

Description

Copies the contents of the source operand (register or memory location) to the destination operand (register) and sign extends the value to 16 or 32 bits. The size of the converted value depends on the operand-size attribute.

Copies the contents of the source operand (register or memory location) to the destination operand (register) and zero extends the value to 16 or 32 bits. The size of the converted value depends on the operand-size attribute.

Operation

DEST ← ZeroExtend(SRC);

Flags Affected

None.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

- #GP(0): If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- #SS(0): If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code): If a page fault occurs.
- #AC(0): If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

- #GP: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- #SS: If a memory operand effective address is outside the SS segment limit.
MOVZX—Move with Zero-Extend (continued)

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
MUL—Unsigned Multiplication of AL, AX, or EAX

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F6 /4</td>
<td>MUL r/m8</td>
<td>unsigned multiply (AX ← AL * r/m8)</td>
</tr>
<tr>
<td>F7 /4</td>
<td>MUL r/m16</td>
<td>unsigned multiply (DX:AX ← AX * r/m16)</td>
</tr>
<tr>
<td>F7 /4</td>
<td>MUL r/m32</td>
<td>unsigned multiply (EDX:EAX ← EAX * r/m32)</td>
</tr>
</tbody>
</table>

Description

Performs an unsigned multiplication of the first operand (destination operand) and the second operand (source operand) and stores the result in the destination operand. The destination operand is an implied operand located in register AL, AX or EAX (depending on the size of the operand); the source operand is located in a general-purpose register or a memory location. The action of this instruction and the location of the result depends on the opcode and the operand size as shown in the following table.

<table>
<thead>
<tr>
<th>Operand Size</th>
<th>Source 1</th>
<th>Source 2</th>
<th>Destination</th>
</tr>
</thead>
<tbody>
<tr>
<td>Byte</td>
<td>AL</td>
<td>r/m8</td>
<td>AX</td>
</tr>
<tr>
<td>Word</td>
<td>AX</td>
<td>r/m16</td>
<td>DX:AX</td>
</tr>
<tr>
<td>Doubleword</td>
<td>EAX</td>
<td>r/m32</td>
<td>EDX:EAX</td>
</tr>
</tbody>
</table>

The AH, DX, or EDX registers (depending on the operand size) contain the high-order bits of the product. If the contents of one of these registers are 0, the CF and OF flags are cleared; otherwise, the flags are set.

Operation

IF byte operation
THEN
AX ← AL * SRC
ELSE (* word or doubleword operation *)
   IF OperandSize = 16
       THEN
           DX:AX ← AX * SRC
       ELSE (* OperandSize = 32 *)
           EDX:EAX ← EAX * SRC
   FI;
FI;

Flags Affected

The OF and CF flags are cleared to 0 if the upper half of the result is 0; otherwise, they are set to 1. The SF, ZF, AF, and PF flags are undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
MUL—Unsigned Multiplication of AL, AX, or EAX (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
NEG—Two's Complement Negation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F6 /3</td>
<td>NEG r/m8</td>
<td>Two's complement negate r/m8</td>
</tr>
<tr>
<td>F7 /3</td>
<td>NEG r/m16</td>
<td>Two's complement negate r/m16</td>
</tr>
<tr>
<td>F7 /3</td>
<td>NEG r/m32</td>
<td>Two's complement negate r/m32</td>
</tr>
</tbody>
</table>

**Description**

Replaces the value of operand (the destination operand) with its two's complement. The destination operand is located in a general-purpose register or a memory location.

**Operation**

IF DEST = 0
  THEN CF ← 0
  ELSE CF ← 1;
FI;
DEST ← − (DEST)

**Flags Affected**

The CF flag cleared to 0 if the source operand is 0; otherwise it is set to 1. The OF, SF, ZF, AF, and PF flags are set according to the result.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults   NaT Register Consumption Abort.
Itanium Mem Faults   VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Protected Mode Exceptions**

#GP(0) If the destination is located in a nonwritable segment.
  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.
NEG—Two’s Complement Negation (continued)

Virtual 8086 Mode Exceptions

#GP(0)      If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)      If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)      If alignment checking is enabled and an unaligned memory reference is made.
NOP—No Operation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>90</td>
<td>NOP</td>
<td>No operation</td>
</tr>
</tbody>
</table>

Description

Performs no operation. This instruction is a one-byte instruction that takes up space in the instruction stream but does not affect the machine context, except the EIP register.

The NOP instruction performs no operation, no registers are accessed and no faults are generated.

Flags Affected

None.

Exceptions (All Operating Modes)

None.
NOT—One's Complement Negation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F6 /2</td>
<td>NOT r/m8</td>
<td>Reverse each bit of r/m8</td>
</tr>
<tr>
<td>F7 /2</td>
<td>NOT r/m16</td>
<td>Reverse each bit of r/m16</td>
</tr>
<tr>
<td>F7 /2</td>
<td>NOT r/m32</td>
<td>Reverse each bit of r/m32</td>
</tr>
</tbody>
</table>

Description
Performs a bitwise NOT operation (1’s complement) on the destination operand and stores the result in the destination operand location. The destination operand can be a register or a memory location.

Operation
DEST ← NOT DEST;

Flags Affected
None.

Additional Itanium System Environment Exceptions
Itanium Reg Faults  NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions
#GP(0)  If the destination operand points to a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions
#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS  If a memory operand effective address is outside the SS segment limit.
NOT—One's Complement Negation (continued)

<table>
<thead>
<tr>
<th>Exception Code</th>
<th>Exception Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP(0)</td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.</td>
</tr>
<tr>
<td>#SS(0)</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made.</td>
</tr>
</tbody>
</table>
OR—Logical Inclusive OR

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0C ib</td>
<td>OR AL,imm8</td>
<td>AL OR imm8</td>
</tr>
<tr>
<td>0D iw</td>
<td>OR AX,imm16</td>
<td>AX OR imm16</td>
</tr>
<tr>
<td>0D id</td>
<td>OR EAX,imm32</td>
<td>EAX OR imm32</td>
</tr>
<tr>
<td>80 /1 ib</td>
<td>OR r/m8,imm8</td>
<td>r/m8 OR imm8</td>
</tr>
<tr>
<td>81 /1 iw</td>
<td>OR r/m16,imm16</td>
<td>r/m16 OR imm16</td>
</tr>
<tr>
<td>81 /1 id</td>
<td>OR r/m32,imm32</td>
<td>r/m32 OR imm32</td>
</tr>
<tr>
<td>83 /1 ib</td>
<td>OR r/m16,imm8</td>
<td>r/m16 OR imm8</td>
</tr>
<tr>
<td>83 /1 ib</td>
<td>OR r/m32,imm8</td>
<td>r/m32 OR imm8</td>
</tr>
<tr>
<td>08 /r</td>
<td>OR r/m8,r8</td>
<td>r/m8 OR r8</td>
</tr>
<tr>
<td>09 /r</td>
<td>OR r/m16,r16</td>
<td>r/m16 OR r16</td>
</tr>
<tr>
<td>09 /r</td>
<td>OR r/m32,r32</td>
<td>r/m32 OR r32</td>
</tr>
<tr>
<td>0A /r</td>
<td>OR r8,r/m8</td>
<td>r8 OR r/m8</td>
</tr>
<tr>
<td>0B /r</td>
<td>OR r16,r/m16</td>
<td>r16 OR r/m16</td>
</tr>
<tr>
<td>0B /r</td>
<td>OR r32,r/m32</td>
<td>r32 OR r/m32</td>
</tr>
</tbody>
</table>

**Description**

Performs a bitwise OR operation on the destination (first) and source (second) operands and stores the result in the destination operand location. The source operand can be an immediate, a register, or a memory location; the destination operand can be a register or a memory location.

**Operation**

DEST ← DEST OR SRC;

**Flags Affected**

The OF and CF flags are cleared; the SF, ZF, and PF flags are set according to the result. The state of the AF flag is undefined.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
OR—Logical Inclusive OR (continued)

Protected Mode Exceptions

#GP(0) If the destination operand points to a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
## OUT—Output to Port

### Description

Copies the value from the second operand (source operand) to the I/O port specified with the destination operand (first operand). The source operand can be register AL, AX, or EAX, depending on the size of the port being accessed (8, 16, or 32 bits, respectively); the destination operand can be a byte-immediate or the DX register. Using a byte immediate allows I/O port addresses 0 to 255 to be accessed; using the DX register as a source operand allows I/O ports from 0 to 65,535 to be accessed.

When accessing an 8-bit I/O port, the opcode determines the port size; when accessing a 16- and 32-bit I/O port, the operand-size attribute determines the port size.

At the machine code level, I/O instructions are shorter when accessing 8-bit I/O ports. Here, the upper eight bits of the port address will be 0.

This instruction is only useful for accessing I/O ports located in the processor’s I/O address space.

I/O transactions are performed after all prior data memory operations. No subsequent data memory operations can pass an I/O transaction.

In the Itanium System Environment, I/O port references are mapped into the 64-bit virtual address pointed to by the IOBase register, with four ports per 4K-byte virtual page. Operating systems can utilize TLBs in the Itanium architecture to grant or deny permission to any four I/O ports. The I/O port space can be mapped into any arbitrary 64-bit physical memory location by operating system code. If CFLG.io is 1 and CPL>IOPL, the TSS is consulted for I/O permission. If CFLG.io is 0 or CPL<=IOPL, permission is granted regardless of the state of the TSS I/O permission bitmap (the bitmap is not referenced).

If the referenced I/O port is mapped to an unimplemented virtual address (via the I/O Base register) or if data translations are disabled (PSR.dt is 0) a GPFault is generated on the referencing OUT instruction.

### Operation

\[
\text{IF } ((\text{PE} = 1) \text{ AND } ((\text{VM} = 1) \text{ OR } (\text{CPL} > \text{IOPL}))) \\
\text{THEN } (* \text{ Protected mode or virtual-8086 mode with CPL} > \text{IOPL} *) \\
\text{IF } (\text{CFLG.io AND Any I/O Permission Bit for I/O port being accessed} = 1) \\
\text{THEN } \#\text{GP}(0); \\
\text{FI;} \\
\text{ELSE } (* \text{ Real-address mode or protected mode with CPL} \leq \text{IOPL} *) \\
(* \text{ or virtual-8086 mode with all I/O permission bits for I/O port cleared } *)
\]

### Opcode Table

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>E6 ib</td>
<td>OUT imm8, AL</td>
<td>Output byte AL to imm8 I/O port address</td>
</tr>
<tr>
<td>E7 ib</td>
<td>OUT imm8, AX</td>
<td>Output word AX to imm8 I/O port address</td>
</tr>
<tr>
<td>E7 ib</td>
<td>OUT imm8, EAX</td>
<td>Output doubleword EAX to imm8 I/O port address</td>
</tr>
<tr>
<td>EE</td>
<td>OUT DX, AL</td>
<td>Output byte AL to I/O port address in DX</td>
</tr>
<tr>
<td>EF</td>
<td>OUT DX, AX</td>
<td>Output word AX to I/O port address in DX</td>
</tr>
<tr>
<td>EF</td>
<td>OUT DX, EAX</td>
<td>Output doubleword EAX to I/O port address in DX</td>
</tr>
</tbody>
</table>
OUT—Output to Port (continued)

FI;
IF (Itanium_System_Environment) THEN
    DEST_VA = IOBase | (Port{15:2}<<12) | Port{11:0};
    DEST_PA = translate(DEST_VA);

    [DEST_PA] ← SRC; (* Writes to selected I/O port *)
FI;

memory_fence();
[DEST_PA] ← SRC; (* Writes to selected I/O port *)
memory_fence();

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB
                   Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data
                   Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data
                   Access Bit Fault, Data Dirty Bit Fault
IA-32_Exception    Debug traps for data breakpoints and single step
IA-32_Exception    Alignment faults
#GP(0)             Referenced Port is to an unimplemented virtual address or PSR.dt is zero.

Protected Mode Exceptions

#GP(0)              If the CPL is greater than (has less privilege) the I/O privilege level (IOPL)
                    and any of the corresponding I/O permission bits in TSS for the I/O port being
                    accessed is 1 and when CFLG.io is 1.

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

#GP(0)              If any of the I/O permission bits in the TSS for the I/O port being accessed is
                    1.
OUTS/OUTSB/OUTSW/OUTSD—Output String to Port

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>6E</td>
<td>OUTS DX, DS:(E)SI</td>
<td>Output byte at address DS:(E)SI to I/O port in DX</td>
</tr>
<tr>
<td>6F</td>
<td>OUTS DX, DS:SI</td>
<td>Output word at address DS:SI to I/O port in DX</td>
</tr>
<tr>
<td>6F</td>
<td>OUTS DX, DS:ESI</td>
<td>Output doubleword at address DS:ESI to I/O port in DX</td>
</tr>
<tr>
<td>6E</td>
<td>OUTSB</td>
<td>Output byte at address DS:(E)SI to I/O port in DX</td>
</tr>
<tr>
<td>6F</td>
<td>OUTSW</td>
<td>Output word at address DS:SI to I/O port in DX</td>
</tr>
<tr>
<td>6F</td>
<td>OUTSD</td>
<td>Output doubleword at address DS:ESI to I/O port in DX</td>
</tr>
</tbody>
</table>

### Description

Copies data from the second operand (source operand) to the I/O port specified with the first operand (destination operand). The source operand is a memory location at the address DS:ESI. (When the operand-size attribute is 16, the SI register is used as the source-index register.) The DS register may be overridden with a segment override prefix.

The destination operand must be the DX register, allowing I/O port addresses from 0 to 65,535 to be accessed. When accessing an 8-bit I/O port, the opcode determines the port size; when accessing a 16- and 32-bit I/O port, the operand-size attribute determines the port size.

The OUTSB, OUTSW and OUTSD mnemonics are synonyms of the byte, word, and doubleword versions of the OUTS instructions. (For the OUTS instruction, “DS:ESI” must be explicitly specified in the instruction.)

After the byte, word, or doubleword is transfer from the memory location to the I/O port, the ESI register is incremented or decremented automatically according to the setting of the DF flag in the EFLAGS register. (If the DF flag is 0, the ESI register is incremented; if the DF flag is 1, the EDI register is decremented.) The ESI register is incremented or decremented by 1 for byte operations, by 2 for word operations, or by 4 for doubleword operations.

The OUTS, OUTSB, OUTSW, and OUTSD instructions can be preceded by the REP prefix for block input of ECX bytes, words, or doublewords. See “REP/REPE/REPZ/REPNE/REPNZ—Repeat String Operation Prefix” on page 3:685 for a description of the REP prefix.

After an OUTS, OUTSB, OUTSW, or OUTSD instruction is executed, the processor waits for the acknowledgment of the OUT transaction before beginning to execute the next instruction. Note that the next instruction may be prefetched, even if the OUT transaction has not completed.

This instruction is only useful for accessing I/O ports located in the processor’s I/O address space.

**I/O transactions are performed after all prior data memory operations. No subsequent data memory operations can pass an I/O transaction.**
OUTS/OUTSB/OUTSW/OUTSD—Output String to Port (continued)

In the Itanium System Environment, I/O port references are mapped into the 64-bit virtual address pointed to by the IOBase register, with four ports per 4K-byte virtual page. Operating systems can utilize TLBs in the Itanium architecture to grant or deny permission to any four I/O ports. The I/O port space can be mapped into any arbitrary 64-bit physical memory location by operating system code. If CFLG.io is 1 and CPL>IOPL, the TSS is consulted for I/O permission. If CFLG.io is 0 or CPL<=IOPL, permission is granted regardless of the state of the TSS I/O permission bitmap (the bitmap is not referenced).

If the referenced I/O port is mapped to an unimplemented virtual address (via the I/O Base register) or if data translations are disabled (PSR.dt is 0) a GPFault is generated on the referencing OUTS instruction.

Operation

IF ((PE = 1) AND ((VM = 1) OR (CPL > IOPL)))
THEN (* Protected mode or virtual-8086 mode with CPL > IOPL *)
  IF (CFLG.io AND Any I/O Permission Bit for I/O port being accessed = 1)
      THEN #GP(0);
  ELSE (* I/O operation is allowed *)
  FI;
ELSE (* I/O operation is allowed *)
FI;

IF (Itanium_System_Environment) THEN
  DEST_VA = IOBase | (Port{15:2}<c12) | Port{11:0};
  DEST_PA = translate(DEST_VA);
  [DEST_PA] ← SRC; (* Writes to selected I/O port *)
FI;
memory_fence();
[DEST_PA] ← SRC; (* Writes to selected I/O port *)
memory_fence();

IF (byte operation)
  THEN IF DF = 0
      THEN (E)DI ← 1;
      ELSE (E)DI ← -1;
  FI;
ELSE IF (word operation)
  THEN IF DF = 0
      THEN DI ← 2;
      ELSE DI ← -2;
  FI;
ELSE (* doubleword operation *)
  THEN IF DF = 0
      THEN EDI ← 4;
      ELSE EDI ← -4;
  FI;
FI;
FI;

Flags Affected

None.
OUTS/OUTSB/OUTSW/OUTSD—Output String to Port (continued)

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
IA-32_Exception  Debug traps for data breakpoints and single step
IA-32_Exception  Alignment faults
#GP(0)  Referenced Port is to an unimplemented virtual address or PSR.dt is zero.

Protected Mode Exceptions

#GP(0)  If the CPL is greater than (has less privilege) the I/O privilege level (IOPL) and any of the corresponding I/O permission bits in TSS for the I/O port being accessed is 1 and when CFLG.io is 1.
If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the limit of the ES segment.
If the ES register contains a null segment selector.
If an illegal memory operand effective address in the ES segments is given.

#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP  If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS  If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0)  If any of the I/O permission bits in the TSS for the I/O port being accessed is 1.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
**POP—Pop a Value from the Stack**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>8F /0</td>
<td>POP m16</td>
<td>Pop top of stack into m16; increment stack pointer</td>
</tr>
<tr>
<td>8F /0</td>
<td>POP m32</td>
<td>Pop top of stack into m32; increment stack pointer</td>
</tr>
<tr>
<td>58+ rd</td>
<td>POP r32</td>
<td>Pop top of stack into r32; increment stack pointer</td>
</tr>
<tr>
<td>1F</td>
<td>POP DS</td>
<td>Pop top of stack into DS; increment stack pointer</td>
</tr>
<tr>
<td>07</td>
<td>POP ES</td>
<td>Pop top of stack into ES; increment stack pointer</td>
</tr>
<tr>
<td>17</td>
<td>POP SS</td>
<td>Pop top of stack into SS; increment stack pointer</td>
</tr>
<tr>
<td>0F A1</td>
<td>POP FS</td>
<td>Pop top of stack into FS; increment stack pointer</td>
</tr>
<tr>
<td>0F A9</td>
<td>POP GS</td>
<td>Pop top of stack into GS; increment stack pointer</td>
</tr>
</tbody>
</table>

**Description**

Loads the value from the top of the procedure stack to the location specified with the destination operand and then increments the stack pointer. The destination operand can be a general-purpose register, memory location, or segment register.

The current address-size attribute for the stack segment and the operand-size attribute determine the amount the stack pointer is incremented (see the “Operation” below). For example, if 32-bit addressing and operands are being used, the ESP register (stack pointer) is incremented by 4 and, if 16-bit addressing and operands are being used, the SP register (stack pointer for 16-bit addressing) is incremented by 2. The B flag in the stack segment’s segment descriptor determines the stack’s address-size attribute.

If the destination operand is one of the segment registers DS, ES, FS, GS, or SS, the value loaded into the register must be a valid segment selector. In protected mode, popping a segment selector into a segment register automatically causes the descriptor information associated with that segment selector to be loaded into the hidden (shadow) part of the segment register and causes the selector and the descriptor information to be validated (see the “Operation” below).

A null value (0000-0003) may be popped into the DS, ES, FS, or GS register without causing a general protection fault. However, any subsequent attempt to reference a segment whose corresponding segment register is loaded with a null value causes a general protection exception (#GP). In this situation, no memory reference occurs and the saved value of the segment register is null.

The POP instruction cannot pop a value into the CS register. To load the CS register, use the RET instruction.

A POP SS instruction inhibits all external interrupts, including the NMI interrupt, and traps until after execution of the next instruction. in the IA-32 System Environment. For the Itanium System Environment, POP SS results in an IA-32_Intercept(SystemFlag) trap after the instruction completes. This operation allows a stack pointer to be loaded into the ESP register with the next instruction (MOV ESP, stack-pointer value) before an interrupt occurs. The LSS instruction offers a more efficient method of loading the SS and ESP registers.
POP—Pop a Value from the Stack (continued)

This action allows sequential execution of POP SS and MOV ESP, EBP instructions without the danger of having an invalid stack during an interrupt. However, use of the LSS instruction is the preferred method of loading the SS and ESP registers.

If the ESP register is used as a base register for addressing a destination operand in memory, the POP instructions computes the effective address of the operand after it increments the ESP register.

The POP ESP instruction increments the stack pointer (ESP) before data at the old top of stack is written into the destination.

Operation

IF StackAddrSize = 32
THEN
  IF OperandSize = 32
  THEN
    DEST ← SS:ESP; (* copy a doubleword *)
    ESP ← ESP + 4;
  ELSE (* OperandSize = 16 *)
    DEST ← SS:ESP; (* copy a word *)
    ESP ← ESP + 2;
  FI;
  ELSE (* StackAddrSize = 16 *)
    IF OperandSize = 16
    THEN
      DEST ← SS:SP; (* copy a word *)
      SP ← SP + 2;
    ELSE (* OperandSize = 32 *)
      DEST ← SS:SP; (* copy a doubleword *)
      SP ← SP + 4;
    FI;
  FI;
FI;

Loading a segment register while in protected mode results in special checks and actions, as described in the following listing. These checks are performed on the segment selector and the segment descriptor it points to.

IF SS is loaded;
THEN
  IF segment selector is null
  THEN #GP(0);
  FI;
  IF segment selector index is outside descriptor table limits
    OR segment selector’s RPL ≠ CPL
    OR segment is not a writable data segment
    OR DPL ≠ CPL
    THEN #GP(selector);
  FI;
  IF segment not marked present
  THEN #SS(selector);
ELSE
  SS ← segment selector;
  SS ← segment descriptor;
FI;
POP—Pop a Value from the Stack (continued)

FI;
IF DS, ES, FS or GS is loaded with non-null selector;
THEN
   IF segment selector index is outside descriptor table limits
   OR segment is not a data or readable code segment
   OR ((segment is a data or nonconforming code segment)
   AND (both RPL and CPL > DPL))
   THEN #GP(selector);
   IF segment not marked present
   THEN #NP(selector);
ELSE
   SegmentRegister ← segment selector;
   SegmentRegister ← segment descriptor;
FI;
FI;
IF DS, ES, FS or GS is loaded with a null selector;
THEN
   SegmentRegister ← null segment selector;
   SegmentRegister ← null segment descriptor;
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Interceptor System Flag Intercept trap for POP SS
Itanium Reg Faults NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB
Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data
Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data
Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If attempt is made to load SS register with null segment selector.
   If the destination operand is in a nonwritable segment.
   If a memory operand effective address is outside the CS, DS, ES, FS, or GS
   segment limit.
   If the DS, ES, FS, or GS register is used to access memory and it contains a
   null segment selector.

#GP(selector) If segment selector index is outside descriptor table limits.
   If the SS register is being loaded and the segment selector’s RPL and the
   segment descriptor’s DPL are not equal to the CPL.
   If the SS register is being loaded and the segment pointed to is a nonwritable
   data segment.
POP—Pop a Value from the Stack (continued)

If the DS, ES, FS, or GS register is being loaded and the segment pointed to is not a data or readable code segment.

If the DS, ES, FS, or GS register is being loaded and the segment pointed to is a data or nonconforming code segment, but both the RPL and the CPL are greater than the DPL.

#SS(0) If the current top of stack is not within the stack segment.

If a memory operand effective address is outside the SS segment limit.

#SS(selector) If the SS register is being loaded and the segment pointed to is marked not present.

#NP If the DS, ES, FS, or GS register is being loaded and the segment pointed to is marked not present.

#PF(fault-code) If a page fault occurs.

#AC(0) If an unaligned memory reference is made while the current privilege level is 3 and alignment checking is enabled.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If an unaligned memory reference is made while alignment checking is enabled.
POPA/POPAD—Pop All General-Purpose Registers

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>61</td>
<td>POPA</td>
<td>Pop DI, SI, BP, BX, DX, CX, and AX</td>
</tr>
<tr>
<td>61</td>
<td>POPAD</td>
<td>Pop EDI, ESI, EBP, EBX, EDX, ECX, and EAX</td>
</tr>
</tbody>
</table>

Description

Pops doublewords (POPAD) or words (POPA) from the procedure stack into the general-purpose registers. The registers are loaded in the following order: EDI, ESI, EBP, EBX, EDX, ECX, and EAX (if the current operand-size attribute is 32) and DI, SI, BP, BX, DX, CX, and AX (if the operand-size attribute is 16). (These instructions reverse the operation of the PUSHA/PUSHAD instructions.) The value on the stack for the ESP or SP register is ignored. Instead, the ESP or SP register is incremented after each register is loaded (see the “Operation” below).

The POPA (pop all) and POPAD (pop all double) mnemonics reference the same opcode. The POPA instruction is intended for use when the operand-size attribute is 16 and the POPAD instruction for when the operand-size attribute is 32. Some assemblers may force the operand size to 16 when POPA is used and to 32 when POPAD is used. Others may treat these mnemonics as synonyms (POPA/POPAD) and use the current setting of the operand-size attribute to determine the size of values to be popped from the stack, regardless of the mnemonic used.

Operation

IF OperandSize = 32 (* instruction = POPAD *)
THEN
  EDI ← Pop();
  ESI ← Pop();
  EBP ← Pop();
  increment ESP by 4 (* skip next 4 bytes of stack *)
  EBX ← Pop();
  EDX ← Pop();
  ECX ← Pop();
  EAX ← Pop();
ELSE (* OperandSize = 16, instruction = POPA *)
  DI ← Pop();
  SI ← Pop();
  BP ← Pop();
  increment ESP by 2 (* skip next 2 bytes of stack *)
  BX ← Pop();
  DX ← Pop();
  CX ← Pop();
  AX ← Pop();
FI;

Flags Affected

None.
**POPA/POPAD—Pop All General-Purpose Registers** (continued)

### Additional Itanium System Environment Exceptions

- **Itanium Reg Faults**
  - NaT Register Consumption Abort.
- **Itanium Mem Faults**
  - VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

### Protected Mode Exceptions

- **#SS(0)**
  - If the starting or ending stack address is not within the stack segment.
- **#PF(fault-code)**
  - If a page fault occurs.

### Real Address Mode Exceptions

- **#GP**
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS**
  - If a memory operand effective address is outside the SS segment limit.

### Virtual 8086 Mode Exceptions

- **#GP(0)**
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)**
  - If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)**
  - If a page fault occurs.
POPF/POPFD—Pop Stack into EFLAGS Register

**Description**

Pops a doubleword (POPFD) from the top of the stack (if the current operand-size attribute is 32) and stores the value in the EFLAGS register or pops a word from the top of the stack (if the operand-size attribute is 16) and stores it in the lower 16 bits of the EFLAGS register. (These instructions reverse the operation of the PUSHF/PUSHFD instructions.)

The POPF (pop flags) and POPFD (pop flags double) mnemonics reference the same opcode. The POPF instruction is intended for use when the operand-size attribute is 16 and the POPFD instruction for when the operand-size attribute is 32. Some assemblers may force the operand size to 16 when POPF is used and to 32 when POPFD is used. Others may treat these mnemonics as synonyms (POPF/POPFD) and use the current setting of the operand-size attribute to determine the size of values to be popped from the stack, regardless of the mnemonic used.

The effect of the POPF/POPFD instructions on the EFLAGS register changes slightly, depending on the mode of operation of the processor. When the processor is operating in protected mode at privilege level 0 (or in real-address mode, which is equivalent to privilege level 0), all the non-reserved flags in the EFLAGS register except the VIP and VIF flags can be modified. The VIP and VIF flags are cleared.

When operating in protected mode, but with a privilege level greater than 0, all the flags can be modified except the IOPL field and the VIP and VIF flags. Here, the IOPL flags are masked and the VIP and VIF flags are cleared.

When operating in virtual-8086 mode, the I/O privilege level (IOPL) must be equal to 3 to use POPF/POPFD instructions and the VM, RF, IOPL, VIP, and VIF flags are masked. If the IOPL is less than 3, the POPF/POPFD instructions cause a general protection exception (#GP).

The IOPL is altered only when executing at privilege level 0. The interrupt flag is altered only when executing at a level at least as privileged as the IOPL. (Real-address mode is equivalent to privilege level 0.) If a POPF/POPFD instruction is executed with insufficient privilege, an exception does not occur, but the privileged bits do not change.

**Operation**

\[
\begin{align*}
\text{OLD_IF} & \leftarrow \text{IF}; \text{OLD_AC} \leftarrow \text{AC}; \text{OLD_TF} \leftarrow \text{TF}; \\
\text{IF CR0.PE} = 0 \text{ (*Real Mode *)} & \\
\text{THEN} & \\
\text{IF OperandSize} = 32; & \\
\text{THEN} & \\
\text{EFLAGS} & \leftarrow \text{Pop();} \\
\text{(* All non-reserved flags except VM, RF, VIP and VIF can be modified; *)} & \\
\text{ELSE (* OperandSize = 16 *)} & \\
\text{EFLAGS}[15:0] & \leftarrow \text{Pop();} \text{ (* All non-reserved flags can be modified; *)} \\
\text{FI;} & \\
\text{ELSE (*In Protected Mode *)} & \\
\text{IF VM=0 (* Not in Virtual-8086 Mode *)} & \\
\text{THEN} & \\
\end{align*}
\]
POPF/POPFD—Pop Stack into EFLAGS Register (continued)

```assembly
IF CPL=0
  THEN
    IF OperandSize = 32;
       THEN
         EFLAGS ← Pop();
         (* All non-reserved flags except VM, RF, VIP and VIF can be *)
         (* modified; *)
         ELSE (* OperandSize = 16 *)
         EFLAGS[15:0] ← Pop(); (* All non-reserved flags can be modified; *)
     FI;
    ELSE (* CPL > 0 *)
    IF OperandSize = 32;
    THEN
       EFLAGS ← Pop()
       (* All non-reserved bits except IOPL, RF, VM, VIP, and VIF can *)
       (* be modified; *)
       (* IOPL is masked *)
    ELSE (* OperandSize = 16 *)
       EFLAGS[15:0] ← Pop();
       (* All non-reserved bits except IOPL can be modified; IOPL is *)
       masked *)
     FI;
    FI;
ELSE (* In Virtual-8086 Mode *)
IF IOPL=3
  THEN
    IF OperandSize=32
      THEN
         EFLAGS ← Pop()
         (* All non-reserved bits except VM, RF, IOPL, VIP, and VIF *)
         (* can be modified; VM, RF, IOPL, VIP, and VIF are masked*)
         ELSE
         EFLAGS[15:0] ← Pop();
         (* All non-reserved bits except IOPL can be modified; IOPL is *)
         (* masked *)
       FI;
      ELSE (* IOPL < 3 *)
      IF CR4.VME = 0
        THEN #GP(0);
      ELSE
         IF ((OperandSize = 32) OR (STACK.TF = 1) OR (EFLAGS.VIP = 1
         AND STACK.IF = 1))
           THEN #GP(0);
         ELSE
            TempFlags ← pop();
            FLAGS ← TempFlags; (*IF and IOPL bits are unchanged*)
            EFLAGS.VIF ← TempFlags.IF;
           FI;
        FI;
      FI;
  FI;
FI;
```
POPF/POPFD—Pop Stack into EFLAGS Register (continued)

\[
\text{IF (Itanium System Environment AND } (AC, \text{ TF } \neq \text{ OLD AC, OLD TF}) \\
\text{ THEN IA-32_\text{Intercept}(System Flag,POPF);} \\
\text{IF (Itanium System Environment AND CFLG.ii AND IF } \neq \text{ OLD IF} \\
\text{ THEN IA-32_\text{Intercept}(System Flag,POPF);} \\
\]

Flags Affected

All flags except the reserved bits.

Additional Itanium System Environment Exceptions

Itanium Reg Faults

- NaT Register Consumption Abort.

Itanium Mem Faults

- VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

IA-32\_Intercept

- System Flag Intercept Trap if CFLG.ii is 1 and the IF flag changes state or if the AC, RF or TF changes state.

Protected Mode Exceptions

- #SS(0) If the top of stack is not within the stack segment.

Real Address Mode Exceptions

- #GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

- #SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

- #GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

  - If the I/O privilege level is less than 3.

  - If an attempt is made to execute the POPF/POPFD instruction with an operand-size override prefix.

- #SS(0) If a memory operand effective address is outside the SS segment limit.
PUSH—Push Word or Doubleword Onto the Stack

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FF /6</td>
<td>PUSH r/m16</td>
<td>Push r/m16</td>
</tr>
<tr>
<td>FF /6</td>
<td>PUSH r/m32</td>
<td>Push r/m32</td>
</tr>
<tr>
<td>50+rw</td>
<td>PUSH r16</td>
<td>Push r16</td>
</tr>
<tr>
<td>50+rd</td>
<td>PUSH r32</td>
<td>Push r32</td>
</tr>
<tr>
<td>6A</td>
<td>PUSH imm8</td>
<td>Push imm8</td>
</tr>
<tr>
<td>68</td>
<td>PUSH imm16</td>
<td>Push imm16</td>
</tr>
<tr>
<td>68</td>
<td>PUSH imm32</td>
<td>Push imm32</td>
</tr>
<tr>
<td>0E</td>
<td>PUSH CS</td>
<td>Push CS</td>
</tr>
<tr>
<td>16</td>
<td>PUSH SS</td>
<td>Push SS</td>
</tr>
<tr>
<td>1E</td>
<td>PUSH DS</td>
<td>Push DS</td>
</tr>
<tr>
<td>06</td>
<td>PUSH ES</td>
<td>Push ES</td>
</tr>
<tr>
<td>0F A0</td>
<td>PUSH FS</td>
<td>Push FS</td>
</tr>
<tr>
<td>0F A8</td>
<td>PUSH GS</td>
<td>Push GS</td>
</tr>
</tbody>
</table>

Description

Decrement the stack pointer and then stores the source operand on the top of the procedure stack. The current address-size attribute for the stack segment and the operand-size attribute determine the amount the stack pointer is decremented (see the “Operation” below). For example, if 32-bit addressing and operands are being used, the ESP register (stack pointer) is decremented by 4 and, if 16-bit addressing and operands are being used, the SP register (stack pointer for 16-bit addressing) is decremented by 2. Pushing 16-bit operands when the stack address-size attribute is 32 can result in a misaligned the stack pointer (that is, the stack pointer not aligned on a doubleword boundary).

The PUSH ESP instruction pushes the value of the ESP register as it existed before the instruction was executed. Thus, if a PUSH instruction uses a memory operand in which the ESP register is used as a base register for computing the operand address, the effective address of the operand is computed before the ESP register is decremented.

In the real-address mode, if the ESP or SP register is 1 when the PUSH instruction is executed, the processor shuts down due to a lack of stack space. No exception is generated to indicate this condition.

Operation

IF StackAddrSize = 32
THEN
  IF OperandSize = 32
  THEN
    ESP ← ESP − 4;
    SS:ESP ← SRC; (* push doubleword *)
  ELSE (* OperandSize = 16*)
    ESP ← ESP − 2;
    SS:ESP ← SRC; (* push word *)
  FI;
ELSE (* StackAddrSize = 16*)
PUSH—Push Word or Doubleword Onto the Stack (continued)

IF OperandSize = 16
    THEN
        SP ← SP − 2;
        SS:SP ← SRC; (* push word *)
    ELSE (* OperandSize = 32*)
        SP ← SP − 4;
        SS:SP ← SRC; (* push doubleword *)
FI;
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults   NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)   If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
         If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#SS(0)   If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)   If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP   If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS   If a memory operand effective address is outside the SS segment limit.
If the new value of the SP or ESP register is outside the stack segment limit.

Virtual 8086 Mode Exceptions

#GP(0)   If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
PUSH—Push Word or Doubleword Onto the Stack (continued)

#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.

Intel Architecture Compatibility

For Intel Architecture processors from the Intel 286 on, the PUSH ESP instruction pushes the value of the ESP register as it existed before the instruction was executed. (This is also true in the real-address and virtual-8086 modes.) For the Intel 8086 processor, the PUSH SP instruction pushes the new value of the SP register (that is the value after it has been decremented by 2).
PUSHA/PUSHAD—Push All General-Purpose Registers

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>60</td>
<td>PUSHA</td>
<td>Push AX, CX, DX, BX, original SP, BP, SI, and DI</td>
</tr>
<tr>
<td>60</td>
<td>PUSHAD</td>
<td>Push EAX, ECX, EDX, EBX, original ESP, EBP, ESI, and EDI</td>
</tr>
</tbody>
</table>

**Description**

Push the contents of the general-purpose registers onto the procedure stack. The registers are stored on the stack in the following order: EAX, ECX, EDX, EBX, ESP (original value), EBP, ESI, and EDI (if the current operand-size attribute is 32) and AX, CX, DX, BX, SP (original value), BP, SI, and DI (if the operand-size attribute is 16). (These instructions perform the reverse operation of the POPA/POPAD instructions.) The value pushed for the ESP or SP register is its value before prior to pushing the first register (see the “Operation” below).

The PUSHA (push all) and PUSHAD (push all double) mnemonics reference the same opcode. The PUSHA instruction is intended for use when the operand-size attribute is 16 and the PUSHAD instruction for when the operand-size attribute is 32. Some assemblers may force the operand size to 16 when PUSHA is used and to 32 when PUSHAD is used. Others may treat these mnemonics as synonyms (PUSHA/PUSHAD) and use the current setting of the operand-size attribute to determine the size of values to be pushed from the stack, regardless of the mnemonic used.

In the real-address mode, if the ESP or SP register is 1, 3, or 5 when the PUSHA/PUSHAD instruction is executed, the processor shuts down due to a lack of stack space. No exception is generated to indicate this condition.

**Operation**

IF OperandSize = 32 (* PUSHA instruction *)
THEN
  Temp ← (ESP);
  Push(EAX);
  Push(ECX);
  Push(EDX);
  Push(EBX);
  Push(Temp);
  Push(EBP);
  Push(ESI);
  Push(EDI);
ELSE (* OperandSize = 16, PUSHA instruction *)
  Temp ← (SP);
  Push(AX);
  Push(CX);
  Push(DX);
  Push(BX);
  Push(Temp);
  Push(BP);
  Push(SI);
  Push(DI);
FI;
PUSHA/PUSHAD—Push All General-Purpose Registers (continued)

Flags Affected

None.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

- #SS(0): If the starting or ending stack address is outside the stack segment limit.
- #PF(fault-code): If a page fault occurs.

Real Address Mode Exceptions

- #GP: If the ESP or SP register contains 7, 9, 11, 13, or 15.

Virtual 8086 Mode Exceptions

- #GP(0): If the ESP or SP register contains 7, 9, 11, 13, or 15.
- #PF(fault-code): If a page fault occurs.
PUSHF/PUSHFD—Push EFLAGS Register onto the Stack

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9C</td>
<td>PUSHF</td>
<td>Push EFLAGS</td>
</tr>
<tr>
<td>9C</td>
<td>PUSHFD</td>
<td>Push EFLAGS</td>
</tr>
</tbody>
</table>

**Description**

Decrement the stack pointer by 4 (if the current operand-size attribute is 32) and push the entire contents of the EFLAGS register onto the procedure stack or decrement the stack pointer by 2 (if the operand-size attribute is 16) push the lower 16 bits of the EFLAGS register onto the stack. (These instructions reverse the operation of the POPF/POPFD instructions.)

When copying the entire EFLAGS register to the stack, bits 16 and 17, called the VM and RF flags, are not copied. Instead, the values for these flags are cleared in the EFLAGS image stored on the stack.

The PUSHF (push flags) and PUSHFD (push flags double) mnemonics reference the same opcode. The PUSHF instruction is intended for use when the operand-size attribute is 16 and the PUSHFD instruction for when the operand-size attribute is 32. Some assemblers may force the operand size to 16 when PUSHF is used and to 32 when PUSHFD is used. Others may treat these mnemonics as synonyms (PUSHF/PUSHFD) and use the current setting of the operand-size attribute to determine the size of values to be pushed from the stack, regardless of the mnemonic used.

When the I/O privilege level (IOPL) is less than 3 in virtual-8086 mode, the PUSHF/PUSHFD instructions causes a general protection exception (#GP). The IOPL is altered only when executing at privilege level 0. The interrupt flag is altered only when executing at a level at least as privileged as the IOPL. (Real-address mode is equivalent to privilege level 0.) If a PUSHF/PUSHFD instruction is executed with insufficient privilege, an exception does not occur, but the privileged bits do not change.

In the real-address mode, if the ESP or SP register is 1, 3, or 5 when the PUSHA/PUSHAD instruction is executed, the processor shuts down due to a lack of stack space. No exception is generated to indicate this condition.

**Operation**

IF VM=0 (* Not in Virtual-8086 Mode *)
THEN
   IF OperandSize = 32
   THEN
      push(EFLAGS AND 0FCFFFFH);
      (* VM and RF EFLAG bits are cleared in image stored on the stack*)
   ELSE
      push(EFLAGS); (* Lower 16 bits only *)
   FI;
ELSE (* In Virtual-8086 Mode *)
   IF IOPL=3
   THEN
      IF OperandSize = 32
      THEN push(EFLAGS AND 0FCFFFFH);
         (* VM and RF EFLAGS bits are cleared in image stored on the stack*)
      ELSE push(EFLAGS); (* Lower 16 bits only *)
   FI;
PUSHF/PUSHFD—Push EFLAGS Register onto the Stack (continued)

Fl;
ELSE (*IOPL < 3*)

IF OperandSize =32 OR CR$.VME=0
    THEN #GP(0); (* Trap to virtual-8086 monitor *)
ELSE
    TempFlags <- FLAGS OR 3000H; (*Set IOPL bits to 11B or IOPL 3 *)
    TempFlags.IF <- EFLAGS.VIF;
    push(TempFlags);
Fl;
Fl;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#SS(0) If the new value of the ESP register is outside the stack segment boundary.

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

#GP(0) If the I/O privilege level is less than 3.
### RCL/Rcr/ROL/ROR—Rotate

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D0 /2</td>
<td>RCL r/m8,1</td>
<td>Rotate 9 bits (CF, r/m8) left once</td>
</tr>
<tr>
<td>D2 /2</td>
<td>RCL r/m8,CL</td>
<td>Rotate 9 bits (CF, r/m8) left CL times</td>
</tr>
<tr>
<td>C0 /2 ib</td>
<td>RCL r/m8,imm8</td>
<td>Rotate 9 bits (CF, r/m8) left imm8 times</td>
</tr>
<tr>
<td>D1 /2</td>
<td>RCL r/m16,1</td>
<td>Rotate 17 bits (CF, r/m16) left once</td>
</tr>
<tr>
<td>D3 /2</td>
<td>RCL r/m16,CL</td>
<td>Rotate 17 bits (CF, r/m16) left CL times</td>
</tr>
<tr>
<td>C1 /2 ib</td>
<td>RCL r/m16,imm8</td>
<td>Rotate 17 bits (CF, r/m16) left imm8 times</td>
</tr>
<tr>
<td>D1 /2</td>
<td>RCL r/m32,1</td>
<td>Rotate 33 bits (CF, r/m32) left once</td>
</tr>
<tr>
<td>D3 /2</td>
<td>RCL r/m32,CL</td>
<td>Rotate 33 bits (CF, r/m32) left CL times</td>
</tr>
<tr>
<td>C1 /2 ib</td>
<td>RCL r/m32,imm8</td>
<td>Rotate 33 bits (CF, r/m32) left imm8 times</td>
</tr>
<tr>
<td>D0 /3</td>
<td>RCR r/m8,1</td>
<td>Rotate 9 bits (CF, r/m8) right once</td>
</tr>
<tr>
<td>D2 /3</td>
<td>RCR r/m8,CL</td>
<td>Rotate 9 bits (CF, r/m8) right CL times</td>
</tr>
<tr>
<td>C0 /3 ib</td>
<td>RCR r/m8,imm8</td>
<td>Rotate 9 bits (CF, r/m8) right imm8 times</td>
</tr>
<tr>
<td>D1 /3</td>
<td>RCR r/m16,1</td>
<td>Rotate 17 bits (CF, r/m16) right once</td>
</tr>
<tr>
<td>D3 /3</td>
<td>RCR r/m16,CL</td>
<td>Rotate 17 bits (CF, r/m16) right CL times</td>
</tr>
<tr>
<td>C1 /3 ib</td>
<td>RCR r/m16,imm8</td>
<td>Rotate 17 bits (CF, r/m16) right imm8 times</td>
</tr>
<tr>
<td>D1 /3</td>
<td>RCR r/m32,1</td>
<td>Rotate 33 bits (CF, r/m32) right once</td>
</tr>
<tr>
<td>D3 /3</td>
<td>RCR r/m32,CL</td>
<td>Rotate 33 bits (CF, r/m32) right CL times</td>
</tr>
<tr>
<td>C1 /3 ib</td>
<td>RCR r/m32,imm8</td>
<td>Rotate 33 bits (CF, r/m32) right imm8 times</td>
</tr>
<tr>
<td>D0 /0</td>
<td>ROL r/m8,1</td>
<td>Rotate 8 bits r/m8 left once</td>
</tr>
<tr>
<td>D2 /0</td>
<td>ROL r/m8,CL</td>
<td>Rotate 8 bits r/m8 left CL times</td>
</tr>
<tr>
<td>C0 /0 ib</td>
<td>ROL r/m8,imm8</td>
<td>Rotate 8 bits r/m8 left imm8 times</td>
</tr>
<tr>
<td>D1 /0</td>
<td>ROL r/m16,1</td>
<td>Rotate 16 bits r/m16 left once</td>
</tr>
<tr>
<td>D3 /0</td>
<td>ROL r/m16,CL</td>
<td>Rotate 16 bits r/m16 left CL times</td>
</tr>
<tr>
<td>C1 /0 ib</td>
<td>ROL r/m16,imm8</td>
<td>Rotate 16 bits r/m16 left imm8 times</td>
</tr>
<tr>
<td>D1 /0</td>
<td>ROL r/m32,1</td>
<td>Rotate 32 bits r/m32 left once</td>
</tr>
<tr>
<td>D3 /0</td>
<td>ROL r/m32,CL</td>
<td>Rotate 32 bits r/m32 left CL times</td>
</tr>
<tr>
<td>C1 /0 ib</td>
<td>ROL r/m32,imm8</td>
<td>Rotate 32 bits r/m32 left imm8 times</td>
</tr>
<tr>
<td>D0 /1</td>
<td>ROR r/m8,1</td>
<td>Rotate 8 bits r/m8 right once</td>
</tr>
<tr>
<td>D2 /1</td>
<td>ROR r/m8,CL</td>
<td>Rotate 8 bits r/m8 right CL times</td>
</tr>
<tr>
<td>C0 /1 ib</td>
<td>ROR r/m8,imm8</td>
<td>Rotate 8 bits r/m16 right imm8 times</td>
</tr>
<tr>
<td>D1 /1</td>
<td>ROR r/m16,1</td>
<td>Rotate 16 bits r/m16 right once</td>
</tr>
<tr>
<td>D3 /1</td>
<td>ROR r/m16,CL</td>
<td>Rotate 16 bits r/m16 right CL times</td>
</tr>
<tr>
<td>C1 /1 ib</td>
<td>ROR r/m16,imm8</td>
<td>Rotate 16 bits r/m16 right imm8 times</td>
</tr>
<tr>
<td>D1 /1</td>
<td>ROR r/m32,1</td>
<td>Rotate 32 bits r/m32 right once</td>
</tr>
<tr>
<td>D3 /1</td>
<td>ROR r/m32,CL</td>
<td>Rotate 32 bits r/m32 right CL times</td>
</tr>
<tr>
<td>C1 /1 ib</td>
<td>ROR r/m32,imm8</td>
<td>Rotate 32 bits r/m32 right imm8 times</td>
</tr>
</tbody>
</table>
RCL/RCR/ROL/ROR—Rotate (continued)

Description

Shifts (rotates) the bits of the first operand (destination operand) the number of bit positions specified in the second operand (count operand) and stores the result in the destination operand. The destination operand can be a register or a memory location; the count operand is an unsigned integer that can be an immediate or a value in the CL register. The processor restricts the count to a number between 0 and 31 by masking all the bits in the count operand except the 5 least-significant bits.

The rotate left (ROL) and rotate through carry left (RCL) instructions shift all the bits toward more-significant bit positions, except for the most-significant bit, which is rotated to the least-significant bit location. The rotate right (ROR) and rotate through carry right (RCR) instructions shift all the bits toward less significant bit positions, except for the least-significant bit, which is rotated to the most-significant bit location.

The RCL and RCR instructions include the CF flag in the rotation. The RCL instruction shifts the CF flag into the least-significant bit and shifts the most-significant bit into the CF flag. The RCR instruction shifts the CF flag into the most-significant bit and shifts the least-significant bit into the CF flag. For the ROL and ROR instructions, the original value of the CF flag is not a part of the result, but the CF flag receives a copy of the bit that was shifted from one end to the other.

The OF flag is defined only for the 1-bit rotates; it is undefined in all other cases. For left rotates, the OF flag is set to the exclusive OR of the CF bit (after the rotate) and the most-significant bit of the result. For right rotates, the OF flag is set to the exclusive OR of the two most-significant bits of the result.

Operation

```
SIZE ← OperandSize
CASE (determine count) OF
    SIZE = 8: tempCOUNT ← (COUNT AND 1FH) MOD 9;
    SIZE = 16: tempCOUNT ← (COUNT AND 1FH) MOD 17;
    SIZE = 32: tempCOUNT ← COUNT AND 1FH;
ESAC;
(* ROL instruction operation *)
WHILE (tempCOUNT ≠ 0) DO
    tempCF ← MSB(DEST);
    DEST ← (DEST * 2) + tempCF;
    tempCOUNT ← tempCOUNT - 1;
OD;
ELIHW;
CF ← tempCF;
IF COUNT = 1 THEN OF ← MSB(DEST) XOR CF;
ELSE OF is undefined;
FI;
(* ROR instruction operation *)
WHILE (tempCOUNT ≠ 0) DO
    tempCF ← LSB(SRC);
    DEST ← (DEST / 2) + (tempCF * 2^SIZE);
```
RCL/RCR/ROL/ROR—Rotate  (continued)

```
tempCOUNT ← tempCOUNT − 1;
OD;
IF COUNT = 1
   THEN OF ← MSB(DEST) XOR MSB − 1(DEST);
   ELSE OF is undefined;
FI;
(* RCL instruction operation *)
WHILE (tempCOUNT ≠ 0)
   DO
      tempCF ← MSB(DEST);
      DEST ← (DEST + tempCF);
      tempCOUNT ← tempCOUNT − 1;
   OD;
ELIHW;
CF ← tempCF;
IF COUNT = 1
   THEN OF ← MSB(DEST) XOR CF;
   ELSE OF is undefined;
FI;
(* RCR instruction operation *)
WHILE (tempCOUNT ≠ 0)
   DO
      tempCF ← LSB(SRC);
      DEST ← (DEST / 2) + (tempCF * 2^SIZE);
      tempCOUNT ← tempCOUNT − 1;
   OD;
IF COUNT = 1
IF COUNT = 1
   THEN OF ← MSB(DEST) XOR MSB − 1(DEST);
   ELSE OF is undefined;
FI;
```

Flags Affected

The CF flag contains the value of the bit shifted into it. The OF flag is affected only for single-bit rotates (see “Description” above); it is undefined for multi-bit rotates. The SF, ZF, AF, and PF flags are not affected.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
RCL/RCR/ROL/ROR—Rotate (continued)

Protected Mode Exceptions

#GP(0) If the source operand is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.

Intel Architecture Compatibility

The 8086 does not mask the rotation count. All Intel Architecture processors from the Intel386™ processor on do mask the rotation count in all operating modes.
RDMSR—Read from Model Specific Register

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 32</td>
<td>RDMSR</td>
<td>Load MSR specified by ECX into EDX:EAX</td>
</tr>
</tbody>
</table>

Description

Loads the contents of a 64-bit model specific register (MSR) specified in the ECX register into registers EDX:EAX. The EDX register is loaded with the high-order 32 bits of the MSR and the EAX register is loaded with the low-order 32 bits. If less than 64 bits are implemented in the MSR being read, the values returned to EDX:EAX in unimplemented bit locations are undefined.

This instruction must be executed at privilege level 0 or in real-address mode; otherwise, a general protection exception #GP(0) will be generated. Specifying a reserved or unimplemented MSR address in ECX will also cause a general protection exception.

The MSRs control functions for testability, execution tracing, performance-monitoring and machine check errors.

The CPUID instruction should be used to determine whether MSRs are supported (EDX[5]=1) before using this instruction.

See model specific instructions for all the MSRs that can be written to with this instruction and their addresses

Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,RDMSR);

EDX:EAX ← MSR[ECX];

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Intercept Mandatory Instruction Intercept.

Protected Mode Exceptions

#GP(0) If the current privilege level is not 0.
If the value in ECX specifies a reserved or unimplemented MSR address.

Real Address Mode Exceptions

#GP If the current privilege level is not 0
If the value in ECX specifies a reserved or unimplemented MSR address.

Virtual 8086 Mode Exceptions

#GP(0) The RDMSR instruction is not recognized in virtual 8086 mode.
RDMSR—Read from Model Specific Register (continued)

**Intel Architecture Compatibility**

The MSRs and the ability to read them with the RDMSR instruction were introduced into the Intel Architecture with the Pentium processor. Execution of this instruction by an Intel Architecture processor earlier than the Pentium processor results in an invalid opcode exception #UD.
RDPMC—Read Performance-Monitoring Counters

Description

Loads the contents of the N-bit performance-monitoring counter specified in the ECX register into registers EDX:EAX. The EDX register is loaded with the high-order N-32 bits of the counter and the EAX register is loaded with the low-order 32 bits.

The RDPMC instruction allows application code running at a privilege level of 1, 2, or 3 to read the performance-monitoring counters if the PCE flag in the CR4 register is set for IA-32 System Environment operation or in the Itanium System Environment if the performance counters have been configured as user level counters. This instruction is provided to allow performance monitoring by application code without incurring the overhead of a call to an operating-system procedure.

The performance-monitoring counters are event counters that can be programmed to count events such as the number of instructions decoded, number of interrupts received, or number of cache loads.

The RDPMC instruction does not serialize instruction execution. That is, it does not imply that all the events caused by the preceding instructions have been completed or that events caused by subsequent instructions have not begun. If an exact event count is desired, software must use a serializing instruction (such as the CPUID instruction) before and/or after the execution of the RDPMC instruction.

The RDPMC instruction can execute in 16-bit addressing mode or virtual 8086 mode; however, the full contents of the ECX register are used to determine the counter to access and a full N-bit result is returned (the low-order 32 bits in the EAX register and the high-order N-32 bits in the EDX register).

Operation

IF (ECX != Implemented Counters) THEN #GP(0)
IF (Itanium System Environment)
THEN
SECURED = PSR.sp || CR4.pce==0;
IF ((PSR.cpl ==0) || (PSR.cpl!=0 && ~PMC(ECX).pm && ~SECURED)))
THEN
EDX:EAX ← PMD(ECX+4);
ELSE
#GP(0)
FI;
ELSE
IF ((CR4.PCE = 1 OR ((CR4.PCE = 0 ) AND (CPL=0)))
THEN
EDX:EAX ← PMD(ECX+4);
ELSE (* CR4.PCE is 0 and CPL is 1, 2, or 3 *)
#GP(0)
FI;
FI;

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 33</td>
<td>RDPMC</td>
<td>Read performance-monitoring counter specified by ECX into EDX:EAX</td>
</tr>
</tbody>
</table>
RDPMC—Read Performance-Monitoring Counters (continued)

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults NaT Register Consumption Abort.

#GP(0) If the current privilege level is not 0 and the selected PMD register’s PM bit is 1, or if PSR.sp is 1.

Protected Mode Exceptions

#GP(0) If the current privilege level is not 0 and the PCE flag in the CR4 register is clear /*In IA-32 System Environment*/.

If the value in the ECX register does not match an implemented performance counter.

Real Address Mode Exceptions

#GP If the PCE flag in the CR4 register is clear /*In the IA-32 System Environment*/.

If the value in the ECX register does not match an implemented performance counter.

Virtual 8086 Mode Exceptions

#GP(0) If the PCE flag in the CR4 register is clear /*In the IA-32 System Environment*/.

If the value in the ECX register does not match an implemented performance counter.
RDTSC—Read Time-Stamp Counter

**Description**

Loads the current value of the processor’s time-stamp counter into the EDX:EAX registers. The time-stamp counter is contained in a 64-bit MSR. The high-order 32 bits of the MSR are loaded into the EDX register, and the low-order 32 bits are loaded into the EAX register. The processor increments the time-stamp counter MSR every clock cycle and resets it to 0 whenever the processor is reset.

In the IA-32 System Environment, the time stamp disable (TSD) flag in register CR4 restricts the use of the RDTSC instruction. When the TSD flag is clear, the RDTSC instruction can be executed at any privilege level; when the flag is set, the instruction can only be executed at privilege level 0. The time-stamp counter can also be read with the RDMSR instruction.

In the Itanium System Environment, PSR.si and CR4.TSD restricts the use of the RDTSC instruction. When PSR.si is clear and CR4.TSD is clear, the RDTSC instruction can be executed at any privilege level; when PSR.si is set or CR4.TSD is set, the instruction can only be executed at privilege level 0.

The RDTSC instruction is not serializing instruction. Thus, it does not necessarily wait until all previous instructions have been executed before reading the counter. Similarly, subsequent instructions may begin execution before the read operation is performed.

This instruction was introduced into the Intel Architecture in the Pentium processor.

**Operation**

```
IF (IA-32 System Environment)
    IF (CR4.TSD = 0) OR ((CR4.TSD = 1) AND (CPL=0))
        THEN
            EDX:EAX ← TimeStampCounter;
        ELSE (* CR4 is 1 and CPL is 1, 2, or 3 *)
            #GP(0)
        FI;
    ELSE /*Itanium System Environment*/
        SECURED = PSR.si || CR4.TSD;
        IF (!SECURED) OR (SECURED AND (CPL=0))
            THEN
                EDX:EAX ← TimeStampCounter;
            ELSE (* CR4 is 1 and CPL is 1, 2, or 3 *)
                #GP(0)
            FI;
        FI;
```

**Flags Affected**

None.

---

**Opcode Instruction Description**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 31</td>
<td>RDTSC</td>
<td>Read time-stamp counter into EDX:EAX</td>
</tr>
</tbody>
</table>
RDTSC—Read Time-Stamp Counter (continued)

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  NaT Register Consumption Abort.

#GP(0)  If PSR.si is 1 or CR4.TSD is 1 and the CPL is greater than 0.

**Protected Mode Exceptions**

#GP(0)  If the TSD flag in register CR4 is set and the CPL is greater than 0.

/*For the IA-32 System Environment only*/

**Real Address Mode Exceptions**

#GP  If the TSD flag in register CR4 is set. /*For the IA-32 System Environment only*/

**Virtual 8086 Mode Exceptions**

#GP(0)  If the TSD flag in register CR4 is set. /*For the IA-32 System Environment only*/
REP/REPE/REPZ/REPNE/REPNZ—Repeat String Operation Prefix

<table>
<thead>
<tr>
<th>Prefix</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3 6C</td>
<td>REP INS r/m8, DX</td>
<td>Input ECX bytes from port DX into ES:[EDI]</td>
</tr>
<tr>
<td>F3 6D</td>
<td>REP INS r/m16,DX</td>
<td>Input ECX words from port DX into ES:[EDI]</td>
</tr>
<tr>
<td>F3 6D</td>
<td>REP INS r/m32,DX</td>
<td>Input ECX doublewords from port DX into ES:[EDI]</td>
</tr>
<tr>
<td>F3 A4</td>
<td>REP MOVS m8,m8</td>
<td>Move ECX bytes from DS:[ESI] to ES:[EDI]</td>
</tr>
<tr>
<td>F3 A5</td>
<td>REP MOVS m16,m16</td>
<td>Move ECX words from DS:[ESI] to ES:[EDI]</td>
</tr>
<tr>
<td>F3 A5</td>
<td>REP MOVS m32,m32</td>
<td>Move ECX doublewords from DS:[ESI] to ES:[EDI]</td>
</tr>
<tr>
<td>F3 6E</td>
<td>REP OUTS DX,r/m8</td>
<td>Output ECX bytes from DS:[ESI] to port DX</td>
</tr>
<tr>
<td>F3 6F</td>
<td>REP OUTS DX,r/m16</td>
<td>Output ECX words from DS:[ESI] to port DX</td>
</tr>
<tr>
<td>F3 6F</td>
<td>REP OUTS DX,r/m32</td>
<td>Output ECX doublewords from DS:[ESI] to port DX</td>
</tr>
<tr>
<td>F3 AC</td>
<td>REP LODS AL</td>
<td>Load ECX bytes from DS:[ESI] to AL</td>
</tr>
<tr>
<td>F3 AD</td>
<td>REP LODS AX</td>
<td>Load ECX words from DS:[ESI] to AX</td>
</tr>
<tr>
<td>F3 AD</td>
<td>REP LODS EAX</td>
<td>Load ECX doublewords from DS:[ESI] to EAX</td>
</tr>
<tr>
<td>F3 AA</td>
<td>REP STOS m8</td>
<td>Fill ECX bytes at ES:[EDI] with AL</td>
</tr>
<tr>
<td>F3 AB</td>
<td>REP STOS m16</td>
<td>Fill ECX words at ES:[EDI] with AX</td>
</tr>
<tr>
<td>F3 AB</td>
<td>REP STOS m32</td>
<td>Fill ECX doublewords at ES:[EDI] with EAX</td>
</tr>
<tr>
<td>F3 A6</td>
<td>REPE CMPS m8,m8</td>
<td>Find nonmatching bytes in ES:[EDI] and DS:[ESI]</td>
</tr>
<tr>
<td>F3 A7</td>
<td>REPE CMPS m16,m16</td>
<td>Find nonmatching words in ES:[EDI] and DS:[ESI]</td>
</tr>
<tr>
<td>F3 A7</td>
<td>REPE CMPS m32,m32</td>
<td>Find nonmatching doublewords in ES:[EDI] and DS:[ESI]</td>
</tr>
<tr>
<td>F3 AE</td>
<td>REPE SCAS m8</td>
<td>Find non-AL byte starting at ES:[EDI]</td>
</tr>
<tr>
<td>F3 AF</td>
<td>REPE SCAS m16</td>
<td>Find non-AX word starting at ES:[EDI]</td>
</tr>
<tr>
<td>F3 AF</td>
<td>REPE SCAS m32</td>
<td>Find non-EAX doubleword starting at ES:[EDI]</td>
</tr>
<tr>
<td>F2 A6</td>
<td>REPNE CMPS m8,m8</td>
<td>Find matching bytes in ES:[EDI] and DS:[ESI]</td>
</tr>
<tr>
<td>F2 A7</td>
<td>REPNE CMPS m16,m16</td>
<td>Find matching words in ES:[EDI] and DS:[ESI]</td>
</tr>
<tr>
<td>F2 A7</td>
<td>REPNE CMPS m32,m32</td>
<td>Find matching doublewords in ES:[EDI] and DS:[ESI]</td>
</tr>
<tr>
<td>F2 AE</td>
<td>REPNE SCAS m8</td>
<td>Find AL, starting at ES:[EDI]</td>
</tr>
<tr>
<td>F2 AF</td>
<td>REPNE SCAS m16</td>
<td>Find AX, starting at ES:[EDI]</td>
</tr>
<tr>
<td>F2 AF</td>
<td>REPNE SCAS m32</td>
<td>Find EAX, starting at ES:[EDI]</td>
</tr>
</tbody>
</table>

Description

Repeats a string instruction the number of times specified in the count register (ECX) or until the indicated condition of the ZF flag is no longer met. The REP (repeat), REPE (repeat while equal), REPNE (repeat while not equal), REPZ (repeat while zero), and REPNZ (repeat while not zero) mnemonics are prefixes that can be added to one of the string instructions. The REP prefix can be added to the INS, OUTS, MOVS, LODS, and STOS instructions, and the REPE, REPNE, REPZ, and REPNZ prefixes can be added to the CMPS and SCAS instructions. (The REPZ and REPNZ prefixes are synonymous forms of the REPE and REPNE prefixes, respectively.) The behavior of the REP prefix is undefined when used with non-string instructions.

The REP prefixes apply only to one string instruction at a time. To repeat a block of instructions, use the LOOP instruction or another looping construct.
REP/REPE/REPZ/REPNE/REPNZ—Repeat String Operation Prefix (continued)

All of these repeat prefixes cause the associated instruction to be repeated until the count in register ECX is decremented to 0 (see the following table). The REPE, REPNE, REPZ, and REPNZ prefixes also check the state of the ZF flag after each iteration and terminate the repeat loop if the ZF flag is not in the specified state. When both termination conditions are tested, the cause of a repeat termination can be determined either by testing the ECX register with a JECXZ instruction or by testing the ZF flag with a JZ, JNZ, and JNE instruction.

Table 1-17. Repeat Conditions

<table>
<thead>
<tr>
<th>Repeat Prefix</th>
<th>Termination Condition 1</th>
<th>Termination Condition 2</th>
</tr>
</thead>
<tbody>
<tr>
<td>REP</td>
<td>ECX=0</td>
<td>None</td>
</tr>
<tr>
<td>REPE/REPZ</td>
<td>ECX=0</td>
<td>ZF=0</td>
</tr>
<tr>
<td>REPNE/REPNZ</td>
<td>ECX=0</td>
<td>ZF=1</td>
</tr>
</tbody>
</table>

When the REPE/REPZ and REPNE/REPNZ prefixes are used, the ZF flag does not require initialization because both the CMPS and SCAS instructions affect the ZF flag according to the results of the comparisons they make.

A repeating string operation can be suspended by an exception or interrupt. When this happens, the state of the registers is preserved to allow the string operation to be resumed upon a return from the exception or interrupt handler. The source and destination registers point to the next string elements to be operated on, the EIP register points to the string instruction, and the ECX register has the value it held following the last successful iteration of the instruction. This mechanism allows long string operations to proceed without affecting the interrupt response time of the system.

When a page fault occurs during CMPS or SCAS instructions that are prefixed with REPNE, the EFLAGS value may NOT be restored to the state prior to the execution of the instruction. Since SCAS and CMPS do not use EFLAGS as an input, the processor can resume the instruction after the page fault handler.

Use the REP INS and REP OUTS instructions with caution. Not all I/O ports can handle the rate at which these instructions execute.

A REP STOS instruction is the fastest way to initialize a large block of memory.

Operation

IF AddressSize = 16
THEN
    use CX for CountReg;
ELSE (* AddressSize = 32 *)
    use ECX for CountReg;
FI;
WHILE CountReg ≠ 0
    DO
        service pending interrupts (if any);
        execute associated string instruction;
        CountReg ← CountReg – 1;
        IF CountReg = 0
            THEN exit WHILE loop
        FI;
    OD
REP/REPE/REPZ/REPNE/REPNZ—Repeat String Operation Prefix

(continued)

IF (repeat prefix is REPZ or REPE) AND (ZF=0)
OR (repeat prefix is REPNZ or REPNE) AND (ZF=1)

THEN exit WHILE loop
FI;
OD;

Flags Affected

None; however, the CMPS and SCAS instructions do set the status flags in the EFLAGS register.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Exceptions (All Operating Modes)

None; however, exceptions can be generated by the instruction a repeat prefix is associated with.
**RET—Return from Procedure**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>C3</td>
<td>RET</td>
<td>Near return to calling procedure</td>
</tr>
<tr>
<td>CB</td>
<td>RET</td>
<td>Far return to calling procedure</td>
</tr>
<tr>
<td>C2 _w</td>
<td>RET _imm16</td>
<td>Near return to calling procedure and pop imm16 bytes from stack</td>
</tr>
<tr>
<td>CA _w</td>
<td>RET _imm16</td>
<td>Far return to calling procedure and pop imm16 bytes from stack</td>
</tr>
</tbody>
</table>

**Description**

Transfers program control to a return address located on the top of the stack. The address is usually placed on the stack by a CALL instruction, and the return is made to the instruction that follows the CALL instruction.

The optional source operand specifies the number of stack bytes to be released after the return address is popped; the default is none. This operand can be used to release parameters from the stack that were passed to the called procedure and are no longer needed.

The RET instruction can be used to execute three different types of returns:

- **Near return** – A return to a calling procedure within the current code segment (the segment currently pointed to by the CS register), sometimes referred to as an intrasegment return.
- **Far return** – A return to a calling procedure located in a different segment than the current code segment, sometimes referred to as an intersegment return.
- **Inter-privilege-level far return** – A far return to a different privilege level than that of the currently executing program or procedure.

The inter-privilege-level return type can only be executed in protected mode.

When executing a near return, the processor pops the return instruction pointer (offset) from the top of the procedure stack into the EIP register and begins program execution at the new instruction pointer. The CS register is unchanged.

When executing a far return, the processor pops the return instruction pointer from the top of the procedure stack into the EIP register, then pops the segment selector from the top of the stack into the CS register. The processor then begins program execution in the new code segment at the new instruction pointer.

The mechanics of an inter-privilege-level far return are similar to an intersegment return, except that the processor examines the privilege levels and access rights of the code and stack segments being returned to determine if the control transfer is allowed to be made. The DS, ES, FS, and GS segment registers are cleared by the RET instruction during an inter-privilege-level return if they refer to segments that are not allowed to be accessed at the new privilege level. Since a stack switch also occurs on an inter-privilege level return, the ESP and SS registers are loaded from the stack.
RET—Return from Procedure (continued)

Operation

(* Near return *)
IF instruction = near return
   THEN;
      IF OperandSize = 32
         THEN
            IF top 12 bytes of stack not within stack limits THEN #SS(0); FI;
            EIP ← Pop();
            ELSE (* OperandSize = 16 *)
               IF top 6 bytes of stack not within stack limits
                  THEN #SS(0)
                  FI;
               tempEIP ← Pop();
               tempEIP ← tempEIP AND 0000FFFFH;
               IF tempEIP not within code segment limits THEN #GP(0); FI;
               EIP ← tempEIP;
         FI;
      IF instruction has immediate operand
         THEN IF StackAddressSize=32
               THEN
                  ESP ← ESP + SRC;
               ELSE (* StackAddressSize=16 *)
                  SP ← SP + SRC;
               FI;
         FI;
      IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
      FI;
(* Real-address mode or virtual-8086 mode *)
IF ((PE = 0) OR (PE = 1 AND VM = 1)) AND instruction = far return
   THEN;
      IF OperandSize = 32
         THEN
            IF top 12 bytes of stack not within stack limits THEN #SS(0); FI;
            EIP ← Pop();
            CS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
            ELSE (* OperandSize = 16 *)
               IF top 6 bytes of stack not within stack limits THEN #SS(0); FI;
               tempEIP ← Pop();
               tempEIP ← tempEIP AND 0000FFFFH;
               IF tempEIP not within code segment limits THEN #GP(0); FI;
               EIP ← tempEIP;
               CS ← Pop(); (* 16-bit pop *)
         FI;
      IF instruction has immediate operand THEN SP ← SP + (SRC AND FFFFH); FI;
      IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);
      FI;
(* Protected mode, not virtual 8086 mode *)
IF (PE = 1 AND VM = 0) AND instruction = far RET
   THEN
      IF OperandSize = 32
         THEN
RET—Return from Procedure (continued)

IF second doubleword on stack is not within stack limits THEN #SS(0); FI;
ELSE (* OperandSize = 16 *)
IF second word on stack is not within stack limits THEN #SS(0); FI;
FI;
IF return code segment selector is null THEN GP(0); FI;
IF return code segment selector addresses descriptor beyond descriptor table limit
THEN GP(selector); FI;
Obtain descriptor to which return code segment selector points from descriptor table
IF return code segment descriptor is not a code segment THEN #GP(selector); FI;
if return code segment selector RPL < CPL THEN #GP(selector); FI;
IF return code segment descriptor is conforming
AND return code segment DPL > return code segment selector RPL
THEN #GP(selector); FI;
IF return code segment descriptor is not present THEN #NP(selector); FI;
IF return code segment selector RPL > CPL
THEN GOTO RETURN-OUTER-PRIVILEGE-LEVEL;
ELSE GOTO RETURN-TO-SAME-PRIVILEGE-LEVEL
FI;
END;FI;

RETURN-SAME-PRIVILEGE-LEVEL:
IF the return instruction pointer is not within their return code segment limit
THEN #GP(0);
FI;
IF OperandSize=32
THEN
EIP ← Pop();
CS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
ESP ← ESP + SRC;
ELSE (* OperandSize=16 *)
EIP ← Pop();
EIP ← EIP AND 0000FFFFH;
CS ← Pop(); (* 16-bit pop *)
ESP ← ESP + SRC;
FI;
IF Itanium System Environment AND PSR.tb THEN IA-32_Exception(Debug);

RETURN-OUTER-PRIVILEGE-LEVEL:
IF top (16 + SRC) bytes of stack are not within stack limits (OperandSize=32)
OR top (8 + SRC) bytes of stack are not within stack limits (OperandSize=16)
THEN #SS(0); FI;
FI;
Read return segment selector;
IF stack segment selector is null THEN #GP(0); FI;
IF return stack segment selector index is not within its descriptor table limits
THEN #GP(selector); FI;
Read segment descriptor pointed to by return segment selector;
IF stack segment selector RPL ≠ RPL of the return code segment selector
OR stack segment is not a writable data segment
OR stack segment descriptor DPL ≠ RPL of the return code segment selector
THEN #GP(selector); FI;
IF stack segment not present THEN #SS(StackSegmentSelector); FI;
RET—Return from Procedure (continued)

IF the return instruction pointer is not within the return code segment limit THEN #GP(0); FI:
CPL ← ReturnCodeSegmentSelector(RPL);
IF OperandSize=32
    THEN
        EIP ← Pop();
        CS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
        (* segment descriptor information also loaded *)
        CS(RPL) ← CPL;
        ESP ← ESP + SRC;
        tempESP ← Pop();
        tempSS ← Pop(); (* 32-bit pop, high-order 16-bits discarded *)
        (* segment descriptor information also loaded *)
        ESP ← tempESP;
        SS ← tempSS;
    ELSE (* OperandSize=16 *)
        EIP ← Pop();
        EIP ← EIP AND 0000FFFFH;
        CS ← Pop(); (* 16-bit pop; segment descriptor information also loaded *)
        CS(RPL) ← CPL;
        ESP ← ESP + SRC;
        tempESP ← Pop();
        tempSS ← Pop(); (* 16-bit pop; segment descriptor information also loaded *)
        (* segment descriptor information also loaded *)
        ESP ← tempESP;
        SS ← tempSS;
    FI;
FOR each of segment register (ES, FS, GS, and DS)
    DO;
    IF segment register points to data or non-conforming code segment
    AND CPL > segment descriptor DPL; (* DPL in hidden part of segment register *)
    THEN (* segment register invalid *)
        SegmentSelector/Descriptor ← 0; (* null segment selector *)
    FI;
    OD;
For each of ES, FS, GS, and DS
    DO
    IF segment descriptor indicates the segment is not a data or readable code segment
    OR if the segment is a data or non-conforming code segment and the segment descriptor’s DPL < CPL or RPL of code segment’s segment selector
    THEN
        segment selector register ← null selector;
    OD;

Flags Affected

None.
RET—Return from Procedure (continued)

Additional Itanium System Environment Exceptions
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
IA-32_Exception  Taken Branch Debug Exception if PSR.tb is 1

Protected Mode Exceptions

#GP(0)  If the return code or stack segment selector null.
         If the return instruction pointer is not within the return code segment limit
#GP(selector)  If the RPL of the return code segment selector is less then the CPL.
         If the return code or stack segment selector index is not within its descriptor table limits.
         If the return code segment descriptor does not indicate a code segment.
         If the return code segment is non-conforming and the segment selector’s DPL is not equal to the RPL of the code segment’s segment selector
         If the return code segment is conforming and the segment selector’s DPL greater than the RPL of the code segment’s segment selector
         If the stack segment is not a writable data segment.
         If the stack segment selector RPL is not equal to the RPL of the return code segment selector.
         If the stack segment descriptor DPL is not equal to the RPL of the return code segment selector.
#SS(0)  If the top bytes of stack are not within stack limits.
         If the return stack segment is not present.
#NP(selector)  If the return code segment is not present.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If an unaligned memory access occurs when the CPL is 3 and alignment checking is enabled.

Real Address Mode Exceptions

#GP  If the return instruction pointer is not within the return code segment limit
#SS  If the top bytes of stack are not within stack limits.

Virtual 8086 Mode Exceptions

#GP(0)  If the return instruction pointer is not within the return code segment limit
#SS(0)  If the top bytes of stack are not within stack limits.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If an unaligned memory access occurs when alignment checking is enabled.
ROL/ROR—Rotate

See entry for RCL/RCR/ROL/ROR.
RSM—Resume from System Management Mode

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F AA</td>
<td>RSM</td>
<td>Resume operation of interrupted program</td>
</tr>
</tbody>
</table>

Description

Returns program control from system management mode (SMM) to the application program or operating system procedure that was interrupted when the processor received an SSM interrupt. The processor’s state is restored from the dump created upon entering SMM. If the processor detects invalid state information during state restoration, it enters the shutdown state. The following invalid information can cause a shutdown:

- Any reserved bit of CR4 is set to 1.
- Any illegal combination of bits in CR0, such as (PG=1 and PE=0) or (NW=1 and CD=0).
- (Intel Pentium and Intel486 only.) The value stored in the state dump base field is not a 32-KByte aligned address.

The contents of the model-specific registers are not affected by a return from SMM.

See Chapter 9 in the *Intel Architecture Software Developer’s Manual, Volume 3* for more information about SMM and the behavior of the RSM instruction.

Operation

IF Itanium System Environment THEN IA-32 Intercept(INST,RSM);
ReturnFromSSM;
ProcessorState ← Restore(SSMDump);

Flags Affected

All.

Additional Itanium System Environment Exceptions

IA-32_Intercept Mandatory Instruction Intercept.

Protected Mode Exceptions

#UD If an attempt is made to execute this instruction when the processor is not in SMM.

Real Address Mode Exceptions

#UD If an attempt is made to execute this instruction when the processor is not in SMM.

Virtual 8086 Mode Exceptions

#UD If an attempt is made to execute this instruction when the processor is not in SMM.
SAHF—Store AH into Flags

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Clocks</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9E</td>
<td>SAHF</td>
<td>2</td>
<td>Loads SF, ZF, AF, PF, and CF from AH into EFLAGS register</td>
</tr>
</tbody>
</table>

**Description**

Loads the SF, ZF, AF, PF, and CF flags of the EFLAGS register with values from the corresponding bits in the AH register (bits 7, 6, 4, 2, and 0, respectively). Bits 1, 3, and 5 of register AH are ignored; the corresponding reserved bits (1, 3, and 5) in the EFLAGS registers are set as shown in the “Operation” below.

**Operation**

\[
\text{EFLAGS}(\text{SF}:\text{ZF}:0:\text{AF}:0:\text{PF}:1:\text{CF}) \leftarrow \text{AH};
\]

**Flags Affected**

The SF, ZF, AF, PF, and CF flags are loaded with values from the AH register. Bits 1, 3, and 5 of the EFLAGS register are set to 1, 0, and 0, respectively.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  NaT Register Consumption Abort.

**Exceptions (All Operating Modes)**

None.
### Shift Instructions

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D0 /4</td>
<td>SAL r/m8,1</td>
<td>Multiply r/m8 by 2, once</td>
</tr>
<tr>
<td>D2 /4</td>
<td>SAL r/m8,CL</td>
<td>Multiply r/m8 by 2, CL times</td>
</tr>
<tr>
<td>C0 /4 ib</td>
<td>SAL r/m8,imm8</td>
<td>Multiply r/m8 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /4</td>
<td>SAL r/m16,1</td>
<td>Multiply r/m16 by 2, once</td>
</tr>
<tr>
<td>D3 /4</td>
<td>SAL r/m16,CL</td>
<td>Multiply r/m16 by 2, CL times</td>
</tr>
<tr>
<td>C1 /4 ib</td>
<td>SAL r/m16,imm8</td>
<td>Multiply r/m16 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /4</td>
<td>SAL r/m32,1</td>
<td>Multiply r/m32 by 2, once</td>
</tr>
<tr>
<td>D3 /4</td>
<td>SAL r/m32,CL</td>
<td>Multiply r/m32 by 2, CL times</td>
</tr>
<tr>
<td>C1 /4 ib</td>
<td>SAL r/m32,imm8</td>
<td>Multiply r/m32 by 2, imm8 times</td>
</tr>
<tr>
<td>D0 /7</td>
<td>SAR r/m8,1</td>
<td>Signed divide* r/m8 by 2, once</td>
</tr>
<tr>
<td>D2 /7</td>
<td>SAR r/m8,CL</td>
<td>Signed divide* r/m8 by 2, CL times</td>
</tr>
<tr>
<td>C0 /7 ib</td>
<td>SAR r/m8,imm8</td>
<td>Signed divide* r/m8 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /7</td>
<td>SAR r/m16,1</td>
<td>Signed divide* r/m16 by 2, once</td>
</tr>
<tr>
<td>D3 /7</td>
<td>SAR r/m16,CL</td>
<td>Signed divide* r/m16 by 2, CL times</td>
</tr>
<tr>
<td>C1 /7 ib</td>
<td>SAR r/m16,imm8</td>
<td>Signed divide* r/m16 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /7</td>
<td>SAR r/m32,1</td>
<td>Signed divide* r/m32 by 2, once</td>
</tr>
<tr>
<td>D3 /7</td>
<td>SAR r/m32,CL</td>
<td>Signed divide* r/m32 by 2, CL times</td>
</tr>
<tr>
<td>C1 /7 ib</td>
<td>SAR r/m32,imm8</td>
<td>Signed divide* r/m32 by 2, imm8 times</td>
</tr>
<tr>
<td>D0 /4</td>
<td>SHL r/m8,1</td>
<td>Multiply r/m8 by 2, once</td>
</tr>
<tr>
<td>D2 /4</td>
<td>SHL r/m8,CL</td>
<td>Multiply r/m8 by 2, CL times</td>
</tr>
<tr>
<td>C0 /4 ib</td>
<td>SHL r/m8,imm8</td>
<td>Multiply r/m8 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /4</td>
<td>SHL r/m16,1</td>
<td>Multiply r/m16 by 2, once</td>
</tr>
<tr>
<td>D3 /4</td>
<td>SHL r/m16,CL</td>
<td>Multiply r/m16 by 2, CL times</td>
</tr>
<tr>
<td>C1 /4 ib</td>
<td>SHL r/m16,imm8</td>
<td>Multiply r/m16 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /4</td>
<td>SHL r/m32,1</td>
<td>Multiply r/m32 by 2, once</td>
</tr>
<tr>
<td>D3 /4</td>
<td>SHL r/m32,CL</td>
<td>Multiply r/m32 by 2, CL times</td>
</tr>
<tr>
<td>C1 /4 ib</td>
<td>SHL r/m32,imm8</td>
<td>Multiply r/m32 by 2, imm8 times</td>
</tr>
<tr>
<td>D0 /5</td>
<td>SHR r/m8,1</td>
<td>Unsigned divide r/m8 by 2, once</td>
</tr>
<tr>
<td>D2 /5</td>
<td>SHR r/m8,CL</td>
<td>Unsigned divide r/m8 by 2, CL times</td>
</tr>
<tr>
<td>C0 /5 ib</td>
<td>SHR r/m8,imm8</td>
<td>Unsigned divide r/m8 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /5</td>
<td>SHR r/m16,1</td>
<td>Unsigned divide r/m16 by 2, once</td>
</tr>
<tr>
<td>D3 /5</td>
<td>SHR r/m16,CL</td>
<td>Unsigned divide r/m16 by 2, CL times</td>
</tr>
<tr>
<td>C1 /5 ib</td>
<td>SHR r/m16,imm8</td>
<td>Unsigned divide r/m16 by 2, imm8 times</td>
</tr>
<tr>
<td>D1 /5</td>
<td>SHR r/m32,1</td>
<td>Unsigned divide r/m32 by 2, once</td>
</tr>
<tr>
<td>D3 /5</td>
<td>SHR r/m32,CL</td>
<td>Unsigned divide r/m32 by 2, CL times</td>
</tr>
<tr>
<td>C1 /5 ib</td>
<td>SHR r/m32,imm8</td>
<td>Unsigned divide r/m32 by 2, imm8 times</td>
</tr>
</tbody>
</table>

Note:
* Not the same form of division as IDIV; rounding is toward negative infinity.
SAL/SAR/SHL/SHR—Shift Instructions (continued)

Description

Shift the bits in the first operand (destination operand) to the left or right by the number of bits specified in the second operand (count operand). Bits shifted beyond the destination operand boundary are first shifted into the CF flag, then discarded. At the end of the shift operation, the CF flag contains the last bit shifted out of the destination operand.

The destination operand can be a register or a memory location. The count operand can be an immediate value or register CL. The count is masked to 5 bits, which limits the count range to from 0 to 31. A special opcode encoding is provide for a count of 1.

The shift arithmetic left (SAL) and shift logical left (SHL) instructions perform the same operation; they shift the bits in the destination operand to the left (toward more significant bit locations). For each shift count, the most significant bit of the destination operand is shifted into the CF flag, and the least significant bit is cleared.

The shift arithmetic right (SAR) and shift logical right (SHR) instructions shift the bits of the destination operand to the right (toward less significant bit locations). For each shift count, the least significant bit of the destination operand is shifted into the CF flag, and the most significant bit is either set or cleared depending on the instruction type. The SHR instruction clears the most significant bit; the SAR instruction sets or clears the most significant bit to correspond to the sign (most significant bit) of the original value in the destination operand. In effect, the SAR instruction fills the empty bit position’s shifted value with the sign of the unshifted value.

The SAR and SHR instructions can be used to perform signed or unsigned division, respectively, of the destination operand by powers of 2. For example, using the SAR instruction shift a signed integer 1 bit to the right divides the value by 2.

Using the SAR instruction to perform a division operation does not produce the same result as the IDIV instruction. The quotient from the IDIV instruction is rounded toward zero, whereas the “quotient” of the SAR instruction is rounded toward negative infinity. This difference is apparent only for negative numbers. For example, when the IDIV instruction is used to divide -9 by 4, the result is -2 with a remainder of -1. If the SAR instruction is used to shift -9 right by two bits, the result is -3 and the “remainder” is +3; however, the SAR instruction stores only the most significant bit of the remainder (in the CF flag).

The OF flag is affected only on 1-bit shifts. For left shifts, the OF flag is cleared to 0 if the most-significant bit of the result is the same as the CF flag (that is, the top two bits of the original operand were the same); otherwise, it is set to 1. For the SAR instruction, the OF flag is cleared for all 1-bit shifts. For the SHR instruction, the OF flag is set to the most-significant bit of the original operand.

Operation

\[
\text{tempCOUNT} \leftarrow \text{COUNT}; \\
\text{tempDEST} \leftarrow \text{DEST}; \\
\text{WHILE (tempCOUNT} \neq 0) \\
\text{DO} \\
\quad \text{IF instruction is SAL or SHL} \\
\quad \text{THEN} \\
\quad \quad \text{CF} \leftarrow \text{MSB(DEST)};
\]
SAL/SAR/SHL/SHR—Shift Instructions (continued)

ELSE (* instruction is SAR or SHR *)
   CF ← LSB(DEST);
FI;
IF instruction is SAL or SHL
   THEN
      DEST ← DEST * 2;
   ELSE
      IF instruction is SAR
         THEN
            DEST ← DEST / 2 (*Signed divide, rounding toward negative infinity*);
         ELSE (* instruction is SHR *)
            DEST ← DEST / 2 (* Unsigned divide *);
      FI;
   FI;
   IF instruction is SAR
      THEN
         OF ← MSB(DEST) XOR CF;
      ELSE (* instruction is SHR *)
         OF ← MSB(tempDEST);
      FI;
   FI;
   IF instruction is SAR
      THEN
         OF ← MSB(DEST) XOR CF;
   ELSE (* instruction is SHR *)
      OF ← MSB(tempDEST);
   FI;
ELSE
   OF ← undefined;
FI;

Flags Affected

The CF flag contains the value of the last bit shifted out of the destination operand; it is undefined for SHL and SHR instructions count is greater than or equal to the size of the destination operand. The OF flag is affected only for 1-bit shifts (see “Description” above); otherwise, it is undefined. The SF, ZF, and PF flags are set according to the result. If the count is 0, the flags are not affected.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
SAL/SAR/SHL/SHR—Shift Instructions (continued)

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.
   If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
   If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while
      the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment
      limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment
      limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.

Intel Architecture Compatibility

The 8086 does not mask the shift count. All Intel Architecture processors from the Intel386 processor on do mask the rotation count in all operating modes.
SBB—Integer Subtraction with Borrow

Description

Adds the source operand (second operand) and the carry (CF) flag, and subtracts the result from the destination operand (first operand). The result of the subtraction is stored in the destination operand. The destination operand can be a register or a memory location; the source operand can be an immediate, a register, or a memory location. The state of the CF flag represents a borrow from a previous subtraction.

When an immediate value is used as an operand, it is sign-extended to the length of the destination operand format.

The SBB instruction does not distinguish between signed or unsigned operands. Instead, the processor evaluates the result for both data types and sets the OF and CF flags to indicate a borrow in the signed or unsigned result, respectively. The SF flag indicates the sign of the signed result.

The SBB instruction is usually executed as part of a multibyte or multiword subtraction in which a SUB instruction is followed by a SBB instruction.

Operation

\[ \text{DEST} \leftarrow \text{DEST} - (\text{SRC} + \text{CF}) \]

Flags Affected

The OF, SF, ZF, AF, PF, and CF flags are set according to the result.

Additional Itanium System Environment Exceptions

Itanium Reg Faults NaT Register Consumption Abort.
SBB—Integer Subtraction with Borrow (continued)

Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions
#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions
#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
SCAS/SCASB/SCASW/SCASD—Scan String Data

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>AE</td>
<td>SCAS ES:(E)DI</td>
<td>Compare AL with byte at ES:(E)DI and set status flags</td>
</tr>
<tr>
<td>AF</td>
<td>SCAS ES:DI</td>
<td>Compare AX with word at ES:DI and set status flags</td>
</tr>
<tr>
<td>AF</td>
<td>SCAS ES:EDI</td>
<td>Compare EAX with doubleword at ES:EDI and set status flags</td>
</tr>
<tr>
<td>AE</td>
<td>SCASB</td>
<td>Compare AL with byte at ES:(E)DI and set status flags</td>
</tr>
<tr>
<td>AF</td>
<td>SCASW</td>
<td>Compare AX with word at ES:DI and set status flags</td>
</tr>
<tr>
<td>AF</td>
<td>SCASD</td>
<td>Compare EAX with doubleword at ES:EDI and set status flags</td>
</tr>
</tbody>
</table>

Description

Compares the byte, word, or double word specified with the source operand with the value in the AL, AX, or EAX register, respectively, and sets the status flags in the EFLAGS register according to the results. The source operand specifies the memory location at the address ES:EDI. (When the operand-size attribute is 16, the DI register is used as the source-index register.) The ES segment cannot be overridden with a segment override prefix.

The SCASB, SCASW, and SCASD mnemonics are synonyms of the byte, word, and doubleword versions of the SCAS instructions. They are simpler to use, but provide no type or segment checking. (For the SCAS instruction, “ES:EDI” must be explicitly specified in the instruction.)

After the comparison, the EDI register is incremented or decremented automatically according to the setting of the DF flag in the EFLAGS register. (If the DF flag is 0, the EDI register is incremented; if the DF flag is 1, the EDI register is decremented.) The EDI register is incremented or decremented by 1 for byte operations, by 2 for word operations, or by 4 for doubleword operations.

The SCAS, SCASB, SCASW, and SCASD instructions can be preceded by the REP prefix for block comparisons of ECX bytes, words, or doublewords. More often, however, these instructions will be used in a LOOP construct that takes some action based on the setting of the status flags before the next comparison is made. See “REP/REPE/REPZ/REPNE /REPNZ—Repeat String Operation Prefix” on page 3:685 for a description of the REP prefix.

Operation

IF (byte comparison)
   THEN
      temp ← AL – SRC;
      SetStatusFlags(temp);
      THEN IF DF = 0
         THEN (E)DI ← 1;
         ELSE (E)DI ← –1;
      FI;
   ELSE IF (word comparison)
      THEN
         temp ← AX – SRC;
         SetStatusFlags(temp)
         THEN IF DF = 0

SCAS/SCASB/SCASW/SCASD—Scan String Data (continued)

THEN DI ← 2;
ELSE DI ← −2;
FI;
ELSE (* doubleword comparison *)
temp ← EAX − SRC;
SetStatusFlags(temp)
THEN IF DF = 0
    THEN EDI ← 4;
    ELSE EDI ← −4;
FI;
FI;

Flags Affected
The OF, SF, ZF, AF, PF, and CF flags are set according to the temporary result of the comparison.

Additional Itanium System Environment Exceptions
Itanium Reg Faults      NaT Register Consumption Abort.
Itanium Mem Faults     VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB
                       Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data
                       Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data
                       Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions
#GP(0)                  If a memory operand effective address is outside the limit of the ES segment.
If the ES register contains a null segment selector.
If an illegal memory operand effective address in the ES segment is given.
#PF(fault-code)          If a page fault occurs.
#AC(0)                  If alignment checking is enabled and an unaligned memory reference is made
while the current privilege level is 3.

Real Address Mode Exceptions
#GP                     If a memory operand effective address is outside the CS, DS, ES, FS, or GS
                       segment limit.
#SS                     If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions
#GP(0)                  If a memory operand effective address is outside the CS, DS, ES, FS, or GS
                       segment limit.
#SS(0)                  If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)         If a page fault occurs.
#AC(0)                  If alignment checking is enabled and an unaligned memory reference is made.
SETcc—Set Byte on Condition

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 97</td>
<td>SETA r/m8</td>
<td>Set byte if above (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>0F 93</td>
<td>SETAE r/m8</td>
<td>Set byte if above or equal (CF=0)</td>
</tr>
<tr>
<td>0F 92</td>
<td>SETB r/m8</td>
<td>Set byte if below (CF=1)</td>
</tr>
<tr>
<td>0F 96</td>
<td>SETBE r/m8</td>
<td>Set byte if below or equal (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 92</td>
<td>SETC r/m8</td>
<td>Set if carry (CF=1)</td>
</tr>
<tr>
<td>0F 94</td>
<td>SETE r/m8</td>
<td>Set byte if equal (ZF=1)</td>
</tr>
<tr>
<td>0F 9F</td>
<td>SETG r/m8</td>
<td>Set byte if greater (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>0F 9D</td>
<td>SETGE r/m8</td>
<td>Set byte if greater or equal (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 9C</td>
<td>SETL r/m8</td>
<td>Set byte if less (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 9E</td>
<td>SETLE r/m8</td>
<td>Set byte if less or equal (ZF=1 or SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 96</td>
<td>SETNA r/m8</td>
<td>Set byte if not above (CF=1 or ZF=1)</td>
</tr>
<tr>
<td>0F 92</td>
<td>SETNAE r/m8</td>
<td>Set byte if not above or equal (CF=1)</td>
</tr>
<tr>
<td>0F 93</td>
<td>SETNB r/m8</td>
<td>Set byte if not below (CF=0)</td>
</tr>
<tr>
<td>0F 97</td>
<td>SETNBE r/m8</td>
<td>Set byte if not below or equal (CF=0 and ZF=0)</td>
</tr>
<tr>
<td>0F 93</td>
<td>SETNC r/m8</td>
<td>Set byte if not carry (CF=0)</td>
</tr>
<tr>
<td>0F 95</td>
<td>SETNE r/m8</td>
<td>Set byte if not equal (ZF=0)</td>
</tr>
<tr>
<td>0F 9E</td>
<td>SETNG r/m8</td>
<td>Set byte if not greater (ZF=1 or SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 9C</td>
<td>SETNGE r/m8</td>
<td>Set byte if not greater or equal (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 9D</td>
<td>SETNL r/m8</td>
<td>Set byte if not less (SF&lt;&gt;OF)</td>
</tr>
<tr>
<td>0F 9F</td>
<td>SETNLE r/m8</td>
<td>Set byte if not less or equal (ZF=0 and SF=OF)</td>
</tr>
<tr>
<td>0F 91</td>
<td>SETNO r/m8</td>
<td>Set byte if not overflow (OF=0)</td>
</tr>
<tr>
<td>0F 9B</td>
<td>SETNP r/m8</td>
<td>Set byte if not parity (PF=0)</td>
</tr>
<tr>
<td>0F 99</td>
<td>SETNS r/m8</td>
<td>Set byte if not sign (SF=0)</td>
</tr>
<tr>
<td>0F 95</td>
<td>SETNZ r/m8</td>
<td>Set byte if not zero (ZF=0)</td>
</tr>
<tr>
<td>0F 90</td>
<td>SETO r/m8</td>
<td>Set byte if overflow (OF=1)</td>
</tr>
<tr>
<td>0F 9A</td>
<td>SETP r/m8</td>
<td>Set byte if parity (PF=1)</td>
</tr>
<tr>
<td>0F 9A</td>
<td>SETPE r/m8</td>
<td>Set byte if parity even (PF=1)</td>
</tr>
<tr>
<td>0F 9B</td>
<td>SETPO r/m8</td>
<td>Set byte if parity odd (PF=0)</td>
</tr>
<tr>
<td>0F 98</td>
<td>SETS r/m8</td>
<td>Set byte if sign (SF=1)</td>
</tr>
<tr>
<td>0F 94</td>
<td>SETZ r/m8</td>
<td>Set byte if zero (ZF=1)</td>
</tr>
</tbody>
</table>

Description

Set the destination operand to the value 0 or 1, depending on the settings of the status flags (CF, SF, OF, ZF, and PF) in the EFLAGS register. The destination operand points to a byte register or a byte in memory. The condition code suffix (cc) indicates the condition being tested for.

The terms “above” and “below” are associated with the CF flag and refer to the relationship between two unsigned integer values. The terms “greater” and “less” are associated with the SF and OF flags and refer to the relationship between two signed integer values.
SETcc—Set Byte on Condition (continued)

Many of the SETcc instruction opcodes have alternate mnemonics. For example, the SETG (set byte if greater) and SETNLE (set if not less or equal) both have the same opcode and test for the same condition: ZF equals 0 and SF equals OF. These alternate mnemonics are provided to make code more intelligible.

Some languages represent a logical one as an integer with all bits set. This representation can be arrived at by choosing the mutually exclusive condition for the SETcc instruction, then decrementing the result. For example, to test for overflow, use the SETNO instruction, then decrement the result.

**Operation**

```plaintext
IF condition
    THEN DEST ← 1
    ELSE DEST ← 0;
FI;
```

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault</td>
</tr>
</tbody>
</table>

**Protected Mode Exceptions**

<table>
<thead>
<tr>
<th>#GP(0)</th>
<th>If the destination is located in a nonwritable segment.</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.</td>
</tr>
<tr>
<td></td>
<td>If the DS, ES, FS, or GS register contains a null segment selector.</td>
</tr>
<tr>
<td>#SS(0)</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.</td>
</tr>
</tbody>
</table>

**Real Address Mode Exceptions**

| #GP       | If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit. |
| #SS       | If a memory operand effective address is outside the SS segment limit. |
SETcc—Set Byte on Condition (continued)

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.

SGDT/SIDT—Store Global/Interrupt Descriptor Table Register

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 01 /0</td>
<td>SGDT m</td>
<td>Store GDTR to m</td>
</tr>
<tr>
<td>0F 01 /1</td>
<td>SIDT m</td>
<td>Store IDTR to m</td>
</tr>
</tbody>
</table>

Description

Stores the contents of the global descriptor table register (GDTR) or the interrupt descriptor table register (IDTR) in the destination operand. The destination operand is a pointer to 6-byte memory location. If the operand-size attribute is 32 bits, the 16-bit limit field of the register is stored in the lower 2 bytes of the memory location and the 32-bit base address is stored in the upper 4 bytes. If the operand-size attribute is 16 bits, the limit is stored in the lower 2 bytes and the 24-bit base address is stored in the third, fourth, and fifth byte, with the sixth byte filled with 0s.

The SGDT and SIDT instructions are useful only in operating-system software; however, they can be used in application programs.

Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,SGDT/SIDT);

IF instruction is IDTR THEN
  IF OperandSize = 16 THEN
    DEST[0:15] ← IDTR(Limit);
    DEST[16:39] ← IDTR(Base); (* 24 bits of base address loaded; *)
    DEST[40:47] ← 0;
    ELSE (* 32-bit Operand Size *)
    DEST[0:15] ← IDTR(Limit);
    DEST[16:47] ← IDTR(Base); (* full 32-bit base address loaded *)
  FI;
ELSE (* instruction is SGDT *)
  IF OperandSize = 16 THEN
    DEST[0:15] ← GDTR(Limit);
    DEST[16:39] ← GDTR(Base); (* 24 bits of base address loaded; *)
    DEST[40:47] ← 0;
    ELSE (* 32-bit Operand Size *)
    DEST[0:15] ← GDTR(Limit);
    DEST[16:47] ← GDTR(Base); (* full 32-bit base address loaded *)
  FI;
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Intercept Instruction Intercept for SIDT and SGDT.
SGDT/ SIDT—Store Global/Interrupt Descriptor Table Register (continued)

Protected Mode Exceptions

#UD If the destination operand is a register.
#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If an unaligned memory access occurs when the CPL is 3 and alignment checking is enabled.

Real Address Mode Exceptions

#UD If the destination operand is a register.
#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#UD If the destination operand is a register.
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If an unaligned memory access occurs when alignment checking is enabled.

Intel Architecture Compatibility

The 16-bit forms of the SGDT and SIDT instructions are compatible with the Intel 286 processor, if the upper 8 bits are not referenced. The Intel 286 processor fills these bits with 1s; the Pentium Pro processor fills these bits with 0s.
SHL/SHR—Shift Instructions

See entry for SAL/SAR/SHL/SHR.
SHLD—Double Precision Shift Left

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F A4</td>
<td>SHLD r/m16,r16,imm8</td>
<td>Shift r/m16 to left imm8 places while shifting bits from r16 in from the right</td>
</tr>
<tr>
<td>0F A5</td>
<td>SHLD r/m16,r16,CL</td>
<td>Shift r/m16 to left CL places while shifting bits from r16 in from the right</td>
</tr>
<tr>
<td>0F A4</td>
<td>SHLD r/m32,r32,imm8</td>
<td>Shift r/m32 to left imm8 places while shifting bits from r32 in from the right</td>
</tr>
<tr>
<td>0F A5</td>
<td>SHLD r/m32,r32,CL</td>
<td>Shift r/m32 to left CL places while shifting bits from r32 in from the right</td>
</tr>
</tbody>
</table>

**Description**

Shifts the first operand (destination operand) to the left the number of bits specified by the third operand (count operand). The second operand (source operand) provides bits to shift in from the right (starting with bit 0 of the destination operand). The destination operand can be a register or a memory location; the source operand is a register. The count operand is an unsigned integer that can be an immediate byte or the contents of the CL register. Only bits 0 through 4 of the count are used, which masks the count to a value between 0 and 31. If the count is greater than the operand size, the result in the destination operand is undefined.

If the count is 1 or greater, the CF flag is filled with the last bit shifted out of the destination operand. For a 1-bit shift, the OF flag is set if a sign change occurred; otherwise, it is cleared. If the count operand is 0, the flags are not affected.

The SHLD instruction is useful for multi-precision shifts of 64 bits or more.

**Operation**

\[
\begin{align*}
\text{COUNT} & \leftarrow \text{COUNT MOD 32}; \\
\text{SIZE} & \leftarrow \text{OperandSize} \\
\text{IF } \text{COUNT} = 0 & \text{ THEN no operation} \\
\text{ELSE} & \\
& \text{IF } \text{COUNT} \geq \text{SIZE} \\
& \text{ THEN (* Bad parameters *)} \\
& \quad \text{DEST is undefined;} \\
& \quad \text{CF, OF, SF, ZF, AF, PF are undefined;} \\
& \text{ELSE (* Perform the shift *)} \\
& \quad \text{CF} \leftarrow \text{BIT}[	ext{DEST}, \text{SIZE} – \text{COUNT}]; \\
& \quad (* \text{Last bit shifted out on exit *}) \text{ FOR } i \leftarrow \text{SIZE} – 1 \text{ DOWNTO COUNT} \text{ DO} \\
& \quad \quad \text{BIT}[(\text{DEST}, i) \leftarrow \text{BIT}[(\text{DEST}, i – \text{COUNT})]; \text{ OD}; \text{ FOR } i \leftarrow \text{COUNT} – 1 \text{ DOWNTO 0}\]
\]
SHLD—Double Precision Shift Left (continued)

```
DO
  BIT[DEST, i] ← BIT[SRC, i − COUNT + SIZE];
OD;
FI;
FI;
```

Flags Affected

If the count is 1 or greater, the CF flag is filled with the last bit shifted out of the destination operand and the SF, ZF, and PF flags are set according to the value of the result. For a 1-bit shift, the OF flag is set if a sign change occurred; otherwise, it is cleared. For shifts greater than 1 bit, the OF flag is undefined. If a shift occurs, the AF flag is undefined. If the count operand is 0, the flags are not affected. If the count is greater than the operand size, the flags are undefined.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

- #GP(0): If the destination is located in a nonwritable segment.
  - If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register contains a null segment selector.
- #SS(0): If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code): If a page fault occurs.
- #AC(0): If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

- #GP: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- #SS: If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

- #GP(0): If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- #SS(0): If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code): If a page fault occurs.
- #AC(0): If alignment checking is enabled and an unaligned memory reference is made.
SHRD—Double Precision Shift Right

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F AC</td>
<td>SHRD r/m16, r16, imm8</td>
<td>Shift r/m16 to right imm8 places while shifting bits from r16 in from the left</td>
</tr>
<tr>
<td>0F AD</td>
<td>SHRD r/m16, r16, CL</td>
<td>Shift r/m16 to right CL places while shifting bits from r16 in from the left</td>
</tr>
<tr>
<td>0F AC</td>
<td>SHRD r/m32, r32, imm8</td>
<td>Shift r/m32 to right imm8 places while shifting bits from r32 in from the left</td>
</tr>
<tr>
<td>0F AD</td>
<td>SHRD r/m32, r32, CL</td>
<td>Shift r/m32 to right CL places while shifting bits from r32 in from the left</td>
</tr>
</tbody>
</table>

Description

Shifts the first operand (destination operand) to the right the number of bits specified by the third operand (count operand). The second operand (source operand) provides bits to shift in from the left (starting with the most significant bit of the destination operand). The destination operand can be a register or a memory location; the source operand is a register. The count operand is an unsigned integer that can be an immediate byte or the contents of the CL register. Only bits 0 through 4 of the count are used, which masks the count to a value between 0 and 31. If the count is greater than the operand size, the result in the destination operand is undefined.

If the count is 1 or greater, the CF flag is filled with the last bit shifted out of the destination operand. For a 1-bit shift, the OF flag is set if a sign change occurred; otherwise, it is cleared. If the count operand is 0, the flags are not affected.

The SHRD instruction is useful for multiprecision shifts of 64 bits or more.

Operation

\[
\begin{align*}
\text{COUNT} & \leftarrow \text{COUNT MOD 32}; \\
\text{SIZE} & \leftarrow \text{OperandSize} \\
\text{IF COUNT} & = 0 \\
& \quad \text{THEN} \\
& \quad \quad \text{no operation} \\
\text{ELSE} & \\
& \quad \quad \text{IF COUNT} \geq \text{SIZE} \\
& \quad \quad \quad \text{THEN} (* \text{ Bad parameters } *) \\
& \quad \quad \quad \quad \text{DEST is undefined;} \\
& \quad \quad \quad \quad \text{CF, OF, SF, ZF, AF, PF are undefined;} \\
& \quad \quad \quad \text{ELSE } (* \text{ Perform the shift } *) \\
& \quad \quad \quad \quad \text{CF} \leftarrow \text{BIT}[\text{DEST, COUNT} - 1]; (* \text{ last bit shifted out on exit } *) \\
& \quad \quad \quad \quad \text{FOR } i \leftarrow 0 \text{ TO SIZE} - 1 - \text{COUNT} \\
& \quad \quad \quad \quad \quad \text{DO} \\
& \quad \quad \quad \quad \quad \quad \text{BIT}[\text{DEST}, i] \leftarrow \text{BIT}[\text{DEST}, i - \text{COUNT}]; \\
& \quad \quad \quad \quad \quad \text{OD;} \\
& \quad \quad \quad \quad \text{FOR } i \leftarrow \text{SIZE} - \text{COUNT} \text{ TO SIZE} - 1 \\
& \quad \quad \quad \quad \quad \text{DO} \\
& \quad \quad \quad \quad \quad \quad \text{BIT}[\text{DEST}, i] \leftarrow \text{BIT}[\text{inBits}, i + \text{COUNT} - \text{SIZE}]; \\
& \quad \quad \quad \quad \quad \text{OD;} \\
& \quad \quad \quad \quad \text{FI;} \\
& \quad \quad \text{FI;} \\
\end{align*}
\]
SHRD—Double Precision Shift Right (continued)

**Flags Affected**

If the count is 1 or greater, the CF flag is filled with the last bit shifted out of the destination operand and the SF, ZF, and PF flags are set according to the value of the result. For a 1-bit shift, the OF flag is set if a sign change occurred; otherwise, it is cleared. For shifts greater than 1 bit, the OF flag is undefined. If a shift occurs, the AF flag is undefined. If the count operand is 0, the flags are not affected. If the count is greater than the operand size, the flags are undefined.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Type</th>
<th>Exceptions</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Reg Faults</td>
<td>NaT Register Consumption Abort.</td>
</tr>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault,</td>
</tr>
<tr>
<td></td>
<td>Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss</td>
</tr>
<tr>
<td></td>
<td>Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access</td>
</tr>
<tr>
<td></td>
<td>Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

**Protected Mode Exceptions**

- **#GP(0)**: If the destination is located in a nonwritable segment.
- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)**: If the DS, ES, FS, or GS register contains a null segment selector.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

- **#GP**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS**: If a memory operand effective address is outside the SS segment limit.

**Virtual 8086 Mode Exceptions**

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made.
SIDT—Store Interrupt Descriptor Table Register

See entry for SGDT/SIDT.
SLDT—Store Local Descriptor Table Register

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 00 /0</td>
<td>SLDT r/m16</td>
<td>Stores segment selector from LDTR in r/m16</td>
</tr>
<tr>
<td>0F 00 /0</td>
<td>SLDT r/m32</td>
<td>Store segment selector from LDTR in low-order 16 bits of r/m32; high-order 16 bits are undefined</td>
</tr>
</tbody>
</table>

Description

Stores the segment selector from the local descriptor table register (LDTR) in the destination operand. The destination operand can be a general-purpose register or a memory location. The segment selector stored with this instruction points to the LDT.

When the destination operand is a 32-bit register, the 16-bit segment selector is copied into the lower 16 bits of the register and the upper 16 bits of the register are cleared to 0s. With the destination operand is a memory location, the segment selector is written to memory as a 16-bit quantity, regardless of the operand size.

The SLDT instruction is only useful in operating-system software; however, it can be used in application programs. Also, this instruction can only be executed in protected mode.

Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,SLDT);
DEST ← LDTR(SegmentSelector);

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Intercept SLDT results in an IA-32 Intercept

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
SLDT—Store Local Descriptor Table Register (continued)

Real Address Mode Exceptions

#UD  The SLDT instruction is not recognized in real address mode.

Virtual 8086 Mode Exceptions

#UD  The SLDT instruction is not recognized in virtual 8086 mode.
**SMSW—Store Machine Status Word**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 01 /4</td>
<td>SMSW r32/m16</td>
<td>Store machine status word in low-order 16 bits of r32/m16; high-order 16 bits of r32 are undefined</td>
</tr>
</tbody>
</table>

**Description**

Stores the machine status word (bits 0 through 15 of control register CR0) into the destination operand. The destination operand can be a 16-bit general-purpose register or a memory location.

When the destination operand is a 32-bit register, the low-order 16 bits of register CR0 are copied into the low-order 16 bits of the register and the upper 16 bits of the register are undefined. With the destination operand a memory location, the low-order 16 bits of register CR0 are written to memory as a 16-bit quantity, regardless of the operand size.

The SMSW instruction is only useful in operating-system software; however, it is not a privileged instruction and can be used in application programs.

This instruction is provided for compatibility with the Intel 286 processor; programs and procedures intended to run on processors more recent than the Intel 286 should use the MOV (control registers) instruction to load the machine status word.

**Operation**

```markdown
IF Itanium System Environment THEN IA-32_Intercept(INST, SMSW);
DEST ← CR0[15:0]; (* MachineStatusWord *);
```

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

IA-32_Intercept  Mandatory Instruction Intercept.

**Protected Mode Exceptions**

- **#GP(0)**  If the destination is located in a nonwritable segment.
- If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  - If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
- **#SS(0)**  If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)**  If a page fault occurs.
- **#AC(0)**  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
SMSW—Store Machine Status Word (continued)

Real Address Mode Exceptions
#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

Virtual 8086 Mode Exceptions
#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
STC—Set Carry Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F9</td>
<td>STC</td>
<td>Set CF flag</td>
</tr>
</tbody>
</table>

**Description**

Sets the CF flag in the EFLAGS register.

**Operation**

\[
CF \leftarrow 1;
\]

**Flags Affected**

The CF flag is set. The OF, ZF, SF, AF, and PF flags are unaffected.

**Exceptions (All Operating Modes)**

None.
STD—Set Direction Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FD</td>
<td>STD</td>
<td>Set DF flag</td>
</tr>
</tbody>
</table>

Description
Sets the DF flag in the EFLAGS register. When the DF flag is set to 1, string operations decrement the index registers (ESI and/or EDI).

Operation
DF ← 1;

Flags Affected
The DF flag is set. The CF, OF, ZF, SF, AF, and PF flags are unaffected.

Operation
DF ← 1;

Exceptions (All Operating Modes)
None.
STI—Set Interrupt Flag

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>FB</td>
<td>STI</td>
<td>Set interrupt flag; interrupts enabled at the end of the next instruction</td>
</tr>
</tbody>
</table>

**Description**

Sets the interrupt flag (IF) in the EFLAGS register. **In the IA-32 System Environment**, after the IF flag is set, the processor begins responding to external maskable interrupts after the next instruction is executed. If the STI instruction is followed by a CLI instruction (which clears the IF flag) the effect of the STI instruction is negated. **In the Itanium System Environment**, the processor will immediately respond do interrupts after STI, unless execution of STI results in a trap or intercept. External interrupts are enabled for IA-32 instructions if PSR.i and (~CFLG.if or EFLAGS.if).

The IF flag and the STI and CLI instruction have no affect on the generation of exceptions and NMI interrupts.

The following decision table indicates the action of the STI instruction (bottom of the table) depending on the processor’s mode of operating and the CPL and IOPL of the currently running program or procedure (top of the table).

```
<table>
<thead>
<tr>
<th>PE</th>
<th>0</th>
<th>1</th>
<th>1</th>
<th>1</th>
</tr>
</thead>
<tbody>
<tr>
<td>VM</td>
<td>X</td>
<td>0</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>CPL</td>
<td>X</td>
<td>≤ IOPL</td>
<td>&gt; IOPL</td>
<td>=3</td>
</tr>
<tr>
<td>IOPL</td>
<td>X</td>
<td>X</td>
<td>X</td>
<td>=3</td>
</tr>
<tr>
<td>IF ← 1</td>
<td>Y</td>
<td>Y</td>
<td>N</td>
<td>Y</td>
</tr>
<tr>
<td>#GP(0)</td>
<td>N</td>
<td>N</td>
<td>Y</td>
<td>N</td>
</tr>
</tbody>
</table>
```

**Notes:**

- X: Don’t care.
- N: Action in Column 1 not taken.
- Y: Action in Column 1 taken.

**Operation**

```
OLD_IF ← IF;
IF PE = 0 (* Executing in real-address mode *)
    THEN
        IF ← 1; (* Set Interrupt Flag *)
    ELSE (* Executing in protected mode or virtual-8086 mode *)
        IF VM = 0 (* Executing in protected mode*)
            THEN
                IF CR4.PVI = 0
                    THEN
                        IF CPL <= IOPL
                            THEN IF ← 1
                        ELSE #GP(0);
                    FI;
            ELSE (*PVI is 1 *)
```

---

_volume: 3
_page_number: 721_
STI—Set Interrupt Flag (continued)

```
IF CPL = 3
THEN STI—Set Interrupt Flag (continued)
    IF IOPL < 3
    THEN
        IF VIP = 0
        THEN VIF <- 1;
        ELSE #GP(0);
        FI;
        ELSE (*IOPL = 3 *)
        IF <- 1;
        FI;
    ELSE (*CPL < 3*)
        IF IOPL < CPL THEN #GP(0); FI;
        ELSE (*CPL = 3*)
        IF <- 1;
        ELSE (*CPL = 3*)
            IF IOPL = 0
            THEN IF <- 1;
            ELSE
                IF CR4.VME = 0
                THEN #GP(0);
                ELSE
                    IF VIP = 1 (*virtual interrupt is pending*)
                    THEN #GP(0);
                    ELSE VIF <- 1;
                    FI;
                    FI;
                FI;
                FI;
            FI;
        FI;
    FI;
ELSE (*Executing in Virtual-8086 Mode*)
    IF IOPL = 3
    THEN IF <- 1;
    ELSE
        IF VIP = 0
        THEN VIF <- 1;
        ELSE #GP(0);
        FI;
        ELSE (*CPL < 3*)
            IF IOPL < CPL THEN #GP(0); FI;
            ELSE (*CPL = 3*)
                IF <- 1;
                ELSE (*CPL = 3*)
                    IF IOPL = 0
                    THEN IF <- 1;
                    ELSE
                        IF CR4.VME = 0
                        THEN #GP(0);
                        ELSE
                            IF VIP = 1 (*virtual interrupt is pending*)
                            THEN #GP(0);
                            ELSE VIF <- 1;
                            FI;
                            FI;
                        FI;
                    FI;
                FI;
            FI;
        FI;
    FI;
```

```
IF Itanium System Environment AND CFLG.ii AND IF != OLD_IF
THEN IA-32_Intercept(System_Flag,STI);
```

**Flags Affected**

The IF flag is set to 1.

**Additional Itanium System Environment Exceptions**

IA-32_Intercept System Flag Intercept Trap if CFLG.ii is 1 and the IF flag changes state.

**Protected Mode Exceptions**

#GP(0) If the CPL is greater (has less privilege) than the IOPL of the current program or procedure.
STI—Set Interrupt Flag (continued)

Real Address Mode Exceptions

None.

Virtual 8086 Mode Exceptions

#GP(0) If the CPL is greater (has less privilege) than the IOPL of the current program or procedure.
STOS/STOSB/STOSW/STOSD—Store String Data

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>AA</td>
<td>STOS ES:(E)DI</td>
<td>Store AL at address ES:(E)DI</td>
</tr>
<tr>
<td>AB</td>
<td>STOS ES:DI</td>
<td>Store AX at address ES:DI</td>
</tr>
<tr>
<td>AB</td>
<td>STOS ES:EDI</td>
<td>Store EAX at address ES:EDI</td>
</tr>
<tr>
<td>AA</td>
<td>STOSB</td>
<td>Store AL at address ES:(E)DI</td>
</tr>
<tr>
<td>AB</td>
<td>STOSW</td>
<td>Store AX at address ES:DI</td>
</tr>
<tr>
<td>AB</td>
<td>STOSD</td>
<td>Store EAX at address ES:EDI</td>
</tr>
</tbody>
</table>

**Description**

Stores a byte, word, or doubleword from the AL, AX, or EAX register, respectively, into the destination operand. The destination operand is a memory location at the address ES:EDI. (When the operand-size attribute is 16, the DI register is used as the source-index register.) The ES segment cannot be overridden with a segment override prefix.

The STOSB, STOSW, and STOSD mnemonics are synonyms of the byte, word, and doubleword versions of the STOS instructions. They are simpler to use, but provide no type or segment checking. (For the STOS instruction, “ES:EDI” must be explicitly specified in the instruction.)

After the byte, word, or doubleword is transfer from the AL, AX, or EAX register to the memory location, the EDI register is incremented or decremented automatically according to the setting of the DF flag in the EFLAGS register. (If the DF flag is 0, the EDI register is incremented; if the DF flag is 1, the EDI register is decremented.) The EDI register is incremented or decremented by 1 for byte operations, by 2 for word operations, or by 4 for doubleword operations.

The STOS, STOSB, STOSW, and STOSD instructions can be preceded by the REP prefix for block loads of ECX bytes, words, or doublewords. More often, however, these instructions are used within a LOOP construct, because data needs to be moved into the AL, AX, or EAX register before it can be stored. See “REP/REPE/REPZ/REPNZ—Repeat String Operation Prefix” on page 3:685 for a description of the REP prefix.

**Operation**

```plaintext
IF (byte store)
THEN
    DEST ← AL;
    THEN IF DF = 0
        THEN (E)DI ← 1;
        ELSE (E)DI ← −1;
    Fi;
ELSE IF (word store)
THEN
    DEST ← AX;
    THEN IF DF = 0
        THEN DI ← 2;
        ELSE DI ← −2;
    Fi;
ELSE (* doubleword store *)
    DEST ← EAX;
    THEN IF DF = 0
```
STOS/STOSB/STOSW/STOSD—Store String Data (continued)

THEN EDI ← 4;
ELSE EDI ← −4;
FI;
FI;
FI;
FI;

Flags Affected
None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults: NaT Register Consumption Abort.
Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If the destination is located in a nonwritable segment.
If a memory operand effective address is outside the limit of the ES segment.
If the ES register contains a null segment selector.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
STR—Store Task Register

Description

Stores the segment selector from the task register (TR) in the destination operand. The destination operand can be a general-purpose register or a memory location. The segment selector stored with this instruction points to the task state segment (TSS) for the currently running task.

When the destination operand is a 32-bit register, the 16-bit segment selector is copied into the lower 16 bits of the register and the upper 16 bits of the register are cleared to 0s. With the destination operand is a memory location, the segment selector is written to memory as a 16-bit quantity, regardless of operand size.

The STR instruction is useful only in operating-system software. It can only be executed in protected mode.

Operation

IF Itanium System Environment THEN IA-32_Intercept(INST,STR);
DEST ← TR(SegmentSelector);

Flags Affected

None.

Additional Itanium System Environment Exceptions

IA-32_Intercept Mandatory Instruction Intercept.

Protected Mode Exceptions

#GP(0) If the destination is a memory operand that is located in a nonwritable segment or if the effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#UD The STR instruction is not recognized in real address mode.

Virtual 8086 Mode Exceptions

#UD The STR instruction is not recognized in virtual 8086 mode.
SUB—Integer Subtraction

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>2C ib</td>
<td>SUB AL,imm8</td>
<td>Subtract imm8 from AL</td>
</tr>
<tr>
<td>2D iw</td>
<td>SUB AX,imm16</td>
<td>Subtract imm16 from AX</td>
</tr>
<tr>
<td>2D id</td>
<td>SUB EAX,imm32</td>
<td>Subtract imm32 from EAX</td>
</tr>
<tr>
<td>80 /5 ib</td>
<td>SUB r/m8,imm8</td>
<td>Subtract imm8 from r/m8</td>
</tr>
<tr>
<td>81 /5 iw</td>
<td>SUB r/m16,imm16</td>
<td>Subtract imm16 from r/m16</td>
</tr>
<tr>
<td>81 /5 id</td>
<td>SUB r/m32,imm32</td>
<td>Subtract imm32 from r/m32</td>
</tr>
<tr>
<td>83 /5 ib</td>
<td>SUB r/m16,imm8</td>
<td>Subtract sign-extended imm8 from r/m16</td>
</tr>
<tr>
<td>83 /5 ib</td>
<td>SUB r/m32,imm8</td>
<td>Subtract sign-extended imm8 from r/m32</td>
</tr>
<tr>
<td>28 /r</td>
<td>SUB r/m8,r8</td>
<td>Subtract r8 from r/m8</td>
</tr>
<tr>
<td>29 /r</td>
<td>SUB r/m16,r16</td>
<td>Subtract r16 from r/m16</td>
</tr>
<tr>
<td>29 /r</td>
<td>SUB r/m32,r32</td>
<td>Subtract r32 from r/m32</td>
</tr>
<tr>
<td>2A /r</td>
<td>SUB r8,r/m8</td>
<td>Subtract r/m8 from r8</td>
</tr>
<tr>
<td>2B /r</td>
<td>SUB r16,r/m16</td>
<td>Subtract r/m16 from r16</td>
</tr>
<tr>
<td>2B /r</td>
<td>SUB r32,r/m32</td>
<td>Subtract r/m32 from r32</td>
</tr>
</tbody>
</table>

Description

Subtracts the second operand (source operand) from the first operand (destination operand) and stores the result in the destination operand. The destination operand can be a register or a memory location; the source operand can be an immediate, register, or memory location. When an immediate value is used as an operand, it is sign-extended to the length of the destination operand format.

The SUB instruction does not distinguish between signed or unsigned operands. Instead, the processor evaluates the result for both data types and sets the OF and CF flags to indicate a borrow in the signed or unsigned result, respectively. The SF flag indicates the sign of the signed result.

Operation

DEST ← DEST – SRC;

Flags Affected

The OF, SF, ZF, AF, PF, and CF flags are set according to the result.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
SUB—Integer Subtraction (continued)

**Protected Mode Exceptions**

#GP(0)  
If the destination is located in a nonwritable segment.

If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0)  
If a memory operand effective address is outside the SS segment limit.

#PF(fault-code)  
If a page fault occurs.

#AC(0)  
If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real Address Mode Exceptions**

#GP  
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS  
If a memory operand effective address is outside the SS segment limit.

**Virtual 8086 Mode Exceptions**

#GP(0)  
If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

#SS(0)  
If a memory operand effective address is outside the SS segment limit.

#PF(fault-code)  
If a page fault occurs.

#AC(0)  
If alignment checking is enabled and an unaligned memory reference is made.
TEST—Logical Compare

### Description

Computes the bit-wise logical AND of first operand (source 1 operand) and the second operand (source 2 operand) and sets the SF, ZF, and PF status flags according to the result. The result is then discarded.

### Operation

\[
\text{TEMP} \leftarrow \text{SRC1 AND SRC2}; \\
\text{SF} \leftarrow \text{MSB}(\text{TEMP}); \\
\text{IF} \ \text{TEMP} = 0 \\
\quad \text{THEN} \ \text{ZF} \leftarrow 0; \\
\quad \text{ELSE} \ \text{ZF} \leftarrow 1; \\
\text{FI}; \\
\text{PF} \leftarrow \text{BitwiseXNOR}(\text{TEMP}[0:7]); \\
\text{CF} \leftarrow 0; \\
\text{OF} \leftarrow 0; \\
({}^{*} \text{AF} \text{ is Undefined} *)
\]

### Flags Affected

The OF and CF flags are cleared to 0. The SF, ZF, and PF flags are set according to the result (see “Operation” above). The state of the AF flag is undefined.

### Additional Itanium System Environment Exceptions

- **Itanium Reg Faults**: NaN Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaN Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
TEST—Logical Compare (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
            If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
UD2—Undefined Instruction

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 0B</td>
<td>UD2</td>
<td>Raise invalid opcode exception</td>
</tr>
</tbody>
</table>

Description
Generates an invalid opcode. This instruction is provided for software testing to explicitly generate an invalid opcode. The opcode for this instruction is reserved for this purpose.

Other than raising the invalid opcode exception, this instruction is the same as the NOP instruction.

Operation
IF Itanium System Environment THEN IA-32_Intercept(INST,0F0B);
#UD (* Generates invalid opcode exception *);

Flags Affected
None.

Additional Itanium System Environment Exceptions
IA-32_Intercept Mandatory Instruction Intercept.

Exceptions (All Operating Modes)
#UD Instruction is guaranteed to raise an invalid opcode exception in all operating modes).
VERR, VERW—Verify a Segment for Reading or Writing

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 00 /4</td>
<td>VERR r/m16</td>
<td>Set ZF=1 if segment specified with r/m16 can be read</td>
</tr>
<tr>
<td>0F 00 /5</td>
<td>VERW r/m16</td>
<td>Set ZF=1 if segment specified with r/m16 can be written</td>
</tr>
</tbody>
</table>

Description

Verifies whether the code or data segment specified with the source operand is readable (VERR) or writable (VERW) from the current privilege level (CPL). The source operand is a 16-bit register or a memory location that contains the segment selector for the segment to be verified. If the segment is accessible and readable (VERR) or writable (VERW), the ZF flag is set; otherwise, the ZF flag is cleared. Code segments are never verified as writable. This check cannot be performed on system segments.

To set the ZF flag, the following conditions must be met:

- The segment selector is not null.
- The selector must denote a descriptor within the bounds of the descriptor table (GDT or LDT).
- The selector must denote the descriptor of a code or data segment (not that of a system segment or gate).
- For the VERR instruction, the segment must be readable; the VERW instruction, the segment must be a writable data segment.
- If the segment is not a conforming code segment, the segment’s DPL must be greater than or equal to (have less or the same privilege as) both the CPL and the segment selector’s RPL.

The validation performed is the same as if the segment were loaded into the DS, ES, FS, or GS register, and the indicated access (read or write) were performed. The selector's value cannot result in a protection exception, enabling the software to anticipate possible segment access problems.

Operation

IF SRC(Offset) > (GDTR(Limit) OR (LDTR(Limit))
THEN
    ZF ← 0
Read segment descriptor;
IF SegmentDescriptor(DescriptorType) = 0 (* system segment *)
    OR (SegmentDescriptor(Type) ≠ conforming code segment)
    AND (CPL > DPL) OR (RPL > DPL)
THEN
    ZF ← 0
ELSE
    IF (((Instruction = VERR) AND (segment = readable))
        OR (((Instruction = VERW) AND (segment = writable))
        THEN
            ZF ← 1;
    FI;
Fi;
VERR, VERW—Verify a Segment for Reading or Writing (continued)

Flags Affected
The ZF flag is set to 1 if the segment is accessible and readable (VERR) or writable (VERW); otherwise, it is cleared to 0.

Additional Itanium System Environment Exceptions
- Itanium Reg Faults: NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions
The only exceptions generated for these instructions are those related to illegal addressing of the source operand.
- #GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
  If the DS, ES, FS, or GS register is used to access memory and it contains a null segment selector.
- #SS(0) If a memory operand effective address is outside the SS segment limit.
- #PF(fault-code) If a page fault occurs.
- #AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions
#UD The VERR and VERW instructions are not recognized in real address mode.

Virtual 8086 Mode Exceptions
#UD The VERR and VERW instructions are not recognized in virtual 8086 mode.
WAIT/FWAIT—Wait

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>9B</td>
<td>WAIT</td>
<td>Check pending unmasked floating-point exceptions.</td>
</tr>
<tr>
<td>9B</td>
<td>FWAIT</td>
<td>Check pending unmasked floating-point exceptions.</td>
</tr>
</tbody>
</table>

**Description**

Causes the processor to check for and handle pending unmasked floating-point exceptions before proceeding. (FWAIT is an alternate mnemonic for the WAIT).

This instruction is useful for synchronizing exceptions in critical sections of code. Coding a WAIT instruction after a floating-point instruction insures that any unmasked floating-point exceptions the instruction may raise are handled before the processor can modify the instruction’s results.

**Operation**

CheckPendingUnmaskedFloatingPointExceptions;

**FPU Flags Affected**

The C0, C1, C2, and C3 flags are undefined.

**Floating-point Exceptions**

None.

**Protected Mode Exceptions**

- #NM MP and TS in CR0 is set.

**Real Address Mode Exceptions**

- #NM MP and TS in CR0 is set.

**Virtual 8086 Mode Exceptions**

- #NM MP and TS in CR0 is set.
WBINVD—Write-Back and Invalidate Cache

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 09</td>
<td>WBINVD</td>
<td>Write-back and flush Internal caches; initiate writing-back and flushing of external caches.</td>
</tr>
</tbody>
</table>

**Description**

Writes back all modified cache lines in the processor’s internal cache to main memory, invalidates (flushes) the internal caches, and issues a special-function bus cycle that directs external caches to also write back modified data.

After executing this instruction, the processor does not wait for the external caches to complete their write-back and flushing operations before proceeding with instruction execution. It is the responsibility of hardware to respond to the cache write-back and flush signals.

The WBINVD instruction is a privileged instruction. When the processor is running in protected mode, the CPL of a program or procedure must be 0 to execute this instruction. This instruction is also a serializing instruction.

In situations where cache coherency with main memory is not a concern, software can use the INVD instruction.

**Operation**

IF Itanium System Environment THEN IA-32_Intercept(INST,WBINVD);

WriteBack(InternalCaches);
Flush(InternalCaches);
SignalWriteBack(ExternalCaches);
SignalFlush(ExternalCaches);
Continue (* Continue execution);

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

IA-32_Intercept Mandatory Instruction Intercept.

**Protected Mode Exceptions**

#GP(0) If the current privilege level is not 0.

**Real Address Mode Exceptions**

None.
WBINVD—Write-Back and Invalidate Cache (continued)

Virtual 8086 Mode Exceptions

#GP(0) The WBINVD instruction cannot be executed at the virtual 8086 mode.

Intel Architecture Compatibility

The WBINVD instruction implementation-dependent; its function may be implemented differently on future Intel Architecture processors. The instruction is not supported on Intel Architecture processors earlier than the Intel486 processor.
WRMSR—Write to Model Specific Register

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 30</td>
<td>WRMSR</td>
<td>Write the value in EDX:EAX to MSR specified by ECX</td>
</tr>
</tbody>
</table>

**Description**

Writes the contents of registers EDX:EAX into the 64-bit model specific register (MSR) specified in the ECX register. The high-order 32 bits are copied from EDX and the low-order 32 bits are copied from EAX. Always set undefined or reserved bits in an MSR to the values previously read.

This instruction must be executed at privilege level 0 or in real-address mode; otherwise, a general protection exception #GP(0) will be generated. Specifying a reserved or unimplemented MSR address in ECX will also cause a general protection exception.

When the WRMSR instruction is used to write to an MTRR, the TLBs are invalidated, including the global entries see the *Intel Architecture Software Developer’s Manual, Volume 3*).

The MSRs control functions for testability, execution tracing, performance-monitoring and machine check errors. See model specific instructions for all the MSRs that can be written to with this instruction and their addresses.

The WRMSR instruction is a serializing instruction.

The CPUID instruction should be used to determine whether MSRs are supported (EDX[5]=1) before using this instruction.

**Operation**

IF Itanium System Environment THEN IA-32_Interceptor(INST,WRMSR);  
MSR[ECX] ← EDX:EAX;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

IA-32_Interceptor      Mandatory Instruction Intercept.

**Protected Mode Exceptions**

#GP(0)            If the current privilege level is not 0.
                    If the value in ECX specifies a reserved or unimplemented MSR address.

**Real Address Mode Exceptions**

#GP              If the current privilege level is not 0
                    If the value in ECX specifies a reserved or unimplemented MSR address.
WRMSR—Write to Model Specific Register (continued)

Virtual 8086 Mode Exceptions

#GP(0) The WRMSR instruction is not recognized in virtual 8086 mode.

Intel Architecture Compatibility

The MSRs and the ability to read them with the WRMSR instruction were introduced into the Intel Architecture with the Pentium processor. Execution of this instruction by an Intel Architecture processor earlier than the Pentium processor results in an invalid opcode exception #UD.
XADD—Exchange and Add

Description

Exchanges the first operand (destination operand) with the second operand (source operand), then loads the sum of the two values into the destination operand. The destination operand can be a register or a memory location; the source operand is a register.

This instruction can be used with a LOCK prefix.

Operation

\[
\text{IF Itanium System Environment AND External Bus Lock Required AND DCR.lc}
\text{ THEN IA-32_Intercept(LOCK,XADD);}
\]

\[
\text{TEMP} \leftarrow \text{SRC + DEST}
\]

\[
\text{SRC} \leftarrow \text{DEST}
\]

\[
\text{DEST} \leftarrow \text{TEMP}
\]

Flags Affected

The CF, PF, AF, SF, ZF, and OF flags are set according to the result stored in the destination operand.

Additional Itanium System Environment Exceptions

- **Itanium Reg Faults**: NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
- **IA-32_Intercept**: Lock Intercept - If an external atomic bus lock is required to complete this operation and DCR.lc is 1, no atomic transaction occurs, this instruction is faulted and an IA-32_Intercept(Lock) fault is generated. The software lock handler is responsible for the emulation of this instruction.

Protected Mode Exceptions

- **#GP(0)**: If the destination is located in a nonwritable segment.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#PF(fault-code)**: If the DS, ES, FS, or GS register contains a null segment selector.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
XADD—Exchange and Add (continued)

**Real Address Mode Exceptions**

#GP        If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS        If a memory operand effective address is outside the SS segment limit.

**Virtual 8086 Mode Exceptions**

#GP(0)     If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)     If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)  If a page fault occurs.
#AC(0)     If alignment checking is enabled and an unaligned memory reference is made.

**Intel Architecture Compatibility**

Intel Architecture processors earlier than the Intel486 processor do not recognize this instruction. If this instruction is used, you should provide an equivalent code sequence that runs on earlier processors.
XCHG—Exchange Register/Memory with Register

### Description

Exchanges the contents of the destination (first) and source (second) operands. The operands can be two general-purpose registers or a register and a memory location. When the operands are two registers, one of the registers must be the EAX or AX register. If a memory operand is referenced, the LOCK# signal is automatically asserted for the duration of the exchange operation, regardless of the presence or absence of the LOCK prefix or of the value of the IOPL.

This instruction is useful for implementing semaphores or similar data structures for process synchronization. (See Chapter 5, *Processor Management and Initialization*, in the *Intel Architecture Software Developer’s Manual, Volume 3* for more information on bus locking.)

The XCHG instruction can also be used instead of the BSWAP instruction for 16-bit operands.

### Operation

\[
\text{IF Itanium System Environment AND External\_Atomic\_Lock\_Required AND DCR.lc}
\text{THEN IA-32\_Intercept(LOCK,XCHG);}
\]

\[
\begin{align*}
\text{TEMP} & \leftarrow \text{DEST} \\
\text{DEST} & \leftarrow \text{SRC} \\
\text{SRC} & \leftarrow \text{TEMP}
\end{align*}
\]

### Flags Affected

None.

### Additional Itanium System Environment Exceptions

- **Itanium Reg Faults**: NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
- **IA-32\_Intercept**: Lock Intercept - If an external atomic bus lock is required to complete this operation and DCR.lc is 1, no atomic transaction occurs, this instruction is faulted and an IA-32\_Intercept(LOCK) fault is generated. The software lock handler is responsible for the emulation of this instruction.
XCHG—Exchange Register/Memory with Register (continued)

Protected Mode Exceptions

#GP(0) If either operand is in a nonwritable segment.
   If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
   If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
XLAT/XLATB—Table Look-up Translation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>D7</td>
<td>XLAT m8</td>
<td>Set AL to memory byte DS:[(E)BX + unsigned AL]</td>
</tr>
<tr>
<td>D7</td>
<td>XLATB</td>
<td>Set AL to memory byte DS:[(E)BX + unsigned AL]</td>
</tr>
</tbody>
</table>

**Description**

Locates a byte entry in a table in memory, using the contents of the AL register as a table index, then copies the contents of the table entry back into the AL register. The index in the AL register is treated as unsigned integer. The XLAT and XLATB instructions get the base address of the table in memory from the DS:EBX registers (or the DS:BX registers when the address-size attribute of 16 bits.) The XLAT instruction allows a different segment register to be specified with a segment override. When assembled, the XLAT and XLATB instructions produce the same machine code.

**Operation**

IF AddressSize = 16
THEN
    AL ← (DS:BX + ZeroExtend(AL))
ELSE (* AddressSize = 32 *)
    AL ← (DS:EBX + ZeroExtend(AL));
FI;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

**Protected Mode Exceptions**

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.

If the DS, ES, FS, or GS register contains a null segment selector.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
XLAT/XLATB—Table Look-up Translation (continued)

Real Address Mode Exceptions

#GP               If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS               If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0)            If a memory operand effective address is outside the CS, DS, ES, FS, or GS segment limit.
#SS(0)            If a memory operand effective address is outside the SS segment limit.
#PF(fault-code)   If a page fault occurs.
#AC(0)            If alignment checking is enabled and an unaligned memory reference is made.
XOR—Logical Exclusive OR

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>34 ib</td>
<td>XOR AL,imm8</td>
<td>AL XOR imm8</td>
</tr>
<tr>
<td>35 iw</td>
<td>XOR AX,imm16</td>
<td>AX XOR imm16</td>
</tr>
<tr>
<td>35 id</td>
<td>XOR EAX,imm32</td>
<td>EAX XOR imm32</td>
</tr>
<tr>
<td>80 /6 ib</td>
<td>XOR r/m8,imm8</td>
<td>r/m8 XOR imm8</td>
</tr>
<tr>
<td>81 /6 iw</td>
<td>XOR r/m16,imm16</td>
<td>r/m16 XOR imm16</td>
</tr>
<tr>
<td>81 /6 id</td>
<td>XOR r/m32,imm32</td>
<td>r/m32 XOR imm32</td>
</tr>
<tr>
<td>83 /6 ib</td>
<td>XOR r/m16,imm8</td>
<td>r/m16 XOR imm8</td>
</tr>
<tr>
<td>83 /6 ib</td>
<td>XOR r/m32,imm8</td>
<td>r/m32 XOR imm8</td>
</tr>
<tr>
<td>30 /r</td>
<td>XOR r/m8,r8</td>
<td>r/m8 XOR r8</td>
</tr>
<tr>
<td>31 /r</td>
<td>XOR r/m16,r16</td>
<td>r/m16 XOR r16</td>
</tr>
<tr>
<td>31 /r</td>
<td>XOR r/m32,r32</td>
<td>r/m32 XOR r32</td>
</tr>
<tr>
<td>32 /r</td>
<td>XOR r8,r/m8</td>
<td>r8 XOR r/m8</td>
</tr>
<tr>
<td>33 /r</td>
<td>XOR r16,r/m16</td>
<td>r8 XOR r/m8</td>
</tr>
<tr>
<td>33 /r</td>
<td>XOR r32,r/m32</td>
<td>r8 XOR r/m8</td>
</tr>
</tbody>
</table>

Description

Performs a bitwise exclusive-OR (XOR) operation on the destination (first) and source (second) operands and stores the result in the destination operand location. The source operand can be an immediate, a register, or a memory location; the destination operand can be a register or a memory location.

Operation

DEST ← DEST XOR SRC;

Flags Affected

The OF and CF flags are cleared; the SF, ZF, and PF flags are set according to the result. The state of the AF flag is undefined.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  NaT Register Consumption Abort.

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
XOR—Logical Exclusive OR (continued)

Protected Mode Exceptions

#GP(0) If the destination operand points to a nonwritable segment.
If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
If the DS, ES, FS, or GS register contains a null segment selector.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made
while the current privilege level is 3.

Real Address Mode Exceptions

#GP If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
#SS If a memory operand effective address is outside the SS segment limit.

Virtual 8086 Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS, or GS
segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
This section lists the IA-32 MMX technology instructions designed to increase performance of multimedia intensive applications.
EMMS—Empty MMX State

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 77</td>
<td>EMMS</td>
<td>Set the FP tag word to empty.</td>
</tr>
</tbody>
</table>

Description

Sets the values of all the tags in the FPU tag word to empty (all ones). This operation marks the MMX technology registers as available, so they can subsequently be used by floating-point instructions. (See Figure 7-11 in the Intel Architecture Software Developer’s Manual, Volume 1, for the format of the FPU tag word.) All other MMX instructions (other than the EMMS instruction) set all the tags in FPU tag word to valid (all zeros).

The EMMS instruction must be used to clear the MMX technology state at the end of all MMX technology routines and before calling other procedures or subroutines that may execute floating-point instructions. If a floating-point instruction loads one of the registers in the FPU register stack before the FPU tag word has been reset by the EMMS instruction, a floating-point stack overflow can occur that will result in a floating-point exception or incorrect result.

Operation

FPUTagWord ← FFFFH;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1.

Protected Mode Exceptions

#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.

Real-Address Mode Exceptions

#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.
MOVD—Move 32 Bits

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 6E /r</td>
<td>MOVD mm, r/m32</td>
<td>Move doubleword from r/m32 to mm.</td>
</tr>
<tr>
<td>0F 7E /r</td>
<td>MOVD r/m32, mm</td>
<td>Move doubleword from mm to r/m32.</td>
</tr>
</tbody>
</table>

**Description**

Copies doubleword from the source operand (second operand) to the destination operand (first operand). Source and destination operands can be MMX technology registers, memory locations, or 32-bit general-purpose registers; however, data cannot be transferred from an MMX technology register to an MMX technology register, from one memory location to another memory location, or from one general-purpose register to another general-purpose register.

When the destination operand is an MMX technology register, the 32-bit source value is written to the low-order 32 bits of the 64-bit MMX technology register and zero-extended to 64 bits (see Figure 2-1). When the source operand is an MMX technology register, the low-order 32 bits of the MMX technology register are written to the 32-bit general-purpose register or 32-bit memory location selected with the destination operand.

**Figure 2-1. Operation of the MOVD Instruction**

```
MOVD m32, mm

63 32 31 0

mm

b3 b2 b1 b0

MOVD mm, r32

63 32 31 0

mm

b3 b2 b1 b0

r32
```

**Operation**

IF DEST is MMX register
THEN
    DEST ← ZeroExtend(SRC);
ELSE (* SRC is MMX register *)
    DEST ← LowOrderDoubleword(SRC);
MOVD—Move 32 Bits (continued)

Flags Affected

None.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

Protected Mode Exceptions

- **#GP(0)**: If the destination operand is in a nonwritable segment.
  - If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#UD**: If EM in CR0 is set.
- **#NM**: If TS in CR0 is set.
- **#MF**: If there is a pending FPU exception.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

- **#GP**: If any part of the operand lies outside of the effective address space from 0 to FFFFH.
- **#UD**: If EM in CR0 is set.
- **#NM**: If TS in CR0 is set.
- **#MF**: If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

- **#GP**: If any part of the operand lies outside of the effective address space from 0 to FFFFH.
- **#UD**: If EM in CR0 is set.
- **#NM**: If TS in CR0 is set.
- **#MF**: If there is a pending FPU exception.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made.
MOVQ—Move 64 Bits

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 6F /r</td>
<td>MOVQ mm, mm/m64</td>
<td>Move quadword from mm/m64 to mm.</td>
</tr>
<tr>
<td>0F 7F /r</td>
<td>MOVQ mm/m64, mm</td>
<td>Move quadword from mm to mm/m64.</td>
</tr>
</tbody>
</table>

Description

Copies quadword from the source operand (second operand) to the destination operand (first operand). (See Figure 2-2.) A source or destination operand can be either an MMX technology register or a memory location; however, data cannot be transferred from one memory location to another memory location. Data can be transferred from one MMX technology register to another MMX technology register.

Figure 2-2. Operation of the MOVQ Instruction

![Figure 2-2. Operation of the MOVQ Instruction](image)

Operation

DEST ← SRC;

Flags Affected

None.

Additional Itanium System Environment Exceptions

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>

Volume 3: IA-32 Intel® MMX™ Technology Instruction Reference
MOVQ—Move 64 Bits (continued)

**Protected Mode Exceptions**

- **#GP(0)** If the destination operand is in a nonwritable segment.
  
  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.

- **#SS(0)** If a memory operand effective address is outside the SS segment limit.

- **#UD** If EM in CR0 is set.

- **#NM** If TS in CR0 is set.

- **#MF** If there is a pending FPU exception.

- **#PF(fault-code)** If a page fault occurs.

- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real-Address Mode Exceptions**

- **#GP** If any part of the operand lies outside of the effective address space from 0 to FFFFH.

- **#UD** If EM in CR0 is set.

- **#NM** If TS in CR0 is set.

- **#MF** If there is a pending FPU exception.

**Virtual-8086 Mode Exceptions**

- **#GP** If any part of the operand lies outside of the effective address space from 0 to FFFFH.

- **#UD** If EM in CR0 is set.

- **#NM** If TS in CR0 is set.

- **#MF** If there is a pending FPU exception.

- **#PF(fault-code)** If a page fault occurs.

- **#AC(0)** If alignment checking is enabled and an unaligned memory reference is made.
PACKSSWB/PACKSSDW—Pack with Signed Saturation

**Description**

Packs and saturates signed words into bytes (PACKSSWB) or signed doublewords into words (PACKSSDW). The PACKSSWB instruction packs 4 signed words from the destination operand (first operand) and 4 signed words from the source operand (second operand) into 8 signed bytes in the destination operand. If the signed value of a word is beyond the range of a signed byte (that is, greater than 7FH or less than 80H), the saturated byte value of 7FH or 80H, respectively, is stored into the destination.

The PACKSSDW instruction packs 2 signed doublewords from the destination operand (first operand) and 2 signed doublewords from the source operand (second operand) into 4 signed words in the destination operand (see Figure 2-3). If the signed value of a doubleword is beyond the range of a signed word (that is, greater than 7FFFFH or less than 8000H), the saturated word value of 7FFFFH or 8000H, respectively, is stored into the destination.

The destination operand for either the PACKSSWB or PACKSSDW instruction must be an MMX technology register; the source operand may be either an MMX technology register or a quadword memory location.

**Figure 2-3. Operation of the PACKSSDW Instruction**

**Operation**

IF instruction is PACKSSWB

THEN

DEST(7..0) ← SaturateSignedWordToSignedByte DEST(15..0);
DEST(15..8) ← SaturateSignedWordToSignedByte DEST(31..16);
DEST(23..16) ← SaturateSignedWordToSignedByte DEST(47..32);
DEST(31..24) ← SaturateSignedWordToSignedByte DEST(63..48);
DEST(39..32) ← SaturateSignedWordToSignedByte SRC(15..0);
DEST(47..40) ← SaturateSignedWordToSignedByte SRC(31..16);
DEST(55..48) ← SaturateSignedWordToSignedByte SRC(47..32);
DEST(63..56) ← SaturateSignedWordToSignedByte SRC(63..48);
PACKSSWB/PACKSSDW—Pack with Signed Saturation (continued)

ELSE (* instruction is PACKSSDW *)
   DEST(15..0) ← SaturateSignedDoublewordToSignedWord DEST(31..0);
   DEST(31..16) ← SaturateSignedDoublewordToSignedWord DEST(63..32);
   DEST(47..32) ← SaturateSignedDoublewordToSignedWord SRC(31..0);
   DEST(63..48) ← SaturateSignedDoublewordToSignedWord SRC(63..32);
FI;

Flags Affected
None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
PACKSSWB/PACKSSDW—Pack with Signed Saturation (continued)

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PACKUSWB—Pack with Unsigned Saturation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 67 /r</td>
<td>PACKUSWB mm, mm/m64</td>
<td>Pack and saturate 4 signed words from mm and 4 signed words from mm/m64 into 8 unsigned bytes in mm.</td>
</tr>
</tbody>
</table>

Description

Packs and saturates 4 signed words from the destination operand (first operand) and 4 signed words from the source operand (second operand) into 8 unsigned bytes in the destination operand (see Figure 2-4). If the signed value of a word is beyond the range of an unsigned byte (that is, greater than FFH or less than 00H), the saturated byte value of FFH or 00H, respectively, is stored into the destination.

The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a quadword memory location.

Figure 2-4. Operation of the PACKUSWB Instruction

Operation

DEST(7..0) ← SaturateSignedWordToUnsignedByte DEST(15..0);
DEST(15..8) ← SaturateSignedWordToUnsignedByte DEST(31..16);
DEST(23..16) ← SaturateSignedWordToUnsignedByte DEST(47..32);
DEST(31..24) ← SaturateSignedWordToUnsignedByte DEST(63..48);
DEST(39..32) ← SaturateSignedWordToUnsignedByte SRC(15..0);
DEST(47..40) ← SaturateSignedWordToUnsignedByte SRC(31..16);
DEST(55..48) ← SaturateSignedWordToUnsignedByte SRC(47..32);
DEST(63..56) ← SaturateSignedWordToUnsignedByte SRC(63..48);

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
PACKUSWB—Pack with Unsigned Saturation (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PADDB/PADDW/PADDD—Packed Add

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F FC /r</td>
<td>PADDB mm, mm/m64</td>
<td>Add packed bytes from mm/m64 to packed bytes in mm.</td>
</tr>
<tr>
<td>0F FD /r</td>
<td>PADDW mm, mm/m64</td>
<td>Add packed words from mm/m64 to packed words in mm.</td>
</tr>
<tr>
<td>0F FE /r</td>
<td>PADDD mm, mm/m64</td>
<td>Add packed doublewords from mm/m64 to packed doublewords in mm.</td>
</tr>
</tbody>
</table>

Description

Adds the individual data elements (bytes, words, or doublewords) of the source operand (second operand) to the individual data elements of the destination operand (first operand). (See Figure 2-5.) If the result of an individual addition exceeds the range for the specified data type (overflows), the result is wrapped around, meaning that the result is truncated so that only the lower (least significant) bits of the result are returned (that is, the carry is ignored).

The destination operand must be an MMX technology register; the source operand can be either an MMX technology register or a quadword memory location.

Figure 2-5. Operation of the PADDW Instruction

The PADDB instruction adds the bytes of the source operand to the bytes of the destination operand and stores the results to the destination operand. When an individual result is too large to be represented in 8 bits, the lower 8 bits of the result are written to the destination operand and therefore the result wraps around.

The PADDW instruction adds the words of the source operand to the words of the destination operand and stores the results to the destination operand. When an individual result is too large to be represented in 16 bits, the lower 16 bits of the result are written to the destination operand and therefore the result wraps around.

The PADDD instruction adds the doublewords of the source operand to the doublewords of the destination operand and stores the results to the destination operand. When an individual result is too large to be represented in 32 bits, the lower 32 bits of the result are written to the destination operand and therefore the result wraps around.
PADDB/PADDW/PADDD—Packed Add (continued)

Note that like the integer ADD instruction, the PADDB, PADDW, and PADDD instructions can operate on either unsigned or signed (two's complement notation) packed integers. Unlike the integer instructions, none of the MMX instructions affect the EFLAGS register. With MMX instructions, there are no carry or overflow flags to indicate when overflow has occurred, so the software must control the range of values or else use the “with saturation” MMX instructions.

**Operation**

IF instruction is PADDB

THEN

\[
\begin{align*}
\text{DEST}(7..0) & \leftarrow \text{DEST}(7..0) + \text{SRC}(7..0); \\
\text{DEST}(15..8) & \leftarrow \text{DEST}(15..8) + \text{SRC}(15..8); \\
\text{DEST}(23..16) & \leftarrow \text{DEST}(23..16) + \text{SRC}(23..16); \\
\text{DEST}(31..24) & \leftarrow \text{DEST}(31..24) + \text{SRC}(31..24); \\
\text{DEST}(39..32) & \leftarrow \text{DEST}(39..32) + \text{SRC}(39..32); \\
\text{DEST}(47..40) & \leftarrow \text{DEST}(47..40) + \text{SRC}(47..40); \\
\text{DEST}(55..48) & \leftarrow \text{DEST}(55..48) + \text{SRC}(55..48); \\
\text{DEST}(63..56) & \leftarrow \text{DEST}(63..56) + \text{SRC}(63..56);
\end{align*}
\]

ELSEIF instruction is PADDW

THEN

\[
\begin{align*}
\text{DEST}(15..0) & \leftarrow \text{DEST}(15..0) + \text{SRC}(15..0); \\
\text{DEST}(31..16) & \leftarrow \text{DEST}(31..16) + \text{SRC}(31..16); \\
\text{DEST}(47..32) & \leftarrow \text{DEST}(47..32) + \text{SRC}(47..32); \\
\text{DEST}(63..48) & \leftarrow \text{DEST}(63..48) + \text{SRC}(63..48);
\end{align*}
\]

ELSE (* instruction is PADDD *)

\[
\begin{align*}
\text{DEST}(31..0) & \leftarrow \text{DEST}(31..0) + \text{SRC}(31..0); \\
\text{DEST}(63..32) & \leftarrow \text{DEST}(63..32) + \text{SRC}(63..32);
\end{align*}
\]

FI;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
PADDB/PADDW/PADDD—Packed Add (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PADDSB/PADDSW—Packed Add with Saturation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F EC /r</td>
<td>PADDSB mm, mm/m64</td>
<td>Add signed packed bytes from mm/m64 to signed packed bytes in mm and saturate.</td>
</tr>
<tr>
<td>0F ED /r</td>
<td>PADDSW mm, mm/m64</td>
<td>Add signed packed words from mm/m64 to signed packed words in mm and saturate.</td>
</tr>
</tbody>
</table>

Description

Adds the individual signed data elements (bytes or words) of the source operand (second operand) to the individual signed data elements of the destination operand (first operand). (See Figure 2-6.) If the result of an individual addition exceeds the range for the specified data type, the result is saturated. The destination operand must be an MMX technology register; the source operand can be either an MMX technology register or a quadword memory location.

Figure 2-6. Operation of the PADDSW Instruction

The PADDSB instruction adds the signed bytes of the source operand to the signed bytes of the destination operand and stores the results to the destination operand. When an individual result is beyond the range of a signed byte (that is, greater than 7FH or less than 80H), the saturated byte value of 7FH or 80H, respectively, is written to the destination operand.

The PADDSW instruction adds the signed words of the source operand to the signed words of the destination operand and stores the results to the destination operand. When an individual result is beyond the range of a signed word (that is, greater than 7FFFH or less than 8000H), the saturated word value of 7FFFH or 8000H, respectively, is written to the destination operand.

Operation

IF instruction is PADDSB

THEN

DEST(7..0) ← SaturateToSignedByte(DEST(7..0) + SRC (7..0)) ;
DEST(15..8) ← SaturateToSignedByte(DEST(15..8) + SRC(15..8) );
DEST(23..16) ← SaturateToSignedByte(DEST(23..16)+ SRC(23..16) );
DEST(31..24) ← SaturateToSignedByte(DEST(31..24) + SRC(31..24) );
DEST(39..32) ← SaturateToSignedByte(DEST(39..32) + SRC(39..32) );
DEST(47..40) ← SaturateToSignedByte(DEST(47..40)+ SRC(47..40) );
DEST(55..48) ← SaturateToSignedByte(DEST(55..48) + SRC(55..48) );
DEST(63..56) ← SaturateToSignedByte(DEST(63..56) + SRC(63..56) );

ELSE { (* instruction is PADDSW * )

Volume 3: IA-32 Intel® MMX™ Technology Instruction Reference
PADDSB/PADDSW—Packed Add with Saturation (continued)

\[
\text{DEST}(15..0) \leftarrow \text{SaturateToSignedWord}(_{\text{DEST}(15..0)} + _{\text{SRC}(15..0)}) \; ; \\
\text{DEST}(31..16) \leftarrow \text{SaturateToSignedWord}(_{\text{DEST}(31..16)} + _{\text{SRC}(31..16)}) \; ; \\
\text{DEST}(47..32) \leftarrow \text{SaturateToSignedWord}(_{\text{DEST}(47..32)} + _{\text{SRC}(47..32)}) \; ; \\
\text{DEST}(63..48) \leftarrow \text{SaturateToSignedWord}(_{\text{DEST}(63..48)} + _{\text{SRC}(63..48)}) \; ; \\
\text{FI};
\]

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.
PADDSB/PADDSW—Packed Add with Saturation (continued)

Virtual-8086 Mode Exceptions

<table>
<thead>
<tr>
<th>Exception Code</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP</td>
<td>If any part of the operand lies outside of the effective address space from 0 to FFFFH.</td>
</tr>
<tr>
<td>#UD</td>
<td>If EM in CR0 is set.</td>
</tr>
<tr>
<td>#NM</td>
<td>If TS in CR0 is set.</td>
</tr>
<tr>
<td>#MF</td>
<td>If there is a pending FPU exception.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made.</td>
</tr>
</tbody>
</table>
PADDUSB/PADDUSW—Packed Add Unsigned with Saturation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F DC  /r</td>
<td>PADDUSB mm, mm/m64</td>
<td>Add unsigned packed bytes from mm/m64 to unsigned packed bytes in mm and saturate.</td>
</tr>
<tr>
<td>0F DD  /r</td>
<td>PADDUSW mm, mm/m64</td>
<td>Add unsigned packed words from mm/m64 to unsigned packed words in mm and saturate.</td>
</tr>
</tbody>
</table>

Description

Adds the individual unsigned data elements (bytes or words) of the packed source operand (second operand) to the individual unsigned data elements of the packed destination operand (first operand). (See Figure 2-7.) If the result of an individual addition exceeds the range for the specified unsigned data type, the result is saturated. The destination operand must be an MMX technology register; the source operand can be either an MMX technology register or a quadword memory location.

Figure 2-7. Operation of the PADDUSB Instruction

The PADDUSB instruction adds the unsigned bytes of the source operand to the unsigned bytes of the destination operand and stores the results to the destination operand. When an individual result is beyond the range of an unsigned byte (that is, greater than FFH), the saturated unsigned byte value of FFH is written to the destination operand.

The PADDUSW instruction adds the unsigned words of the source operand to the unsigned words of the destination operand and stores the results to the destination operand. When an individual result is beyond the range of an unsigned word (that is, greater than FFFFH), the saturated unsigned word value of FFFFH is written to the destination operand.
PADDUSB/PADDUSW—Packed Add Unsigned with Saturation (continued)

Operation

IF instruction is PADDUSB
THEN
  DEST(7..0) ← SaturateToUnsignedByte(DEST(7..0) + SRC(7..0));
  DEST(15..8) ← SaturateToUnsignedByte(DEST(15..8) + SRC(15..8));
  DEST(23..16) ← SaturateToUnsignedByte(DEST(23..16) + SRC(23..16));
  DEST(31..24) ← SaturateToUnsignedByte(DEST(31..24) + SRC(31..24));
  DEST(39..32) ← SaturateToUnsignedByte(DEST(39..32) + SRC(39..32));
  DEST(47..40) ← SaturateToUnsignedByte(DEST(47..40) + SRC(47..40));
  DEST(55..48) ← SaturateToUnsignedByte(DEST(55..48) + SRC(55..48));
  DEST(63..56) ← SaturateToUnsignedByte(DEST(63..56) + SRC(63..56));
ELSE { (* instruction is PADDUSW *)
  DEST(15..0) ← SaturateToUnsignedWord(DEST(15..0) + SRC(15..0));
  DEST(31..16) ← SaturateToUnsignedWord(DEST(31..16) + SRC(31..16));
  DEST(47..32) ← SaturateToUnsignedWord(DEST(47..32) + SRC(47..32));
  DEST(63..48) ← SaturateToUnsignedWord(DEST(63..48) + SRC(63..48));
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
PADDUSB/PADDUSW—Packed Add Unsigned with Saturation (continued)

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PAND—Logical AND

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F DB /r</td>
<td>PAND mm, mm/m64</td>
<td>AND quadword from mm/m64 to quadword in mm.</td>
</tr>
</tbody>
</table>

**Description**

Performs a bitwise logical AND operation on the quadword source (second) and destination (first) operands and stores the result in the destination operand location (see Figure 2-8). The source operand can be an MMX technology register or a quadword memory location; the destination operand must be an MMX technology register. Each bit of the result of the PAND instruction is set to 1 if the corresponding bits of the operands are both 1; otherwise it is made zero.

**Figure 2-8. Operation of the PAND Instruction**

<table>
<thead>
<tr>
<th>Destination (mm)</th>
<th>Source (mm/m64)</th>
<th>Result</th>
</tr>
</thead>
<tbody>
<tr>
<td>11111111111110000000000001011011011011000100011101110111011111</td>
<td>0001000011011001010100000011000100011110111011110001010110010101</td>
<td>&amp;</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Destination (mm)</th>
<th>Source (mm/m64)</th>
<th>Result</th>
</tr>
</thead>
<tbody>
<tr>
<td>0001000011011000000000000100100010101000101010000001010100010101</td>
<td>0001000011011001010100000011000100011110111011110001010110010101</td>
<td>&amp;</td>
</tr>
</tbody>
</table>

**Operation**

DEST ← DEST AND SRC;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- Itanium Mem Faults: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
PAND—Logical AND (continued)

**Protected Mode Exceptions**

- **#GP(0)**: If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
- **#SS(0)**: If a memory operand effective address is outside the SS segment limit.
- **#UD**: If EM in CR0 is set.
- **#NM**: If TS in CR0 is set.
- **#MF**: If there is a pending FPU exception.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real-Address Mode Exceptions**

- **#GP**: If any part of the operand lies outside of the effective address space from 0 to FFFFH.
- **#UD**: If EM in CR0 is set.
- **#NM**: If TS in CR0 is set.
- **#MF**: If there is a pending FPU exception.

**Virtual-8086 Mode Exceptions**

- **#GP**: If any part of the operand lies outside of the effective address space from 0 to FFFFH.
- **#UD**: If EM in CR0 is set.
- **#NM**: If TS in CR0 is set.
- **#MF**: If there is a pending FPU exception.
- **#PF(fault-code)**: If a page fault occurs.
- **#AC(0)**: If alignment checking is enabled and an unaligned memory reference is made.
PANDN—Logical AND NOT

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F DF /r</td>
<td>PANDN mm, mm/m64</td>
<td>AND quadword from mm/m64 to NOT quadword in mm.</td>
</tr>
</tbody>
</table>

**Description**

Performs a bitwise logical NOT on the quadword destination operand (first operand). Then, the instruction performs a bitwise logical AND operation on the inverted destination operand and the quadword source operand (second operand). (See Figure 2-9.) Each bit of the result of the AND operation is set to one if the corresponding bits of the source and inverted destination bits are one; otherwise it is set to zero. The result is stored in the destination operand location.

The source operand can be an MMX technology register or a quadword memory location; the destination operand must be an MMX technology register.

**Figure 2-9. Operation of the PANDN Instruction**

```
PANDN mm, mm/m64
```

```
~
```

```
&
```

<table>
<thead>
<tr>
<th>mm</th>
<th>11111111111110000000000000000101101101010011101111000100010001000</th>
</tr>
</thead>
<tbody>
<tr>
<td>m/m64</td>
<td>11111111111110000000000000000101101101010011101111000100010001000</td>
</tr>
<tr>
<td>mm</td>
<td>11111111111110000000000000000101101101010011101111000100010001000</td>
</tr>
</tbody>
</table>

**Operation**

```C
DEST ← (NOT DEST) AND SRC;
```

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
PANDN—Logical AND NOT (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PCMPEQB/PCMPEQW/PCMPEQD—Packed Compare for Equal

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 74</td>
<td>PCMPEQB mm, mm/m64</td>
<td>Compare packed bytes in mm/m64 with packed bytes in mm for equality.</td>
</tr>
<tr>
<td>0F 75</td>
<td>PCMPEQW mm, mm/m64</td>
<td>Compare packed words in mm/m64 with packed words in mm for equality.</td>
</tr>
<tr>
<td>0F 76</td>
<td>PCMPEQD mm, mm/m64</td>
<td>Compare packed doublewords in mm/m64 with packed doublewords in mm for equality.</td>
</tr>
</tbody>
</table>

Description

Compares the individual data elements (bytes, words, or doublewords) in the destination operand (first operand) to the corresponding data elements in the source operand (second operand). (See Figure 2-10.) If a pair of data elements are equal, the corresponding data element in the destination operand is set to all ones; otherwise, it is set to all zeros. The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a 64-bit memory location.

Figure 2-10. Operation of the PCMPEQW Instruction

The PCMPEQB instruction compares the bytes in the destination operand to the corresponding bytes in the source operand, with the bytes in the destination operand being set according to the results.

The PCMPEQW instruction compares the words in the destination operand to the corresponding words in the source operand, with the words in the destination operand being set according to the results.

The PCMPEQD instruction compares the doublewords in the destination operand to the corresponding doublewords in the source operand, with the doublewords in the destination operand being set according to the results.
Operation

IF instruction is PCMPEQB
THEN
    IF DEST(7..0) = SRC(7..0)
        THEN DEST(7 0) ← FFH;
        ELSE DEST(7..0) ← 0;
        * Continue comparison of second through seventh bytes in DEST and SRC *
    IF DEST(63..56) = SRC(63..56)
        THEN DEST(63..56) ← FFH;
        ELSE DEST(63..56) ← 0;
ELSE IF instruction is PCMPEQW
THEN
    IF DEST(15..0) = SRC(15..0)
        THEN DEST(15..0) ← FFFFH;
        ELSE DEST(15..0) ← 0;
        * Continue comparison of second and third words in DEST and SRC *
    IF DEST(63..48) = SRC(63..48)
        THEN DEST(63..48) ← FFFFH;
        ELSE DEST(63..48) ← 0;
ELSE (* instruction is PCMPEQD *)
    IF DEST(31..0) = SRC(31..0)
        THEN DEST(31..0) ← FFFFFFFFH;
        ELSE DEST(31..0) ← 0;
        IF DEST(63..32) = SRC(63..32)
            THEN DEST(63..32) ← FFFFFFFFH;
            ELSE DEST(63..32) ← 0;
FI;

Flags Affected

None:

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
PCMPEQB/PCMPEQW/PCMPEQD—Packed Compare for Equal (continued)

#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PCMPGTB/PCMPGTW/PCMPGTD—Packed Compare for Greater Than

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 64 /r</td>
<td>PCMPGTB mm, mm/m64</td>
<td>Compare packed bytes in mm with packed bytes in mm/m64 for greater value.</td>
</tr>
<tr>
<td>0F 65 /r</td>
<td>PCMPGTW mm, mm/m64</td>
<td>Compare packed words in mm with packed words in mm/m64 for greater value.</td>
</tr>
<tr>
<td>0F 66 /r</td>
<td>PCMPGTD mm, mm/m64</td>
<td>Compare packed doublewords in mm with packed doublewords in mm/m64 for greater value.</td>
</tr>
</tbody>
</table>

Description

Compare the individual signed data elements (bytes, words, or doublewords) in the destination operand (first operand) to the corresponding signed data elements in the source operand (second operand). (See Figure 2-11.) If a data element in the destination operand is greater than its corresponding data element in the source operand, the data element in the destination operand is set to all ones; otherwise, it is set to all zeros. The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a 64-bit memory location.

Figure 2-11. Operation of the PCMPGTW Instruction

The PCMPGTB instruction compares the signed bytes in the destination operand to the corresponding signed bytes in the source operand, with the bytes in the destination operand being set according to the results.

The PCMPGTW instruction compares the signed words in the destination operand to the corresponding signed words in the source operand, with the words in the destination operand being set according to the results.

The PCMPGTD instruction compares the signed doublewords in the destination operand to the corresponding signed doublewords in the source operand, with the doublewords in the destination operand being set according to the results.
PCMPGTB/PCMPGTW/PCMPGTD—Packed Compare for Greater Than (continued)

Operation

IF instruction is PCMPGTB
THEN
  IF DEST(7..0) > SRC(7..0)
  THEN DEST(7 0) ← FFH;
  ELSE DEST(7..0) ← 0;
  * Continue comparison of second through seventh bytes in DEST and SRC *
  IF DEST(63..56) > SRC(63..56)
  THEN DEST(63..56) ← FFH;
  ELSE DEST(63..56) ← 0;
ELSE IF instruction is PCMPGTW
THEN
  IF DEST(15..0) > SRC(15..0)
  THEN DEST(15..0) ← FFFFH;
  ELSE DEST(15..0) ← 0;
  * Continue comparison of second and third bytes in DEST and SRC *
  IF DEST(63..48) > SRC(63..48)
  THEN DEST(63..48) ← FFFFH;
  ELSE DEST(63..48) ← 0;
ELSE { (* instruction is PCMPGTD *)
  IF DEST(31..0) > SRC(31..0)
  THEN DEST(31..0) ← FFFFFFFFH;
  ELSE DEST(31..0) ← 0;
  IF DEST(63..32) > SRC(63..32)
  THEN DEST(63..32) ← FFFFFFFFH;
  ELSE DEST(63..32) ← 0;
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
PCMPGTB/PCMPGTW/PCMPGTD—Packed Compare for Greater Than (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PMADDWD—Packed Multiply and Add

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F F5 /r</td>
<td>PMADDWD mm, mm/m64</td>
<td>Multiply the packed words in \textit{mm} by the packed words in \textit{mm/m64}. Add the 32-bit pairs of results and store in \textit{mm} as doubleword</td>
</tr>
</tbody>
</table>

**Description**

Multiplies the individual signed words of the destination operand by the corresponding signed words of the source operand, producing four signed, doubleword results (see Figure 2-12). The two doubleword results from the multiplication of the high-order words are added together and stored in the upper doubleword of the destination operand; the two doubleword results from the multiplication of the low-order words are added together and stored in the lower doubleword of the destination operand. The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a 64-bit memory location.

The PMADDWD instruction wraps around to 80000000H only when all four words of both the source and destination operands are 8000H.

**Figure 2-12. Operation of the PMADDWD Instruction**

\[
\begin{align*}
\text{DEST}(31..0) & \leftarrow (\text{DEST}(15..0) \times \text{SRC}(15..0)) + (\text{DEST}(31..16) \times \text{SRC}(31..16)); \\
\text{DEST}(63..32) & \leftarrow (\text{DEST}(47..32) \times \text{SRC}(47..32)) + (\text{DEST}(63..48) \times \text{SRC}(63..48));
\end{align*}
\]

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault.
PMADDWD—Packed Multiply and Add (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS
segment limit.

#SS(0) If a memory operand effective address is outside the SS segment limit.

#UD If EM in CR0 is set.

#NM If TS in CR0 is set.

#MF If there is a pending FPU exception.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made
while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to
FFFFH.

#UD If EM in CR0 is set.

#NM If TS in CR0 is set.

#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to
FFFFH.

#UD If EM in CR0 is set.

#NM If TS in CR0 is set.

#MF If there is a pending FPU exception.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PMULHW—Packed Multiply High

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F E5 /r</td>
<td>PMULHW mm, mm/m64</td>
<td>Multiply the signed packed words in mm by the signed packed words in mm/m64, then store the high-order word of each doubleword result in mm.</td>
</tr>
</tbody>
</table>

**Description**

Multiplies the four signed words of the source operand (second operand) by the four signed words of the destination operand (first operand), producing four signed, doubleword, intermediate results (see Figure 2-13). The high-order word of each intermediate result is then written to its corresponding word location in the destination operand. The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a 64-bit memory location.

**Figure 2-13. Operation of the PMULHW Instruction**

<table>
<thead>
<tr>
<th>PMULHW mm, mm/m64</th>
</tr>
</thead>
<tbody>
<tr>
<td>mm</td>
</tr>
<tr>
<td>mm/m64</td>
</tr>
<tr>
<td>mm</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>0111000111000111000111000111</th>
</tr>
</thead>
<tbody>
<tr>
<td>1000000000000000000000000000</td>
</tr>
<tr>
<td>0000001000000000000000000000</td>
</tr>
<tr>
<td>0111000111000111000111000111</td>
</tr>
</tbody>
</table>

**Operation**

DEST(15..0) ← HighOrderWord(DEST(15..0) × SRC(15..0));  
DEST(31..16) ← HighOrderWord(DEST(31..16) × SRC(31..16));  
DEST(47..32) ← HighOrderWord(DEST(47..32) × SRC(47..32));  
DEST(63..48) ← HighOrderWord(DEST(63..48) × SRC(63..48));

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault</td>
</tr>
<tr>
<td></td>
<td>Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
PMULHW—Packed Multiply High (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PMULLW—Packed Multiply Low

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F D5 /r</td>
<td>PMULLW mm, mm/m64</td>
<td>Multiply the packed words in mm with the packed words in mm/m64, then store the low-order word of each doubleword result in mm.</td>
</tr>
</tbody>
</table>

**Description**

Multiplies the four signed or unsigned words of the source operand (second operand) with the four signed or unsigned words of the destination operand (first operand), producing four doubleword intermediate results (see Figure 2-14). The low-order word of each intermediate result is then written to its corresponding word location in the destination operand. The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a 64-bit memory location.

**Figure 2-14. Operation of the PMULLW Instruction**

<table>
<thead>
<tr>
<th>PMULLW mm, mm/m64</th>
</tr>
</thead>
</table>
| \[
\begin{array}{c}
\text{mm} \\
\text{mm/m64} \\
\text{mm}
\end{array}
\begin{array}{c}
0111000111000111 \\
1000000000000000 \\
1000000000000000
\end{array}
\begin{array}{c}
\star \\
\star \\
\star \\
\star \\
\downarrow \\
\downarrow \\
\downarrow \\
\downarrow \\
0001110000000000
\end{array}
\]

**Operation**

\[
\begin{align*}
\text{DEST}(15..0) & \leftarrow \text{LowOrderWord}(\text{DEST}(15..0) \times \text{SRC}(15..0)); \\
\text{DEST}(31..16) & \leftarrow \text{LowOrderWord}(\text{DEST}(31..16) \times \text{SRC}(31..16)); \\
\text{DEST}(47..32) & \leftarrow \text{LowOrderWord}(\text{DEST}(47..32) \times \text{SRC}(47..32)); \\
\text{DEST}(63..48) & \leftarrow \text{LowOrderWord}(\text{DEST}(63..48) \times \text{SRC}(63..48));
\end{align*}
\]

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

<table>
<thead>
<tr>
<th>Itanium Reg Faults</th>
<th>Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.</th>
</tr>
</thead>
<tbody>
<tr>
<td>Itanium Mem Faults</td>
<td>VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault</td>
</tr>
</tbody>
</table>
PMULLW—Packed Multiply Low (continued)

**Protected Mode Exceptions**

- **#GP(0)**  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
- **#SS(0)**  If a memory operand effective address is outside the SS segment limit.
- **#UD**  If EM in CR0 is set.
- **#NM**  If TS in CR0 is set.
- **#MF**  If there is a pending FPU exception.
- **#PF(fault-code)**  If a page fault occurs.
- **#AC(0)**  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real-Address Mode Exceptions**

- **#GP**  If any part of the operand lies outside of the effective address space from 0 to FFFFH.
- **#UD**  If EM in CR0 is set.
- **#NM**  If TS in CR0 is set.
- **#MF**  If there is a pending FPU exception.

**Virtual-8086 Mode Exceptions**

- **#GP**  If any part of the operand lies outside of the effective address space from 0 to FFFFH.
- **#UD**  If EM in CR0 is set.
- **#NM**  If TS in CR0 is set.
- **#MF**  If there is a pending FPU exception.
- **#PF(fault-code)**  If a page fault occurs.
- **#AC(0)**  If alignment checking is enabled and an unaligned memory reference is made.
POR—Bitwise Logical OR

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F EB /r</td>
<td>POR mm, mm/m64</td>
<td>OR quadword from mm/m64 to quadword in mm.</td>
</tr>
</tbody>
</table>

**Description**

Performs a bitwise logical OR operation on the quadword source (second) and destination (first) operands and stores the result in the destination operand location (see Figure 2-15). The source operand can be an MMX technology register or a quadword memory location; the destination operand must be an MMX technology register. Each bit of the result is made 0 if the corresponding bits of both operands are 0; otherwise the bit is set to 1.

**Figure 2-15. Operation of the POR Instruction.**

![Figure 2-15](image)

**Operation**

DEST ← DEST OR SRC;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- **Itanium Mem Faults**: VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
POR—Bitwise Logical OR (continued)

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#UD    If EM in CR0 is set.
#NM    If TS in CR0 is set.
#MF    If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP    If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD    If EM in CR0 is set.
#NM    If TS in CR0 is set.
#MF    If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP    If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD    If EM in CR0 is set.
#NM    If TS in CR0 is set.
#MF    If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
PSLLW/PSLD/PSLLQ—Packed Shift Left Logical

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F F1 /r</td>
<td>PSLLW mm, mm/m64</td>
<td>Shift words in mm left by amount specified in mm/m64, while shifting in zeros.</td>
</tr>
<tr>
<td>0F 71 /6, ib</td>
<td>PSLLW mm, imm8</td>
<td>Shift words in mm left by imm8, while shifting in zeros.</td>
</tr>
<tr>
<td>0F F2 /r</td>
<td>PSLLD mm, mm/m64</td>
<td>Shift doublewords in mm left by amount specified in mm/m64, while shifting in zeros.</td>
</tr>
<tr>
<td>0F 72 /6 ib</td>
<td>PSLLD mm, imm8</td>
<td>Shift doublewords in mm by imm8, while shifting in zeros.</td>
</tr>
<tr>
<td>0F F3 /r</td>
<td>PSLLQ mm, mm/m64</td>
<td>Shift mm left by amount specified in mm/m64, while shifting in zeros.</td>
</tr>
<tr>
<td>0F 73 /6 ib</td>
<td>PSLLQ mm, imm8</td>
<td>Shift mm left by imm8, while shifting in zeros.</td>
</tr>
</tbody>
</table>

Description

Shifts the bits in the data elements (words, doublewords, or quadword) in the destination operand (first operand) to the left by the number of bits specified in the unsigned count operand (second operand). (See Figure 2-16.) The result of the shift operation is written to the destination operand. As the bits in the data elements are shifted left, the empty low-order bits are cleared (set to zero). If the value specified by the count operand is greater than 15 (for words), 31 (for doublewords), or 63 (for a quadword), then the destination operand is set to all zeros.

The destination operand must be an MMX technology register; the count operand can be either an MMX technology register, a 64-bit memory location, or an 8-bit immediate.

The PSLLW instruction shifts each of the four words of the destination operand to the left by the number of bits specified in the count operand; the PSLLD instruction shifts each of the two doublewords of the destination operand; and the PSLLQ instruction shifts the 64-bit quadword in the destination operand. As the individual data elements are shifted left, the empty low-order bit positions are filled with zeros.

Figure 2-16. Operation of the PSLLW Instruction
PSLLW/PSLLD/PSLLQ—Packed Shift Left Logical (continued)

**Operation**

IF instruction is PSLLW

THEN

\[
\text{DEST}(15..0) \leftarrow \text{DEST}(15..0) \ll \text{COUNT};
\]
\[
\text{DEST}(31..16) \leftarrow \text{DEST}(31..16) \ll \text{COUNT};
\]
\[
\text{DEST}(47..32) \leftarrow \text{DEST}(47..32) \ll \text{COUNT};
\]
\[
\text{DEST}(63..48) \leftarrow \text{DEST}(63..48) \ll \text{COUNT};
\]

ELSE IF instruction is PSLLD

THEN {

\[
\text{DEST}(31..0) \leftarrow \text{DEST}(31..0) \ll \text{COUNT};
\]
\[
\text{DEST}(63..32) \leftarrow \text{DEST}(63..32) \ll \text{COUNT};
\]

ELSE (* instruction is PSLLQ *)

\[
\text{DEST} \leftarrow \text{DEST} \ll \text{COUNT};
\]

FI;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Protected Mode Exceptions**

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#UD  If EM in CR0 is set.

#NM  If TS in CR0 is set.

#MF  If there is a pending FPU exception.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

**Real-Address Mode Exceptions**

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.

#UD  If EM in CR0 is set.

#NM  If TS in CR0 is set.

#MF  If there is a pending FPU exception.
Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.

#UD If EM in CR0 is set.

#NM If TS in CR0 is set.

#MF If there is a pending FPU exception.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PSRAW/PSRAD—Packed Shift Right Arithmetic

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F E1 /r</td>
<td>PSRAW mm, mm/m64</td>
<td>Shift words in mm right by amount specified in mm/m64 while shifting in sign bits.</td>
</tr>
<tr>
<td>0F 71 /4 ib</td>
<td>PSRAW mm, imm8</td>
<td>Shift words in mm right by imm8 while shifting in sign bits</td>
</tr>
<tr>
<td>0F E2 /r</td>
<td>PSRAD mm, mm/m64</td>
<td>Shift doublewords in mm right by amount specified in mm/m64 while shifting in sign bits.</td>
</tr>
<tr>
<td>0F 72 /4 ib</td>
<td>PSRAD mm, imm8</td>
<td>Shift doublewords in mm right by imm8 while shifting in sign bits.</td>
</tr>
</tbody>
</table>

Description

Shifts the bits in the data elements (words or doublewords) in the destination operand (first operand) to the right by the amount of bits specified in the unsigned count operand (second operand). (See Figure 2-17.) The result of the shift operation is written to the destination operand. The empty high-order bits of each element are filled with the initial value of the sign bit of the data element. If the value specified by the count operand is greater than 15 (for words) or 31 (for doublewords), each destination data element is filled with the initial value of the sign bit of the element.

The destination operand must be an MMX technology register; the count operand (source operand) can be either an MMX technology register, a 64-bit memory location, or an 8-bit immediate.

The PSRAW instruction shifts each of the four words in the destination operand to the right by the number of bits specified in the count operand; the PSRAD instruction shifts each of the two doublewords in the destination operand. As the individual data elements are shifted right, the empty high-order bit positions are filled with the sign value.

Figure 2-17. Operation of the PSRAW Instruction

![Figure 2-17. Operation of the PSRAW Instruction](image-url)
PSRAW/PSRAD—Packed Shift Right Arithmetic (continued)

Operation

IF instruction is PSRAW
THEN
    DEST(15..0) ← SignExtend (DEST(15..0) >> COUNT);
    DEST(31..16) ← SignExtend (DEST(31..16) >> COUNT);
    DEST(47..32) ← SignExtend (DEST(47..32) >> COUNT);
    DEST(63..48) ← SignExtend (DEST(63..48) >> COUNT);
ELSE (*instruction is PSRAD*)
    DEST(31..0) ← SignExtend (DEST(31..0) >> COUNT);
    DEST(63..32) ← SignExtend (DEST(63..32) >> COUNT);
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#UD    If EM in CR0 is set.
#NM    If TS in CR0 is set.
#MF    If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.
PSRAW/PSRAD—Packed Shift Right Arithmetic (continued)

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
**PSRLW/PSRLD/PSRLQ—Packed Shift Right Logical**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F D1 /r</td>
<td>PSRLW mm, mm/m64</td>
<td>Shift words in mm right by amount specified in mm/m64 while shifting in zeros.</td>
</tr>
<tr>
<td>0F 71 /2 ib</td>
<td>PSRLW mm, imm8</td>
<td>Shift words in mm right by imm8.</td>
</tr>
<tr>
<td>0F D2 /r</td>
<td>PSRLD mm, mm/m64</td>
<td>Shift doublewords in mm right by amount specified in mm/m64 while shifting in zeros.</td>
</tr>
<tr>
<td>0F 72 /2 ib</td>
<td>PSRLD mm, imm8</td>
<td>Shift doublewords in mm right by imm8.</td>
</tr>
<tr>
<td>0F D3 /r</td>
<td>PSRLQ mm, mm/m64</td>
<td>Shift mm right by amount specified in mm/m64 while shifting in zeros.</td>
</tr>
<tr>
<td>0F 73 /2 ib</td>
<td>PSRLQ mm, imm8</td>
<td>Shift mm right by imm8 while shifting in zeros.</td>
</tr>
</tbody>
</table>

**Description**

Shifts the bits in the data elements (words, doublewords, or quadword) in the destination operand (first operand) to the right by the number of bits specified in the unsigned count operand (second operand). (See Figure 2-18.) The result of the shift operation is written to the destination operand. As the bits in the data elements are shifted right, the empty high-order bits are cleared (set to zero). If the value specified by the count operand is greater than 15 (for words), 31 (for doublewords), or 63 (for a quadword), then the destination operand is set to all zeros.

The destination operand must be an MMX technology register; the count operand can be either an MMX technology register, a 64-bit memory location, or an 8-bit immediate.

The PSRLW instruction shifts each of the four words of the destination operand to the right by the number of bits specified in the count operand; the PSRLD instruction shifts each of the two doublewords of the destination operand; and the PSRLQ instruction shifts the 64-bit quadword in the destination operand. As the individual data elements are shifted right, the empty high-order bit positions are filled with zeros.

**Figure 2-18. Operation of the PSRLW Instruction**
PSRLW/PSRLD/PSRLQ—Packed Shift Right Logical (continued)

Operation

IF instruction is PSRLW
   THEN {
      DEST(15..0) ← DEST(15..0) >> COUNT;
      DEST(31..16) ← DEST(31..16) >> COUNT;
      DEST(47..32) ← DEST(47..32) >> COUNT;
      DEST(63..48) ← DEST(63..48) >> COUNT;
   }
ELSE IF instruction is PSRLD
   THEN {
      DEST(31..0) ← DEST(31..0) >> COUNT;
      DEST(63..32) ← DEST(63..32) >> COUNT;
   }
ELSE (* instruction is PSRLQ *)
   DEST ← DEST >> COUNT;
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#UD    If EM in CR0 is set.
#NM    If TS in CR0 is set.
#MF    If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.
PSRLW/PSRLD/PSRLQ—Packed Shift Right Logical (continued)

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PSUBB/PSUBW/PSUBD—Packed Subtract

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 80 /r</td>
<td>PSUBB mm, mm/m64</td>
<td>Subtract packed bytes in mm/m64 from packed bytes in mm.</td>
</tr>
<tr>
<td>0F 81 /r</td>
<td>PSUBW mm, mm/m64</td>
<td>Subtract packed words in mm/m64 from packed words in mm.</td>
</tr>
<tr>
<td>0F 82 /r</td>
<td>PSUBD mm, mm/m64</td>
<td>Subtract packed doublewords in mm/m64 from packed doublewords in mm.</td>
</tr>
</tbody>
</table>

Description

Subtracts the individual data elements (bytes, words, or doublewords) of the source operand (second operand) from the individual data elements of the destination operand (first operand). (See Figure 2-19.) If the result of a subtraction exceeds the range for the specified data type (overflows), the result is wrapped around, meaning that the result is truncated so that only the lower (least significant) bits of the result are returned (that is, the carry is ignored).

The destination operand must be an MMX technology register; the source operand can be either an MMX technology register or a quadword memory location.

Figure 2-19. Operation of the PSUBW Instruction

The PSUBB instruction subtracts the bytes of the source operand from the bytes of the destination operand and stores the results to the destination operand. When an individual result is too large to be represented in 8 bits, the lower 8 bits of the result are written to the destination operand and therefore the result wraps around.

The PSUBW instruction subtracts the words of the source operand from the words of the destination operand and stores the results to the destination operand. When an individual result is too large to be represented in 16 bits, the lower 16 bits of the result are written to the destination operand and therefore the result wraps around.

The PSUBD instruction subtracts the doublewords of the source operand from the doublewords of the destination operand and stores the results to the destination operand. When an individual result is too large to be represented in 32 bits, the lower 32 bits of the result are written to the destination operand and therefore the result wraps around.
PSUBB/PSUBW/PSUBD—Packed Subtract (continued)

Note that like the integer SUB instruction, the PSUBB, PSUBW, and PSUBD instructions can operate on either unsigned or signed (two's complement notation) packed integers. Unlike the integer instructions, none of the MMX instructions affect the EFLAGS register. With MMX instructions, there are no carry or overflow flags to indicate when overflow has occurred, so the software must control the range of values or else use the “with saturation” MMX instructions.

Operation

IF instruction is PSUBB
THEN

\[
\begin{align*}
\text{DEST}(7..0) & \leftarrow \text{DEST}(7..0) - \text{SRC}(7..0); \\
\text{DEST}(15..8) & \leftarrow \text{DEST}(15..8) - \text{SRC}(15..8); \\
\text{DEST}(23..16) & \leftarrow \text{DEST}(23..16) - \text{SRC}(23..16); \\
\text{DEST}(31..24) & \leftarrow \text{DEST}(31..24) - \text{SRC}(31..24); \\
\text{DEST}(39..32) & \leftarrow \text{DEST}(39..32) - \text{SRC}(39..32); \\
\text{DEST}(47..40) & \leftarrow \text{DEST}(47..40) - \text{SRC}(47..40); \\
\text{DEST}(55..48) & \leftarrow \text{DEST}(55..48) - \text{SRC}(55..48); \\
\text{DEST}(63..56) & \leftarrow \text{DEST}(63..56) - \text{SRC}(63..56); \\
\end{align*}
\]

ELSE IF instruction is PSUBW
THEN

\[
\begin{align*}
\text{DEST}(15..0) & \leftarrow \text{DEST}(15..0) - \text{SRC}(15..0); \\
\text{DEST}(31..16) & \leftarrow \text{DEST}(31..16) - \text{SRC}(31..16); \\
\text{DEST}(47..32) & \leftarrow \text{DEST}(47..32) - \text{SRC}(47..32); \\
\text{DEST}(63..48) & \leftarrow \text{DEST}(63..48) - \text{SRC}(63..48); \\
\end{align*}
\]

ELSE { (* instruction is PSUBD *)

\[
\begin{align*}
\text{DEST}(31..0) & \leftarrow \text{DEST}(31..0) - \text{SRC}(31..0); \\
\text{DEST}(63..32) & \leftarrow \text{DEST}(63..32) - \text{SRC}(63..32); \\
\end{align*}
\]

FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
PSUBB/PSUBW/PSUBD—Packed Subtract (continued)

Protected Mode Exceptions

#GP(0) If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0) If a memory operand effective address is outside the SS segment limit.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PSUBSB/PSUBSW—Packed Subtract with Saturation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F E8 /r</td>
<td>PSUBSB mm, mm/m64</td>
<td>Subtract signed packed bytes in mm/m64 from signed packed bytes in mm and saturate.</td>
</tr>
<tr>
<td>0F E9 /r</td>
<td>PSUBSW mm, mm/m64</td>
<td>Subtract signed packed words in mm/m64 from signed packed words in mm and saturate.</td>
</tr>
</tbody>
</table>

**Description**

Subtracts the individual signed data elements (bytes or words) of the source operand (second operand) from the individual signed data elements of the destination operand (first operand). (See Figure 2-20.) If the result of a subtraction exceeds the range for the specified data type, the result is saturated. The destination operand must be an MMX technology register; the source operand can be either an MMX technology register or a quadword memory location.

**Figure 2-20. Operation of the PSUBSW Instruction**

The PSUBSB instruction subtracts the signed bytes of the source operand from the signed bytes of the destination operand and stores the results to the destination operand. When an individual result is beyond the range of a signed byte (that is, greater than 7FH or less than 80H), the saturated byte value of 7FH or 80H, respectively, is written to the destination operand.

The PSUBSW instruction subtracts the signed words of the source operand from the signed words of the destination operand and stores the results to the destination operand. When an individual result is beyond the range of a signed word (that is, greater than 7FFFH or less than 8000H), the saturated word value of 7FFFH or 8000H, respectively, is written to the destination operand.
PSUBSB/PSUBSW—Packed Subtract with Saturation (continued)

Operation

IF instruction is PSUBSB
THEN
  DEST(7..0) ← SaturateToSignedByte(DEST(7..0) – SRC(7..0));
  DEST(15..8) ← SaturateToSignedByte(DEST(15..8) – SRC(15..8));
  DEST(23..16) ← SaturateToSignedByte(DEST(23..16) – SRC(23..16));
  DEST(31..24) ← SaturateToSignedByte(DEST(31..24) – SRC(31..24));
  DEST(39..32) ← SaturateToSignedByte(DEST(39..32) – SRC(39..32));
  DEST(47..40) ← SaturateToSignedByte(DEST(47..40) – SRC(47..40));
  DEST(55..48) ← SaturateToSignedByte(DEST(55..48) – SRC(55..48));
  DEST(63..56) ← SaturateToSignedByte(DEST(63..56) – SRC(63..56));
ELSE (* instruction is PSUBSW *)
  DEST(15..0) ← SaturateToSignedWord(DEST(15..0) – SRC(15..0));
  DEST(31..16) ← SaturateToSignedWord(DEST(31..16) – SRC(31..16));
  DEST(47..32) ← SaturateToSignedWord(DEST(47..32) – SRC(47..32));
  DEST(63..48) ← SaturateToSignedWord(DEST(63..48) – SRC(63..48));
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#UD   If EM in CR0 is set.
#NM   If TS in CR0 is set.
#MF   If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
PSUBSB/PSUBSW—Packed Subtract with Saturation (continued)

Real-Address Mode Exceptions

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.

#UD  If EM in CR0 is set.

#NM  If TS in CR0 is set.

#MF  If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.

#UD  If EM in CR0 is set.

#NM  If TS in CR0 is set.

#MF  If there is a pending FPU exception.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
PSUBUSB/PSUBUSW—Packed Subtract Unsigned with Saturation

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F D8 /r</td>
<td>PSUBUSB mm, mm/m64</td>
<td>Subtract unsigned packed bytes in mm/m64 from unsigned packed bytes in mm and saturate.</td>
</tr>
<tr>
<td>0F D9 /r</td>
<td>PSUBUSW mm, mm/m64</td>
<td>Subtract unsigned packed words in mm/m64 from unsigned packed words in mm and saturate.</td>
</tr>
</tbody>
</table>

**Description**

Subtracts the individual unsigned data elements (bytes or words) of the source operand (second operand) from the individual unsigned data elements of the destination operand (first operand). (See Figure 2-21.) If the result of an individual subtraction exceeds the range for the specified unsigned data type, the result is saturated. The destination operand must be an MMX technology register; the source operand can be either an MMX technology register or a quadword memory location.

**Figure 2-21. Operation of the PSUBUSB Instruction**

The PSUBUSB instruction subtracts the unsigned bytes of the source operand from the unsigned bytes of the destination operand and stores the results to the destination operand. When an individual result is less than zero (a negative value), the saturated unsigned byte value of 00H is written to the destination operand.

The PSUBUSW instruction subtracts the unsigned words of the source operand from the unsigned words of the destination operand and stores the results to the destination operand. When an individual result is less than zero (a negative value), the saturated unsigned word value of 0000H is written to the destination operand.
PSUBUSB/PSUBUSW—Packed Subtract Unsigned with Saturation (continued)

Operation

IF instruction is PSUBUSB
THEN

    DEST(7..0) ← SaturateToUnsignedByte (DEST(7..0) – SRC(7..0));
    DEST(15..8) ← SaturateToUnsignedByte (DEST(15..8) – SRC(15..8));
    DEST(23..16) ← SaturateToUnsignedByte (DEST(23..16) – SRC(23..16));
    DEST(31..24) ← SaturateToUnsignedByte (DEST(31..24) – SRC(31..24));
    DEST(39..32) ← SaturateToUnsignedByte (DEST(39..32) – SRC(39..32));
    DEST(47..40) ← SaturateToUnsignedByte (DEST(47..40) – SRC(47..40));
    DEST(55..48) ← SaturateToUnsignedByte (DEST(55..48) – SRC(55..48));
    DEST(63..56) ← SaturateToUnsignedByte (DEST(63..56) – SRC(63..56));

ELSE { (* instruction is PSUBUSW *)

    DEST(15..0) ← SaturateToUnsignedWord (DEST(15..0) – SRC(15..0));
    DEST(31..16) ← SaturateToUnsignedWord (DEST(31..16) – SRC(31..16));
    DEST(47..32) ← SaturateToUnsignedWord (DEST(47..32) – SRC(47..32));
    DEST(63..48) ← SaturateToUnsignedWord (DEST(63..48) – SRC(63..48));

FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.

Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.

#SS(0)  If a memory operand effective address is outside the SS segment limit.

#UD  If EM in CR0 is set.

#NM  If TS in CR0 is set.

#MF  If there is a pending FPU exception.

#PF(fault-code)  If a page fault occurs.

#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
PSUBUSB/PSUBUSW—Packed Subtract Unsigned with Saturation (continued)

Real-Address Mode Exceptions

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP  If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made.
PUNPCKHBW/PUNPCKHWD/PUNPCKHDQ—Unpack High Packed Data

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 68 /r</td>
<td>PUNPCKHBW mm, mm/m64</td>
<td>Interleave high-order bytes from mm and mm/m64 into mm.</td>
</tr>
<tr>
<td>0F 69 /r</td>
<td>PUNPCKHWD mm, mm/m64</td>
<td>Interleave high-order words from mm and mm/m64 into mm.</td>
</tr>
<tr>
<td>0F 6A /r</td>
<td>PUNPCKHDQ mm, mm/m64</td>
<td>Interleave high-order doublewords from mm and mm/m64 into mm.</td>
</tr>
</tbody>
</table>

Description

Unpacks and interleaves the high-order data elements (bytes, words, or doublewords) of the destination operand (first operand) and source operand (second operand) into the destination operand (see Figure 2-22). The low-order data elements are ignored. The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a 64-bit memory location. When the source data comes from a memory operand, the full 64-bit operand is accessed from memory, but the instruction uses only the high-order 32 bits.

Figure 2-22. High-order Unpacking and Interleaving of Bytes with the PUNPCKHBW Instruction

The PUNPCKHBW instruction interleaves the four high-order bytes of the source operand and the four high-order bytes of the destination operand and writes them to the destination operand.

The PUNPCKHWD instruction interleaves the two high-order words of the source operand and the two high-order words of the destination operand and writes them to the destination operand.

The PUNPCKHDQ instruction interleaves the high-order doubleword of the source operand and the high-order doubleword of the destination operand and writes them to the destination operand.

If the source operand is all zeros, the result (stored in the destination operand) contains zero extensions of the high-order data elements from the original value in the destination operand. With the PUNPCKHBW instruction the high-order bytes are zero extended (that is, unpacked into unsigned words), and with the PUNPCKHWD instruction, the high-order words are zero extended (unpacked into unsigned doublewords).
PUNPCKHBW/PUNPCKHWD/PUNPCKHDQ—Unpack High Packed Data
(continued)

Operation

IF instruction is PUNPCKHBW
THEN
  DEST(7..0) ← DEST(39..32);
  DEST(15..8) ← SRC(39..32);
  DEST(23..16) ← DEST(47..40);
  DEST(31..24) ← SRC(47..40);
  DEST(39..32) ← DEST(55..48);
  DEST(47..40) ← SRC(55..48);
  DEST(55..48) ← DEST(63..56);
  DEST(63..56) ← SRC(63..56);
ELSE IF instruction is PUNPCKHWD
THEN
  DEST(15..0) ← DEST(47..32);
  DEST(31..16) ← SRC(47..32);
  DEST(47..32) ← DEST(63..48);
  DEST(63..48) ← SRC(63..48);
ELSE (* instruction is PUNPCKHDQ *)
  DEST(31..0) ← DEST(63..32)
  DEST(63..32) ← SRC(63..32);
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#UD  If EM in CR0 is set.
#NM  If TS in CR0 is set.
#MF  If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
PUNPCKHBW/PUNPCKHWD/PUNPCKHDQ—Unpack High Packed Data
(continued)

Real-Address Mode Exceptions
#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions
#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.
#UD If EM in CR0 is set.
#NM If TS in CR0 is set.
#MF If there is a pending FPU exception.
#PF(fault-code) If a page fault occurs.
#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PUNPCKLBW/PUNPCKLWD/PUNPCKLDQ—Unpack Low Packed Data

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F 60 /r</td>
<td>PUNPCKLBW mm, mm/m32</td>
<td>Interleave low-order bytes from mm and mm/m64 into mm.</td>
</tr>
<tr>
<td>0F 61 /r</td>
<td>PUNPCKLWD mm, mm/m32</td>
<td>Interleave low-order words from mm and mm/m64 into mm.</td>
</tr>
<tr>
<td>0F 62 /r</td>
<td>PUNPCKLDQ mm, mm/m32</td>
<td>Interleave low-order doublewords from mm and mm/m64 into mm.</td>
</tr>
</tbody>
</table>

Description

Unpacks and interleaves the low-order data elements (bytes, words, or doublewords) of the destination and source operands into the destination operand (see Figure 2-23). The destination operand must be an MMX technology register; the source operand may be either an MMX technology register or a memory location. When source data comes from an MMX technology register, the upper 32 bits of the register are ignored. When the source data comes from a memory, only 32-bits are accessed from memory.

Figure 2-23. Low-order Unpacking and Interleaving of Bytes with the PUNPCKLBW Instruction

The PUNPCKLBW instruction interleaves the four low-order bytes of the source operand and the four low-order bytes of the destination operand and writes them to the destination operand.

The PUNPCKLWD instruction interleaves the two low-order words of the source operand and the two low-order words of the destination operand and writes them to the destination operand.

The PUNPCKLDQ instruction interleaves the low-order doubleword of the source operand and the low-order doubleword of the destination operand and writes them to the destination operand.

If the source operand is all zeros, the result (stored in the destination operand) contains zero extensions of the high-order data elements from the original value in the destination operand. With the PUNPCKLBW instruction the low-order bytes are zero extended (that is, unpacked into unsigned words), and with the PUNPCKLWD instruction, the low-order words are zero extended (unpacked into unsigned doublewords).
PUNPCKLBW/PUNPCKLWD/PUNPCKLDQ—Unpack Low Packed Data (continued)

Operation

IF instruction is PUNPCKLBW
THEN
  DEST(63..56) ← SRC(31..24);
  DEST(55..48) ← DEST(31..24);
  DEST(47..40) ← SRC(23..16);
  DEST(39..32) ← DEST(23..16);
  DEST(31..24) ← SRC(15..8);
  DEST(23..16) ← DEST(15..8);
  DEST(15..8) ← SRC(7..0);
  DEST(7..0) ← DEST(7..0);
ELSE IF instruction is PUNPCKLWD
THEN
  DEST(63..48) ← SRC(31..16);
  DEST(47..32) ← DEST(31..16);
  DEST(31..16) ← SRC(15..0);
  DEST(15..0) ← DEST(15..0);
ELSE (* instruction is PUNPCKLDQ *)
  DEST(63..32) ← SRC(31..0);
  DEST(31..0) ← DEST(31..0);
FI;

Flags Affected

None.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
Itanium Mem Faults  VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Protected Mode Exceptions

#GP(0)  If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.
#SS(0)  If a memory operand effective address is outside the SS segment limit.
#UD    If EM in CR0 is set.
#NM    If TS in CR0 is set.
#MF    If there is a pending FPU exception.
#PF(fault-code)  If a page fault occurs.
#AC(0)  If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.
PUNPCKLBW/PUNPCKLWD/PUNPCKLDQ—Unpack Low Packed Data (continued)

Real-Address Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.

#UD If EM in CR0 is set.

#NM If TS in CR0 is set.

#MF If there is a pending FPU exception.

Virtual-8086 Mode Exceptions

#GP If any part of the operand lies outside of the effective address space from 0 to FFFFH.

#UD If EM in CR0 is set.

#NM If TS in CR0 is set.

#MF If there is a pending FPU exception.

#PF(fault-code) If a page fault occurs.

#AC(0) If alignment checking is enabled and an unaligned memory reference is made.
PXOR—Logical Exclusive OR

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F EF /r</td>
<td>PXOR mm, mm/m64</td>
<td>XOR quadword from mm/m64 to quadword in mm.</td>
</tr>
</tbody>
</table>

**Description**

Performs a bitwise logical exclusive-OR (XOR) operation on the quadword source (second) and destination (first) operands and stores the result in the destination operand location (see Figure 2-24). The source operand can be an MMX technology register or a quadword memory location; the destination operand must be an MMX technology register. Each bit of the result is 1 if the corresponding bits of the two operands are different; each bit is 0 if the corresponding bits of the operands are the same.

**Figure 2-24. Operation of the PXOR Instruction**

```
    0001000011011001010100000011000100011110111011110001010110010101
    1110111100100001010100000011010010101011011001110110001011100010

   ^   

    0001000011011001010100000011000100011110111011110001010110010101
    1110111100100001010100000011010010101011011001110110001011100010
```

**Operation**

DEST ← DEST XOR SRC;

**Flags Affected**

None.

**Additional Itanium System Environment Exceptions**

- **Itanium Reg Faults**
  - Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Abort.
- **Itanium Mem Faults**
  - VHPT Data Fault, Nested TLB Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
PXOR—Logical Exclusive OR (continued)

**Protected Mode Exceptions**

<table>
<thead>
<tr>
<th>Exception Code</th>
<th>Condition</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP(0)</td>
<td>If a memory operand effective address is outside the CS, DS, ES, FS or GS segment limit.</td>
</tr>
<tr>
<td>#SS(0)</td>
<td>If a memory operand effective address is outside the SS segment limit.</td>
</tr>
<tr>
<td>#UD</td>
<td>If EM in CR0 is set.</td>
</tr>
<tr>
<td>#NM</td>
<td>If TS in CR0 is set.</td>
</tr>
<tr>
<td>#MF</td>
<td>If there is a pending FPU exception.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made while the current privilege level is 3.</td>
</tr>
</tbody>
</table>

**Real-Address Mode Exceptions**

<table>
<thead>
<tr>
<th>Exception Code</th>
<th>Condition</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP</td>
<td>If any part of the operand lies outside of the effective address space from 0 to FFFFH.</td>
</tr>
<tr>
<td>#UD</td>
<td>If EM in CR0 is set.</td>
</tr>
<tr>
<td>#NM</td>
<td>If TS in CR0 is set.</td>
</tr>
<tr>
<td>#MF</td>
<td>If there is a pending FPU exception.</td>
</tr>
</tbody>
</table>

**Virtual-8086 Mode Exceptions**

<table>
<thead>
<tr>
<th>Exception Code</th>
<th>Condition</th>
</tr>
</thead>
<tbody>
<tr>
<td>#GP</td>
<td>If any part of the operand lies outside of the effective address space from 0 to FFFFH.</td>
</tr>
<tr>
<td>#UD</td>
<td>If EM in CR0 is set.</td>
</tr>
<tr>
<td>#NM</td>
<td>If TS in CR0 is set.</td>
</tr>
<tr>
<td>#MF</td>
<td>If there is a pending FPU exception.</td>
</tr>
<tr>
<td>#PF(fault-code)</td>
<td>If a page fault occurs.</td>
</tr>
<tr>
<td>#AC(0)</td>
<td>If alignment checking is enabled and an unaligned memory reference is made.</td>
</tr>
</tbody>
</table>
3.1 IA-32 Streaming SIMD Extension Instructions

This section lists the IA-32 Streaming SIMD Extension instructions designed to increase performance of IA-32 3D and floating-point intensive applications. For details on Streaming SIMD Extension please refer to the Intel Architecture Software Developer’s Manual.

3.2 About the Intel® Architecture Streaming SIMD Extensions

The Streaming SIMD Extensions for the Intel Architecture (IA) accelerates performance of 3D graphics applications over the current P6 generation of the Pentium Pro, Pentium II and Pentium III processors. The programming model is similar to the MMX technology model except that instructions now operate on new packed floating-point data types which contain four single-precision floating-point numbers.

The Streaming SIMD Extensions introduces new general purpose floating-point instructions, which operate on a new set of eight 128-bit Streaming SIMD Extension registers. This gives the programmer the ability to develop algorithms that can finely mix packed single-precision floating-point and integer using both Streaming SIMD Extension and MMX instructions respectively. In addition to these instructions, Streaming SIMD Extensions also provides new instructions to control cacheability of all MMX technology data types. These include ability to stream data into and from the processor while minimizing pollution of the caches and the ability to prefetch data before it is actually used. The main focus of packed floating-point instructions is the acceleration of 3D geometry. The new definition also contains additional SIMD Integer instructions to accelerate 3D rendering and video encoding and decoding. Together with the cacheability control instruction, this combination enables the development of new algorithms that can significantly accelerate 3D graphics.

The new Streaming SIMD Extension state requires OS support for saving and restoring the new state during a context switch. A new set of extended FSAVE/FRSTOR instructions will permit saving/restoring new and existing state for applications and OS. To make use of these new instructions, an application must verify that the processor supports Streaming SIMD Extensions extensions and the operating system supports this new extension. If both the extension and support is enabled, then the software application can use the new features.

The Streaming SIMD Extension instruction set is fully compatible with all software written for Intel Architecture microprocessors. All existing software continues to run correctly, without modification, on microprocessors that incorporate the Streaming SIMD Extensions, as well as in the presence of existing and new applications that incorporate this technology.
3.3 Single Instruction Multiple Data

The Streaming SIMD Extensions uses the Single Instruction Multiple Data (SIMD) technique. This technique speeds up software performance by processing multiple data elements in parallel, using a single instruction. The Streaming SIMD Extensions supports operations on packed single-precision floating-point data types, and the additional SIMD Integer instructions support operations on packed quadrate data types (byte, word, or double-word). This approach was chosen because most 3D graphics and DSP applications have the following characteristics:

- Inherently parallel
- Wide dynamic range, hence floating-point based
- Regular and re-occurring memory access patterns
- Localized re-occurring operations performed on the data
- Data independent control flow

Streaming SIMD Extensions is 100% compatible with the IEEE Standard 754 for Binary Floating-point Arithmetic. The Streaming SIMD Extension instructions are accessible from all IA execution modes: Protected mode, Real address mode, and Virtual 8086 mode.

New Features

Streaming SIMD Extensions provides the following new features, while maintaining backward compatibility with all existing Intel Architecture microprocessors, IA applications and operating systems.

- New data type
- Eight Streaming SIMD Extension registers
- Enhanced instruction set

Streaming SIMD Extensions can enhance the performance of applications that use these features.

3.4 New Data Types

The principal data type of the Streaming SIMD Extensions is a packed single-precision floating-point operand, specifically:

- Four 32-bit single-precision (SP) floating-point numbers (Figure 3-1).

The SIMD Integer instructions will operate on the packed byte, word or doubleword data types. The prefetch instruction works on typeless data of size 32 bytes or greater.

Figure 3-1. Packed Single-FP Data Type

<table>
<thead>
<tr>
<th>127</th>
<th>96</th>
<th>95</th>
<th>65</th>
<th>63</th>
<th>32</th>
<th>31</th>
<th>0</th>
</tr>
</thead>
</table>

Packed Single-FP
3.5 **Streaming SIMD Extension Registers**

The Streaming SIMD Extensions provides eight 128-bit general purpose registers, each of which can be directly addressed. These registers are new state, and require support from the operating system to use them.

The Streaming SIMD Extension registers can hold packed 128-bit data. The Streaming SIMD Extension instructions access the Streaming SIMD Extension registers directly using the registers names XMM0 to XMM7 (Figure 3-2).

Streaming SIMD Extension registers can be used to perform calculation on data. They cannot be used to address memory; addressing is accomplished by using the integer registers and existing IA addressing modes.

The contents of Streaming SIMD Extension registers are cleared upon reset.

There is a new control/status register MXCSR which is used to mask/unmask numerical exception handling, to set rounding modes, to set flush-to-zero mode, and to view status flags.

**Figure 3-2. Streaming SIMD Extension Register Set**

<p>| | | | |</p>
<table>
<thead>
<tr>
<th></th>
<th></th>
<th></th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>XMM7</td>
<td>XMM6</td>
<td>XMM5</td>
<td>XMM4</td>
</tr>
<tr>
<td>XMM3</td>
<td>XMM2</td>
<td>XMM1</td>
<td>XMM0</td>
</tr>
</tbody>
</table>

3.6 **Extended Instruction Set**

The Streaming SIMD Extensions supplies a rich set of instructions that operate on either all or the least significant pairs of packed data operands, in parallel. The packed instructions operate on a pair of operands as shown in Figure 3-3 while scalar instructions always operate on the least significant pair of the two operands as shown in Figure 3-4; for scalar operations, the three upper components from the first operand are passed through to the destination. In general, the address of a memory operand has to be aligned on a 16-byte boundary for all instructions, except for unaligned loads and stores.
3.6.1 Instruction Group Review

3.6.1.1 Arithmetic Instructions

Packed/Scalar Addition and Subtraction
The ADDPS (Add packed single-precision floating-point) and SUBPS (Subtract packed single-precision floating-point) instructions add or subtract four pairs of packed single-precision floating-point operands.

The ADDSS (Add scalar single-precision floating-point) and SUBSS (Subtract scalar single-precision floating-point) instructions add or subtract the least significant pair of packed single-precision floating-point operands; the upper three fields are passed through from the source operand.

Packed/Scalar Multiplication and Division
The MULPS (Multiply packed single-precision floating-point) instruction multiplies four pairs of packed single-precision floating-point operands.
The MULSS (Multiply scalar single-precision floating-point) instruction multiplies the least significant pair of packed single-precision floating-point operands; the upper three fields are passed through from the source operand.

The DIVPS (Divide packed single-precision floating-point) instruction divides four pairs of packed single-precision floating-point operands.

The DIVSS (Divide scalar single-precision floating-point) instruction divides the least significant pair of packed single-precision floating-point operands; the upper three fields are passed through from the source operand.

**Packed/Scalar Square Root**

The SQRTPS (Square root packed single-precision floating-point) instruction returns the square root of the packed four single-precision floating-point numbers from the source to a destination register.

The SQRTSS (Square root scalar single-precision floating-point) instruction returns the square root of the least significant component of the packed single-precision floating-point numbers from source to a destination register; the upper three fields are passed through from the source operand.

**Packed Maximum/Minimum**

The MAXPS (Maximum packed single-precision floating-point) instruction returns the maximum of each pair of packed single-precision floating-point numbers into the destination register.

The MAXSS (Maximum scalar single-precision floating-point) instruction returns the maximum of the least significant pair of packed single-precision floating-point numbers into the destination register; the upper three fields are passed through from the source operand to the destination register.

The MINPS (Minimum packed single-precision floating-point) instruction returns the minimum of each pair of packed single-precision floating-point numbers into the destination register.

The MINSS (Minimum scalar single-precision floating-point) instruction returns the minimum of the least significant pair of packed single-precision floating-point numbers into the destination register; the upper three fields are passed through from the source operand to the destination register.

### 3.6.1.2 Logical Instructions

The ANDPS (Bit-wise packed logical AND for single-precision floating-point) instruction returns a bitwise AND between the two operands.

The ANDNPS (Bit-wise packed logical AND NOT for single-precision floating-point) instruction returns a bitwise AND NOT between the two operands.

The ORPS (Bit-wise packed logical OR for single-precision floating-point) instruction returns a bitwise OR between the two operands.

The XORPS (Bit-wise packed logical XOR for single-precision floating-point) instruction returns a bitwise XOR between the two operands.
3.6.1.3 Compare Instructions

The CMPPS (Compare packed single-precision floating-point) instruction compares four pairs of packed single-precision floating-point numbers using the immediate operand as a predicate, returning per SP field an all “1” 32-bit mask or an all “0” 32-bit mask as a result. The instruction supports a full set of 12 conditions: equal, less than, less than equal, greater than, greater than or equal, unordered, not equal, not less than, not less than or equal, not greater than, not greater than or equal, ordered.

The CMPSS (Compare scalar single-precision floating-point) instruction compares the least significant pairs of packed single-precision floating-point numbers using the immediate operand as a predicate (same as CMPPS), returning per SP field an all “1” 32-bit mask or an all “0” 32-bit mask as a result.

The COMISS (Compare scalar single-precision floating-point ordered and set EFLAGS) instruction compares the least significant pairs of packed single-precision floating-point numbers and sets the ZF,PF,CF bits in the EFLAGS register (the OF, SF and AF bits are cleared).

The UCOMISS (Unordered compare scalar single-precision floating-point ordered and set EFLAGS) instruction compares the least significant pairs of packed single-precision floating-point numbers and sets the ZF,PF,CF bits in the EFLAGS register as described above (the OF, SF and AF bits are cleared).

3.6.1.4 Shuffle Instructions

The SHUFPS (Shuffle packed single-precision floating-point) instruction is able to shuffle any of the packed four single-precision floating-point numbers from one source operand to the lower two destination fields; the upper two destination fields are generated from a shuffle of any of the four SP FP numbers from the second source operand (Figure 3-5). By using the same register for both sources, SHUFPS can return any combination of the four SP FP numbers from this register.

Figure 3-5. Packed Shuffle Operation

![Figure 3-5. Packed Shuffle Operation](image)

The UNPCKHPS (Unpacked high packed single-precision floating-point) instruction performs an interleaved unpack of the high-order data elements of first and second packed single-precision floating-point operands. It ignores the lower half part of the sources (Figure 3-6). When unpacking from a memory operand, the full 128-bit operand is accessed from memory but only the high order 64 bits are utilized by the instruction.
The UNPCKLPS (Unpacked low packed single-precision floating-point) instruction performs an interleaved unpack of the low-order data elements of first and second packed single-precision floating-point operands. It ignores the higher half part of the sources (Figure 3-7). When unpacking from a memory operand, the full 128-bit operand is accessed from memory but only the low order 64 bits are utilized by the instruction.

3.6.1.5 Conversion Instructions

These instructions support packed and scalar conversions between 128-bit Streaming SIMD Extension registers and either 64-bit integer MMX technology registers or 32-bit integer IA-32 registers. The packed versions behave identically to original MMX instructions, in the presence of x87-FP instructions, including:

- Transition from x87-FP to MMX technology (TOS=0, FP valid bits set to all valid).
- MMX instructions write ones (1’s) to the exponent part of the corresponding x87-FP register.
- Use of EMMS for transition from MMX technology to x87-FP.

The CVTPI2PS (Convert packed 32-bit integer to packed single-precision floating-point) instruction converts two 32-bit signed integers in a MMX technology register to the two least significant single-precision floating-point numbers; when the conversion is inexact, the rounded value according to the rounding mode in MXCSR is returned. The upper two significant numbers in the destination register are retained.
The CVTSI2SS (Convert scalar 32-bit integer to scalar single-precision floating-point) instruction converts a 32-bit signed integer in a MMX technology register to the least significant single-precision floating-point number; when the conversion is inexact, the rounded value according to the rounding mode in MXCSR is returned. The upper three significant numbers in the destination register are retained.

The CVTPS2PI (Convert packed single-precision floating-point to packed 32-bit integer) instruction converts the two least significant single-precision floating-point numbers to two 32-bit signed integers in a MMX technology register; when the conversion is inexact, the rounded value according to the rounding mode in MXCSR is returned. The CVTTPS2PI (Convert truncate packed single-precision floating-point to packed 32-bit integer) instruction is similar to CVTPS2PI except if the conversion is inexact, in which case the truncated result is returned.

The CVTSS2SI (Convert scalar single-precision floating-point to a 32-bit integer) instruction converts the least significant single-precision floating-point number to a 32-bit signed integer in an Intel Architecture 32-bit integer register; when the conversion is inexact, the rounded value according to the rounding mode in MXCSR is returned. The CVTTSS2SI (Convert truncate scalar single-precision floating-point to scalar 32-bit integer) instruction is similar to CVTSS2SI except if the conversion is inexact, the truncated result is returned.

### 3.6.1.6 Data Movement Instructions

The MOVAPS (Move aligned packed single-precision floating-point) instruction transfers 128-bits of packed data from memory to Streaming SIMD Extension registers and vice versa, or between Streaming SIMD Extension registers. The memory address is aligned to 16-byte boundary; if not then a general protection exception will occur.

The MOVUPS (Move unaligned packed single-precision floating-point) instruction transfers 128-bits of packed data from memory to Streaming SIMD Extension registers and vice versa, or between Streaming SIMD Extension registers. No assumption is made for alignment.

The MOVHPS (Move aligned high packed single-precision floating-point) instruction transfers 64-bits of packed data from memory to the upper two fields of a Streaming SIMD Extension register and vice versa. The lower field is left unchanged.

The MOVLPS (Move aligned low packed single-precision floating-point) instruction transfers 64-bits of packed data from memory to the lower two fields of a Streaming SIMD Extension register and vice versa. The upper field is left unchanged.

The MOVMSKPS (Move mask packed single-precision floating-point) instruction transfers the most significant bit of each of the four packed single-precision floating-point number to an IA integer register. This 4-bit value can then be used as a condition to perform branching.

The MOVSS (Move scalar single-precision floating-point) instruction transfers a single 32-bit floating-point number from memory to a Streaming SIMD Extension register or vice versa, and between registers.
3.6.1.7 State Management Instructions

The LDMXCSR (Load Streaming SIMD Extension Control and Status Register) instruction loads the Streaming SIMD Extension control and status register from memory. STMXCSR (Store Streaming SIMD Extension Control and Status Register) instruction stores the Streaming SIMD Extension control and status word to memory.

The FXSAVE instruction saves FP and MMX technology state and Streaming SIMD Extension state to memory. Unlike FSAVE, FXSAVE does not clear the x87-FP state. FXRSTOR loads FP and MMX technology state and Streaming SIMD Extension state from memory.

3.6.1.8 Additional SIMD Integer Instructions

Similar to the conversions instructions discussed in Section 3.6.1.5, these SIMD Integer instructions also behave identically to original MMX instructions, in the presence of x87-FP instructions.

The PAVGB/PAVGW (Average unsigned source sub-operands, without incurring a loss in precision) instructions add the unsigned data elements of the source operand to the unsigned data elements of the destination register. The results of the add are then each independently right shifted right by one bit position. The high order bits of each element are filled with the carry bits of the sums. To prevent cumulative round-off errors, an averaging is performed. The low order bit of each final shifted result is set to 1 if at least one of the two least significant bits of the intermediate unshifted shifted sum is 1.

The PEXTRW (Extract 16-bit word from MMX technology register) instruction moves the word in a MMX technology register selected by the two least significant bits of the immediate operand to the lower half of a 32-bit integer register; the upper word in the integer register is cleared.

The PINSRW (Insert 16-bit word into MMX technology register) instruction moves the lower word in a 32-bit integer register or 16-bit word from memory into one of the four word locations in a MMX technology register, selected by the two least significant bits of the immediate operand.

The PMAXUB/PMAXSW (Maximum of packed unsigned integer bytes or signed integer words) instruction returns the maximum of each pair of packed elements into the destination register.

The PMINUB/PMINSW (Minimum of packed unsigned integer bytes or signed integer words) instructions returns the minimum of each pair of packed data elements into the destination register.

The PMOVMSKB (Move Byte Mask from MMX technology register) instruction returns an 8-bit mask formed of the most significant bits of each byte of its source operand in a MMX technology register to an IA integer register.

The PMULHUW (Unsigned high packed integer word multiply in MMX technology register) instruction performs an unsigned multiply on each word field of the two source MMX technology registers, returning the high word of each result to a MMX technology register.

The PSADBW (Sum of absolute differences) instruction computes the absolute difference for each pair of sub-operand byte sources and then accumulates the 8 differences into a single 16-bit result.

The PSHUFW (Shuffle packed integer word in MMX technology register) instruction performs a full shuffle of any source word field to any result word field, using an 8-bit immediate operand.
3.6.1.9 Cacheability Control Instructions

Data referenced by a programmer can have temporal (data will be used again) or spatial (data will be in adjacent locations, e.g. same cache line) locality. Some multimedia data types, such as the display list in a 3D graphics application, are referenced once and not reused in the immediate future. We will refer to this data type as non-temporal data. Thus the programmer does not want the application’s cached code and data to be overwritten by this non-temporal data. The cacheability control instructions enable the programmer to control caching so that non-temporal accesses will minimize cache pollution.

In addition, the execution engine needs to be fed such that it does not become stalled waiting for data. Streaming SIMD Extension instructions allow the programmer to prefetch data long before it’s final use. These instructions are not architectural since they do not update any architectural state, and are specific to each implementation. The programmer may have to tune his application for each implementation to take advantage of these instructions. These instructions merely provide a hint to the hardware, and they will not generate exceptions or faults. Excessive use of prefetch instructions may be throttled by the processor.

The following four instructions provide hints to the cache hierarchy which enables the data to be prefetched to different levels of the cache hierarchy and avoid polluting cache with non-temporal data.

The MASKMOVQ (Non-temporal byte mask store of packed integer in a MMX technology register) instruction stores data from a MMX technology register to the location specified by the EDI register. The most significant bit in each byte of the second MMX technology mask register is used to selectively write the data of the first register on a per-byte basis. The instruction is implicitly weakly-ordered, with all of the characteristics of the WC memory type; successive non-temporal stores may not write memory in program-order, do not write-allocate (i.e. the processor will not fetch the corresponding cache line into the cache hierarchy, prior to performing the store), write combine/collapse, and minimize cache pollution.

The MOVNTQ (Non-temporal store of packed integer in a MMX technology register) instruction stores data from a MMX technology register to memory. The instruction is implicitly weakly-ordered, does not write-allocate and minimizes cache pollution.

The MOVNTPS (Non-temporal store of packed single-precision floating-point) instruction stores data from a Streaming SIMD Extension register to memory. The memory address must be aligned to a 16-byte boundary; if it is not aligned, a general protection exception will occur. The instruction is implicitly weakly-ordered, does not write-allocate and minimizes cache pollution.

The main difference between a non-temporal store and a regular cacheable store is in the write-allocation policy. The memory type of the region being written to can override the non-temporal hint, leading to the following considerations:

- If the programmer specifies a non-temporal store to uncacheable memory, then the store behaves like an uncacheable store; the non-temporal hint is ignored and the memory type for the region is retained. Uncacheable as referred to here means that the region being written to has been mapped with either a UC or WP memory type. If the memory region has been mapped as WB, WT or WC, the non-temporal store will implement weakly-ordered (WC) semantic behavior.
If the programmer specifies a non-temporal store to cacheable memory, two cases may result:

- If the data is present in the cache hierarchy, the instruction will ensure consistency. A given processor may choose different ways to implement this; some examples include: updating data in-place in the cache hierarchy while preserving the memory type semantics assigned to that region, or evicting the data from the caches and writing the new non-temporal data to memory (with WC semantics).
- If the data is not present in the cache hierarchy, and the destination region is mapped as WB, WT or WC, the transaction will be weakly ordered, and is subject to all WC memory semantics. The non-temporal store will not write allocate. Different implementations may choose to collapse and combine these stores.

In general, WC semantics require software to ensure coherence, with respect to other processors and other system agents (such as graphics cards). Appropriate use of synchronization and a fencing operation (see SFENCE, below) must be performed for producer-consumer usage models. Fencing ensures that all system agents have global visibility of the stored data; for instance, failure to fence may result in a written cache line staying within a processor, and the line would not be visible to other agents. For processors which implement non-temporal stores by updating data in-place that already resides in the cache hierarchy, the destination region should also be mapped as WC. Otherwise if mapped as WB or WT, there is the potential for speculative processor reads to bring the data into the caches; in this case, non-temporal stores would then update in place, and data would not be flushed from the processor by a subsequent fencing operation.

- The memory type visible on the bus in the presence of memory type aliasing is implementation specific. As one possible example, the memory type written to the bus may reflect the memory type for the first store to this line, as seen in program order; other alternatives are possible. This behavior should be considered reserved, and dependency on the behavior of any particular implementation risks future incompatibility.

The PREFETCH (Load 32 or greater number of bytes) instructions load either non-temporal data or temporal data in the specified cache level. This access and the cache level are specified as a hint. The prefetch instructions do not affect functional behavior of the program and will be implementation specific.

The SFENCE (Store Fence) instruction guarantees that every store instruction that precedes the store fence instruction in program order is globally visible before any store instruction which follows the fence. The SFENCE instruction provides an efficient way of ensuring ordering between routines that produce weakly-ordered results and routines that consume this data.

### 3.7 IEEE Compliance

Streaming SIMD Extension floating-point computation is IEEE-754 compliant except when the control word is set to flush to zero mode. IEEE-754 compliance includes support for single-precision signed infinities, QNaNs, SNaNs, integer indefinite, signed zeros, denormals, masked and unmasked exceptions. Single-precision floating-point values are represented identically both internally and in memory, and are of the following form:

<table>
<thead>
<tr>
<th>Sign</th>
<th>Exponent</th>
<th>Significand</th>
</tr>
</thead>
<tbody>
<tr>
<td>31</td>
<td>30...23</td>
<td>22...0</td>
</tr>
</tbody>
</table>
This is a change from x87 floating-point which internally represents all numbers in 80-bit extended format. This change implies that x87-FP libraries re-written to use Streaming SIMD Extension instructions may not produce results that are identical to the those of the x87-FP implementation.

Real Numbers and Floating-point Formats.

This section describes how real numbers are represented in floating-point format in the processor. It also introduces terms such as normalized numbers, denormalized numbers, biased exponents, signed zeros, and NaNs. Readers who are already familiar with floating-point processing techniques and the IEEE standards may wish to skip this section.

### 3.7.1 Real Number System

As shown in Figure 3-8, the real-number system comprises the continuum of real numbers from minus infinity (−∞) to plus infinity (+∞).

**Figure 3-8. Binary Real Number System**

Because the size and number of registers that any computer can have is limited, only a subset of the real-number continuum can be used in real-number calculations. As shown at the bottom of Figure 3-1, the subset of real numbers that a particular processor supports represents an approximation of the real number system. The range and precision of this real-number subset is determined by the format that the processor uses to represent real numbers.
3.7.1.1 **Floating-point Format**

To increase the speed and efficiency of real-number computations, computers typically represent real numbers in a binary floating-point format. In this format, a real number has three parts: a sign, a significand, and an exponent. Figure 3-9 shows the binary floating-point format that Streaming SIMD Extension data uses. This format conforms to the IEEE standard.

The sign is a binary value that indicates whether the number is positive (0) or negative (1). The significand has two parts: a 1-bit binary integer (also referred to as the J-bit) and a binary fraction. The J-bit is often not represented, but instead is an implied value. The exponent is a binary integer that represents the base-2 power that the significand is raised to.

**Figure 3-9. Binary Floating-point Format**

![Binary Floating-point Format Diagram]

<table>
<thead>
<tr>
<th>Notation</th>
<th>Value</th>
</tr>
</thead>
<tbody>
<tr>
<td>Ordinary Decimal</td>
<td>178.125</td>
</tr>
<tr>
<td>Scientific Decimal</td>
<td>1.78125E10</td>
</tr>
<tr>
<td>Scientific Binary</td>
<td>1.0110010001E211</td>
</tr>
<tr>
<td>Scientific Binary (Biased Exponent)</td>
<td>1.0110010001E210000110</td>
</tr>
<tr>
<td>Single Format (Normalized)</td>
<td>Sign</td>
</tr>
<tr>
<td></td>
<td>0</td>
</tr>
</tbody>
</table>

**Table 3-1. Real Number Notation**

3.7.1.2 **Normalized Numbers**

In most cases, the processor represents real numbers in normalized form. This means that except for zero, the significand is always made up of an integer of 1 and the following fraction:

1.fff...ff

For values less than 1, leading zeros are eliminated. (For each leading zero eliminated, the exponent is decremented by one.)
Representing numbers in normalized form maximizes the number of significant digits that can be accommodated in a significand of a given width. To summarize, a normalized real number consists of a normalized significand that represents a real number between 1 and 2 and an exponent that specifies the number’s binary point.

### 3.7.1.3 Biased Exponent

The processor represents exponents in a biased form. This means that a constant is added to the actual exponent so that the biased exponent is always a positive number. The value of the biasing constant depends on the number of bits available for representing exponents in the floating-point format being used. The biasing constant is chosen so that the smallest normalized number can be reciprocated without overflow.

### 3.7.1.4 Real Number and Non-Number Encodings

A variety of real numbers and special values can be encoded in the processor’s floating-point format. These numbers and values are generally divided into the following classes:

- Signed zeros
- Denormalized finite numbers
- Normalized finite numbers
- Signed infinities
- NaNs
- Indefinite numbers

(The term NaN stands for “Not a Number.”)

**Figure 3-10** shows how the encodings for these numbers and non-numbers fit into the real number continuum. The encodings shown here are for the IEEE single-precision (32-bit) format, where the term “S” indicates the sign bit, “E” the biased exponent, and “F” the fraction. (The exponent values are given in decimal.)

The processor can operate on and/or return any of these values, depending on the type of computation being performed. The following sections describe these number and non-number classes.

### 3.7.1.5 Signed Zeros

Zero can be represented as a +0 or a −0 depending on the sign bit. Both encodings are equal in value. The sign of a zero result depends on the operation being performed and the rounding mode being used. Signed zeros have been provided to aid in implementing interval arithmetic. The sign of a zero may indicate the direction from which underflow occurred, or it may indicate the sign of an infinity that has been reciprocated.
### 3.7.1.6 Normalized and Denormalized Finite Numbers

Non-zero, finite numbers are divided into two classes: normalized and denormalized. The normalized finite numbers comprise all the non-zero finite values that can be encoded in a normalized real number format between zero and $\infty$. In the format shown in Figure 3-10, this group of numbers includes all the numbers with biased exponents ranging from 1 to 254 (unbiased, the exponent range is from $-126$ to $+127$).

Figure 3-10. Real Numbers and NaNs

<table>
<thead>
<tr>
<th>NaN</th>
<th>−∞</th>
<th>Normalized Finite</th>
<th>0</th>
<th>+0</th>
<th>+Normalized Finite</th>
<th>+∞</th>
</tr>
</thead>
<tbody>
<tr>
<td>S E F</td>
<td>1 0 0</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td></td>
<td></td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>−Denormalized Finite</th>
<th>+Denormalized Finite</th>
</tr>
</thead>
<tbody>
<tr>
<td>−∞</td>
<td>+∞</td>
</tr>
</tbody>
</table>

<table>
<thead>
<tr>
<th>Real Number and NaN Encodings For 32-bit Floating-point Format</th>
</tr>
</thead>
<tbody>
<tr>
<td>S E F</td>
</tr>
<tr>
<td>1 0 0</td>
</tr>
<tr>
<td>1 0 0.XXX</td>
</tr>
<tr>
<td>1 1...254</td>
</tr>
<tr>
<td>1 255</td>
</tr>
<tr>
<td>X 255</td>
</tr>
<tr>
<td>X 255</td>
</tr>
</tbody>
</table>

Notes
1. Sign bit ignored
2. Fractions must be non-zero

When real numbers become very close to zero, the normalized-number format can no longer be used to represent the numbers. This is because the range of the exponent is not large enough to compensate for shifting the binary point to the right to eliminate leading zeros.

When the biased exponent is zero, smaller numbers can only be represented by making the integer bit (and perhaps other leading bits) of the significand zero. The numbers in this range are called denormalized (or tiny) numbers. The use of leading zeros with denormalized numbers allows smaller numbers to be represented. However, this denormalization causes a loss of precision (the number of significant bits in the fraction is reduced by the leading zeros).

When performing normalized floating-point computations, a processor normally operates on normalized numbers and produces normalized numbers as results. Denormalized numbers represent an underflow condition.

A denormalized number is computed through a technique called gradual underflow. Table 3-2 gives an example of gradual underflow in the denormalization process. Here the single-real format is being used, so the minimum exponent (unbiased) is $-126$. The true result in this example requires an exponent of $-129$ (unbiased) in order to have a normalized number. Since $-129$ is beyond the allowable exponent range, the result is denormalized by inserting leading zeros until the minimum exponent of $-126$ is reached.
In the extreme case, all the significant bits are shifted out to the right by leading zeros, creating a zero result.

The processor deals with denormal values in the following ways:

- It avoids creating denormals by normalizing numbers whenever possible.
- It provides the floating-point underflow exception to permit programmers to detect cases when denormals are created.
- It provides the floating-point denormal-operand exception to permit procedures or programs to detect when denormals are being used as source operands for computations.

### 3.7.1.7 Signed Infinities

The two infinities, \(+\infty\) and \(–\infty\), represent the maximum positive and negative real numbers, respectively, that can be represented in the floating-point format. Infinity is always represented by a zero significand (fraction and integer bit) and the maximum biased exponent allowed in the specified format (for example, 255 for the single-real format).

The signs of infinities are observed, and comparisons are possible. Infinities are always interpreted in the affine sense; that is, \(–\infty\) is less than any finite number and \(+\infty\) is greater than any finite number. Arithmetic on infinities is always exact. Exceptions are generated only when the use of an infinity as a source operand constitutes an invalid operation.

Whereas denormalized numbers represent an underflow condition, the two infinity numbers represent the result of an overflow condition. Here, the normalized result of a computation has a biased exponent greater than the largest allowable exponent for the selected result format.

### 3.7.1.8 NaNs

Since NaNs are non-numbers, they are not part of the real number line. In Figure 3-10, the encoding space for NaNs in the processor floating-point formats is shown above the ends of the real number line. This space includes any value with the maximum allowable biased exponent and a non-zero fraction. (The sign bit is ignored for NaNs.)

The IEEE standard defines two classes of NaN: quiet NaNs (QNaNs) and signaling NaNs (SNaNs). A QNaN is a NaN with the most significant fraction bit set; an SNaN is a NaN with the most significant fraction bit clear. QNaNs are allowed to propagate through most arithmetic operations without signaling an exception. SNaNs generally signal an invalid-operation exception whenever they appear as operands in arithmetic operations. Exceptions, as well as detailed information on how the processor handles NaNs, are discussed in Section 3.7.2.
3.7.1.9 Indefinite

In response to a masked invalid-operation floating-point exceptions, the indefinite value QNAN is produced. The integer indefinite, which can be produced during conversion from single-precision floating-point to 32-bit integer, is defined to be 80000000H.

3.7.2 Operating on NaNs

As was described in Section 3.7.1.8, Streaming SIMD Extension supports two types of NaNs: SNaNs and QNaNs. An SNaN is any NaN value with its most-significant fraction bit set to 0 and at least one other fraction bit set to 1. (If all the fraction bits are set to 0, the value is an $\infty$.) A QNaN is any NaN value with the most-significant fraction bit set to 1. The sign bit of a NaN is not interpreted.

As a general rule, when a QNaN is used in one or more arithmetic floating-point instructions, it is allowed to propagate through a computation. An SNaN on the other hand causes a floating-point invalid-operation exception to be signaled. SNaNs are typically used to trap or invoke an exception handler.

The invalid operation exception has a flag and a mask bit associated with it in MXCSR. The mask bit determines how the an SNaN value is handled. If the invalid operation mask bit is set, the SNaN is converted to a QNaN by setting the most-significant fraction bit of the value to 1. The result is then stored in the destination operand and the invalid operation flag is set. If the invalid operation mask is clear, an invalid operation fault is signaled and no result is stored in the destination operand.

When a real operation or exception delivers a QNaN result, the value of the result depends on the source operands, as shown in Table 3-3. The exceptions to the behavior described in Table 3-3 are the MINPS and MAXPS instructions. If only one source is a NaN for these instructions, the Src2 operand (either NaN or real value) is written to the result; this differs from the behavior for other instructions as defined in Table 3-3, which is to always write the NaN to the result, regardless of which source operand contains the NaN. This approach for MINPS/MAXPS allows NaN data to be screened out of the bounds-checking portion of an algorithm. If instead of this behavior, it is required that the NaN source operand be returned, the min/max functionality can be emulated using a sequence of instructions: comparison followed by AND, ANDN and OR.

In general Src1 and Src2 relate to an Streaming SIMD Extension instruction as follows:

```
ADDPS Src1, Src2/m128
```

Except for the rules given at the beginning of this section for encoding SNaNs and QNaNs, software is free to use the bits in the significand of a NaN for any purpose. Both SNaNs and QNaNs can be encoded to carry and store data, such as diagnostic information.
### 3.8 Data Formats

#### 3.8.1 Memory Data Formats

The Intel Architecture Streaming SIMD Extension introduces a new packed 128-bit data type which consists of 4 single-precision floating-point numbers. The 128 bits are numbered 0 through 127. Bit 0 is the least significant bit (LSB), and bit 127 is the most significant bit (MSB).

Bytes in the new data type format have consecutive memory addresses. The ordering is always little endian, that is, the bytes with the lower addresses are less significant than the bytes with the higher addresses.

#### 3.8.2 Streaming SIMD Extension Register Data Formats

Values in Streaming SIMD Extension registers have the same format as a 128-bit quantity in memory. They have two data access modes: 128-bit access mode and 32-bit access mode. The data type corresponds directly to the single-precision format in the IEEE standard. Table 3-4 gives the precision and range of this data type. Only the fraction part of the significand is encoded. The integer is assumed to be 1 for all numbers except 0 and denormalized finite numbers. The exponent of the single-precision data type is encoded in biased format. The biasing constant is 127 for the single-precision format.

---

<table>
<thead>
<tr>
<th>Source Operands</th>
<th>NaN Result (invalid operation exception is masked)</th>
</tr>
</thead>
<tbody>
<tr>
<td>An SNaN and a QNaN.</td>
<td>Src1 NaN (converted to QNaN if Src1 is an SNaN).</td>
</tr>
<tr>
<td>Two SNaNs.</td>
<td>Src1 NaN (converted to QNaN)</td>
</tr>
<tr>
<td>Two QNaNs.</td>
<td>Src1 QNaN</td>
</tr>
<tr>
<td>An SNaN and a real value.</td>
<td>The SNaN converted into a QNaN.</td>
</tr>
<tr>
<td>A QNaN and a real value.</td>
<td>The QNaN source operand.</td>
</tr>
<tr>
<td>An SNaN/QNaN value (for instructions which take only one operand i.e. RCPPS, RCPSS, RSQRTPS, RSQRTPS)</td>
<td>The SNaN converted into a QNaN/the source QNaN.</td>
</tr>
<tr>
<td>Neither source operand is a NaN and a floating-point invalid-operation exception is signaled.</td>
<td>The default QNaN real indefinite.</td>
</tr>
</tbody>
</table>
Table 3-5 shows the encodings for all the classes of real numbers (that is, zero, denormalized-finite, normalized-finite, and ∞) and NaNs for the single-real data-type. It also gives the format for the real indefinite value, which is a QNaN encoding that is generated by several Streaming SIMD Extension instructions in response to a masked floating-point invalid-operation exception.

Table 3-5. Real Number and NaN Encodings

<table>
<thead>
<tr>
<th>Class</th>
<th>Sign</th>
<th>Biased Exponent</th>
<th>Significand</th>
</tr>
</thead>
<tbody>
<tr>
<td></td>
<td></td>
<td></td>
<td>Integer^1</td>
</tr>
<tr>
<td>Positive</td>
<td>+∞</td>
<td>0</td>
<td>1</td>
</tr>
<tr>
<td>+Normals</td>
<td>0</td>
<td>11..10</td>
<td>1</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>+Denormals</td>
<td>0</td>
<td>00..00</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>+Zero</td>
<td>0</td>
<td>00..00</td>
<td>0</td>
</tr>
<tr>
<td>Negative</td>
<td>–Zero</td>
<td>1</td>
<td>00..00</td>
</tr>
<tr>
<td>–Denormals</td>
<td>1</td>
<td>00..00</td>
<td>0</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>–Normals</td>
<td>1</td>
<td>00..01</td>
<td>1</td>
</tr>
<tr>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>–∞</td>
<td>1</td>
<td>11..11</td>
<td>1</td>
</tr>
<tr>
<td>NaNs</td>
<td>SNaN</td>
<td>X</td>
<td>1</td>
</tr>
<tr>
<td>QNaN</td>
<td>X</td>
<td>11..11</td>
<td>1</td>
</tr>
<tr>
<td>Real Indefinite</td>
<td>QNaN</td>
<td>1</td>
<td>1</td>
</tr>
</tbody>
</table>

When storing real values in memory, single-real values are stored in 4 consecutive bytes in memory. The 128-bit access mode is used for 128-bit memory accesses, 128-bit transfers between Streaming SIMD Extension registers, and all logical, unpack and arithmetic instructions. The 32-bit access mode is used for 32-bit memory access, 32-bit transfers between Streaming SIMD Extension registers, and all arithmetic instructions.

There are sixty-eight new instructions in Streaming SIMD Extension instruction set. This chapter describes the packed and scalar floating-point instructions in alphabetical order, with a full description of each instruction. The last two sections of this chapter describe the SIMD Integer instructions and the cacheability control instructions.
3.9 Instruction Formats

The nature of Streaming SIMD Extension allows the use of existing instruction formats. Instructions use the ModR/M format and are preceded by the 0F prefix byte. In general, operations are not duplicated to provide two directions (i.e. separate load and store variants).

3.10 Instruction Prefixes

The Streaming SIMD Extension instruction uses prefixes as specified in Table 3-6, Table 3-7, and Table 3-8. The effect of multiple prefixes (more than one prefix from a group) is unpredictable and may vary from processor to processor.

Applying a prefix, in a manner not defined in this document, is considered reserved behavior. For example, Table 3-6 shows general behavior for most Streaming SIMD Extension instructions; however, the application of a prefix (Repeat, Repeat NE, Operand Size) is reserved for the following instructions:

ANDPS, ANDNPS, COMISS, FXRSTOR, FXSAVE, ORPS, LDMXCSR, MOVAPS, MOVSXPS, MOVLPS, MOVMSKPS, MOVNTPS, MOVUPS, SHUFPS, STMXCSR, UCOMISS, UNPCKHPS, UNPCKLPS, XORPS.

Table 3-6. Streaming SIMD Extension Instruction Behavior with Prefixes

<table>
<thead>
<tr>
<th>Prefix Type</th>
<th>Effect on Streaming SIMD Extension Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>Address Size Prefix (67H)</td>
<td>Affects Streaming SIMD Extension instructions with memory operand Ignored by Streaming SIMD Extension instructions without memory operand.</td>
</tr>
<tr>
<td>Operand Size (66H)</td>
<td>Reserved and may result in unpredictable behavior.</td>
</tr>
<tr>
<td>Segment Override (2EH,36H,3EH,26H,64H,65H)</td>
<td>Affects Streaming SIMD Extension instructions with mem.operand Ignored by Streaming SIMD Extension instructions without mem operand</td>
</tr>
<tr>
<td>Repeat Prefix (F3H)</td>
<td>Affects Streaming SIMD Extension instructions</td>
</tr>
<tr>
<td>Repeat NE Prefix(F2H)</td>
<td>Reserved and may result in unpredictable behavior.</td>
</tr>
<tr>
<td>Lock Prefix (0F0H)</td>
<td>Generates invalid opcode exception.</td>
</tr>
</tbody>
</table>

Table 3-7. SIMD Integer Instructions – Behavior with Prefixes

<table>
<thead>
<tr>
<th>Prefix Type</th>
<th>Effect on Intel® MMX™ Instructions</th>
</tr>
</thead>
<tbody>
<tr>
<td>Address Size Prefix (67H)</td>
<td>Affects Intel® MMX™ instructions with mem. operand Ignored by Intel® MMX™ instructions without mem. operand.</td>
</tr>
<tr>
<td>Operand Size (66H)</td>
<td>Reserved and may result in unpredictable behavior.</td>
</tr>
<tr>
<td>Segment Override (2EH,36H,3EH,26H,64H,65H)</td>
<td>Affects Intel® MMX™ instructions with mem. operand Ignored by Intel® MMX™ instructions without mem operand</td>
</tr>
<tr>
<td>Repeat Prefix (F3H)</td>
<td>Reserved and may result in unpredictable behavior.</td>
</tr>
<tr>
<td>Repeat NE Prefix(F2H)</td>
<td>Reserved and may result in unpredictable behavior.</td>
</tr>
<tr>
<td>Lock Prefix (0F0H)</td>
<td>Generates invalid opcode exception.</td>
</tr>
</tbody>
</table>
3.11 Reserved Behavior and Software Compatibility

In many register and memory layout descriptions, certain bits are marked as reserved. When bits are marked as reserved, it is essential for compatibility with future processors that software treat these bits as having a future, though unknown, effect. The behavior of reserved bits should be regarded as not only reserved, but unpredictable. In general, reserved behavior may also be applied in other areas. Software should follow these guidelines in dealing with reserved behavior:

- Do not depend on the states of any reserved fields when testing the values of registers which contain such bits. Mask out the reserved fields before testing.
- Do not depend on the states of any reserved fields when storing to memory or to a register.
- Do not depend on the ability to retain information written into any reserved fields.
- When loading a register, always load the reserved fields with the values indicated in the documentation, if any, or reload them with values previously read from the same register.

Note: Avoid any software dependency upon the reserved state/behavior. Depending upon reserved behavior will make the software dependent upon the unspecified manner in which the processor handles this behavior and risks incompatibility with future processors.

3.12 Notations

Besides opcodes, two kinds of notations are found which both describe information found in the ModR/M byte:

1. /digit: (digit between 0 and 7) indicates that the instruction uses only the r/m (register and memory) operand. The reg field contains the digit that provides an extension to the instruction’s opcode.

2. /r: indicates that the ModR/M byte of an instruction contains both a register operand and an r/m operand.

In addition, the following abbreviations are used:

- r32: Intel Architecture 32-bit integer register.
- xmm/m128: Indicates a 128-bit multimedia register or a 128-bit memory location.
- xmm/m64: Indicates a 128-bit multimedia register or a 64-bit memory location.
- **xmm/m32**: Indicates a 128-bit multimedia register or a 32-bit memory location.
- **mm/m64**: Indicates a 64-bit multimedia register or a 64-bit memory location.
- **imm8**: Indicates an immediate 8-bit operand.
- **ib**: Indicates that an immediate byte operand follows the opcode, ModR/M byte or scaled-indexing byte.

When there is ambiguity, xmm1 indicates the first source operand and xmm2 the second source operand.

Table 3-9 describes the naming conventions used in the Streaming SIMD Extension instruction mnemonics.

### Table 3-9. Key to Streaming SIMD Extension Naming Convention

<table>
<thead>
<tr>
<th>Mnemonic</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>PI</td>
<td>Packed integer qword (e.g. mm0)</td>
</tr>
<tr>
<td>PS</td>
<td>Packed single FP (e.g. xmm0)</td>
</tr>
<tr>
<td>SI</td>
<td>Scalar integer (e.g. eax)</td>
</tr>
<tr>
<td>SS</td>
<td>Scalar single-FP (e.g. low 32 bits of xmm0)</td>
</tr>
</tbody>
</table>
ADDPS: Packed Single-FP Add

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F.58,./r</td>
<td>ADDPS xmm1, xmm2/m128</td>
<td>Add packed SP FP numbers from XMM2/Mem to XMM1.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{xmm1}_{[31-0]} & = \text{xmm1}_{[31-0]} + \text{xmm2/m128}_{[31-0]}; \\
\text{xmm1}_{[63-32]} & = \text{xmm1}_{[63-32]} + \text{xmm2/m128}_{[63-32]}; \\
\text{xmm1}_{[95-64]} & = \text{xmm1}_{[95-64]} + \text{xmm2/m128}_{[95-64]}; \\
\text{xmm1}_{[127-96]} & = \text{xmm1}_{[127-96]} + \text{xmm2/m128}_{[127-96]};
\end{align*}
\]

**Description:** The ADDPS instruction adds the packed SP FP numbers of both their operands.

**Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** Overflow, Underflow, Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set. #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(XEDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set. #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0)

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
ADDSS: Scalar Single-FP Add

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3.0F.58, /r</td>
<td>ADDSS xmm1, xmm2/m32</td>
<td>Add the lower SP FP number from XMM2/Mem to XMM1.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
xmm1[31-0] &= xmm1[31-0] + xmm2/m32[31-0]; \\
xmm1[63-32] &= xmm1[63-32]; \\
xmm1[95-64] &= xmm1[95-64]; \\
xmm1[127-96] &= xmm1[127-96];
\end{align*}
\]

**Description:** The ADDSS instruction adds the lower SP FP numbers of both their operands; the upper 3 fields are passed through from xmm1.

**FP Exceptions:** None.

**Numeric Exceptions:** Overflow, Underflow, Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
ANDNPS: Bit-wise Logical And Not for Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,55,/r</td>
<td>ANDNPS xmm1, xmm2/m128</td>
<td>Invert the 128 bits in XMM1 and then AND the result with 128 bits from XMM2/Mem.</td>
</tr>
</tbody>
</table>

**Operation:**
\[
\text{xmm1}[127-0] = \neg(xmm1[127-0]) \& \text{xmm2/m128}[127-0];
\]

**Description:**
The ANDNPS instructions returns a bit-wise logical AND between the complement of XMM1 and XMM2/Mem.

**FP Exceptions:**
General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:**
None

**Protected Mode Exceptions:**

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
#SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**
Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set.

**Virtual 8086 Mode Exceptions:**
Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault

Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments:**
The usage of Repeat Prefixes (F2H, F3H) with ANDNPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with ANDNPS risks incompatibility with future processors.
ANDPS: Bit-wise Logical And for Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,54,/r</td>
<td>ANDPS xmm1, xmm2/m128</td>
<td>Logical AND of 128 bits from XMM2/Mem to XMM1 register.</td>
</tr>
</tbody>
</table>

**Operation:**

```
xmm1[127-0] &= xmm2/m128[127-0];
```

**Description:** The ANDPS instruction returns a bit-wise logical AND between XMM1 and XMM2/Mem.

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments:** The usage of Repeat Prefixes (F2H, F3H) with ANDPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with ANDPS risks incompatibility with future processors.
CMPPS: Packed Single-FP Compare

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,C2,/r,ib</td>
<td>CMPPS xmm1, xmm2/m128, imm8</td>
<td>Compare packed SP FP numbers from XMM2/Mem to packed SP FP numbers in XMM1 register using imm8 as predicate.</td>
</tr>
</tbody>
</table>

**Operation:**

```
switch (imm8) {
    case eq:   op = eq;
    case lt:   op = lt;
    case le:   op = le;
    case unord: op = unord;
    case neq:  op = neq;
    case nlt:  op = nlt;
    case nle:  op = nle;
    case ord:  op = ord;
    default:   Reserved;
}
```

```
cmp0 = op(xmm1[31-0],xmm2/m128[31-0]);
cmp1 = op(xmm1[63-32],xmm2/m128[63-32]);
cmp2 = op(xmm1[95-64],xmm2/m128[95-64]);
cmp3 = op(xmm1[127-96],xmm2/m128[127-96]);
```

```
xmm1[31-0]   = (cmp0) ? 0xffffffff : 0x00000000;
xmm1[63-32]  = (cmp1) ? 0xffffffff : 0x00000000;
xmm1[95-64]  = (cmp2) ? 0xffffffff : 0x00000000;
xmm1[127-96] = (cmp3) ? 0xffffffff : 0x00000000;
```

**Description:** For each individual pairs of SP FP numbers, the CMPPS instruction returns an all “1” 32-bit mask or an all “0” 32-bit mask, using the comparison predicate specified by imm8; note that a subsequent computational instruction which uses this mask as an input operand will not generate a fault, since a mask of all “0”s corresponds to a FP value of +0.0 and a mask of all “1”s corresponds to a FP value of -qNaN. Some of the comparisons can be achieved only through software emulation. For these comparisons the programmer must swap the operands, copying registers when necessary to protect the data that will now be in the destination, and then perform the compare using a different predicate. The predicate to be used for these emulations is listed in under the heading “Emulation”. The following table shows the different comparison types:
FP Exceptions: General protection exception if not aligned on 16-byte boundary, regardless of segment.

Numeric Exceptions: Invalid if sNaN operand, invalid if qNaN and predicate as listed in above table, denormal.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set.
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);
- #UD if CPUID.XMM(EDX bit 25) = 0.

Real Address Mode Exceptions:

-Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
-#UD if CR0.EM = 1;
-#NM if TS bit in CR0 is set;
-#XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);
-#UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);
-#UD if CR4.OSXMMEXCPT(bit 9) = 0;
-#UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
Comments: Compilers and assemblers should implement the following 2-operand pseudo-ops in addition to the 3-operand CMPPS instruction:

<table>
<thead>
<tr>
<th>Pseudo-Op</th>
<th>Implementation</th>
</tr>
</thead>
<tbody>
<tr>
<td>CMPEQPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 0</td>
</tr>
<tr>
<td>CMPLTPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 1</td>
</tr>
<tr>
<td>CMPLEPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 2</td>
</tr>
<tr>
<td>CMPUNORDPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 3</td>
</tr>
<tr>
<td>CMPNEQPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 4</td>
</tr>
<tr>
<td>CMPNLTPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 5</td>
</tr>
<tr>
<td>CMPNLEPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 6</td>
</tr>
<tr>
<td>CMPORDPS xmm1, xmm2</td>
<td>CMPPS xmm1,xmm2, 7</td>
</tr>
</tbody>
</table>

The greater-than relations not implemented in hardware require more than one instruction to emulate in software and therefore should not be implemented as pseudo-ops. (For these, the programmer should reverse the operands of the corresponding less than relations and use move instructions to ensure that the mask is moved to the correct destination register and that the source operand is left intact.)

Bits 7-4 of the immediate field are reserved. Different processors may handle them differently. Usage of these bits risks incompatibility with future processors.
CMPSS: Scalar Single-FP Compare

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3.0F.C2./r,ib</td>
<td>CMPSS xmm1, xmm2/m32, imm8</td>
<td>Compare lowest SP FP number from XMM2/Mem to lowest SP FP number in XMM1 register using imm8 as predicate.</td>
</tr>
</tbody>
</table>

Operation:

```c
switch (imm8) {
    case eq:      op = eq;
    case lt:      op = lt;
    case le:      op = le;
    case unord:   op = unord;
    case neq:     op = neq;
    case nlt:     op = nlt;
    case nle:     op = nle;
    case ord:     op = ord;
    default:      Reserved;
}
```

```c
cmp0 = op(xmm1[31-0],xmm2/m32[31-0]);
```

```c
xmm1[31-0] = (cmp0) ? 0xffffffff : 0x00000000;
```

```c
xmm1[63-32] = xmm1[63-32];
```

```c
xmm1[95-64] = xmm1[95-64];
```

```c
xmm1[127-96] = xmm1[127-96];
```

Description: For the lowest pair of SP FP numbers, the CMPSS instruction returns an all “1” 32-bit mask or an all “0” 32-bit mask, using the comparison predicate specified by imm8; the values for the upper three pairs of SP FP numbers are not compared. Note that a subsequent computational instruction which uses this mask as an input operand will not generate a fault, since a mask of all “0”’s corresponds to a FP value of +0.0 and a mask of all “1”’s corresponds to a FP value of -qNaN. Some of the comparisons can be achieved only through software emulation. For these comparisons the programmer must swap the operands, copying registers when necessary to protect the data that will now be in the destination, and then perform the compare using a different predicate. The predicate to be used for these emulations is listed in under the heading “Emulation”. The following table shows the different comparison types:
FP Exceptions: None.

Numeric Exceptions: Invalid if sNaN operand, invalid if qNaN and predicate as listed in above table, denormal.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true (CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD if an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Real Address Mode Exceptions:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
Comments: Compilers and assemblers should implement the following 2-operand pseudo-ops in addition to the 3-operand CMPSS instruction:

<table>
<thead>
<tr>
<th>Pseudo-Op</th>
<th>Implementation</th>
</tr>
</thead>
<tbody>
<tr>
<td>CMPEQSS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 0</td>
</tr>
<tr>
<td>CMPLTSS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 1</td>
</tr>
<tr>
<td>CMPLESS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 2</td>
</tr>
<tr>
<td>CMPUNORDSS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 3</td>
</tr>
<tr>
<td>CMPNEQSS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 4</td>
</tr>
<tr>
<td>CMPLTSS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 5</td>
</tr>
<tr>
<td>CMPNLESS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 6</td>
</tr>
<tr>
<td>CMPORDSS xmm1, xmm2</td>
<td>CMPSS xmm1,xmm2, 7</td>
</tr>
</tbody>
</table>

The greater-than relations not implemented in hardware require more than one instruction to emulate in software and therefore should not be implemented as pseudo-ops. (For these, the programmer should reverse the operands of the corresponding less than relations and use move instructions to ensure that the mask is moved to the correct destination register and that the source operand is left intact.)

Bits 7-4 of the immediate field are reserved. Different processors may handle them differently. Usage of these bits risks incompatibility with future processors.
COMISS: Scalar Ordered Single-FP Compare and set EFLAGS

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,2F,/r</td>
<td>COMISS xmm1, xmm2/m32</td>
<td>Compare lower SP FP number in XMM1 register with lower SP FP number in XMM2/Mem and set the status flags accordingly</td>
</tr>
</tbody>
</table>

Operation:

```c
switch (xmm1[31-0] <> xmm2/m32[31-0]) {

    OF, SF, AF = 000;

    case UNORDERED:     ZF, PF, CF = 111;
    case GREATER_THAN:  ZF, PF, CF = 000;
    case LESS_THAN:     ZF, PF, CF = 001;
    case EQUAL:         ZF, PF, CF = 100;

}
```

Description: The COMISS instructions compare two SP FP numbers and sets the ZF,PF,CF bits in the EFLAGS register as described above. Although the data type is packed single-FP, only the lower SP numbers are compared. In addition, the OF, SF and AF bits in the EFLAGS register are zeroed out. The unordered predicate is returned if either source operand is a NaN (qNaN or sNaN).

FP Exceptions: None.

Numeric Exceptions: Invalid (if SNaN or QNaN operands), Denormal. Integer EFLAGS values will not be updated in the presence of unmasked numeric exceptions.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.
COMISS: Scalar Ordered Single-FP Compare and set EFLAGS (continued)

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments: COMISS differs from UCOMISS in that it signals an invalid numeric exception when a source operand is either a qNaN or sNaN; UCOMISS signals invalid only if a source operand is an sNaN. The usage of Repeat (F2H, F3H) and Operand-Size (66H) prefixes with COMISS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with COMISS risks incompatibility with future processors.
### CVTPI2PS: Packed Signed INT32 to Packed Single-FP Conversion

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,2A/r</td>
<td>CVTPI2PS xmm, mm/m64</td>
<td>Convert two 32-bit signed integers from MM/Mem to two SP FP.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{ xmm}[31-0] & = (\text{float}) (\text{mm/m64}[31-0]); \\
\text{ xmm}[63-32] & = (\text{float}) (\text{mm/m64}[63-32]); \\
\text{ xmm}[95-64] & = \text{ xmm}[95-64]; \\
\text{ xmm}[127-96] & = \text{ xmm}[127-96];
\end{align*}
\]

**Description:** The CVTPI2PS instruction converts signed 32-bit integers to SP FP numbers; when the conversion is inexact, rounding is done according to MXCSR.

**FP Exceptions:** None.

**Numeric Exceptions:** Precision.

**Protected Mode Exceptions:**

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments; 
#SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception; #AC for unaligned memory reference; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFFF; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception; #AC for unaligned memory reference; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
Comments: This instruction behaves identically to original MMX instructions, in the presence of x87-FP instructions:

- Transition from x87-FP to MMX technology (TOS=0, FP valid bits set to all valid).
- MMX instructions write ones (1's) to the exponent part of the corresponding x87-FP register.

However, the use of a memory source operand with this instruction will not result in the above transition from x87-FP to MMX technology.

Prioritization for fault and assist behavior for CVTPI2PS is as follows:

Memory source
1. Invalid opcode (CR0.EM=1)
2. DNA (CR0.TS=1)
3. #SS or #GP, for limit violation
4. #PF, page fault
5. Streaming SIMD Extension numeric fault (i.e. precision)

Register source
1. Invalid opcode (CR0.EM=1)
2. DNA (CR0.TS=1)
3. #MF, pending x87-FP fault signalled
4. After returning from #MF, x87-FP->MMX technology transition
5. Streaming SIMD Extension numeric fault (i.e. precision)
CVTPS2PI: Packed Single-FP to Packed INT32 Conversion

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,2D,./r</td>
<td>CVTPS2PI mm, xmm/m64</td>
<td>Convert lower 2 SP FP from XMM/Mem to 2 32-bit signed integers in MM using rounding specified by MXCSR.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{mm}[31-0] & = (\text{int}) (\text{xmm/m64}[31-0]); \\
\text{mm}[63-32] & = (\text{int}) (\text{xmm/m64}[63-32]);
\end{align*}
\]

**Description:**

The CVTPS2PI instruction converts the lower 2 SP FP numbers in xmm/m64 to signed 32-bit integers in mm; when the conversion is inexact, the value rounded according to the MXCSR is returned. If the converted result(s) is/are larger than the maximum signed 32 bit value, the Integer Indefinite value (0x80000000) will be returned.

**FP Exceptions:** None.

**Numeric Exceptions:** Invalid, Precision.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EF = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception;
- #AC for unaligned memory reference.

To enable #AC exceptions, three conditions must be true: CR0.AM is set; EFLAGS.AC is set; current CPL is 3.

- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EF = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception;

#XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);

#UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);

#UD if CRCR4.OSFXSR(bit 9) = 0;

#UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments:**

This instruction behaves identically to original MMX instructions, in the presence of x87-FP instructions, including:
CVTPS2PI: Packed Single-FP to Packed INT32 Conversion (continued)

- Transition from x87-FP to MMX technology (TOS=0, FP valid bits set to all valid).
- MMX instructions write ones (1’s) to the exponent part of the corresponding x87-FP register.

Prioritization for fault and assist behavior for CVTPS2PI is as follows:

Memory source
1. Invalid opcode (CR0.EM=1)
2. DNA (CR0.TS=1)
3. #MF, pending x87-FP fault signalled
4. After returning from #MF, x87-FP->MMX technology transition
5. #SS or #GP, for limit violation
6. #PF, page fault
7. Streaming SIMD Extension numeric fault (i.e. invalid, precision)

Register source
1. Invalid opcode (CR0.EM=1)
2. DNA (CR0.TS=1)
3. #MF, pending x87-FP fault signalled
4. After returning from #MF, x87-FP->MMX technology transition
5. Streaming SIMD Extension numeric fault (i.e. precision)
CVTSI2SS: Scalar signed INT32 to Single-FP Conversion

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3,0F,2A,/r</td>
<td>CVTSI2SS xmm, r/m32</td>
<td>Convert one 32-bit signed integer from Integer Reg/Mem to one SP FP.</td>
</tr>
</tbody>
</table>

**Operation:**

- $\text{xmm}[31-0] = (\text{float})(r/m32)$;
- $\text{xmm}[63-32] = \text{xmm}[63-32]$;
- $\text{xmm}[95-64] = \text{xmm}[95-64]$;
- $\text{xmm}[127-96] = \text{xmm}[127-96]$;

**Description:** The CVTSI2SS instruction converts a signed 32-bit integer from memory or from a 32-bit integer register to a SP FP number; when the conversion is inexact, rounding is done according to the MXCSR.

**FP Exceptions:** None.

**Numeric Exceptions:** Precision.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults  VHTP Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
CVTSS2SI: Scalar Single-FP to Signed INT32 Conversion

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3.0F.2D,/r</td>
<td>CVTSS2SI r32, xmm/m32</td>
<td>Convert one SP FP from XMM/Mem to one 32 bit signed integer using rounding mode specified by MXCSR, and move the result to an integer register.</td>
</tr>
</tbody>
</table>

**Operation:**
\[ r32 = \text{(int)} (\text{xmm/m32}[31-0]) \]

**Description:**
The CVTSS2SI instruction converts a SP FP number to a signed 32-bit integer and returns it in the 32-bit integer register; when the conversion is inexact, the rounded value according to the MXCSR is returned. If the converted result is larger than the maximum signed 32 bit integer, the Integer Indefinite value (0x80000000) will be returned.

**FP Exceptions:** None.

**Numeric Exceptions:** Invalid, Precision.

**Protected Mode Exceptions:**
- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true: CR0.AM is set; EFLAGS.AC is set; current CPL is 3;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**
- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**
- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**
- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
CVTTPS2PI: Packed Single-FP to Packed INT32 Conversion (truncate)

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,2C,/r</td>
<td>CVTTPS2PI mm, xmm/m64</td>
<td>Convert lower 2 SP FP from XMM/Mem to 2 32-bit signed integers in MM using truncate.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
mm[31-0] & = (\text{int}) (xmm/m64[31-0]); \\
mm[63-32] & = (\text{int}) (xmm/m64[63-32]);
\end{align*}
\]

**Description:** The CVTTPS2PI instruction converts the lower 2 SP FP numbers in xmm/m64 to 2 32-bit signed integers in mm; if the conversion is inexact, the truncated result is returned. If the converted result(s) is/are larger than the maximum signed 32 bit value, the Integer Indefinite value (0x80000000) will be returned.

**FP Exceptions:** None.

**Numeric Exceptions:** Invalid, Precision.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true: CR0.AM is set; EFLAGS.AC is set; current CPL is 3;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception;

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
CVTTPS2PI: Packed Single-FP to Packed INT32 Conversion (truncate) (continued)

Comments: This instruction behaves identically to original MMX instructions, in the presence of x87-FP instructions, including:

- Transition from x87-FP to MMX technology (TOS=0, FP valid bits set to all valid).
- MMX instructions write ones (1’s) to the exponent part of the corresponding x87-FP register.

Prioritization for fault and assist behavior for CVTTPS2PI is as follows:

Memory source
1. Invalid opcode (CR0.EM=1)
2. DNA (CR0.TS=1)
3. #MF, pending x87-FP fault signalled
4. After returning from #MF, x87-FP->MMX technology transition
5. #SS or #GP, for limit violation
6. #PF, page fault
7. Streaming SIMD Extension numeric fault (i.e. invalid, precision)

Register source
1. Invalid opcode (CR0.EM=1)
2. DNA (CR0.TS=1)
3. #MF, pending x87-FP fault signalled
4. After returning from #MF, x87-FP->MMX technology transition
5. Streaming SIMD Extension numeric fault (i.e. precision)
CVTTSS2SI: Scalar Single-FP to signed INT32 Conversion (truncate)

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3,0F,2C,/r</td>
<td>CVTTSS2SI r32, xmm/m32</td>
<td>Convert lowest SP FP from XMM/Mem to one 32 bit signed integer using truncate, and move the result to an integer register.</td>
</tr>
</tbody>
</table>

**Operation:**
\[ r32 = \text{(int)}(\text{xmm/m32}[31-0]) \];

**Description:**
The CVTTSS2SI instruction converts a SP FP number to a signed 32-bit integer and returns it in the 32-bit integer register; if the conversion is inexact, the truncated result is returned. If the converted result is larger than the maximum signed 32 bit value, the Integer Indefinite value (0x80000000) will be returned.

**FP Exceptions:** None.

**Numeric Exceptions:** Invalid, Precision.

**Protected Mode Exceptions:**
- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**
Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**
Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**
- Itanium Reg Faults
- Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults
- VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
**DIVPS: Packed Single-FP Divide**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,5E,/r</td>
<td>DIVPS xmm1, xmm2/m128</td>
<td>Divide packed SP FP numbers in XMM1 by XMM2/Mem</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{xmm1}[31-0] & = \text{xmm1}[31-0] / (\text{xmm2/m128}[31-0]); \\
\text{xmm1}[63-32] & = \text{xmm1}[63-32] / (\text{xmm2/m128}[63-32]); \\
\text{xmm1}[95-64] & = \text{xmm1}[95-64] / (\text{xmm2/m128}[95-64]); \\
\text{xmm1}[127-96] & = \text{xmm1}[127-96] / (\text{xmm2/m128}[127-96]);
\end{align*}
\]

**Description:** The DIVPS instruction divides the packed SP FP numbers of both their operands.

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** Overflow, Underflow, Invalid, Divide by Zero, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.df is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
DIVSS: Scalar Single-FP Divide

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3,0F,5E,/r</td>
<td>DIVSS xmm1, xmm2/m32</td>
<td>Divide lower SP FP numbers in XMM1 by XMM2/Mem</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{xmm1}[31-0] &= \text{xmm1}[31-0] / (\text{xmm2}/m32[31-0]); \\
\text{xmm1}[63-32] &= \text{xmm1}[63-32]; \\
\text{xmm1}[95-64] &= \text{xmm1}[95-64]; \\
\text{xmm1}[127-96] &= \text{xmm1}[127-96];
\end{align*}
\]

**Description:** The DIVSS instructions divide the lowest SP FP numbers of both operands; the upper 3 fields are passed through from xmm1.

**FP Exceptions:** None.

**Numeric Exceptions:** Overflow, Underflow, Invalid, Divide by Zero, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #XM for an unmasked Streaming SIMD Extension numeric exception.

To enable #AC exceptions, three conditions must be true:
- CR0.AM is set;
- EFLAGS.AC is set;
- current CPL is 3;
- #XM for an unmasked Streaming SIMD Extension numeric exception.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EV = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode;
- #AC for unaligned memory reference if the current privilege level is 3;
- #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
FXRSTOR: Restore FP and Intel® MMX™ State and Streaming SIMD Extension State

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,AE,/1</td>
<td>FXRSTOR m512byte</td>
<td>Load FP/Intel® MMX™ and Streaming SIMD Extension state from m512byte.</td>
</tr>
</tbody>
</table>

**Operation:** FP and MMX state and Streaming SIMD Extension state = m512byte;

**Description:** The FXRSTOR instruction reloads the FP and MMX technology state and Streaming SIMD Extension state (environment and registers) from the memory area defined by m512byte. This data should have been written by a previous FXSAVE.

The FP and MMX technology and Streaming SIMD Extension environment and registers consist of the following data structure (little-endian byte order as arranged in memory, with byte offset into row described by right column):

<table>
<thead>
<tr>
<th>15</th>
<th>14</th>
<th>13</th>
<th>12</th>
<th>11</th>
<th>10</th>
<th>9</th>
<th>8</th>
<th>7</th>
<th>6</th>
<th>5</th>
<th>4</th>
<th>3</th>
<th>2</th>
<th>1</th>
<th>0</th>
</tr>
</thead>
<tbody>
<tr>
<td>Resvd</td>
<td>CS</td>
<td>IP</td>
<td>FOP</td>
<td>FTW</td>
<td>FSW</td>
<td>FCW</td>
<td>0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>MXCSR</td>
<td>Resvd</td>
<td>DS</td>
<td>DP</td>
<td>16</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST0/MM0</td>
<td></td>
<td></td>
<td></td>
<td>32</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST1/MM1</td>
<td></td>
<td></td>
<td></td>
<td>48</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST2/MM2</td>
<td></td>
<td></td>
<td></td>
<td>64</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST3/MM3</td>
<td></td>
<td></td>
<td></td>
<td>80</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST4/MM4</td>
<td></td>
<td></td>
<td></td>
<td>96</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST5/MM5</td>
<td></td>
<td></td>
<td></td>
<td>112</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST6/MM6</td>
<td></td>
<td></td>
<td></td>
<td>128</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td>ST7/MM7</td>
<td></td>
<td></td>
<td></td>
<td>144</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM0</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>160</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM1</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>176</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM2</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>192</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM3</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>208</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM4</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>224</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM5</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>240</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM6</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>256</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>XMM7</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>272</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>288</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>304</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>320</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>336</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>352</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>368</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>384</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>400</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>416</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>432</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
<tr>
<td>Reserved</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td>448</td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</tbody>
</table>
Three fields in the floating-point save area contain reserved bits that are not indicated in the table:

- **FOP:** The lower 11-bits contain the opcode, upper 5-bits are reserved.
- **IP & DP:** 32-bit mode: 32-bit IP-offset. 16-bit mode: lower 16-bits are IP-offset and upper 16-bits are reserved.

If the MXCSR state contains an unmasked exception with corresponding status flag also set, loading it will not result in a floating-point error condition being asserted; only the next occurrence of this unmasked exception will result in the error condition being asserted.

Some bits of MXCSR (bits 31-16 and bit 6) are defined as reserved and cleared; attempting to write a non-zero value to these bits will result in a general protection exception.

FXRSTOR does not flush pending x87-FP exceptions, unlike FRSTOR. To check and raise exceptions when loading a new operating environment, use FWAIT after FXRSTOR.

The Streaming SIMD Extension fields in the save image (XMM0-XMM7 and MXCSR) may not be loaded into the processor if the CR4.OSFXSR bit is not set. This CR4 bit must be set in order to enable execution of Streaming SIMD Extension instructions.

**FP Exceptions:** If #AC exception detection is disabled, a general protection exception is signalled if the address is not aligned on 16-byte boundary. Note that if #AC is enabled (and CPL is 3), signalling of #AC is not guaranteed and may vary with implementation; in all implementations where #AC is not signalled, a general protection fault will instead be signalled. In addition, the width of the alignment check when #AC is enabled may also vary with implementation; for instance, for a given implementation #AC might be signalled for a 2-byte misalignment, whereas #GP might be signalled for all other misalignments (4/8/16-byte). Invalid opcode exception if instruction is preceded by a LOCK override prefix. General protection fault if reserved bits of MXCSR are loaded with non-zero values.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments; #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #NM if CR0.EM = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #NM if CR0.EM = 1; #NM if TS bit in CR0 is set.
FXRSTOR: Restore FP and Intel® MMX™ State and Streaming SIMD Extension State (continued)

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults  VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Notes:

- State saved with FXSAVE and restored with FRSTOR (and vice versa) will result in incorrect restoration of state in the processor. The address size prefix will have the usual effect on address calculation but will have no effect on the format of the FXRSTOR image.

- The use of Repeat (F2H, F3H) and Operand Size (66H) prefixes with FXRSTOR is reserved. Different processor implementations may handle this prefix differently. Use of this prefix with FXRSTOR risks incompatibility with future processors.
**FXSAVE: Store FP and Intel® MMX™ State and Streaming SIMD Extension State**

**Opcode | Instruction | Description**
---|---|---
0F,AE,/0 | FXSAVE m512byte | Store FP and Intel® MMX™ state and Streaming SIMD Extension state to m512byte.

**Operation:**
m512byte = FP and MMX state and Streaming SIMD Extension state;

**Description:**
The FXSAVE instruction writes the current FP and MMX technology state and Streaming SIMD Extension state (environment and registers) to the specified destination defined by m512byte. It does this without checking for pending unmasked floating-point exceptions, similar to the operation of FNSAVE. Unlike the FSAVE/FNSAVE instructions, the processor retains the contents of the FP and MMX technology state and Streaming SIMD Extension state in the processor after the state has been saved. This instruction has been optimized to maximize floating-point save performance. The save data structure is as follows (little-endian byte order as arranged in memory, with byte offset into row described by right column):

| 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
Rsrvd | CS | IP | FOP | FTW | FSW | FCW | 0 |
Reserved | MXCSR | Rsrvd | DS | DP | 16 |
Reserved | ST0/MM0 | 32 |
Reserved | ST1/MM1 | 48 |
Reserved | ST2/MM2 | 64 |
Reserved | ST3/MM3 | 80 |
Reserved | ST4/MM4 | 96 |
Reserved | ST5/MM5 | 112 |
Reserved | ST6/MM6 | 128 |
Reserved | ST7/MM7 | 144 |
XMM0 | | 160 |
XMM1 | | 176 |
XMM2 | | 192 |
XMM3 | | 208 |
XMM4 | | 224 |
XMM5 | | 240 |
XMM6 | | 256 |
XMM7 | | 272 |
Reserved | | 288 |
Reserved | | 304 |
Reserved | | 320 |
Reserved | | 336 |
Reserved | | 352 |
Reserved | | 368 |
Reserved | | 384 |
Three fields in the floating-point save area contain reserved bits that are not indicated in the table:

- FOP: The lower 11-bits contain the opcode, upper 5-bits are reserved.
- 16-bit mode: lower 16-bits are IP-offset and upper 16-bits are reserved.

The FXSAVE instruction is used when an operating system needs to perform a context switch or when an exception handler needs to use the FP and MMX technology and Streaming SIMD Extension units. It cannot be used by an application program to pass a "clean" FP state to a procedure, since it retains the current state. An application must explicitly execute an FINIT instruction after FXSAVE to provide for this functionality.

All of the x87-FP fields retain the same internal format as in FSAVE except for FTW.

Unlike FSAVE, FXSAVE saves only the FTW valid bits rather than the entire x87-FP FTW field. The FTW bits are saved in a non-TOS relative order, which means that FR0 is always saved first, followed by FR1, FR2 and so forth. As an example, if TOS=4 and only ST0, ST1 and ST2 are valid, FSAVE saves the FTW field in the following format:

<table>
<thead>
<tr>
<th>ST3</th>
<th>ST2</th>
<th>ST1</th>
<th>ST0</th>
<th>ST7</th>
<th>ST6</th>
<th>ST5</th>
<th>ST4 (TOS=4)</th>
</tr>
</thead>
<tbody>
<tr>
<td>FR7</td>
<td>FR6</td>
<td>FR5</td>
<td>FR4</td>
<td>FR3</td>
<td>FR2</td>
<td>FR1</td>
<td>FR0</td>
</tr>
<tr>
<td>11</td>
<td>xx</td>
<td>xx</td>
<td>xx</td>
<td>11</td>
<td>11</td>
<td>11</td>
<td>11</td>
</tr>
</tbody>
</table>

where xx is one of (00, 01, 10). (11) indicates an empty stack elements, and the 00, 01, and 10 indicate Valid, Zero, and Special, respectively. In this example, FXSAVE would save the following vector:

<table>
<thead>
<tr>
<th>FR7</th>
<th>FR6</th>
<th>FR5</th>
<th>FR4</th>
<th>FR3</th>
<th>FR2</th>
<th>FR1</th>
<th>FR0</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>1</td>
<td>1</td>
<td>1</td>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0</td>
</tr>
</tbody>
</table>

The FSAVE format for FTW can be recreated from the FTW valid bits and the stored 80-bit FP data (assuming the stored data was not the contents of MMX technology registers) using the following table:

<table>
<thead>
<tr>
<th>Exponent all 1’s</th>
<th>Exponent all 0’s</th>
<th>Fraction all 0’s</th>
<th>J and M bits</th>
<th>FTW valid bit</th>
<th>x87 FTW</th>
</tr>
</thead>
<tbody>
<tr>
<td>0</td>
<td>0</td>
<td>0</td>
<td>0x</td>
<td>1</td>
<td>Special</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>0</td>
<td>1x</td>
<td>1</td>
<td>Valid</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>00</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>0</td>
<td>1</td>
<td>0</td>
<td>10</td>
<td>1</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>0</td>
<td>0x</td>
<td>1</td>
<td>Special</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>0</td>
<td>1x</td>
<td>1</td>
<td>Special</td>
</tr>
<tr>
<td>0</td>
<td>1</td>
<td>1</td>
<td>00</td>
<td>1</td>
<td>Zero</td>
</tr>
</tbody>
</table>
The J-bit is defined to be the 1-bit binary integer to the left of the decimal place in the significand. The M-bit is defined to be the most significant bit of the fractional portion of the significand (i.e. the bit immediately to the right of the decimal place).

When the M-bit is the most significant bit of the fractional portion of the significand, it must be 0 if the fraction is all 0’s.

If the FXSAVE instruction is immediately preceded by an FP instruction which does not use a memory operand, then the FXSAVE instruction does not write/update the DP field, in the FXSAVE image.

MXCSR holds the contents of the Streaming SIMD Extension Control/Status Register. See the LDMXCSR instruction for a full description of this field.

The fields XMM0-XMM7 contain the content of registers XMM0-XMM7 in exactly the same format as they exist in the registers.

The Streaming SIMD Extension fields in the save image (XMM0-XMM7 and MXCSR) may not be loaded into the processor if the CR4.OSFXSR bit is not set. This CR4 bit must be set in order to enable execution of Streaming SIMD Extension instructions.

The destination m512byte is assumed to be aligned on a 16-byte boundary. If m512byte is not aligned on a 16-byte boundary, FXSAVE generates a general protection exception.

**FP Exceptions:** If #AC exception detection is disabled, a general protection exception is signalled if the address is not aligned on 16-byte boundary. Note that if #AC is enabled (and CPL is 3), signalling of #AC is not guaranteed and may vary with implementation; in all implementations where #AC is not signalled, a general protection fault will instead be signalled. In addition, the width of the alignment check when #AC is enabled may also vary with implementation; for instance, for a given implementation #AC might be signalled for a 2-byte misalignment, whereas #GP might be signalled for all other misalignments (4/8/16-byte). Invalid opcode exception if instruction is preceded by a LOCK override prefix.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #NM if CR0.EA = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).
FXSAVE: Store FP and Intel® MMX™ State and Streaming SIMD Extension State
(continued)

Real Address Mode Exceptions:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #NM if CR0.EM = 1; #NM if TS bit in CR0 is set.

Virtual 8086 ModeExceptions:

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

Additional Itanium System Environment Exceptions

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data Not Present Fault, Data page consumption abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Notes:

State saved with FXSAVE and restored with FRSTOR (and vice versa) will result in incorrect restoration of state in the processor. The address size prefix will have the usual effect on address calculation but will have no effect on the format of the FXSAVE image.

If there is a pending unmasked FP exception at the time FXSAVE is executed, the sequence of FXSAVE-FWAIT-FXRSTOR will result in incorrect state in the processor. The FWAIT instruction causes the processor to check and handle pending unmasked FP exceptions. Since the processor does not clear the FP state with FXSAVE (unlike FSAVE), the exception is handled but that fact is not reflected in the saved image. When the image is reloaded using FXRSTOR, the exception bits in FSW will be incorrectly reloaded.

The use of Repeat (F2H, F3H) and Operand Size (66H) prefixes with FXSAVE is reserved. Different processor implementations may handle this prefix differently. Use of these prefixes with FXSAVE risks incompatibility with future processors.
LDMXCSR: Load Streaming SIMD Extension Control/Status

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,AE,/2</td>
<td>LDMXCSR m32</td>
<td>Load Streaming SIMD Extension control/status word from m32.</td>
</tr>
</tbody>
</table>

Operation:  

MXCSR = m32;

Description:  
The MXCSR control/status register is used to enable masked/unmasked exception handling, to set rounding modes, to set flush-to-zero mode, and to view exception status flags. The following figure shows the format and encoding of the fields in MXCSR.

```
   31-16  15  10  5  0
   Reserved  FZ  RC  RC  PM  UM  OM  ZM  DM  IM  Revd  PE  UE  OE  ZE  DE  IE
```

Bits 5-0 indicate whether a Streaming SIMD Extension numerical exception has been detected. They are “sticky” flags, and can be cleared by using the LDMXCSR instruction to write zeroes to these fields. If a LDMXCSR instruction clears a mask bit and sets the corresponding exception flag bit, an exception will not be immediately generated. The exception will occur only upon the next Streaming SIMD Extension to cause this type of exception. Streaming SIMD Extension uses only one exception flag for each exception. There is no provision for individual exception reporting within a packed data type. In situations where multiple identical exceptions occur within the same instruction, the associated exception flag is updated and indicates that at least one of these conditions happened. These flags are cleared upon reset.

Bits 12-7 configure numerical exception masking; an exception type is masked if the corresponding bit is set and it is unmasked if the bit is clear. These enables are set upon reset, meaning that all numerical exceptions are masked.

Bits 14-13 encode the rounding-control, which provides for the common round-to-nearest mode, as well as directed rounding and true chop. Rounding control affects the arithmetic instructions and certain conversion instructions. The encoding for RC is as follows:

```
<table>
<thead>
<tr>
<th>Rounding Mode</th>
<th>RC Field</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>Round to nearest (even)</td>
<td>00B</td>
<td>Rounded result is the closest to the infinitely precise result. If two values are equally close, the result is the even value (that is, the one with the least-significant bit of zero).</td>
</tr>
<tr>
<td>Round down (to minus infinity)</td>
<td>01B</td>
<td>Rounded result is close to but no greater than the infinitely precise result.</td>
</tr>
<tr>
<td>Round up (toward positive infinity)</td>
<td>10B</td>
<td>Rounded result is close to but no less than the infinitely precise result.</td>
</tr>
<tr>
<td>Round toward zero (truncate)</td>
<td>11B</td>
<td>Rounded result is close to but no greater in absolute value than the infinitely precise result.</td>
</tr>
</tbody>
</table>
```

The rounding-control is set to round to nearest upon reset.

Bit 15 (FZ) is used to turn on the Flush To Zero mode (bit is set). Turning on the Flush To Zero mode has the following effects during underflow situations:

- Zero results are returned with the sign of the true result.
- Precision and underflow exception flags are set.
LDMXCSR: Load Streaming SIMD Extension Control/Status (continued)

The IEEE mandated masked response to underflow is to deliver the denormalized result (i.e., gradually underflow); consequently, the flush to zero mode is not compatible with IEEE Std. 754. It is provided primarily for performance reasons. At the cost of a slight precision loss, faster execution can be achieved for applications where underflows are common. Unmasking the underflow exception takes precedence over Flush To Zero mode; this means that an exception handler will be invoked for a Streaming SIMD Extension instruction that generates an underflow condition while this exception is unmasked, regardless of whether flush to zero is enabled.

The other bits of MXCSR (bits 31-16 and bit 6) are defined as reserved and cleared; attempting to write a non-zero value to these bits, using either the FXRSTOR or LDMXCSR instructions, will result in a general protection exception.

The linear address corresponds to the address of the least-significant byte of the referenced memory data.

**FP Exceptions**: General protection fault if reserved bits are loaded with non-zero values.

**Numeric Exceptions**: None

**Protected Mode Exceptions**:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set. #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true (CR0.AM is set; EFLAGS.AC is set; current CPL is 3);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions**:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions**:

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault. #AC for unaligned memory reference.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments**: The usage of Repeat (F2H, F3H) and Operand Size (66H) prefixes with LDMXCSR is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with LDMXCSR risks incompatibility with future processors.
MAXPS: Packed Single-FP Maximum

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,5F,/r</td>
<td>MAXPS xmm1, xmm2/m128</td>
<td>Return the maximum SP FP numbers between XMM2/Mem and XMM1.</td>
</tr>
</tbody>
</table>

Operation:

\[
\begin{align*}
\text{xmm1}[31-0] &= (\text{xmm1}[31-0] == \text{NAN}) \ ? \ \text{xmm2}[31-0] : \\
& \quad (\text{xmm2}[31-0] == \text{NAN}) \ ? \ \text{xmm2}[31-0] : \\
& \quad (\text{xmm1}[31-0] > \text{xmm2/m128}[31-0]) \ ? \ \text{xmm1}[31-0] \\
\text{xmm2/m128}[31-0]; \\
\text{xmm1}[63-32] &= (\text{xmm1}[63-32] == \text{NAN}) \ ? \ \text{xmm2}[63-32] : \\
& \quad (\text{xmm2}[63-32] == \text{NAN}) \ ? \ \text{xmm2}[63-32] : \\
& \quad (\text{xmm1}[63-32] > \text{xmm2/m128}[63-32]) \ ? \ \text{xmm1}[63-32] \\
\text{xmm2/m128}[63-32]; \\
\text{xmm1}[95-64] &= (\text{xmm1}[95-64] == \text{NAN}) \ ? \ \text{xmm2}[95-64] : \\
& \quad (\text{xmm2}[95-64] == \text{NAN}) \ ? \ \text{xmm2}[95-64] : \\
& \quad (\text{xmm1}[95-64] > \text{xmm2/m128}[95-64]) \ ? \ \text{xmm1}[95-64] \\
\text{xmm2/m128}[95-64]; \\
\text{xmm1}[127-96] &= (\text{xmm1}[127-96] == \text{NAN}) \ ? \ \text{xmm2}[127-96] : \\
& \quad (\text{xmm2}[127-96] == \text{NAN}) \ ? \ \text{xmm2}[127-96] : \\
& \quad (\text{xmm1}[127-96] > \text{xmm2/m128}[127-96]) \ ? \ \text{xmm1}[127-96] \\
\text{xmm2/m128}[127-96];
\end{align*}
\]

Description: The MAXPS instruction returns the maximum SP FP numbers from XMM1 and XMM2/Mem. If the values being compared are both zeros, source2 (xmm2/m128) would be returned. If source2 (xmm2/m128) is an sNaN, this sNaN is forwarded unchanged to the destination (i.e. a quieted version of the sNaN is not returned).

FP Exceptions: General protection exception if not aligned on 16-byte boundary, regardless of segment.

Numeric Exceptions: Invalid (including qNaN source operand), Denormal.

Protected Mode Exceptions:

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments; #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.
MAXPS: Packed Single-FP Maximum (continued)

Real Address Mode Exceptions:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:  Note that if only one source is a NaN for these instructions, the Src2 operand (either NaN or real value) is written to the result; this differs from the behavior for other instructions as defined in Table 3-3, which is to always write the NaN to the result, regardless of which source operand contains the NaN. This approach for MAXPS allows compilers to use the MAXPS instruction for common C conditional constructs. If instead of this behavior, it is required that the NaN source operand be returned, the min/max functionality can be emulated using a sequence of instructions: comparison followed by AND, ANDN and OR.
MAXSS: Scalar Single-FP Maximum

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3,0F,5F/r</td>
<td>MAXSS xmm1, xmm2/m32</td>
<td>Return the maximum SP FP number between the lower SP FP numbers from XMM2/Mem and XMM1.</td>
</tr>
</tbody>
</table>

**Operation:**
\[
\begin{align*}
\text{xmm1}[31-0] &= (\text{xmm1}[31-0] == \text{NAN}) \ ? \ \text{xmm2}[31-0] : \\
&\quad (\text{xmm2}[31-0] == \text{NAN}) \ ? \ \text{xmm2}[31-0] : \\
&\quad (\text{xmm1}[31-0] > \text{xmm2/m32}[31-0]) \ ? \ \text{xmm1}[31-0] : \ \text{xmm2/m32}[31-0]; \\
\text{xmm1}[63-32] &= \text{xmm1}[63-32]; \\
\text{xmm1}[95-64] &= \text{xmm1}[95-64]; \\
\text{xmm1}[127-96] &= \text{xmm1}[127-96];
\end{align*}
\]

**Description:**
The MAXSS instruction returns the maximum SP FP number from the lower SP FP numbers of XMM1 and XMM2/Mem; the upper 3 fields are passed through from xmm1. If the values being compared are both zeros, source2 (xmm2/m128) would be returned. If source2 (xmm2/m128) is an sNaN, this sNaN is forwarded unchanged to the destination (i.e. a quieted version of the sNaN is not returned).

**FP Exceptions:** None

**Numeric Exceptions:** Invalid (including qNaN source operand), Denormal.

**Protected Mode Exceptions:**
- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EF = 1;
- #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**
- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EF = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**
- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.
MAXSS: Scalar Single-FP Maximum (continued)

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:
Note that if only one source is a NaN for these instructions, the Src2 operand (either NaN or real value) is written to the result; this differs from the behavior for other instructions as defined in Table 3-3, which is to always write the NaN to the result, regardless of which source operand contains the NaN. The upper three operands are still bypassed from the src1 operand, as in all other scalar operations. This approach for MAXSS allows compilers to use the MAXSS instruction for common C conditional constructs. If instead of this behavior, it is required that the NaN source operand be returned, the min/max functionality can be emulated using a sequence of instructions: comparison followed by AND, ANDN and OR.
MINPS: Packed Single-FP Minimum

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,5D,/r</td>
<td>MINPS xmm1, xmm2/m128</td>
<td>Return the minimum SP numbers between XMM2/Mem and XMM1.</td>
</tr>
</tbody>
</table>

Operation:

\[
\begin{align*}
\text{xmm1}_{[31-0]} &= \begin{cases} 
\text{xmm1}_{[31-0]} & \text{if NaN} \\
\text{xmm2}_{[31-0]} & \text{if NaN} \\
\text{xmm1}_{[31-0]} & \text{if less than} \\
\text{xmm2/m128}_{[31-0]} & \text{otherwise}
\end{cases} \\
\text{xmm1}_{[63-32]} &= \begin{cases} 
\text{xmm1}_{[63-32]} & \text{if NaN} \\
\text{xmm2}_{[63-32]} & \text{if NaN} \\
\text{xmm1}_{[63-32]} & \text{if less than} \\
\text{xmm2/m128}_{[63-32]} & \text{otherwise}
\end{cases} \\
\text{xmm1}_{[95-64]} &= \begin{cases} 
\text{xmm1}_{[95-64]} & \text{if NaN} \\
\text{xmm2}_{[95-64]} & \text{if NaN} \\
\text{xmm1}_{[95-64]} & \text{if less than} \\
\text{xmm2/m128}_{[95-64]} & \text{otherwise}
\end{cases} \\
\text{xmm1}_{[127-96]} &= \begin{cases} 
\text{xmm1}_{[127-96]} & \text{if NaN} \\
\text{xmm2}_{[127-96]} & \text{if NaN} \\
\text{xmm1}_{[127-96]} & \text{if less than} \\
\text{xmm2/m128}_{[127-96]} & \text{otherwise}
\end{cases}
\end{align*}
\]

Description: The MINPS instruction returns the minimum SP FP numbers from XMM1 and XMM2/Mem. If the values being compared are both zeros, source2 (xmm2/m128) would be returned. If source2 (xmm2/m128) is an sNaN, this sNaN is forwarded unchanged to the destination (i.e. a quieted version of the sNaN is not returned).

FP Exceptions: General protection exception if not aligned on 16-byte boundary, regardless of segment.

Numeric Exceptions: Invalid (including qNaN source operand), Denormal.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set. #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.
MINPS: Packed Single-FP Minimum (continued)

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:

Note that if only one source is a NaN for these instructions, the Src2 operand (either NaN or real value) is written to the result; this differs from the behavior for other instructions as defined in Table 3-3, which is to always write the NaN to the result, regardless of which source operand contains the NaN. This approach for MINPS allows compilers to use the MINPS instruction for common C conditional constructs. If instead of this behavior, it is required that the NaN source operand be returned, the min/max functionality can be emulated using a sequence of instructions: comparison followed by AND, ANDN and OR.
## MINSS: Scalar Single-FP Minimum

### Opcode

<table>
<thead>
<tr>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>MINSS xmm1, xmm2/m32</td>
<td>Return the minimum SP FP number between the lowest SP FP numbers from XMM2/Mem and XMM1.</td>
</tr>
</tbody>
</table>

### Operation:

\[
\text{xmm1}[31-0] = (\text{xmm1}[31-0] == \text{NAN}) \ ? \ \text{xmm2}[31-0] : \\
(\text{xmm2}[31-0] == \text{NAN}) \ ? \ \text{xmm2}[31-0] : \\
(\text{xmm1}[31-0] < \text{xmm2}/m32[31-0]) \ ? \ \text{xmm1}[31-0] : \ \text{xmm2}/m32[31-0];
\]

\[
\text{xmm1}[63-32] = \text{xmm1}[63-32];
\]

\[
\text{xmm1}[95-64] = \text{xmm1}[95-64];
\]

\[
\text{xmm1}[127-96] = \text{xmm1}[127-96];
\]

### Description:

The MINSS instruction returns the minimum SP FP number from the lower SP FP numbers from XMM1 and XMM2/Mem; the upper 3 fields are passed through from xmm1. If the values being compared are both zeros, source2 (xmm2/m128) would be returned. If source2 (xmm2/m128) is an sNaN, this sNaN is forwarded unchanged to the destination (i.e. a quieted version of the sNaN is not returned).

### FP Exceptions:

None

### Numeric Exceptions:

Invalid (including qNaN source operand), Denormal.

### Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference.

To enable #AC exceptions, three conditions must be true: CR0.AM is set; EFLAGS.AC is set; current CPL is 3; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

### Real Address Mode Exceptions:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0); #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

### Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode; #PF (fault-code) for a page fault; #AC for unaligned memory references.
MINSS: Scalar Single-FP Minimum (continued)

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments: Note that if only one source is a NaN for these instructions, the Src2 operand (either NaN or real value) is written to the result; this differs from the behavior for other instructions as defined in Table 3-3, which is to always write the NaN to the result, regardless of which source operand contains the NaN. The upper three operands are still bypassed from the src1 operand, as in all other scalar operations. This approach for MINSS allows compilers to use the MINSS instruction for common C conditional constructs. If instead of this behavior, it is required that the NaN source operand be returned, the min/max functionality can be emulated using a sequence of instructions: comparison followed by AND, ANDN and OR.
MOVAPS: Move Aligned Four Packed Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,28,/r</td>
<td>MOVAPS xmm1, xmm2/m128</td>
<td>Move 128 bits representing 4 packed SP data from XMM2/Mem to XMM1 register.</td>
</tr>
<tr>
<td>0F,29,/r</td>
<td>MOVAPS xmm2/m128, xmm1</td>
<td>Move 128 bits representing 4 packed SP from XMM1 register to XMM2/Mem.</td>
</tr>
</tbody>
</table>

Operation:

```c
if (destination == xmm1) {
    if (source == m128) {
        // load instruction
        xmm1[127-0] = m128;
    }
    else {
        // move instruction
        xmm2[127-0] = xmm1[127-0];
    }
}
else {
    if (destination == m128) {
        // store instruction
        m128 = xmm1[127-0];
    }
    else {
        // move instruction
        xmm2[127-0] = xmm1[127-0];
    }
}
```

Description: The linear address corresponds to the address of the least-significant byte of the referenced memory data. When a memory address is indicated, the 16 bytes of data at memory location m128 are loaded or stored. When the register-register form of this operation is used, the content of the 128-bit source register is copied into 128-bit destination register.

FP Exceptions: General protection exception if not aligned on 16-byte boundary, regardless of segment.

Numeric Exceptions: None
MOVAPS: Move Aligned Four Packed Single-FP (continued)

Protected Mode Exceptions:

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments; #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Real Address Mode Exceptions:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:

MOVAPS should be used when dealing with 16-byte aligned SP FP numbers. If the data is not known to be aligned, MOVUPS should be used instead of MOVAPS. The usage of this instruction should be limited to the cases where the aligned restriction is easy to meet. Processors that support Streaming SIMD Extension will provide optimal aligned performance for the MOVAPS instruction.

The usage of Repeat Prefixes (F2H, F3H) with MOVAPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with MOVAPS risks incompatibility with future processors.
MOVHLPS: Move High to Low Packed Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,12,/r</td>
<td>MOVHLPS xmm1, xmm2</td>
<td>Move 64 bits representing higher two SP operands from XMM2 to lower two fields of XMM1 register.</td>
</tr>
</tbody>
</table>

**Operation:**  // move instruction

\[
\text{xmm1}[127-64] = \text{xmm1}[127-64]; \\
\text{xmm1}[63-0] = \text{xmm2}[127-64];
\]

**Description:** The upper 64-bits of the source register xmm2 are loaded into the lower 64-bits of the 128-bit register xmm1 and the upper 64-bits of xmm1 are left unchanged.

**FP Exceptions:** None

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

- #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1

**Comments:** The usage of Repeat (F2H, F3H) and Operand Size (66H) prefixes with MOVHLPS is reserved. Different processor implementations may handle these prefixes differently. Usage of these prefixes with MOVHLPS risks incompatibility with future processors.
MOVHPS: Move High Packed Single-FP

### Opcode Instruction Description

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,16,/r</td>
<td>MOVHPS xmm, m64</td>
<td>Move 64 bits representing two SP operands from Mem to upper two fields of XMM register.</td>
</tr>
<tr>
<td>0F,17,/r</td>
<td>MOVHPS m64, xmm</td>
<td>Move 64 bits representing two SP operands from upper two fields of XMM register to Mem.</td>
</tr>
</tbody>
</table>

#### Operation:

```c
if (destination == xmm) {
    // load instruction
    xmm[127-64] = m64;
    xmm[31-0] = xmm[31-0];
    xmm[63-32] = xmm[63-32];
}
else {
    // store instruction
    m64 = xmm[127-64];
}
```

#### Description: The linear address corresponds to the address of the least-significant byte of the referenced memory data. When the load form of this operation is used, m64 is loaded into the upper 64-bits of the 128-bit register xmm and the lower 64-bits are left unchanged.

#### FP Exceptions: None

#### Numeric Exceptions: None

#### Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

#### Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

#### Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode; #PF (fault-code) for a page fault; #AC for unaligned memory reference if the current privilege level is 3.
MOVHPS: Move High Packed Single-FP (continued)

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault

Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Comments: The usage of Repeat Prefixes (F2H, F3H) with MOVHPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with MOVHPS risks incompatibility with future processors.
MOVLHPS: Move Low to High Packed Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,16,/r</td>
<td>MOVLHPS xmm1, xmm2</td>
<td>Move 64 bits representing lower two SP operands from XMM2 to upper two fields of XMM1 register.</td>
</tr>
</tbody>
</table>

**Operation:**
```
// move instruction
xmm1[127-64] = xmm2[63-0];
xmm1[63-0] = xmm1[63-0];
```

**Description:** The lower 64-bits of the source register xmm2 are loaded into the upper 64-bits of the 128-bit register xmm1 and the lower 64-bits of xmm1 are left unchanged.

**FP Exceptions:** None

**Numeric Exceptions:** None

**Protected Mode Exceptions:**
- #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**
- #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**
- Same exceptions as in Real Address Mode.

**Additional Itanium System Environment Exceptions**
- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1

**Comments:**

**Example:**
The usage of Repeat (F2H, F3H) and Operand Size (66H) prefixes with MOVLHPS is reserved. Different processor implementations may handle these prefixes differently. Usage of these prefixes with MOVLHPS risks incompatibility with future processors.
MOVLPS: Move Low Packed Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,12,/r</td>
<td>MOVLPS xmm, m64</td>
<td>Move 64 bits representing two SP operands from Mem to lower two fields of XMM register.</td>
</tr>
<tr>
<td>0F,13,/r</td>
<td>MOVLPS m64, xmm</td>
<td>Move 64 bits representing two SP operands from lower two fields of XMM register to Mem.</td>
</tr>
</tbody>
</table>

Operation:

```c
if (destination == xmm) {
    // load instruction
    xmm[63-0]   = m64;
    xmm[95-64]  = xmm[95-64];
    xmm[127-96] = xmm[127-96];
}
else {
    // store instruction
    m64 = xmm[63-0];
}
```

Description: The linear address corresponds to the address of the least-significant byte of the referenced memory data. When the load form of this operation is used, m64 is loaded into the lower 64-bits of the 128-bit register xmm and the upper 64-bits are left unchanged.

FP Exceptions: None

Numeric Exceptions: None

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3);
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

Real Address Mode Exceptions:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode;
- #PF (fault-code) for a page fault;
- #AC for unaligned memory reference if the current privilege level is 3.
MOVLPS: Move Low Packed Single-FP (continued)

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Comments: The usage of Repeat Prefixes (F2H, F3H) with MOVLPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with MOVLPS risks incompatibility with future processors.
**MOVMSKPS: Move Mask to Integer**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,50,./r</td>
<td>MOVMSKPS r32, xmm</td>
<td>Move the single mask to r32.</td>
</tr>
</tbody>
</table>

**Operation:**

- \( r32[3] = \text{xmm}[127]; \)
- \( r32[2] = \text{xmm}[95]; \)
- \( r32[1] = \text{xmm}[63]; \)
- \( r32[0] = \text{xmm}[31]; \)
- \( r32[7-4] = 0x0; \)
- \( r32[15-8] = 0x00; \)
- \( r32[31-16] = 0x0000; \)

**Description:** The MOVMSKPS instruction returns to the integer register \( r32 \) a 4-bit mask formed of the most significant bits of each SP FP number of its operand.

**FP Exceptions:** None

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

- #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception.;
- #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- #UD if CR0.EM = 1; #NM if TS bit in CR0 is set.; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault

**Comments:** The usage of Repeat Prefixes (F2H, F3H) with MOVMSKPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with MOVMSKPS risks incompatibility with future processors.
MOVSS: Move Scalar Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3.0F,10.10/r</td>
<td>MOVSS xmm1, xmm2/m32</td>
<td>Move 32 bits representing one scalar SP operand from XMM2/Mem to XMM1 register.</td>
</tr>
<tr>
<td>F3.0F,11.10/r</td>
<td>MOVSS xmm2/m32, xmm1</td>
<td>Move 32 bits representing one scalar SP operand from XMM1 register to XMM2/Mem.</td>
</tr>
</tbody>
</table>

Operation:
if (destination == xmm1) {
  if (source == m32) {
    // load instruction
    xmm1[31-0] = m32;
    xmm1[63-32] = 0x00000000;
    xmm1[95-64] = 0x00000000;
    xmm1[127-96] = 0x00000000;
  }
  else {
    // move instruction
    xmm1[31-0] = xmm2[31-0];
    xmm1[63-32] = xmm1[63-32];
    xmm1[95-64] = xmm1[95-64];
    xmm1[127-96] = xmm1[127-96];
  }
}
else {
  if (destination == m32) {
    // store instruction
    m32 = xmm1[31-0];
  }
  else {
    // move instruction
    xmm2[31-0] = xmm1[31-0];
    xmm2[63-32] = xmm2[63-32];
    xmm2[95-64] = xmm2[95-64];
MOVSS: Move Scalar Single-FP (continued)

```c
    xmm2[127-96] = xmm2[127-96];  
```

Description: The linear address corresponds to the address of the least-significant byte of the referenced memory data. When a memory address is indicated, the 4 bytes of data at memory location m32 are loaded or stored. When the load form of this operation is used, the 32-bits from memory are copied into the lower 32 bits of the 128-bit register xmm, the 96 most significant bits being cleared.

FP Exceptions: None

Numeric Exceptions: None

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EF = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EF = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault
MOVUPS: Move Unaligned Four Packed Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,10,\r</td>
<td>MOVUPS xmm1, xmm2/m128</td>
<td>Move 128 bits representing four SP data from XMM2/Mem to XMM1 register.</td>
</tr>
<tr>
<td>0F,11,\r</td>
<td>MOVUPS xmm2/m128, xmm1</td>
<td>Move 128 bits representing four SP data from XMM1 register to XMM2/Mem.</td>
</tr>
</tbody>
</table>

**Operation:**

```c
if (destination == xmm1) {
    if (source == m128) {
        // load instruction
        xmm1[127-0] = m128;
    }
    else {
        // move instruction
        xmm1[127-0] = xmm2[127-0];
    }
} else {
    if (destination == m128) {
        // store instruction
        m128 = xmm1[127-0];
    } else {
        // move instruction
        xmm2[127-0] = xmm1[127-0];
    }
}
```

**Description:** The linear address corresponds to the address of the least-significant byte of the referenced memory data. When a memory address is indicated, the 16 bytes of data at memory location m128 are loaded to the 128-bit multimedia register xmm or stored from the 128-bit multimedia register xmm. When the register-register form of this operation is used, the content of the 128-bit source register is copied into 128-bit register xmm. No assumption is made about alignment.

**FP Exceptions:** None

**Numeric Exceptions:** None
MOVUPS: Move Unaligned Four Packed Single-FP (continued)

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #AC for an unaligned memory reference if the current privilege level is 3;
- #NM if TS bit in CR0 is set.

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set.

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode;
- #AC for an unaligned memory reference if the current privilege level is 3;
- #PF (fault-code) for a page fault.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

Comments:

MOVUPS should be used with SP FP numbers when that data is known to be unaligned. The usage of this instruction should be limited to the cases where the aligned restriction is hard or impossible to meet. Streaming SIMD Extension implementations guarantee optimum unaligned support for MOVUPS. Efficient Streaming SIMD Extension applications should mainly rely on MOVAPS, not MOVUPS, when dealing with aligned data.

The usage of Repeat-NE Prefix (F2H) and Operand Size Prefix (66H) with MOVUPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with MOVUPS risks incompatibility with future processors.

A linear address of the 128 bit data access, while executing in 16-bit mode, that overlaps the end of a 16-bit segment is not allowed and is defined as reserved behavior. Different processor implementations may/may not raise a GP fault in this case if the segment limit has been exceeded; additionally, the address that spans the end of the segment may/may not wrap around to the beginning of the segment.
MULPS: Packed Single-FP Multiply

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,59,/r</td>
<td>MULPS xmm1, xmm2/m128</td>
<td>Multiply packed SP FP numbers in XMM2/Mem to XMM1.</td>
</tr>
</tbody>
</table>

**Operation:**
- \( \text{xmm1}[31-0] = \text{xmm1}[31-0] \times \text{xmm2/m128}[31-0]; \)
- \( \text{xmm1}[63-32] = \text{xmm1}[63-32] \times \text{xmm2/m128}[63-32]; \)
- \( \text{xmm1}[95-64] = \text{xmm1}[95-64] \times \text{xmm2/m128}[95-64]; \)
- \( \text{xmm1}[127-96] = \text{xmm1}[127-96] \times \text{xmm2/m128}[127-96]; \)

**Description:** The MULPS instructions multiply the packed SP FP numbers of both their operands.

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** Overflow, Underflow, Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0).

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0).

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults  VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
MULSS: Scalar Single-FP Multiply

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3.0F,59,/r</td>
<td>MULSS xmm1 xmm2/m32</td>
<td>Multiply the lowest SP FP number in XMM2/Mem to XMM1.</td>
</tr>
</tbody>
</table>

\[
\begin{align*}
xmm1[31-0] & = xmm1[31-0] * xmm2/m32[31-0]; \\
xmm1[63-32] & = xmm1[63-32]; \\
xmm1[95-64] & = xmm1[95-64]; \\
xmm1[127-96] & = xmm1[127-96]; 
\end{align*}
\]

**Description:** The MULSS instructions multiply the lowest SP FP numbers of both their operands; the upper 3 fields are passed through from xmm1.

**FP Exceptions:** None

**Numeric Exceptions:** Overflow, Underflow, Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0).

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0).

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
**ORPS: Bit-wise Logical OR for Single-FP Data**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,66,r</td>
<td>ORPS xmm1, xmm2/m128</td>
<td>OR 128 bits from XMM2/Mem to XMM1 register.</td>
</tr>
</tbody>
</table>

**Operation:**
\[ \text{xmm1}[127:0] \lor \text{xmm2/m128}[127:0] \]

**Description:** The ORPS instructions return a bit-wise logical OR between xmm1 and xmm2/mem.

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**
- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set.

**Real Address Mode Exceptions:**
- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set.

**Virtual 8086 Mode Exceptions:**
- Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**
- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments:** The usage of Repeat Prefixes (F2H, F3H) with ORPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with ORPS risks incompatibility with future processors.
RCPPS: Packed Single-FP Reciprocal

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,53,/r</td>
<td>RCPPS xmm1, xmm2/m128</td>
<td>Return a packed approximation of the reciprocal of XMM2/Mem.</td>
</tr>
</tbody>
</table>

Operation:

\[
\begin{align*}
\text{xmm1}[31-0] & = \text{approx} \left( \frac{1.0}{\text{xmm2/m128}[31-0]} \right); \\
\text{xmm1}[63-32] & = \text{approx} \left( \frac{1.0}{\text{xmm2/m128}[63-32]} \right); \\
\text{xmm1}[95-64] & = \text{approx} \left( \frac{1.0}{\text{xmm2/m128}[95-64]} \right); \\
\text{xmm1}[127-96] & = \text{approx} \left( \frac{1.0}{\text{xmm2/m128}[127-96]} \right);
\end{align*}
\]

Description: RCPPS returns an approximation of the reciprocal of the SP FP numbers from xmm2/m128. The relative error for this approximation is Error, which satisfies:

\[|\text{Error}| \leq 1.5 \times 2^{-12}\]

FP Exceptions: General protection exception if not aligned on 16-byte boundary, regardless of segment.

Numeric Exceptions: None.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set.

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set.

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault

Itanium Mem Faults  VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments: RCPPS is not affected by the rounding control in MXCSR. Denormal inputs are treated as zeros (of the same sign) and tiny results are always flushed to zero, with the sign of the operand.

Results are guaranteed not to be tiny, and therefore not flushed to zero, for input values x which satisfy

\[|x| \leq 1.11111111010000000000_b \times 2^{125}\]

For input values x which satisfy
flush-to-zero might or might not occur, depending on the implementation (this interval contains 6144 + 3072 = 9216 single precision floating-point numbers).

Results are guaranteed to be tiny, and therefore flushed to zero, for input values \( x \) which satisfy

\[
|x| \leq 1.00000000000110000000001 \times 2^{126}
\]

The decimal approximations of the single precision numbers that delimit the three intervals specified above, are as follows:

\[
\begin{align*}
1.1111111110100000000000_{B} \times 2^{125} & \approx 8.5039437 \times 10^{37} \\
1.1111111111010000000001_{B} \times 2^{125} & \approx 8.5039443 \times 10^{37} \\
1.0000000000110000000000_{B} \times 2^{126} & \approx 4.2550872 \times 10^{37} \\
1.0000000000110000000001_{B} \times 2^{126} & \approx 4.2550877 \times 10^{37}
\end{align*}
\]

The hexadecimal representations of the single precision numbers that delimit the three intervals specified above, are as follows:

\[
\begin{align*}
1.1111111111010000000000_{B} \times 2^{125} & = 0x7e7fe800 \\
1.1111111111100000000001_{B} \times 2^{125} & = 0x7e7fe801 \\
1.0000000000110000000000_{B} \times 2^{126} & = 0x7e800c00 \\
1.0000000000110000000001_{B} \times 2^{126} & = 0x7e800c01
\end{align*}
\]
RCPSS: Scalar Single-FP Reciprocal

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3.0F,53,/r</td>
<td>RCPSS xmm1, xmm2/m32</td>
<td>Return an approximation of the reciprocal of the lower SP FP number in XMM2/Mem.</td>
</tr>
</tbody>
</table>

Operation:

\[
\begin{align*}
\text{xmm1}[31-0] & = \text{approx } (1.0/(\text{xmm2/m32}[31-0])); \\
\text{xmm1}[63-32] & = \text{xmm1}[63-32]; \\
\text{xmm1}[95-64] & = \text{xmm1}[95-64]; \\
\text{xmm1}[127-96] & = \text{xmm1}[127-96];
\end{align*}
\]

Description:

RCPSS returns an approximation of the reciprocal of the lower SP FP number from xmm2/m32; the upper 3 fields are passed through from xmm1. The relative error for this approximation is Error, which satisfies:

\[|\text{Error}| \leq 1.5 \times 2^{-12} \]

Numeric Exceptions: None.

Protected Mode Exceptions:

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments; 
#SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EF = 1; #AC for unaligned memory reference if the current privilege level is 3; #NM if TS bit in CR0 is set.

Real Address Mode Exceptions:

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EF = 1; #NM if TS bit in CR0 is set.

Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:

RCPSS is not affected by the rounding control in MXCSR. Denormal inputs are treated as zeros (of the same sign) and tiny results are always flushed to zero, with the sign of the operand.

Results are guaranteed not to be tiny, and therefore not flushed to zero, for input values x which satisfy

\[|x| \leq 1.111111111010000000000B \times 2^{125}\]

For input values x which satisfy
flush-to-zero might or might not occur, depending on the implementation (this interval contains
6144 + 3072 = 9216 single precision floating-point numbers).

Results are guaranteed to be tiny, and therefore flushed to zero, for input values \( x \) which satisfy
\[
|x| \leq 1.00000000000110000000001 \times 2^{126}
\]

The decimal approximations of the single precision numbers that delimit the three intervals
specified above, are as follows:

1.111111111111110000000000001 \times 2^{125} \approx 8.5039437 \times 10^{37}

1.111111111111110000000000001 \times 2^{125} \approx 8.5039443 \times 10^{37}

1.00000000000110000000000 \times 2^{126} \approx 4.2550872 \times 10^{37}

1.00000000000110000000001 \times 2^{126} \approx 4.2550877 \times 10^{37}

The hexadecimal representations of the single precision numbers that delimit the three intervals
specified above, are as follows:

1.111111111111110000000000001 \times 2^{125} = 0x7e7fe800

1.111111111111110000000000001 \times 2^{125} = 0x7e7fe801

1.00000000000110000000000 \times 2^{126} = 0x7e8000c0

1.00000000000110000000001 \times 2^{126} = 0x7e8000c1
RSQRTPS: Packed Single-FP Square Root Reciprocal

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F.52./r</td>
<td>RSQRTPS xmm1, xmm2/m128</td>
<td>Return a packed approximation of the square root of the reciprocal of XMM2/Mem.</td>
</tr>
</tbody>
</table>

**Operation:**
- \( \text{xmm1}[31-0] = \text{approx} \left( \frac{1}{\sqrt{\text{xmm2/m128}[31-0]}} \right) \);
- \( \text{xmm1}[63-32] = \text{approx} \left( \frac{1}{\sqrt{\text{xmm2/m128}[63-32]}} \right) \);
- \( \text{xmm1}[95-64] = \text{approx} \left( \frac{1}{\sqrt{\text{xmm2/m128}[95-64]}} \right) \);
- \( \text{xmm1}[127-96] = \text{approx} \left( \frac{1}{\sqrt{\text{xmm2/m128}[127-96]}} \right) \);

**Description:**
RSQRTPS returns an approximation of the reciprocal of the square root of the SP FP numbers from xmm2/m128. The relative error for this approximation is Error, which satisfies:

\[ |\text{Error}| \leq 1.5 \times 2^{-12} \]

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**
- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**
- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**
- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**
- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments:** RSQRTPS is not affected by the rounding control in MXCSR. Denormal inputs are treated as zeros (of the same sign).
RSQRTSS: Scalar Single-FP Square Root Reciprocal

Operation:
\[
\begin{align*}
xmm1[31-0] &= \text{approx } (1.0/\sqrt{xmm2/m32[31-0]}) \\
xmm1[63-32] &= xmm1[63-32] \\
xmm1[95-64] &= xmm1[95-64] \\
xmm1[127-96] &= xmm1[127-96]
\end{align*}
\]

Description: RSQRTSS returns an approximation of the reciprocal of the square root of the lowest SP FP number from xmm2/m32; the upper 3 fields are passed through from xmm1. The relative error for this approximation is Error, which satisfies:

\[|Error| \leq 1.5 \times 2^{-12}\]

Numeric Exceptions: None.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set.

Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults
- Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults
- VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:

Example: RSQRTSS is not affected by the rounding control in MXCSR. Denormal inputs are treated as zeros (of the same sign).
SHUFPS: Shuffle Single-FP

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,C6,/r, ib</td>
<td>SHUFPS xmm1, xmm2/m128, imm8</td>
<td>Shuffle Single.</td>
</tr>
</tbody>
</table>

**Operation:**

```c
fp_select  = (imm8 >> 0) & 0x3;
xmm1[31-0]   = (fp_select == 0) ? xmm1[31-0]   :
                   (fp_select == 1) ? xmm1[63-32]  :
                   (fp_select == 2) ? xmm1[95-64]  :
                   xmm1[127-96];

fp_select  = (imm8 >> 2) & 0x3;
xmm1[63-32]  = (fp_select == 0) ? xmm1[31-0]   :
                   (fp_select == 1) ? xmm1[63-32]  :
                   (fp_select == 2) ? xmm1[95-64]  :
                   xmm1[127-96];

fp_select  = (imm8 >> 4) & 0x3;
xmm1[95-64]  = (fp_select == 0) ? xmm2/m128[31-0]   :
                   (fp_select == 1) ? xmm2/m128[63-32]  :
                   (fp_select == 2) ? xmm2/m128[95-64]  :
                   xmm2/m128[127-96];

fp_select  = (imm8 >> 6) & 0x3;
xmm1[127-96] = (fp_select == 0) ? xmm2/m128[31-0]   :
                   (fp_select == 1) ? xmm2/m128[63-32]  :
                   (fp_select == 2) ? xmm2/m128[95-64]  :
                   xmm2/m128[127-96];
```

**Description:** The SHUFPS instruction is able to shuffle any of the four SP FP numbers from xmm1 to the lower 2 destination fields; the upper 2 destination fields are generated from a shuffle of any of the four SP FP numbers from xmm2/m128. By using the same register for both sources, SHUFPS can return any combination of the four SP FP numbers from this register. Bits 0 and 1 of the immediate field are used to select which of the four input SP FP numbers will be put in the first SP FP number of the result; bits 3 and 2 of the immediate field are used to select which of the four input SP FP will be put in the second SP FP number of the result; etc.
SHUFPS: Shuffle Single-FP (continued)

Example:

```
+-----+-----+-----+-----+
|    X4  |    X3  |    X2  |    X1  |
+-----+-----+-----+-----+   xmm1
       |    Y4  |    Y3  |    Y2  |    Y1  |
+-----+-----+-----+-----+   xmm2/m128
       |   {Y4 ... Y1} |   {Y4 ... Y1} |
+-----+-----+-----+-----+   xmm1

{xmm1} {Y4 ... Y1} {xmm1} {X4 ... X1}
```

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments:** The usage of Repeat Prefixes (F2H, F3H) with SHUFPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with SHUFPS risks incompatibility with future processors.
SQRTPS: Packed Single-FP Square Root

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,51./r</td>
<td>SQRTPS xmm1, xmm2/m128</td>
<td>Square Root of the packed SP FP numbers in XMM2/Mem.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{xmm1}[31-0] & = \sqrt{\text{xmm2/m128}[31-0]}; \\
\text{xmm1}[63-32] & = \sqrt{\text{xmm2/m128}[63-32]}; \\
\text{xmm1}[95-64] & = \sqrt{\text{xmm2/m128}[95-64]}; \\
\text{xmm1}[127-96] & = \sqrt{\text{xmm2/m128}[127-96]};
\end{align*}
\]

**Description:** The SQRTPS instruction returns the square root of the packed SP FP numbers from xmm2/m128.

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0);
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults
- Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults
- VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
**SQRTSS: Scalar Single-FP Square Root**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3.0F.51,/r</td>
<td>SQRTSS xmm1, xmm2/m32</td>
<td>Square Root of the lower SP FP number in XMM2/Mem.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{xmm1}[31-0] & = \text{sqrt} \ (\text{xmm2/m32}[31-0]) ; \\
\text{xmm1}[63-32] & = \text{xmm1}[63-32] ; \\
\text{xmm1}[95-64] & = \text{xmm1}[95-64] ; \\
\text{xmm1}[127-96] & = \text{xmm1}[127-96] ;
\end{align*}
\]

**Description:** The SQRTSS instructions return the square root of the lowest SP FP numbers of their operand.

**FP Exceptions:** None

**Numeric Exceptions:** Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CR CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 1); #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT = 0); #UD if CR CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults
- Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults
- VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
STMXCSR: Store Streaming SIMD Extension Control/Status

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,AE,/3</td>
<td>STMXCSR m32</td>
<td>Store Streaming SIMD Extension control/status word to m32.</td>
</tr>
</tbody>
</table>

**Operation:**  

\[ m32 = MXCSR; \]

**Description:**  
The MXCSR control/status register is used to enable masked/unmasked exception handling, to set rounding modes, to set flush-to-zero mode, and to view exception status flags. Refer to LDMXCSR for a description of the format of MXCSR. The linear address corresponds to the address of the least-significant byte of the referenced memory data. The reserved bits in the MXCSR are stored as zeroes.

**FP Exceptions:** None.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set. #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3); #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault. #AC for unaligned memory reference.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  NaT Register Consumption Fault
- Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Comments:**

The usage of Repeat (F2H, F3H) andOperand Size (66H) prefixes with STMXCSR is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with STMXCSR risks incompatibility with future processors.
SUBPS: Packed Single-FP Subtract

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,5C,r</td>
<td>SUBPS xmm1 xmm2/m128</td>
<td>Subtract packed SP FP numbers in XMM2/Mem from XMM1.</td>
</tr>
</tbody>
</table>

**Operation:**

- \( \text{xmm1}[31-0] = \text{xmm1}[31-0] - \text{xmm2/m128}[31-0] \)
- \( \text{xmm1}[63-32] = \text{xmm1}[63-32] - \text{xmm2/m128}[63-32] \)
- \( \text{xmm1}[95-64] = \text{xmm1}[95-64] - \text{xmm2/m128}[95-64] \)
- \( \text{xmm1}[127-96] = \text{xmm1}[127-96] - \text{xmm2/m128}[127-96] \)

**Description:** The SUBPS instruction subtracts the packed SP FP numbers of both their operands.

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** Overflow, Underflow, Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault;

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
**SUBSS: Scalar Single-FP Subtract**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>F3,0F,5C, /r</td>
<td>SUBSS xmm1, xmm2/m32</td>
<td>Subtract the lower SP FP numbers in XMM2/Mem from XMM1.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{xmm1}[31-0] &= \text{xmm1}[31-0] - \text{xmm2}/m32[31-0]; \\
\text{xmm1}[63-32] &= \text{xmm1}[63-32]; \\
\text{xmm1}[95-64] &= \text{xmm1}[95-64]; \\
\text{xmm1}[127-96] &= \text{xmm1}[127-96];
\end{align*}
\]

**Description:** The SUBSS instruction subtracts the lower SP FP numbers of both their operands.

**FP Exceptions:** None.

**Numeric Exceptions:** Overflow, Underflow, Invalid, Precision, Denormal.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3);
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFF; #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
UCOMISS: Unordered Scalar Single-FP Compare and Set EFLAGS

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,2E/r</td>
<td>UCOMISS xmm1, xmm2/m32</td>
<td>Compare lower SP FP number in XMM1 register with lower SP FP number in XMM2/Mem and set the status flags accordingly.</td>
</tr>
</tbody>
</table>

**Operation:**

```c
switch (xmm1[31-0] <> xmm2/m32[31-0]) {
    OF, SF, AF = 000;
    case UNORDERED:     ZF, PF, CF = 111;
    case GREATER_THAN:  ZF, PF, CF = 000;
    case LESS_THAN:     ZF, PF, CF = 001;
    case EQUAL:         ZF, PF, CF = 100;
}
```

**Description:**

The UCOMISS instructions compare the two lowest scalar SP FP numbers and sets the ZF,PF,CF bits in the EFLAGS register as described above. In addition, the OF, SF and AF bits in the EFLAGS register are zeroed out. The unordered predicate is returned if either source operand is a NaN (qNaN or sNaN).

**FP Exceptions:** None.

**Numeric Exceptions:** Invalid (if SNaN operands), Denormal. Integer EFLAGS values will not be updated in the presence of unmasked numeric exceptions.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3);
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #XM for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =1);
- #UD for an unmasked Streaming SIMD Extension numeric exception (CR4.OSXMMEXCPT =0);
- #UD if CR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.
UCOMISS: Unordered Scalar Single-FP Compare and Set EFLAGS (continued)

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:

UCOMISS differs from COMISS in that it signals an invalid numeric exception when a source operand is an sNaN; COMISS signals invalid if a source operand is either a qNaN or an sNaN.

The usage of Repeat (F2H, F3H) and Operand-Size prefixes with UCOMISS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with UCOMISS risks incompatibility with future processors.
UNPCKHPS: Unpack High Packed Single-FP Data

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,15,/r</td>
<td>UNPCKHPS xmm1, xmm2/m128</td>
<td>Interleaves SP FP numbers from the high halves of XMM1 and XMM2/Mem into XMM1 register.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{xmm1}[31-0] &= \text{xmm1}[95-64]; \\
\text{xmm1}[63-32] &= \text{xmm2/m128}[95-64]; \\
\text{xmm1}[95-64] &= \text{xmm1}[127-96]; \\
\text{xmm1}[127-96] &= \text{xmm2/m128}[127-96];
\end{align*}
\]

**Description:** The UNPCKHPS instruction performs an interleaved unpack of the high-order data elements of XMM1 and XMM2/Mem. It ignores the lower half of the sources.

**Example:**

![Diagram of UNPCKHPS operation](image)

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments; #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.
UNPCKHPS: Unpack High Packed Single-FP Data (continued)

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults  VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not
               Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data
               Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments:  When unpacking from a memory operand, an implementation may decide to fetch only the
               appropriate 64 bits. Alignment to 16-byte boundary and normal segment checking will still be
               enforced.

The usage of Repeat Prefixes (F2H, F3H) with UNPCKHPS is reserved. Different processor
implementations may handle this prefix differently. Usage of this prefix with UNPCKHPS risks
incompatibility with future processors.
UNPCKLPS: Unpack Low Packed Single-FP Data

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,14,/r</td>
<td>UNPCKLPS xmm1, xmm2/m128</td>
<td>Interleaves SP FP numbers from the low halves of XMM1 and XMM2/Mem into XMM1 register.</td>
</tr>
</tbody>
</table>

Operation:

\[
\begin{align*}
\text{xmm1}[31-0] &= \text{xmm1}[31-0]; \\
\text{xmm1}[63-32] &= \text{xmm2/m128}[31-0]; \\
\text{xmm1}[95-64] &= \text{xmm1}[63-32]; \\
\text{xmm1}[127-96] &= \text{xmm2/m128}[63-32]; \\
\end{align*}
\]

Description: The UNPCKLPS instruction performs an interleaved unpack of the low-order data elements of XMM1 and XMM2/Mem. It ignores the upper half part of the sources.

Example:

\[
\begin{array}{cccc}
  & X4 & X3 & X2 & X1 \\
\text{xmm1} & | & | & | \\
  & Y4 & Y3 & Y2 & Y1 \\
\text{xmm2/m128} & | & | & | \\
  & Y2 & X2 & Y1 & X1 \\
\end{array}
\]

FP Exceptions: General protection exception if not aligned on 16-byte boundary, regardless of segment.

Numeric Exceptions: None.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

Virtual 8086 Mode Exceptions:

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.
UNPCKLPS: Unpack Low Packed Single-FP Data (continued)

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

Comments: When unpacking from a memory operand, an implementation may decide to fetch only the appropriate 64 bits. Alignment to 16-byte boundary and normal segment checking will still be enforced.

The usage of Repeat Prefixes (F2H, F3H) with UNPCKLPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with UNPCKLPS risks incompatibility with future processors.
XORPS: Bit-wise Logical Xor for Single-FP Data

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,57./r</td>
<td>XORPS xmm1, xmm2/m128</td>
<td>XOR 128 bits from XMM2/Mem to XMM1 register.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\text{xmm}[127-0] \oplus= \text{xmm/m128}[127-0];
\]

**Description:**

The XORPS instruction returns a bit-wise logical XOR between XMM1 and XMM2/Mem.

**FP Exceptions:** General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault

**Comments:**

The usage of Repeat Prefixes (F2H, F3H) with XORPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with XORPS risks incompatibility with future processors.

### 3.13 SIMD Integer Instruction Set Extensions

Additional new SIMD Integer instructions have been added to accelerate the performance of 3D graphics, video decoding and encoding and other applications. These instructions operate on the MMX technology registers and on 64-bit memory operands.
PAVGB/PAVGW: Packed Average

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,E0, /r</td>
<td>PAVGB mm1,mm2/m64</td>
<td>Average with rounding packed unsigned bytes from MM2/Mem to packed bytes in MM1 register.</td>
</tr>
<tr>
<td>0F,E3, /r</td>
<td>PAVGW mm1, mm2/m64</td>
<td>Average with rounding packed unsigned words from MM2/Mem to packed words in MM1 register.</td>
</tr>
</tbody>
</table>

Operation:

```c
if (instruction == PAVGB) {
    x[0] = mm1[7-0]  y[0] = mm2/m64[7-0];
    x[1] = mm1[15-8] y[1] = mm2/m64[15-8];
    x[5] = mm1[47-40] y[5] = mm2/m64[47-40];
    x[7] = mm1[63-56] y[7] = mm2/m64[63-56];

    for (i = 0; i < 8; i++) {
        temp[i] = zero_ext(x[i], 8) + zero_ext(y[i], 8);
        res[i] = (temp[i] +1) >> 1;
    }
    mm1[7-0] = res[0];
    ...
    mm1[63-56] = res[7];
}
else if (instruction == PAVGW) {
    x[0] = mm1[15-0]  y[0] = mm2/m64[15-0];
    x[1] = mm1[31-16] y[1] = mm2/m64[31-16];

    for (i = 0; i < 4; i++) {
```

Volume 3: IA-32 Streaming SIMD Extension Instruction Reference
PAVGB/PAVGW: Packed Average (continued)

```
        temp[i] = zero_ext(x[i], 16) + zero_ext(y[i], 16);
        res[i] = (temp[i] +1) >> 1;
    }
    mm1[15-0] = res[0];
    ...
    mm1[63-48] = res[3];
```

**Description:**
The PAVG instructions add the unsigned data elements of the source operand to the unsigned data elements of the destination register, along with a carry-in. The results of the add are then each independently right shifted by one bit position. The high order bits of each element are filled with the carry bits of the corresponding sum.

The destination operand is a MMX technology register. The source operand can either be a MMX technology register or a 64-bit memory operand.

The PAVGB instruction operates on packed unsigned bytes and the PAVGW instruction operates on packed unsigned words.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set.
- #MF if there is a pending FPU exception;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true (CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode: #PF(fault-code) for a page fault; #AC for unaligned memory references (if the current privilege level is 3).

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults  VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PEXTRW: Extract Word

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,C5, /r, ib</td>
<td>PEXTRW r32, mm, imm8</td>
<td>Extract the word pointed to by imm8 from MM and move it to a 32-bit integer register.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\text{sel} = \text{imm8} \& 0x3; \\
\text{mm\_temp} = (\text{mm} \gg (\text{sel} \times 16)) \& 0xffff; \\
\text{r}[15-0] = \text{mm\_temp}[15-0]; \\
\text{r}[31-16] = 0x0000;
\]

**Description:** The PEXTRW instruction moves the word in MM selected by the two least significant bits of imm8 to the lower half of a 32-bit integer register.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF(fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set. #MF if there is a pending FPU exception.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set. #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1
PINSRW: Insert Word

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,C4, r,ib</td>
<td>PINSRW mm, r32/m16, imm8</td>
<td>Insert the word from the lower half of r32 or from Mem16 into the position in MM pointed to by imm8 without touching the other words.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\text{sel} = \text{imm8} & 0x3;
\]

\[
\text{mask} = (\text{sel} == 0)? 0x000000000000ffff : \\
(\text{sel} == 1)? 0x00000000ffff0000 : \\
(\text{sel} == 2)? 0x0000ffff00000000 : 0xffff000000000000;
\]

\[
\text{mm} = (\text{mm} & \sim\text{mask}) | ((\text{m16/r32}[15-0] \ll (\text{sel} \times 16)) & \text{mask});
\]

**Description:** The PINSRW instruction loads a word from the lower half of a 32-bit integer register (or from memory) and inserts it in the MM destination register at a position defined by the two least significant bits of the imm8 constant. The insertion is done in such a way that the three other words from the destination register are left untouched.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set. #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults   Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults   VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PMAXSW: Packed Signed Integer Word Maximum

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,EE, /r</td>
<td>PMAXSW mm1, mm2/m64</td>
<td>Return the maximum words between MM2/Mem and MM1.</td>
</tr>
</tbody>
</table>

Operation:

\[
\begin{align*}
mm1[15-0] &= (mm1[15-0] > mm2/m64[15-0]) \ ? mm1[15-0] : mm2/m64[15-0]; \\
mm1[31-16] &= (mm1[31-16] > mm2/m64[31-16]) \ ? mm1[31-16] : mm2/m64[31-16]; \\
mm1[47-32] &= (mm1[47-32] > mm2/m64[47-32]) \ ? mm1[47-32] : mm2/m64[47-32]; \\
mm1[63-48] &= (mm1[63-48] > mm2/m64[63-48]) \ ? mm1[63-48] : mm2/m64[63-48]; \\
\end{align*}
\]

Description: The PMAXSW instruction returns the maximum between the four signed words in MM1 and MM2/Mem.

Numeric Exceptions: None.

Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

Real Address Mode Exceptions:

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set. #MF if there is a pending FPU exception.

Virtual 8086 Mode Exceptions:

- Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault; #AC for unaligned memory reference if the current privilege level is 3.

Additional Itanium System Environment Exceptions

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PMAxUB: Packed Unsigned Integer Byte Maximum

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,DE, /r</td>
<td>PMAxUB mm1, mm2/m64</td>
<td>Return the maximum bytes between MM2/Mem and MM1.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
    mm1[7-0] &= (mm1[7-0] > mm2/m64[7-0]) \cdot mm1[7-0] : mm2/m64[7-0]; \\
    mm1[15-8] &= (mm1[15-8] > mm2/m64[15-8]) \cdot mm1[15-8] : mm2/m64[15-8]; \\
    mm1[23-16] &= (mm1[23-16] > mm2/m64[23-16]) \cdot mm1[23-16] : mm2/m64[23-16]; \\
    mm1[31-24] &= (mm1[31-24] > mm2/m64[31-24]) \cdot mm1[31-24] : mm2/m64[31-24]; \\
    mm1[39-32] &= (mm1[39-32] > mm2/m64[39-32]) \cdot mm1[39-32] : mm2/m64[39-32]; \\
    mm1[47-40] &= (mm1[47-40] > mm2/m64[47-40]) \cdot mm1[47-40] : mm2/m64[47-40]; \\
    mm1[55-48] &= (mm1[55-48] > mm2/m64[55-48]) \cdot mm1[55-48] : mm2/m64[55-48]; \\
    mm1[63-56] &= (mm1[63-56] > mm2/m64[63-56]) \cdot mm1[63-56] : mm2/m64[63-56];
\end{align*}
\]

**Description:**
The PMAxUB instruction returns the maximum between the eight unsigned words in MM1 and MM2/Mem.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EEM = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH;
- #UD if CR0.EEM = 1;
- #NM if TS bit in CR0 is set.
- #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault;
- #AC for unaligned memory reference if the current privilege level is 3.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults  VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PMINSW: Packed Signed Integer Word Minimum

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F, EA, /r</td>
<td>PMINSW mm1, mm2/m64</td>
<td>Return the minimum words between MM2/Mem and MM1.</td>
</tr>
</tbody>
</table>

**Operation:**
- \( mm1[15-0] = (mm1[15-0] < mm2/m64[15-0]) \ ? mm1[15-0] : mm2/m64[15-0]; \)
- \( mm1[31-16] = (mm1[31-16] < mm2/m64[31-16]) \ ? mm1[31-16] : mm2/m64[31-16]; \)
- \( mm1[47-32] = (mm1[47-32] < mm2/m64[47-32]) \ ? mm1[47-32] : mm2/m64[47-32]; \)
- \( mm1[63-48] = (mm1[63-48] < mm2/m64[63-48]) \ ? mm1[63-48] : mm2/m64[63-48]; \)

**Description:**
The PMINSW instruction returns the minimum between the four signed words in MM1 and MM2/Mem.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**
- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception

To enable #AC exceptions, three conditions must be true (CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**
- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFFH;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set.
- #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**
- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault;
- #AC for unaligned memory reference if the current privilege level is 3.

**Additional Itanium System Environment Exceptions**
- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PMINUB: Packed Unsigned Integer Byte Minimum

**Operation:**

\[
\begin{align*}
\text{mm1}[7-0] &= (\text{mm1}[7-0] < \text{mm2/m64}[7-0]) \ ? \ \text{mm1}[7-0] : \text{mm2/m64}[7-0]; \\
\text{mm1}[15-8] &= (\text{mm1}[15-8] < \text{mm2/m64}[15-8]) \ ? \ \text{mm1}[15-8] : \text{mm2/m64}[15-8]; \\
\text{mm1}[23-16] &= (\text{mm1}[23-16] < \text{mm2/m64}[23-16]) \ ? \ \text{mm1}[23-16] : \text{mm2/m64}[23-16]; \\
\text{mm1}[31-24] &= (\text{mm1}[31-24] < \text{mm2/m64}[31-24]) \ ? \ \text{mm1}[31-24] : \text{mm2/m64}[31-24]; \\
\text{mm1}[39-32] &= (\text{mm1}[39-32] < \text{mm2/m64}[39-32]) \ ? \ \text{mm1}[39-32] : \text{mm2/m64}[39-32]; \\
\text{mm1}[47-40] &= (\text{mm1}[47-40] < \text{mm2/m64}[47-40]) \ ? \ \text{mm1}[47-40] : \text{mm2/m64}[47-40]; \\
\text{mm1}[55-48] &= (\text{mm1}[55-48] < \text{mm2/m64}[55-48]) \ ? \ \text{mm1}[55-48] : \text{mm2/m64}[55-48]; \\
\text{mm1}[63-56] &= (\text{mm1}[63-56] < \text{mm2/m64}[63-56]) \ ? \ \text{mm1}[63-56] : \text{mm2/m64}[63-56];
\end{align*}
\]

**Description:**
The PMINUB instruction returns the minimum between the eight unsigned words in MM1 and MM2/Mem.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EF = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFH;
- #UD if CR0.EF = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode;
- #PF(fault-code) for a page fault;
- #AC for unaligned memory reference if the current privilege level is 3.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults
- Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults
- VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PMOVMSKB: Move Byte Mask To Integer

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,D7,/r</td>
<td>PMOVMSKB r32, mm</td>
<td>Move the byte mask of MM to r32.</td>
</tr>
</tbody>
</table>

**Operation:**

r32[7] = mm[63];  r32[6] = mm[55];  
r32[5] = mm[47];  r32[4] = mm[39];  
r32[3] = mm[31];  r32[2] = mm[23];  
r32[1] = mm[15];  r32[0] = mm[7];  
r32[31-8] = 0x000000;

**Description:** The PMOVMSKB instruction returns a 8-bit mask formed of the most significant bits of each byte of its source operand.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;  
#SS(0) for an illegal address in the SS segment;  
#PF (fault-code) for a page fault;  
#UD if CR0.EM = 1;  
#NM if TS bit in CR0 is set;  
#MF if there is a pending FPU exception;  
#AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set;  
EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH;  
#UD if CR0.EM = 1;  
#NM if TS bit in CR0 is set;  
#MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode;  
#PF (fault-code) for a page fault;  
#AC for unaligned memory reference if the current privilege level is 3.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1
PMULHUW: Packed Multiply High Unsigned

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0Fh, E4, /r</td>
<td>PMULHUW mm1, mm2/m64</td>
<td>Multiply the packed unsigned words in MM1 register with the packed unsigned words in MM2/Mem, then store the high-order 16 bits of the results in MM1.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{mm1}[15-0] &= (\text{mm1}[15-0] \times \text{mm2/m64}[15-0])[31-16]; \\
\text{mm1}[31-16] &= (\text{mm1}[31-16] \times \text{mm2/m64}[31-16])[31-16]; \\
\text{mm1}[47-32] &= (\text{mm1}[47-32] \times \text{mm2/m64}[47-32])[31-16]; \\
\text{mm1}[63-48] &= (\text{mm1}[63-48] \times \text{mm2/m64}[63-48])[31-16];
\end{align*}
\]

**Description:** The PMULHUW instruction multiplies the four unsigned words in the destination operand with the four unsigned words in the source operand. The high-order 16 bits of the 32-bit intermediate results are written to the destination operand.

**Numeric Exceptions:** None.

**Protected Mode Exceptions**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions**

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault; #AC for unaligned memory reference if the current privilege level is 3.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PSADBW: Packed Sum of Absolute Differences

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,F6, /r</td>
<td>PSADBW mm1,mm2/m64</td>
<td>Absolute difference of packed unsigned bytes from MM2/Mem and MM1; these differences are then summed to produce a word result.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{temp1} &= \text{ABS}(\text{mm1}[7-0] - \text{mm2}/\text{m64}[7-0]); \\
\text{temp2} &= \text{ABS}(\text{mm1}[15-8] - \text{mm2}/\text{m64}[15-8]); \\
\text{temp3} &= \text{ABS}(\text{mm1}[23-16] - \text{mm2}/\text{m64}[23-16]); \\
\text{temp4} &= \text{ABS}(\text{mm1}[31-24] - \text{mm2}/\text{m64}[31-24]); \\
\text{temp5} &= \text{ABS}(\text{mm1}[39-32] - \text{mm2}/\text{m64}[39-32]); \\
\text{temp6} &= \text{ABS}(\text{mm1}[47-40] - \text{mm2}/\text{m64}[47-40]); \\
\text{temp7} &= \text{ABS}(\text{mm1}[55-48] - \text{mm2}/\text{m64}[55-48]); \\
\text{temp8} &= \text{ABS}(\text{mm1}[63-56] - \text{mm2}/\text{m64}[63-56]); \\
\text{mm1}[15:0] &= \text{temp1} + \text{temp2} + \text{temp3} + \text{temp4} + \text{temp5} + \text{temp6} + \text{temp7} + \text{temp8}; \\
\text{mm1}[31:16] &= 0x00000000; \\
\text{mm1}[47:32] &= 0x00000000; \\
\text{mm1}[63:48] &= 0x00000000; \\
\end{align*}
\]

**Description:**
The PSADBW instruction computes the absolute value of the difference of unsigned bytes for mm1 and mm2/m64. These differences are then summed to produce a word result in the lower 16-bit field; the upper 3 words are cleared.

The destination operand is a MMX technology register. The source operand can either be a MMX technology register or a 64-bit memory operand.

**Numeric Exceptions:** None

**Protected Mode Exceptions**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception.
PSADBW: Packed Sum of Absolute Differences (continued)

Virtual 8086 Mode Exceptions

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

Additional Itanium System Environment Exceptions

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
PSHUFW: Packed Shuffle Word

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,70, /r, ib</td>
<td>PSHUFW mm1, mm2/m64, imm8</td>
<td>Shuffle the words in MM2/Mem based on the encoding in imm8 and store in MM1.</td>
</tr>
</tbody>
</table>

**Operation:**

\[
\begin{align*}
\text{mm1}[15-0] & = (\text{mm2/m64} \gg (\text{imm8}[1-0] \times 16)) [15-0] \\
\text{mm1}[31-16] & = (\text{mm2/m64} \gg (\text{imm8}[3-2] \times 16)) [15-0] \\
\text{mm1}[47-32] & = (\text{mm2/m64} \gg (\text{imm8}[5-4] \times 16)) [15-0] \\
\text{mm1}[63-48] & = (\text{mm2/m64} \gg (\text{imm8}[7-6] \times 16)) [15-0]
\end{align*}
\]

**Description:**
The PSHUF instruction uses the imm8 operand to select which of the four words in MM2/Mem will be placed in each of the words in MM1. Bits 1 and 0 of imm8 encode the source for destination word 0 (MM1[15-0]), bits 3 and 2 encode for word 1, bits 5 and 4 encode for word 2, and bits 7 and 6 encode for word 3 (MM1[63-48]). Similarly, the two bit encoding represents which source word is to be used, e.g. an binary encoding of 10 indicates that source word 2 (MM2/Mem[47-32]) will be used.

**Numeric Exceptions:** None.

**Protected Mode Exceptions:**

#GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments; #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault; #AC for unaligned memory reference if the current privilege level is 3.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
Itanium Mem Faults VHPF Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault
3.14 Cacheability Control Instructions

This section describes the cacheability control instructions which enable an application writer to minimize data access latency and cache pollution.
MASKMOVQ: Byte Mask Write

## Operation:

if (mm2[7])  m64[edi] = mm1[7-0];
if (mm2[15]) m64[edi+1] = mm1[15-8];
if (mm2[23]) m64[edi+2] = mm1[23-16];
if (mm2[31]) m64[edi+3] = mm1[31-24];
if (mm2[39]) m64[edi+4] = mm1[39-32];
if (mm2[47]) m64[edi+5] = mm1[47-40];
if (mm2[55]) m64[edi+6] = mm1[55-48];
if (mm2[63]) m64[edi+7] = mm1[63-56];

## Description:

Data is stored from the mm1 register to the location specified by the di/edi register (using DS segment). The size of the store address depends on the address-size attribute. The most significant bit in each byte of the mask register mm2 is used to selectively write the data (0 = no write, 1 = write), on a per-byte basis. Behavior with a mask of all zeroes is as follows:

- No data will be written to memory. However, transition from FP to MMX technology state (if necessary) will occur, irrespective of the value of the mask.
- For memory references, a zero byte mask does not prevent addressing faults (i.e. #GP, #SS) from being signalled.
- Signalling of page faults (#PF) is implementation-specific.
- #UD, #NM, #MF, and #AC faults are signalled irrespective of the value of the mask.
- Signalling of breakpoints (code or data) is not guaranteed; different processor implementations may signal or not signal these breakpoints.
- If the destination memory region is mapped as UC or WP, enforcement of associated semantics for these memory types is not guaranteed (i.e. is reserved) and is implementation-specific. Dependency on the behavior of a specific implementation in this case is not recommended, and may lead to future incompatibility.

The Mod field of the ModR/M byte must be 11, or an Invalid Opcode Exception will result.

## Numeric Exceptions:

None

## Protected Mode Exceptions:

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF (fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #MF if there is a pending FPU exception;
- #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).
MASKMOVQ: Byte Mask Write (continued)

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1

**Comments:**

MASKMOVQ can be used to improve performance for algorithms which need to merge data on a byte granularity. MASKMOVQ should not cause a read for ownership; doing so generates unnecessary bandwidth since data is to be written directly using the byte-mask without allocating old data prior to the store. Similar to the Streaming SIMD Extension non-temporal store instructions, MASKMOVQ minimizes pollution of the cache hierarchy. MASKMOVQ implicitly uses weakly-ordered, write-combining stores (WC). See Section 3.6.1.9 for further information about non-temporal stores.

As a consequence of the resulting weakly-ordered memory consistency model, a fencing operation such as SFENCE should be used if multiple processors may use different memory types to read/write the same memory location specified by edi.

This instruction behaves identically to MMX instructions, in the presence of x87-FP instructions: transition from x87-FP to MMX technology (TOS=0, FP valid bits set to all valid).

MASMOVQ ignores the value of CR4.OSFXSR. Since it does not affect the new Streaming SIMD Extension state, they will not generate an invalid exception if CR4.OSFXSR = 0.
MOVNTPS: Move Aligned Four Packed Single-FP Non-temporal

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,2B, /r</td>
<td>MOVNTPS m128, xmm</td>
<td>Move 128 bits representing four packed SP FP data from XMM register to Mem, minimizing pollution in the cache hierarchy.</td>
</tr>
</tbody>
</table>

**Operation:**

\[ m_{128} = \text{xmm}; \]

**Description:**
The linear address corresponds to the address of the least-significant byte of the referenced memory data. This store instruction minimizes cache pollution.

**FP Exceptions:**
General protection exception if not aligned on 16-byte boundary, regardless of segment.

**Numeric Exceptions:**
None

**Protected Mode Exceptions:**
- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment;
- #PF(fault-code) for a page fault;
- #UD if CR0.EM = 1;
- #NM if TS bit in CR0 is set;
- #UD if CRCR4.OSFXSR(bit 9) = 0;
- #UD if CPUID.XMM(EDX bit 25) = 0.

**Real Address Mode Exceptions:**

Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #UD if CRCR4.OSFXSR(bit 9) = 0; #UD if CPUID.XMM(EDX bit 25) = 0.

**Virtual 8086 Mode Exceptions:**
Same exceptions as in Real Address Mode; #PF(fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**
- Itanium Reg Faults: Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults: VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Comments:**
MOVTNPS should be used when dealing with 16-byte aligned single-precision FP numbers. MOVNTPS minimizes pollution in the cache hierarchy. As a consequence of the resulting weakly-ordered memory consistency model, a fencing operation should be used if multiple processors may use different memory types to read/write the memory location. See Section 3.6.1.9 for further information about non-temporal stores.

The usage of Repeat Prefixes(F2H, F3H) with MOVNTPS is reserved. Different processor implementations may handle this prefix differently. Usage of this prefix with MOVNTPS risks incompatibility with future processors.
MOVNTQ: Move 64 Bits Non-temporal

**Opcode Instruction Description**

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,E7,/r</td>
<td>MOVNTQ m64, mm</td>
<td>Move 64 bits representing integer operands (8b, 16b, 32b) from MM register to memory, minimizing pollution within cache hierarchy.</td>
</tr>
</tbody>
</table>

**Operation:** 

m64 = mm;

**Description:** The linear address corresponds to the address of the least-significant byte of the referenced memory data. This store instruction minimizes cache pollution.

**Numeric Exceptions:** None

**Protected Mode Exceptions:**

- #GP(0) for an illegal memory operand effective address in the CS, DS, ES, FS or GS segments;
- #SS(0) for an illegal address in the SS segment; #PF (fault-code) for a page fault; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception; #AC for unaligned memory reference. To enable #AC exceptions, three conditions must be true(CR0.AM is set; EFLAGS.AC is set; current CPL is 3).

**Real Address Mode Exceptions:**

- Interrupt 13 if any part of the operand would lie outside of the effective address space from 0 to 0FFFFH; #UD if CR0.EM = 1; #NM if TS bit in CR0 is set; #MF if there is a pending FPU exception.

**Virtual 8086 Mode Exceptions:**

- Same exceptions as in Real Address Mode; #AC for unaligned memory reference if the current privilege level is 3; #PF (fault-code) for a page fault.

**Additional Itanium System Environment Exceptions**

- Itanium Reg Faults  Disabled FP Register Fault if PSR.dfl is 1, NaT Register Consumption Fault
- Itanium Mem Faults VHPT Data Fault, Data TLB Fault, Alternate Data TLB Fault, Data Page Not Present Fault, Data NaT Page Consumption Abort, Data Key Miss Fault, Data Key Permission Fault, Data Access Rights Fault, Data Access Bit Fault, Data Dirty Bit Fault

**Comments:** MOVNTQ minimizes pollution in the cache hierarchy. As a consequence of the resulting weakly-ordered memory consistency model, a fencing operation should be used if multiple processors may use different memory types to read/write the memory location. See Section 3.6.1.9 for further information about non-temporal stores.

MOVNTQ ignores the value of CR4.OSFXSR. Since it does not affect the new Streaming SIMD Extension state, they will not generate an invalid exception if CR4.OSFXSR = 0.
PREFETCH: Prefetch

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F,18,/1</td>
<td>PREFETCHT0 m8</td>
<td>Move data specified by address closer to the processor using the t0 hint.</td>
</tr>
<tr>
<td>0F,18,/2</td>
<td>PREFETCHT1 m8</td>
<td>Move data specified by address closer to the processor using the t1 hint.</td>
</tr>
<tr>
<td>0F,18,/3</td>
<td>PREFETCHT2 m8</td>
<td>Move data specified by address closer to the processor using the t2 hint.</td>
</tr>
<tr>
<td>0F,18,/0</td>
<td>PREFETCHNTA m8</td>
<td>Move data specified by address closer to the processor using the nta hint.</td>
</tr>
</tbody>
</table>

Operation: \texttt{fetch (m8)};

Description: If there are no excepting conditions, the prefetch instruction fetches the line containing the addresses byte to a location in the cache hierarchy specified by a locality hint. If the line is already present in the cache hierarchy at a level closer to the processor, no data movement occurs. The bits 5:3 of the ModR/M byte specify locality hints as follows:

- Temporal data (t0) - prefetch data into all cache levels.
- Temporal with respect to first level cache (t1) – prefetch data in all cache levels except 0th cache level.
- Temporal with respect to second level cache (t2) – prefetch data in all cache levels, except 0th and 1st cache levels.
- Non-temporal with respect to all cache levels (nta) – prefetch data into non-temporal cache structure.

Locality hints do not affect the functional behavior of the program. They are implementation dependent, and can be overloaded or ignored by an implementation. The prefetch instruction does not cause any exceptions (except for code breakpoints), does not affect program behavior and may be ignored by the implementation. The amount of data prefetched is implementation dependent. It will however be a minimum of 32 bytes. Prefetches to uncacheable memory (UC or WC memory types) will be ignored. Additional ModRM encodings, besides those specified above, are defined to be reserved and the use of reserved encodings risks future incompatibility.

Numeric Exceptions: None

Protected Mode Exceptions: None

Real Address Mode Exceptions: None

Virtual 8086 Mode Exceptions: None

Additional Itanium System Environment Exceptions: None

Comments: This instruction is merely a hint. If executed, this instruction moves data closer to the processor in anticipation of future use. The performance of these instructions in application code can be implementation specific. To achieve maximum speedup, code tuning might be necessary for each implementation. The non temporal hint also minimizes pollution of useful cache data.

PREFETCH instructions ignore the value of CR4.OSFXSR. Since they do not affect the new Streaming SIMD Extension state, they will not generate an invalid exception if CR4.OSFXSR = 0.
SFENCE: Store Fence

<table>
<thead>
<tr>
<th>Opcode</th>
<th>Instruction</th>
<th>Description</th>
</tr>
</thead>
<tbody>
<tr>
<td>0F AE /7</td>
<td>SFENCE</td>
<td>Guarantees that every store instruction that precedes in program order the store fence instruction is globally visible before any store instruction which follows the fence is globally visible.</td>
</tr>
</tbody>
</table>

**Operation:**

```c
while (!(preceding_stores_globally_visible)) wait();
```

**Description:**

Weakly ordered memory types can enable higher performance through such techniques as out-of-order issue, write-combining, and write-collapsing. Memory ordering issues can arise between a producer and a consumer of data and there are a number of common usage models which may be affected by weakly ordered stores: (1) library functions, which use weakly ordered memory to write results (2) compiler-generated code, which also benefit from writing weakly-ordered results, and (3) hand-written code. The degree to which a consumer of data knows that the data is weakly ordered can vary for these cases. As a result, the SFENCE instruction provides a performance-efficient way of ensuring ordering between routines that produce weakly-ordered results and routines that consume this data.

SFENCE uses the following ModRM encoding:

- Mod (7:6) = 11B
- Reg/Opcode (5:3) = 111B
- R/M (2:0) = 000B

All other ModRM encodings are defined to be reserved, and use of these encodings risks incompatibility with future processors.

**Numeric Exceptions:** None

**Protected Mode Exceptions:** None

**Real Address Mode Exceptions:** None

**Virtual 8086 Mode Exceptions:** None

**Additional Itanium System Environment Exceptions:** None

**Comments:** SFENCE ignores the value of CR4.OSFXSR. SFENCE will not generate an invalid exception if CR4.OSFXSR = 0
Entries in this index are described by the volume number and page or range of pages where the entries can be found. The volume number appears to the left of the colon. The page or range of pages appears to the right of the colon. A range of pages is separated by a hyphen.

**Numerics**

32-bit virtual addressing 2:59, 2:60
pointer “swizzling” model 2:60
sign-extension model 2:60
zero-extension model 2:59, 2:60

user NaT collection register (UNAT – AR 36) 1:26
application register state 1:19
application register model 1:21
ignored fields 1:20
ignored register 1:19, 1:20, 1:29
read-only register 1:20, 1:25, 2:106
reserved fields 1:20, 2:83
reserved register 1:19, 1:20
reserved value 1:20
arithmetic instructions 1:45, 3:324, 3:863
atomic operations 2:237
atomicity 2:57, 2:77, 2:238

**A**

abort 2:79, 2:406
interruption priorities 2:92
machine check abort 2:44, 2:489-2:491
PSR.mc bit is 0 2:82
reset abort 2:481
abort handling 2:491
access rights, segment descriptor 3:600
acquire semantics 1:66, 2:70, 2:238, 2:375
address space model 2:425, 2:429
addressable units 1:30
advanced load address table (See ALAT) 1:57, 1:62, 1:144, 2:416
data speculation 1:57, 1:58, 1:62, 1:144, 2:445
related instructions 1:58
application programming model 1:41
application register
  compare and exchange value register (CCV – AR 32) 1:26
  epilog count register (EC – AR 66) 1:27
  floating-point status register (FPSR – AR 40) 1:26
  IA-32 time stamp counter (TSC) 1:26, 1:110, 2:27, 2:469
  interval time counter (ITC – AR 44) 1:26
  kernel registers (KR 0-7 – AR 0-7) 1:25
  loop count register (LC – AR 65) 1:27
  previous function state (PFS – AR 64) 1:27
  register stack configuration register (RSC – AR 16) 1:25
  RSE backing store pointer (BSP – AR 17) 1:25
  RSE NaT collection register (RNAT – AR 19) 1:26
  firmware boot flow 2:481
  boot sequence 2:9, 2:481
  boot flow 2:10, 2:481
  bootstrap processor (BSP) 2:481
  br.ia instruction 1:10, 1:103, 1:105, 2:458
  be bit 1:31
  PSR.be 2:86, 3:343, 3:349
  RSC.be 2:123
  biased exponent 3:823-3:826, 3:829
  bit field and shift instructions 1:46, 1:47
  boot flow 2:10, 2:481
  firmware boot flow 2:481
  boot sequence 2:9, 2:481
  boot flow 2:10, 2:481
  bootstrap processor (BSP) 2:481
Index

branch prediction instructions 1:15, 1:70, 1:71, 3:320
branch prediction hints 1:70, 1:168
modulo-scheduled loop support 1:68
brl instruction 2:449
bundles 1:11, 1:31, 1:32, 1:34, 1:132, 1:133, 3:257
byte ordering 1:30, 1:31

C

cache synchronization 2:69
cache write policy attribute 2:66
cacheability and coherency attribute 2:65
cacheable 2:63, 2:64, 2:65, 2:66
cacheable pages 2:66
uncacheable pages 2:66
causality 2:390
obeying causality 2:390
character strings 1:76
clrrrb.pr instruction 1:68, 2:422, 3:316, 3:321
coalescing attribute 2:66
coaesed pages 2:67
coherency 1:125, 2:65, 2:238, 3:576, 3:735
compare types 1:49, 1:50
normal compare 1:49
parallel compare 1:73, 1:164, 1:165, 3:327
unconditional compare 1:49
computational models 1:200
constant register 2:415
countext switching 1:43, 2:144, 2:422, 2:424, 2:477

data access bit 2:57, 2:59, 2:90, 2:91
data breakpoint register matching 2:246
DBR.addr 2:246
DBR.mask 2:247
trap code B bits 2:247
data breakpoint registers (DBR) 2:133, 2:134

data space switching 2:424
non-local control transfer 2:422
performance monitor 2:144, 2:145
RSE backing store 1:26, 2:130
thread switch within the same address space 2:424
control flow optimization 1:155
control flow optimizations 1:163
multiple values for one variable or register 1:165
multiway branches 1:165
parallel compares 1:163-1:165
control registers (CR) 2:16
banked general registers 2:34, 2:35, 2:61
default control register (DCR – CR0) 2:25, 2:26
external interrupt control registers 2:34, 2:104, 2:105, 2:216, 2:467
global control registers 2:25
interrupt control registers 2:29
interrupt faulting address (IFA – CR20) 2:31
interrupt function state (IFS – CR23) 2:33
interrupt hash address (IHA – CR25) 2:34
interrupt immediate (IIM – CR24) 2:34
interrupt instruction bundle pointer (IIP – CR19) 2:30
interrupt instruction previous address (IIPA – CR22) 2:32, 2:33
interrupt processor status register (IPSR – CR16) 2:29
interrupt status register (ISR – CR17) 2:29
interrupt status register fields 2:29
interrupt TLB insertion register (ITIR – CR21) 2:32
interrupt vector address (IVA – CR2) 2:28
interval time counter (ITC – AR44) 2:27
interval timer match register (ITM – CR1) 2:27
ITIR fields 2:32
page table address (PTA – CR8) 2:28
control registers, moving values to and from 3:636
control speculative 1:12, 2:415, 2:445
corrected machine check (CMC) 2:109, 2:283, 2:489, 2:490
corrected machine check interrupt (CMCI) 2:489
CPUID registers 1:29
cross-modifying code 2:400
current frame marker (CFM) 1:19, 1:23, 1:36, 1:42, 2:13, 2:121
size of frame (sof) 1:42
size of locals (sol) 1:42
cycles per instructions (CPI) 2:477

D

data access bit 2:57, 2:59, 2:90, 2:91
data breakpoint register matching 2:246
DBR.addr 2:246
DBR.mask 2:247
trap code B bits 2:247
data breakpoint registers (DBR) 2:133, 2:134

Index-2
Index

data debug 2:57, 2:90, 2:91
data dependencies 1:57, 1:140-1:142, 2:13, 2:378, 2:383
data dependency 1:14, 1:141, 1:142, 2:13, 2:382-2:384
data key miss fault 2:128, 2:149
data key permission 2:57, 2:59, 2:90, 2:91
data NaT page consumption 2:57, 2:59, 2:72, 2:89-2:91
data page not present 2:57, 2:59, 2:89, 2:91
data prefetch
    load instructions 1:52
    semaphore instructions 1:53, 3:303
data serialization 2:14, 2:15, 3:335, 3:336
    dependency violation 1:36, 1:38, 2:448
    instruction group 1:35-1:39, 1:67
    register dependencies 1:35-1:39
    WAR dependency 1:38, 3:350
division operations
    double precision – divide 1:199
    double precision – square root 1:99
DMA 1:17, 2:403
    edge sensitive interrupt messages 2:114
popping 3:665
    popping on return from interrupt 3:579
    pushing 3:673
    pushing on interrupts 3:565
    saving 3:695
    status flags 3:418, 3:588, 3:704, 3:729
exception deferral 1:55, 2:45, 2:90, 2:91
    combined hardware/software deferral 2:443, 2:444
    exception deferral of control speculative loads 2:443
    hardware-only deferral 2:443, 2:444
    software-only deferral 2:443, 2:444
exception indicator 2:72
exception qualification 2:89
execution unit type 1:24, 1:32, 1:33, 3:257
extended instructions 1:34, 3:257
extensible firmware interface (See EFI) 2:451, 2:481
external interrupt (INT) 2:92, 2:101
    control register usage examples 2:467
    external (I/O) devices 2:98
    external interrupt (INT)
    external interrupt architecture 2:463
    external interrupt masking 2:103, 2:464
    external interrupt sampling 2:104
    external interrupt states 2:101
    inactive 2:465
    in-service/none pending 2:466
    in-service/one pending 2:466
    internal processor interrupts 2:99
    interrupt acknowledge (INTA) cycle 2:114
    interrupt enabling 2:102
    interrupt masking 2:103
    interrupt priorities 2:102, 2:463
    interrupt registers 2:16
    interrupt sources 2:98, 2:104, 2:114, 2:464
    locally connected devices 2:98
    pending 2:82, 2:99, 2:102, 2:103, 2:465, 2:466
    external interrupt control registers 2:34, 2:104, 2:105, 2:216, 2:464, 2:467
    Local ID (LID – CR64) 2:105
    external task priority (XTP) 2:110, 2:114
    XTP cycle 2:114
    XTP register 2:465

E
edge- and level-sensitive interrupts 2:114
    boot services 2:483, 2:488
    EFI boot manager 2:482
    EFI procedure calls 2:488
    EFI system partition 2:482, 2:483
    runtime services 2:488
EFLAGS register
    condition codes 3:415, 3:458, 3:463
    flags affected by instructions 3:366
    loading 3:599
    fault suppression 2:88
    fence 1:66, 2:69, 2:70
    operations 1:66, 2:69, 2:70
    semantics 1:66, 2:69, 2:70, 2:238, 2:239
<table>
<thead>
<tr>
<th>Index</th>
<th>Intel® Itanium™ Architecture Software Developer’s Manual, Rev 2.0</th>
</tr>
</thead>
<tbody>
<tr>
<td>firmware address space</td>
<td>2:258</td>
</tr>
<tr>
<td>firmware entrypoint</td>
<td>2:283</td>
</tr>
<tr>
<td>firmware entrypoints</td>
<td>2:257</td>
</tr>
<tr>
<td>firmware interface table (FIT)</td>
<td>2:261</td>
</tr>
<tr>
<td>firmware model</td>
<td>2:482</td>
</tr>
<tr>
<td>firmware procedure</td>
<td>2:283</td>
</tr>
<tr>
<td>floating-point applications</td>
<td>1:193</td>
</tr>
<tr>
<td>execution bandwidth</td>
<td>1:194</td>
</tr>
<tr>
<td>execution latency</td>
<td>1:193</td>
</tr>
<tr>
<td>memory bandwidth</td>
<td>1:195</td>
</tr>
<tr>
<td>memory latency</td>
<td>1:194</td>
</tr>
<tr>
<td>performance limiters</td>
<td>1:193</td>
</tr>
<tr>
<td>floating-point architecture</td>
<td>1:11, 1:15, 1:77</td>
</tr>
<tr>
<td>arithmetic instructions</td>
<td>1:90, 3:324, 3:863</td>
</tr>
<tr>
<td>integer multiply and add instructions</td>
<td>1:93, 1:94</td>
</tr>
<tr>
<td>memory access instructions</td>
<td>1:83</td>
</tr>
<tr>
<td>non-arithmetic instructions</td>
<td>1:92</td>
</tr>
<tr>
<td>register to/from general register transfer instructions</td>
<td>1:89</td>
</tr>
<tr>
<td>floating-point programming model</td>
<td>1:77</td>
</tr>
<tr>
<td>data types and formats</td>
<td>1:77</td>
</tr>
<tr>
<td>floating-point register encodings</td>
<td>1:78</td>
</tr>
<tr>
<td>floating-point register format</td>
<td>1:77, 1:78</td>
</tr>
<tr>
<td>floating-point status register</td>
<td>1:24, 1:26, 1:80, 1:81, 1:93</td>
</tr>
<tr>
<td>real types</td>
<td>1:77</td>
</tr>
<tr>
<td>floating-point status register (FPSR)</td>
<td>1:26, 1:37, 1:80, 1:93, 2:451</td>
</tr>
<tr>
<td>floating-point system software</td>
<td></td>
</tr>
<tr>
<td>floating-point exception handling</td>
<td>2:451, 2:453</td>
</tr>
<tr>
<td>FP precision</td>
<td>1:197</td>
</tr>
<tr>
<td>FP subfield handling</td>
<td>1:203</td>
</tr>
<tr>
<td>FPU</td>
<td></td>
</tr>
<tr>
<td>checking for pending FPU exceptions</td>
<td>3:734</td>
</tr>
<tr>
<td>constants</td>
<td>3:487</td>
</tr>
<tr>
<td>existence of</td>
<td>3:428</td>
</tr>
<tr>
<td>floating-point format</td>
<td>3:822, 3:823</td>
</tr>
<tr>
<td>initialization</td>
<td>3:481</td>
</tr>
<tr>
<td>floating-point register (FR)</td>
<td></td>
</tr>
<tr>
<td>high FP reg fault</td>
<td>3:360</td>
</tr>
<tr>
<td>low FP reg fault</td>
<td>3:359</td>
</tr>
<tr>
<td>floating-point register set</td>
<td>1:195</td>
</tr>
<tr>
<td>floating-point software assistance (FP SWA)</td>
<td>2:451</td>
</tr>
<tr>
<td>SWA faults</td>
<td>2:451, 2:452, 2:454</td>
</tr>
<tr>
<td>SWA traps</td>
<td>2:451, 2:452, 2:453</td>
</tr>
<tr>
<td>FPU control word</td>
<td></td>
</tr>
<tr>
<td>loading</td>
<td>3:489, 3:491</td>
</tr>
<tr>
<td>RC field</td>
<td>3:482, 3:487, 3:521</td>
</tr>
<tr>
<td>restoring</td>
<td>3:508</td>
</tr>
<tr>
<td>saving</td>
<td>3:510, 3:526</td>
</tr>
<tr>
<td>storing</td>
<td>3:524</td>
</tr>
<tr>
<td>FPU data pointer</td>
<td>3:491, 3:508, 3:510, 3:526</td>
</tr>
<tr>
<td>FPU flag, CPUID instruction</td>
<td>3:428</td>
</tr>
<tr>
<td>FPU instruction pointer</td>
<td>3:491, 3:508, 3:510, 3:526</td>
</tr>
<tr>
<td>FPU last opcode</td>
<td>3:491, 3:508, 3:510, 3:526</td>
</tr>
<tr>
<td>FPU status word</td>
<td></td>
</tr>
<tr>
<td>condition code flags</td>
<td>3:460, 3:476, 3:536, 3:538, 3:541</td>
</tr>
<tr>
<td>FPU flags affected by instructions</td>
<td>3:366</td>
</tr>
<tr>
<td>loading</td>
<td>3:491</td>
</tr>
<tr>
<td>restoring</td>
<td>3:508</td>
</tr>
<tr>
<td>saving</td>
<td>3:510, 3:526, 3:528</td>
</tr>
<tr>
<td>TOP field</td>
<td>3:480</td>
</tr>
<tr>
<td>FPU tag word</td>
<td>3:491, 3:508, 3:510, 3:526</td>
</tr>
<tr>
<td>G</td>
<td></td>
</tr>
<tr>
<td>gate interception</td>
<td>2:215</td>
</tr>
<tr>
<td>general register (GR)</td>
<td></td>
</tr>
<tr>
<td>NaT bit</td>
<td>1:21, 1:134, 1:146, 1:147</td>
</tr>
<tr>
<td>global TLB purge operations</td>
<td>2:69</td>
</tr>
<tr>
<td>H</td>
<td></td>
</tr>
<tr>
<td>hardware debugger</td>
<td>2:136</td>
</tr>
<tr>
<td>I</td>
<td></td>
</tr>
<tr>
<td>i bit</td>
<td></td>
</tr>
<tr>
<td>I/O port space model</td>
<td>2:240, 2:241</td>
</tr>
<tr>
<td>physical I/O port addressing</td>
<td>2:243</td>
</tr>
<tr>
<td>virtual I/O port addressing</td>
<td>2:241</td>
</tr>
<tr>
<td>IA-32 application execution model</td>
<td>1:103</td>
</tr>
<tr>
<td>IA-32 operating mode transitions</td>
<td>1:106</td>
</tr>
<tr>
<td>instruction set execution in the Itanium architecture</td>
<td>1:104</td>
</tr>
<tr>
<td>instruction set modes</td>
<td>1:103</td>
</tr>
<tr>
<td>instruction set transitions</td>
<td>1:105, 2:215, 2:240</td>
</tr>
<tr>
<td>IA-32 application register state model</td>
<td>1:107</td>
</tr>
<tr>
<td>IA-32 application EFLAG register</td>
<td>1:116</td>
</tr>
<tr>
<td>IA-32 floating-point registers</td>
<td>1:117, 1:118</td>
</tr>
<tr>
<td>IA-32 general purpose registers</td>
<td>1:107, 1:108, 1:110</td>
</tr>
<tr>
<td>IA-32 instruction pointer</td>
<td>1:111</td>
</tr>
<tr>
<td>IA-32 MMX technology registers</td>
<td>1:122</td>
</tr>
<tr>
<td>IA-32 segment registers</td>
<td>1:111</td>
</tr>
<tr>
<td>IA-32 streaming SIMD extension registers</td>
<td>1:109, 1:122</td>
</tr>
<tr>
<td>IA-32 application support</td>
<td>2:250</td>
</tr>
<tr>
<td>procedure calls between Itanium and IA-32 instruction sets</td>
<td>2:459</td>
</tr>
<tr>
<td>transitioning between Itanium and IA-32 instruction sets</td>
<td>2:457</td>
</tr>
<tr>
<td>IA-32 architecture</td>
<td>1:5, 1:17, 2:5, 3:5, 3:359</td>
</tr>
<tr>
<td>IA-32 architecture handlers</td>
<td>2:460</td>
</tr>
<tr>
<td>IA-32 vectors that need Itanium-based OS support</td>
<td>1:2461</td>
</tr>
<tr>
<td>shared Itanium/IA-32 exception vectors</td>
<td>2:460</td>
</tr>
<tr>
<td>unique IA-32 exception vectors</td>
<td>2:460</td>
</tr>
<tr>
<td>unique Itanium exception vectors</td>
<td>2:460</td>
</tr>
<tr>
<td>IA-32 compatible bus transactions</td>
<td>2:251</td>
</tr>
<tr>
<td>IA-32 current privilege level</td>
<td>2:218</td>
</tr>
<tr>
<td>IA-32 fault and trap handling</td>
<td>2:215</td>
</tr>
<tr>
<td>IA-32 faults</td>
<td>3:359</td>
</tr>
<tr>
<td>IA-32 floating-point exceptions</td>
<td>2:456</td>
</tr>
<tr>
<td>IA-32 GPFault</td>
<td>3:359</td>
</tr>
<tr>
<td>IA-32 I/O instructions</td>
<td>2:244</td>
</tr>
<tr>
<td>IA-32 instruction behavior</td>
<td>2:215, 2:227</td>
</tr>
<tr>
<td>IA-32 instruction format</td>
<td>3:360</td>
</tr>
<tr>
<td>IA-32 instruction summary</td>
<td>2:228</td>
</tr>
<tr>
<td>IA-32 interruption</td>
<td>2:94, 2:95, 2:248</td>
</tr>
<tr>
<td>IA-32 interruption priorities and classes</td>
<td>2:95</td>
</tr>
<tr>
<td>IA-32 interruption vector</td>
<td>2:189, 2:248</td>
</tr>
<tr>
<td>IA-32 memory ordering</td>
<td>2:238, 2:392</td>
</tr>
<tr>
<td>IA-32 MMX technology instructions</td>
<td>1:122, 3:747</td>
</tr>
<tr>
<td>IA-32 numeric exception model</td>
<td>2:250</td>
</tr>
<tr>
<td>IA-32 physical memory references</td>
<td>2:235</td>
</tr>
<tr>
<td>IA-32 privileged system resources</td>
<td>2:215</td>
</tr>
<tr>
<td>IA-32 processes during a context switch</td>
<td>2:226</td>
</tr>
<tr>
<td>entering IA-32 processes</td>
<td>2:226</td>
</tr>
<tr>
<td>exiting IA-32 processes</td>
<td>2:227</td>
</tr>
<tr>
<td>IA-32 segmentation</td>
<td>1:124, 2:333</td>
</tr>
<tr>
<td>IA-32 streaming SIMD extension instructions</td>
<td>1:123, 3:811</td>
</tr>
<tr>
<td>IA-32 system and control register behavior</td>
<td>2:215</td>
</tr>
<tr>
<td>IA-32 system EFLAG register</td>
<td>2:219</td>
</tr>
<tr>
<td>IA-32 system environment</td>
<td>1:5, 1:9, 1:10, 1:17, 2:5, 2:9, 3:5</td>
</tr>
<tr>
<td>IA-32 system register mapping</td>
<td>2:216</td>
</tr>
<tr>
<td>IA-32 system registers</td>
<td>2:222</td>
</tr>
<tr>
<td>IA-32 control registers</td>
<td>2:222</td>
</tr>
<tr>
<td>IA-32 debug registers</td>
<td>2:225</td>
</tr>
<tr>
<td>IA-32 machine check registers</td>
<td>2:226</td>
</tr>
<tr>
<td>IA-32 memory type range registers (MTRRs)</td>
<td>2:225</td>
</tr>
<tr>
<td>IA-32 model specific and test registers</td>
<td>2:225</td>
</tr>
<tr>
<td>IA-32 performance monitor registers</td>
<td>2:226</td>
</tr>
<tr>
<td>IA-32 system segment registers</td>
<td>2:217</td>
</tr>
<tr>
<td>IA-32 TLB forward progress requirements</td>
<td>2:234</td>
</tr>
<tr>
<td>IA-32 trap code</td>
<td>2:189</td>
</tr>
<tr>
<td>IA-32 usage of Itanium registers</td>
<td>1:126</td>
</tr>
<tr>
<td>ALAT</td>
<td>1:126</td>
</tr>
<tr>
<td>NaN/NaNVal response for IA-32 instructions</td>
<td>1:126</td>
</tr>
<tr>
<td>register stack engine</td>
<td>1:126, 3:597</td>
</tr>
<tr>
<td>IA-32 virtual memory references</td>
<td>2:234</td>
</tr>
<tr>
<td>protection keys</td>
<td>2:234</td>
</tr>
<tr>
<td>region identifiers</td>
<td>2:234</td>
</tr>
<tr>
<td>TLB access bit</td>
<td>2:234</td>
</tr>
<tr>
<td>TLB dirty bit</td>
<td>2:234</td>
</tr>
<tr>
<td>IA-32 virtual memory support</td>
<td>2:215</td>
</tr>
<tr>
<td>IEEE considerations</td>
<td>1:94</td>
</tr>
<tr>
<td>additions beyond the IEEE standard</td>
<td>1:100</td>
</tr>
<tr>
<td>arithmetic operations</td>
<td>1:100, 3:826</td>
</tr>
<tr>
<td>floating-point interruptions</td>
<td>1:94</td>
</tr>
<tr>
<td>inexact</td>
<td>1:97, 1:99, 2:453, 2:456</td>
</tr>
<tr>
<td>integer invalid operations</td>
<td>1:100</td>
</tr>
<tr>
<td>mandated operations deferred to software</td>
<td>1:100</td>
</tr>
<tr>
<td>NaNs</td>
<td>1:78, 1:100</td>
</tr>
<tr>
<td>overflow</td>
<td>1:97, 1:98, 2:452, 2:455</td>
</tr>
<tr>
<td>tinnness</td>
<td>1:99</td>
</tr>
<tr>
<td>underflow</td>
<td>1:97, 1:99, 2:452, 2:456</td>
</tr>
<tr>
<td>IEEE floating-point exception filter</td>
<td>2:451, 2:454</td>
</tr>
<tr>
<td>denormal/normal operand exception (fault)</td>
<td>2:455</td>
</tr>
<tr>
<td>divide by zero exception (fault)</td>
<td>2:455</td>
</tr>
<tr>
<td>inexact exception (trap)</td>
<td>2:456</td>
</tr>
<tr>
<td>invalid operation exception (fault)</td>
<td>2:455</td>
</tr>
<tr>
<td>overflow exception (trap)</td>
<td>2:455</td>
</tr>
<tr>
<td>underflow exception (trap)</td>
<td>2:456</td>
</tr>
<tr>
<td>ANSI/IEEE-754 standard compliant</td>
<td>1:193</td>
</tr>
<tr>
<td>if-conversion</td>
<td>1:157</td>
</tr>
<tr>
<td>illegal dependency fault</td>
<td>1:38, 2:149, 2:448</td>
</tr>
<tr>
<td>illegal operation fault</td>
<td>1:19, 1:20, 1:39, 2:149, 3:257</td>
</tr>
<tr>
<td>implicit serialization</td>
<td>2:13</td>
</tr>
<tr>
<td>indefinite</td>
<td>description of 3:827</td>
</tr>
<tr>
<td>real 3:829</td>
<td></td>
</tr>
<tr>
<td>infinity, floating-point format</td>
<td>3:826</td>
</tr>
<tr>
<td>in-flight resources</td>
<td>2:15</td>
</tr>
<tr>
<td>INIT flows</td>
<td>2:491</td>
</tr>
<tr>
<td>initialization event (INIT)</td>
<td>2:489</td>
</tr>
<tr>
<td>initialization interrupts</td>
<td>2:80, 2:98, 2:406</td>
</tr>
<tr>
<td>PALE_INIT</td>
<td>2:80, 2:92</td>
</tr>
<tr>
<td>inserting/purging of translations</td>
<td>2:425</td>
</tr>
<tr>
<td>instruction breakpoint register matching</td>
<td>2:247</td>
</tr>
<tr>
<td>IBR.addr</td>
<td>2:247</td>
</tr>
<tr>
<td>IBR.mask</td>
<td>2:247</td>
</tr>
<tr>
<td>instruction breakpoints (IBR)</td>
<td>2:133, 2:134</td>
</tr>
<tr>
<td>instruction classes</td>
<td>3:336, 3:350, 3:351</td>
</tr>
<tr>
<td>instruction dependencies</td>
<td>1:140</td>
</tr>
<tr>
<td>control dependencies</td>
<td>1:65, 1:140, 2:384</td>
</tr>
<tr>
<td>data dependencies</td>
<td>1:57, 1:141-1:143, 2:13</td>
</tr>
<tr>
<td>instruction encoding</td>
<td>1:32</td>
</tr>
<tr>
<td>bundles</td>
<td>1:11, 1:31, 1:32, 1:34, 1:132, 1:133, 3:257</td>
</tr>
<tr>
<td>instruction slots</td>
<td>1:32, 1:33, 3:257</td>
</tr>
<tr>
<td>template</td>
<td>1:32-1:34, 1:133, 3:257, 3:258</td>
</tr>
<tr>
<td>instruction field names</td>
<td>3:259, 3:262</td>
</tr>
<tr>
<td>instruction format</td>
<td>1:132, 3:260</td>
</tr>
<tr>
<td>instruction interception</td>
<td>2:215</td>
</tr>
<tr>
<td>instruction pointer (IP)</td>
<td>1:19, 1:22, 2:84, 2:408, 2:479</td>
</tr>
</tbody>
</table>
### Instruction Serialization


### Instruction Set

- architecture (ISA) 1:5, 2:5, 3:5
- set features 1:11
- transition model overview 1:10
- slots 1:32, 1:33, 3:257
- slot mapping 1:33, 3:258
- stream alignment 1:167
- fetching 1:167
- type 1:32, 3:257, 3:697
  - ALU (A) 3:258
  - branch (B) 3:258
  - floating-point (F) 3:258
  - integer (I) 1:133, 3:258
  - memory (M) 1:133, 3:258
- instruction/data TLB miss 2:57-2:59
- instruction computation instructions 1:44
  - 32-bit addresses and integers 1:46
  - arithmetic instructions 1:45, 3:324, 3:863
  - bit field and shift instructions 1:46, 1:47
  - large constants 1:47
  - logical instructions 1:45
- integer/floating-point conversion 1:203
- processor interrupt (IPI) 2:97, 2:99, 2:110, 2:471
  - interrupt message 2:111, 2:491
  - data fields 2:111, 2:113
  - interrupt acknowledge (INTA) 2:110
  - execution environment 2:407
  - heavy weight interruptions 2:411, 2:413
  - interruption handling 2:79, 2:82, 2:85, 2:86, 2:410
  - interruption register state 2:408
  - lightweight interruptions 2:410
  - nested interruptions 2:413
  - resource serialization 2:409, 2:410
- interruption model 2:82, 2:247
- interruption priorities 2:92, 2:95
- registers 2:216, 2:406, 2:408
- vector address (IVA) 2:406
- vector table (IVT) 2:79, 2:96, 2:406
- vectors 2:85, 2:96, 2:147, 2:151, 2:406
- vector definition 2:148
- aborts 2:79, 2:89, 2:92, 2:406
- interruption handling during instruction execution 2:82
- interruption programming model 2:81
- IVA-based interruption 2:85, 2:96, 2:406
- PAL-based interruption 2:85, 2:405
- IPI ordering 2:113
- ISR setting 2:147
- Itanium architecture 1:1, 1:5, 1:9
- data mem faults 3:360
- instruction set 1:17, 3:430, 3:597, 3:598
- expressing parallelism 1:132
- format 3:464, 3:539
- interruption set 1:17, 3:430, 3:597, 3:598
- syntax 1:132, 3:336
- interruption mem faults 3:360
- interruption system environment 1:5, 1:9, 1:10, 1:17, 2:5, 2:9, 2:10, 2:11
- Itanium-based firmware 1:5, 1:17, 2:5, 3:5

### J

- jmpe instruction 1:10, 1:103, 1:105

### L

- Lamport's algorithm 2:397, 2:398
- ld.c.clr instruction 1:51, 1:60-1:62, 2:73, 2:74
- ld.c.clr.acq instruction 1:51, 1:60-1:62, 2:73, 2:74
ldf.a instruction 1:51, 1:57, 1:61, 1:62
ldf.c instruction 1:51, 1:57, 1:62, 2:73
ldf.s instruction 1:51, 1:54, 1:56, 2:415
ldf.sa instruction 1:51, 1:61, 1:62, 2:415
ldfp.a instruction 1:51, 1:57, 1:59, 1:61, 1:62
ldfp.c instruction 1:51, 1:57
ldfp.c.clr instruction 1:51, 1:61, 1:62
ldfp.c.nc instruction 1:51, 1:61, 1:62
ldfp.s instruction 1:51, 1:54, 1:56, 2:415
ldfp.sa instruction 1:51, 1:61, 1:62, 2:415
level sensitive external interrupts 2:114
loadrs field 1:44, 2:122, 2:125
logical instructions 1:45
long branch handler 2:447
loop support 1:68, 1:171, 1:174
capacity limitations 1:185
conflicts in the ALAT 1:186
counted loop 1:69, 1:171, 1:176, 1:177
counted loop branches 1:176
epilog 1:68, 1:174, 1:179
epilog count register (EC) 1:27
explicit prolog and epilog 1:190
implementing reductions 1:189
induction variable 1:172
iteration interval (II) 1:173
kernel 1:68, 1:174, 1:175, 1:179
kernel iteration 1:174
kernel loop 1:174
loop count application register (LC) 1:69, 1:171
loop unrolling 1:137, 1:172, 1:187
loop unrolling prior to software pipelining 1:187
loops with predicated instructions 1:182
multiple-exit loops 1:183
prolog 1:68, 1:174, 1:179
redundant load elimination in loops 1:192
register rotation 1:15, 1:174, 1:175
software pipelining and advanced loads 1:185
software pipelining considerations 1:185
software-pipelined loop branches 1:176, 1:177
source iteration 1:174
source loop 1:174
while loop 1:70, 1:178, 1:180, 3:686, 3:687
M
machine check abort (See MCA)
machine check abort flows
  machine check abort handling in OS 2:491
  machine check handling in PAL 2:490
  machine check handling in SAL 2:490
machine check aborts 2:481
machine checks 2:268
major opcode 1:34, 3:257-3:259
master boot record 2:483
mc bit
  PSR.mc 2:82, 2:85, 2:86, 2:102, 2:408, 3:345, 3:349
MCA 2:481
PALE_CHECK 2:79, 2:92
memory acceptance fence 2:473
memory access control 1:204
  allocation control 1:63, 1:205
data prefetch 1:205
load-pair instructions 1:204
memory access instructions 1:51, 1:62, 2:375
memory access ordering 1:65, 2:70
memory ordering instructions 1:66
memory ordering rules 1:66
memory addressing model 1:30, 1:123
memory alignment 2:236
memory attribute 2:45, 2:63, 2:64, 2:73-2:75, 3:429
effects of memory attributes on advanced/check loads 2:73
effects of memory attributes on memory reference instructions 2:72
memory attribute transition 2:74
physical addressing memory attribute 2:64
virtual addressing memory attribute 2:63, 2:74
memory dependency 1:35, 2:69
read-after-write 1:35, 1:38, 1:65, 2:69
write-after-read 1:35, 1:38, 1:65, 2:69
write-after-write 1:35, 1:38, 1:65, 2:69
memory endianness 1:124
memory fence 1:66, 2:392
memory fences 2:113, 2:378
memory hierarchy 1:63
  hierarchy control and hints 1:62
memory consistency 1:65, 3:924-3:926
memory mapped I/O model 2:241, 2:474
memory model 1:123, 2:233
  acquire semantics 1:66, 2:70, 2:238, 2:375
memory ordering interactions 1:125
memory ordering model 2:241, 2:378, 2:392
memory ordering semantics 1:66, 2:378, 2:424
release semantics 1:66, 2:69, 2:238, 2:239, 2:375
memory ordering fence 1:66
memory reference 1:139, 1:140, 2:38
memory synchronization 2:392
mf instruction 1:66, 2:113, 2:378
Min/Max/AMin/AMax 1:202
multimedia instructions 1:11, 1:16, 1:41, 1:72
data arrangement 1:74
parallel arithmetic 1:72, 1:73
parallel shifts 1:73
multimedia support 1:16
multiple address space (MAS) 1:16, 2:37, 2:425, 2:426
multiple status fields 1:200
multiply-add instruction 1:198
multiprocessor (MP)
multiprocessor instruction cache coherency 2:238
multiprocessor TLB coherency 2:235

N
NaN
description of 3:824, 3:826
encoding of 3:825, 3:829
operating on 3:827
SNaNs vs. QNaNs 3:826
testing for 3:536
NaN (not a thing) 1:131
NaT page consumption fault 2:72
NaTPage attribute 2:72
NaNval (not a thing value) 1:22
non-access instructions 2:87
non-cacheable memory 2:69
non-programmer-visible state 2:378
non-speculative memory references 1:139, 2:62
data prefetch hint 1:140
loads from memory 1:139
stores to memory 1:139
non-temporal hint 1:205
NOP instruction 3:649
no-recovery model 2:88, 2:89
normalized finite number 3:823, 3:825
normalized numbers 1:79, 3:822, 3:823, 3:825
not a thing attribute (NaTPage) 2:72

O
OLR 2:283
operand screening support 1:202
operating environments 1:9, 1:10
optimization of memory references 1:148
data interference 1:149, 1:150
loop optimization 1:152
minimizing check code 1:152
optimizing code size 1:150
using post-increment loads and stores 1:151
ordered instruction 2:376, 2:380
ordered cacheable operations 2:389
ordering semantics 1:66, 2:69, 2:70, 2:381
unordered 1:66, 2:69, 2:70, 2:238, 2:375, 2:380
OS boot flow sample code 2:495
OS kernel 2:483, 2:485
OS loader 2:482, 2:483
overflow 1:14, 1:97, 1:98, 2:452, 2:455

P
tenpoints 2:256
procedures 2:256
PAL power on/reset 2:263
PALE_RESET 2:79
PAL procedure calling conventions 2:288
PAL procedure calls 2:486
PAL procedures 2:285, 2:481, 2:486-2:488
stacked PAL call 2:487
stacked registers 1:136, 2:486, 2:487
static PAL call 2:486
PAL self-test control word 2:267
PAL_MC_RESUME 2:276
PAL_PREFETCH_VISIBILITY 2:354
PAL_PROC_GET_FEATURES 2:355
PAL-based interrupt states 2:100
PALE_CHECK 2:268
PALE_INIT 2:277
PALE_RESET 2:263
performance monitor events 2:142
performance monitor code sequences 2:143
performance monitor configuration (PMC) 2:137, 2:139
performance monitor data (PMD) 2:137, 2:477
performance monitor data registers (PMD) 1:19, 1:28
performance monitor interrupt service routine 2:143
performance monitor overflow registers 2:141
Index

performance monitor registers 2:137, 2:139, 2:477
performance monitoring mechanisms 2:477
physical addressing 2:61, 2:64, 2:76, 2:488, 3:429, 3:637
pk bit 2:427
platform management interrupt (PMI) 2:92, 2:279, 2:405, 2:481, 2:489, 2:492
PMI flows 2:492
population count 1:76, 3:279
power management 2:80, 2:281, 2:492
NORMAL 1:164, 3:541, 3:905, 3:907
predicate register (PR)
predicate register transfers 1:50
cache pollution reduction 1:160
downward code motion 1:159, 1:160
guidelines for removing branches 1:162
instruction prefetch hints 1:168
instruction scheduling 1:140, 1:142, 1:156
off-path predication 1:158
optimizing program performance using predication 1:157
performance costs of branches 1:155
predication considerations 1:160
predication in the Itanium architecture 1:156
prediction resources 1:71, 1:155, 1:156
upward code motion 1:159
preservation of floating-point state in the OS 2:419
preserved 2:283
preserved registers 2:415, 2:420
preserving ALAT coherency 2:420
privilege levels 1:22, 2:13, 3:597, 3:598, 3:688
current privilege level (CPL) 2:13, 3:732
privilege level transfer 1:76
processor status register (PSR) 2:13, 2:16, 2:18, 2:139, 2:408
processor status register fields 2:19
processor status register instructions 2:18
privileged operation fault 2:149
procedure 1:41-1:43
br.call 1:67, 1:69, 3:318, 3:319
branches and hints 1:136
loops and software pipelining 1:137
rotating registers 1:23, 1:137
stacked register 1:42, 2:486, 2:487
processor abstraction layer (See PAL)
processor caches 2:75, 2:375
processor identifiers (CPUID) 1:19
processor identification registers 1:29
processor interrupt block 2:110-2:112, 2:471
processor min-state save area 2:274
processor ordered 2:238
processor state 2:289
system state 2:13, 2:15, 2:16
processor state parameter 2:271
processor status register (PSR) 2:13, 2:16, 2:18, 2:139, 2:408
programmed I/O 2:401, 2:402
protection key registers (PKR) 2:16, 2:48
pseudo-code functions 3:249
Q
QNaN
description of 3:826
operating on 3:827
qualified exception deferral 2:91
R
RAR (read-after-read) dependency 3:335
RAW (read-after-write) dependency 3:335
reader of a resource 3:335
recovery model 2:88, 2:89
region identifier (RID) 2:38, 2:48, 2:425
region register (RR) 2:48, 2:425
register dependency 1:35, 1:37
read-after-write (RAW) 1:35
write-after-read (WAR) 1:35
write-after-write (WAW) 1:35
register file transfers 1:74
register preservation 2:415
preservation at different points in the OS 2:418
preservation of stacked registers in the OS 2:418
preserving floating-point registers 2:416
preserving general registers 2:416
register rotation 1:15, 1:23, 1:174, 1:175
initializing rotating predicates 1:50, 1:176
clean partition 2:120, 2:127
dirty partition 2:120, 2:127
invalid partition 2:120, 2:127
register stack instructions 1:43
register stack operation 1:41
register stack engine (See RSE)
release semantics 1:66, 2:69, 2:238, 2:239, 2:375
release stores 2:376, 2:378, 2:389, 2:390
reserved 1:19, 1:20, 2:97, 2:284
RSE byte order 1:23
RSE control instructions 2:125, 2:126
RSE initialization 2:132
RSE internal state 2:119
RSE interruptions 2:127
RSE mode 1:25, 2:122
RSE operation instructions and state modification 2:122
RSE privilege level 1:25, 2:123
S
SAL procedure calls 2:487
SALE_ENTRY 2:265
scratch 2:284
scratch registers 2:81, 2:415, 2:420
self test state parameter 2:266
self-modifying code 2:399
semaphore 3:303
behavior of uncacheable and misaligned semaphores 2:377
semaphore instructions 1:35, 1:53, 2:376, 3:303
semaphore operations 1:53, 2:237, 2:378, 2:388
sequential consistency (SC)
SC system 2:392
sequential semantics 2:70
inter-processor interrupt messages 2:70, 2:111-2:113
sequential pages 2:70
single address space (SAS) 1:16, 2:37, 2:425, 2:427, 2:429
single instruction multiple data (SIMD) 3:812
single stepping 2:88
sof field
CFM.sof 2:84, 2:120-2:122, 2:126, 2:127, 2:129
software pipelining 1:11, 1:15, 1:137, 1:173, 1:185, 1:187
sol field
CFM.sol 2:122, 2:127, 2:129
special instruction notations 3:263
special use registers 2:415
advanced load check 1:58, 1:59, 1:146, 3:305
advanced load example 1:145
always-defer model 2:88
check load 1:51, 1:57, 1:59-1:61, 1:145, 1:146, 2:73, 2:74
combining data and control speculation 1:148
control speculation 1:12, 1:13, 1:53-1:56, 1:61, 1:134, 1:143, 1:146, 1:147, 2:445
control speculation example 1:147
count speculative load 1:12, 1:146-1:148
recovery code example 1:145
speculation attributes 2:67
speculation check 1:54, 1:58, 1:148, 3:283, 3:305
speculation considerations 1:149
speculation model in the itanium architecture 1:143, 1:144
speculation recovery code 2:445
speculation related exception handlers 2:445
speculative load exceptions 2:89
speculatively accessible 2:68
speculatively inaccessible 2:68
unaligned handler 2:445
speculative advanced load 1:148
spin lock 2:393, 2:394
square root operations 1:198


st.spill instruction 1:51, 1:62, 2:69


stacked calling convention 2:284


delecroned 2:129

stacked general registers 1:21, 2:117, 2:416

state mappings 3:359

static calling convention 2:284

static general registers 1:21, 2:117, 2:416


store buffers 2:378, 2:385, 2:387


streaming SIMD extension technology 1:104, 3:359, 3:430

subpaging 2:440, 2:441


supervisor accesses 2:236

system abstractions (See SAL)

system architecture features 1:16, 2:11

- support for multiple address space operating systems 1:16
- support for single address space operating systems 1:16

system performance and scalability 1:17

system security and supportability 1:17

system calls 2:420, 2:421, 2:422

system descriptors 2:217

system flag interception 2:215

system memory model 2:233

system register model 2:17, 2:215

IA-32 state 1:108, 2:215, 2:216


undefined 1:108, 2:216

unmodified 1:108, 1:109, 2:216, 3:582

system register resources 2:13, 2:15, 2:16

T


template 1:32, 1:33, 1:34, 1:133, 3:257, 3:258

temporal hint 1:205, 3:927


- page not present vector 2:97, 2:152, 2:440


- TLB miss handlers 2:59, 2:436, 2:439

- TLB purges 2:40, 2:42, 2:44

- translation insertion format 2:45

- VHPT translation vector 2:96, 2:152, 2:438

TLB entry, invalidating (flushing) 3:578


translation caches (TCs) 2:431

TR insertion 2:431

TR purge 2:429, 2:432

translation lookaside buffer (See TLB)

translation registers (TRs) 2:429

TR insertion 2:430

TR purge 2:429, 2:430, 2:431


U

UC memory attribute 2:242

unaligned reference handler 2:445-2:447

uncacheable 2:63-2:65, 3:927

 uncachable pages 2:66


undefined behavior 1:39

underflow 1:14, 1:97, 1:99, 2:452, 2:456

unimplemented addresses 2:62, 2:63

- unimplemented physical address bits 2:61, 2:62

- unimplemented virtual address bits 2:62

unnormalized numbers 1:79

unordered semantics 2:375

unsupported data reference handler 2:447, 2:448

user mask (UM) 1:19, 1:28
## Index

### V
  - TLB and VHPT search faults 2:59
  - TLB/VHPT search 2:58
  - translation searching 2:57
  - VHPT configuration 2:51
  - VHPT searching 2:52
  - VHPT short format 2:52
  - VHPT short-format index 2:54, 2:55
  - VHPT updates 2:436
- virtual addressing 2:37, 2:38, 2:63, 2:74, 2:485, 2:488
- virtual aliasing 2:60
- virtual hash page table (See VHPT)
- virtual region number (VRN) 2:38, 2:62, 2:425
- virtualized interrupt flag 2:219
- VME extensions 2:219, 2:224

### W
- WAR (write-after-read) dependency 3:335
- WAW (write-after-write) dependency 3:335
- write BSPSTORE 2:131
- write-back and invalidate caches 3:735
- writer of a resource 3:335

### X

### Z
- zero, floating-point format 3:824