# int<sub>el</sub>.

# Intel<sup>®</sup> Processor Identification and the CPUID Instruction

**Application Note 485** 

July 2004

Document Number: 241618-027

INFORMATION IN THIS DOCUMENT IS PROVIDED IN CONNECTION WITH INTEL<sup>®</sup> PRODUCTS. NO LICENSE, EXPRESS OR IMPLIED, BY ESTOPPEL OR OTHERWISE, TO ANY INTELLECTUAL PROPERTY RIGHTS IS GRANTED BY THIS DOCUMENT. EXCEPT AS PROVIDED IN INTEL'S TERMS AND CONDITIONS OF SALE FOR SUCH PRODUCTS, INTEL ASSUMES NO LIABILITY WHATSOEVER, AND INTEL DISCLAIMS ANY EXPRESS OR IMPLIED WARRANTY, RELATING TO SALE AND/OR USE OF INTEL PRODUCTS INCLUDING LIABILITY OR WARRANTIES RELATING TO FITNESS FOR A PARTICULAR PURPOSE, MERCHANTABILITY, OR INFRINGEMENT OF ANY PATENT, COPYRIGHT OR OTHER INTELLECTUAL PROPERTY RIGHT. Intel products are not intended for use in medical, life saving, or life sustaining applications.

Intel may make changes to specifications and product descriptions at any time, without notice.

Designers must not rely on the absence or characteristics of any features or instructions marked "reserved" or "undefined." Intel reserves these for future definition and shall have no responsibility whatsoever for conflicts or incompatibilities arising from future changes to them.

The Intel® processors may contain design defects or errors known as errata which may cause the product to deviate from published specifications. Current characterized errata are available on request.

Contact your local Intel sales office or your distributor to obtain the latest specifications and before placing your product order.

Intel, Pentium, Celeron, Intel NetBurst, Intel Xeon, Pentium II Xeon, Pentium III Xeon, Intel SpeedStep, OverDrive, MMX, Intel486, Intel386, IntelDX2, and the Intel logo are trademarks or registered trademarks of Intel Corporation or its subsidiaries in the United States and other countries.

\*Other names and brands may be claimed as the property of others.

Copyright © 1993-2004, Intel Corporation

# int<sub>e</sub>l.

### **Contents**

| 1  | Introdu                                | ction9                                                   |  |
|----|----------------------------------------|----------------------------------------------------------|--|
|    | 1.1                                    | Update Support9                                          |  |
| 2  | Detecti                                | ng the CPUID Instruction11                               |  |
| 3  | Output                                 | of the CPUID Instruction13                               |  |
|    | 3.1                                    | Vendor ID String                                         |  |
|    | 3.2                                    | Processor Signature                                      |  |
|    | 3.3                                    | Feature Flags                                            |  |
|    | 3.4                                    | Extended Feature Flags                                   |  |
|    | 3.5                                    | SYSENTER/SYSEXIT – SEP Features Bit                      |  |
|    | 3.6                                    | Cache Size, Format and TLB Information26                 |  |
|    | 3.7                                    | Pentium <sup>®</sup> 4 Processor, Model 0 Output Example |  |
| 4  | Proces                                 | sor Serial Number                                        |  |
|    | 4.1                                    | Presence of Processor Serial Number                      |  |
|    | 4.2                                    | Forming the 96-bit Processor Serial Number               |  |
| 5  | Brand I                                | D and Brand String                                       |  |
|    | 5.1                                    | Brand ID                                                 |  |
|    | 5.2                                    | Brand String                                             |  |
| 6  | Usage                                  | Guidelines                                               |  |
| 7  | Proper                                 | Identification Sequence                                  |  |
| 8  | Usage Program Examples                 |                                                          |  |
| 9  | Alternate Method of Detecting Features |                                                          |  |
| 10 | Denormals Are Zero43                   |                                                          |  |
| 11 | Operating Frequency                    |                                                          |  |



### **Figures**

| Figure 1. Flag Register Evolution                               | 11 |
|-----------------------------------------------------------------|----|
| Figure 2. CPUID Instruction Outputs                             |    |
| Figure 3. EDX Register after RESET                              | 17 |
| Figure 4. Processor Signature Format on Intel386™ Processors    | 19 |
| Figure 5. Flow of Processor get_cpu_type Procedure              | 38 |
| Figure 6. Flow of Processor Identification Extraction Procedure |    |

### **Tables**

| Table 1. Information Returned by the CPUID Instruction                                 | 15 |
|----------------------------------------------------------------------------------------|----|
| Table 2. Processor Type (Bit Positions 13 and 12)                                      | 18 |
| Table 3. Intel386™ Processor Signatures                                                |    |
| Table 4. Intel486 <sup>™</sup> and Subsequent Processor Signatures                     | 20 |
| Table 5. Feature Flag Values Reported in the EDX Register                              | 22 |
| Table 6. Feature Flag Values Reported in the ECX Register                              | 24 |
| Table 7. Extended Feature Flag Values Reported in the EDX Register                     | 25 |
| Table 8. Descriptor Formats                                                            |    |
| Table 9. Descriptor Decode Values                                                      |    |
| Table 10. Pentium <sup>®</sup> 4 Processor, Model 0 with 256-KB L2 Cache CPUID (EAX=2) |    |
| _/                                                                                     | 29 |
| Table 11. Brand ID, CPUID (EAX=1) Return Values in EBX (bits 7 through 0)              |    |
| Table 12. Processor Brand String Feature                                               | 34 |

### int<sub>el</sub>.

### **Revision History**

| Revision | Description                                                                                                                                                                                                                                                                                                    | Date  |  |
|----------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|--|
| -001     | Original Issue.                                                                                                                                                                                                                                                                                                |       |  |
| -002     | Modified Table 4, Intel486™ and Pentium® Processor Signatures.                                                                                                                                                                                                                                                 |       |  |
| -003     | Updated to accommodate new processor versions. Program examples modified for ease of use, section added discussing BIOS recognition for OverDrive® processors and feature flag information updated.                                                                                                            | 09/94 |  |
| -004     | Updated with Pentium Pro and OverDrive processors information. Modified Table 1, Table 3, and Table 5. Inserted Table 6, Table 8, Table 9. Inserted Sections 3.4. and 3.5.                                                                                                                                     | 12/95 |  |
| -005     | Added Figures 1 and 3. Added Footnotes 1 and 2. Modified Figure 2. Added Assembly code example in Section 4. Modified Tables 3, 5 and 7. Added two bullets in Section 5.0. Modified cpuid3b.ASM and cpuid3b.C programs to determine if processor features MMX <sup>™</sup> technology. Modified Figure 6.0.    |       |  |
| -006     | Modified Table 3. Added reserved for future member of P6 family of processors entry. Modified table header to reflect Pentium II processor family. Modified Table 5. Added SEP bit definition. Added Section 3.5. Added Section 3.7 and Table 9. Corrected references of P6 family to reflect correct usage.   |       |  |
|          | Modified cpuid3a.asm, cpuid3b.asm and cpuid3.c example code sections to check for SEP feature bit and to check for, and identify, the Pentium II processor. Added additional disclaimer related to designers and errata.                                                                                       |       |  |
| - 007    | Modified Table 2. Added Pentium II processor, model 5 entry. Modified<br>existing Pentium II processor entry to read "Pentium II processor, model 3".<br>Modified Table 5. Added additional feature bits, PAT and FXSR. Modified<br>Table 7. Added entries 44h and 45h.                                        |       |  |
|          | Removed the note "Do not assume a value of 1 in a feature flag indicates that a given feature is present. For future feature flags, a value of 1 may indicate that the specific feature is not present" in section 4.0.                                                                                        |       |  |
|          | Modified cpuid3b.asm and cpuid3.c example code section to check for, and identify, the Pentium II processor, model 5. Modified existing Pentium II processor code to print Pentium II processor, model 3.                                                                                                      |       |  |
| - 008    | Added note to identify Intel <sup>®</sup> Celeron <sup>®</sup> processor, model 5 in section 3.2.<br>Modified Table 2. Added Celeron processor and Pentium <sup>®</sup> OverDrive <sup>®</sup><br>processor with MMX <sup>™</sup> technology entry. Modified Table 5. Added additional<br>feature bit, PSE-36. |       |  |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Celeron processor.                                                                                                                                                                                                              |       |  |
| -009     | Added note to identify Pentium II Xeon <sup>™</sup> processor in section 3.2. Modified Table 2. Added Pentium II Xeon processor entry.                                                                                                                                                                         |       |  |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Pentium II Xeon processor.                                                                                                                                                                                                      |       |  |
| -010     | No Changes                                                                                                                                                                                                                                                                                                     |       |  |

| Revision | Description                                                                                                                                                                                                                                                                                                                                                                                                            |       |  |  |
|----------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|--|--|
| -011     | Modified Table 2. Added Celeron processor, model 6 entry.                                                                                                                                                                                                                                                                                                                                                              | 12/98 |  |  |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Celeron processor, model 6.                                                                                                                                                                                                                                                                                                             |       |  |  |
| -012     | Modified Figure 1 to add the reserved information for the Intel386 processors.<br>Modified Figure 2. Added the Processor serial number information returned<br>when the CPUID instruction is executed with EAX=3. Modified Table 1. Added<br>the Processor serial number parameter. Modified Table 2. Added the Pentium<br>III processor and Pentium III Xeon processor. Added Section 4 "Processor<br>serial number". |       |  |  |
|          | Modified cpuid3a.asm, cpuid3b.asm and cpuid3.c example code to check for and identify the Pentium III processor and the Pentium III Xeon processor.                                                                                                                                                                                                                                                                    |       |  |  |
| -013     | Modified Figure 2. Added the Brand ID information returned when the CPUID instruction is executed with EAX=1. Added section 5 "Brand ID". Added Table 10 that shows the defined Brand ID values.                                                                                                                                                                                                                       | 10/99 |  |  |
|          | Modified cpuid3a.asm, cpuid3b.asm and cpuid3.c example code to check for<br>and identify the Pentium III processor, model 8 and the Pentium III Xeon<br>processor, model 8.                                                                                                                                                                                                                                            |       |  |  |
| -014     | Modified Table 4. Added Celeron processor, model 8                                                                                                                                                                                                                                                                                                                                                                     | 03/00 |  |  |
| -015     | Modified Table 4. Added Pentium III Xeon processor, model A. Modified Table 9, Added the 8-way set associative 1M, and 8-way set associative 2M cache descriptor entries.                                                                                                                                                                                                                                              |       |  |  |
| -016     | Revised Figure 2 to include the Extended Family and Extended Model when CPUID is executed with EAX=1.                                                                                                                                                                                                                                                                                                                  |       |  |  |
|          | Added section 6 which describes the Brand String.                                                                                                                                                                                                                                                                                                                                                                      |       |  |  |
|          | Added section 10 Alternate Method of Detecting Features and sample code .                                                                                                                                                                                                                                                                                                                                              |       |  |  |
|          | Added the Pentium 4 processor signature to Table 4.                                                                                                                                                                                                                                                                                                                                                                    |       |  |  |
|          | Added new feature flags (SSE2, SS and TM) to Table 5.                                                                                                                                                                                                                                                                                                                                                                  |       |  |  |
|          | Added new cache descriptors to Table 9.                                                                                                                                                                                                                                                                                                                                                                                |       |  |  |
|          | Removed Pentium Pro cache descriptor example.                                                                                                                                                                                                                                                                                                                                                                          |       |  |  |
| -017     | Modified Figure 2 to include additional features reported by the Pentium 4 processors.                                                                                                                                                                                                                                                                                                                                 |       |  |  |
|          | Modified Table 9 to include additional Cache and TLB descriptors defined by the Intel® NetBurst <sup>™</sup> microarchitecture.                                                                                                                                                                                                                                                                                        |       |  |  |
|          | Added Section 10 and program Example 5 which describes how to detect if a processor supports the DAZ feature.                                                                                                                                                                                                                                                                                                          |       |  |  |
|          | Added Section 11 and program Example 6 which describes a method of calculating the actual operating frequency of the processor.                                                                                                                                                                                                                                                                                        |       |  |  |
| -018     | Changed the second 66h cache descriptor in Table 7 to 68h.                                                                                                                                                                                                                                                                                                                                                             |       |  |  |
|          | Added the 83h cache descriptor to Table 7.                                                                                                                                                                                                                                                                                                                                                                             |       |  |  |
|          | Added the Pentium III processor, model B, processor signature and the Intel Xeon processor, processor signature to Table 4.                                                                                                                                                                                                                                                                                            |       |  |  |
|          | Modified Table 4 to include the extended family and extended model fields.                                                                                                                                                                                                                                                                                                                                             |       |  |  |
|          | Modified Table 1 to include the information returned by the extended CPUID functions.                                                                                                                                                                                                                                                                                                                                  |       |  |  |

# int<sub>el</sub>,

| Revision | Description                                                                                                                                                  |       |
|----------|--------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|
| -019     | Changed to use registered trademark for Intel® Celeron® throughout entire document.                                                                          |       |
|          | Modified Table 11 to include new Brand ID values supported by the Intel® processors with Intel NetBurst® microarchitecture.                                  |       |
|          | Added Hyper-Threading Technology Flag to Table 5 and Logical Processor Count to Table 1.                                                                     |       |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for and identify<br>Intel® processors based on the updated Brand ID values contained in Table<br>11. |       |
| -020     | Modified Table 9 to include new Cache Descriptor values supported by the Intel processors with Intel NetBurst microarchitecture.                             | 03/02 |
|          | Modified Table 11 to include new Brand ID values supported by the Intel processors with Intel NetBurst microarchitecture.                                    |       |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for and identify<br>Intel® processors based on the updated Brand ID values contained in Table<br>11. |       |
| -021     | Modified Table 4 to include additional processors that return a processor signature with a value in the family code equal to 0Fh.                            |       |
|          | Modified Table 9 to include new Cache Descriptor values supported by various Intel processors.                                                               |       |
|          | Modified Table 11 to include new Brand ID values supported by the Intel processors with Intel NetBurst microarchitecture.                                    |       |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for and identify<br>Intel processors based on the updated Brand ID values contained in Table 11.     |       |
| -022     | Modified Table 9 with correct Cache Descriptor descriptions.                                                                                                 |       |
|          | Modified Table 5 with new feature flags returned in EDX.                                                                                                     |       |
|          | Added Table 6 the feature flags returned in ECX.                                                                                                             |       |
|          | Modified Table 4, broke out the processors with family 'F' by model numbers.                                                                                 |       |
| -023     | Modified Table 4, added the Intel® Pentium® M processor.                                                                                                     | 03/03 |
|          | Modified Table 5 with new feature flags returned in EDX.                                                                                                     |       |
|          | Modified Table 6 the feature flags returned in ECX.                                                                                                          |       |
|          | Modified Table 9 with correct Cache Descriptor descriptions.                                                                                                 |       |
| -024     | Corrected feature flag definitions in Table 6 for bits 7 and 8.                                                                                              | 11/03 |

### int<sub>el</sub>.

| Revision | Description                                                                                                                                                                                                                                                                                                      | Date  |  |
|----------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|--|
| -025     | Modified Table 1 to add Deterministic Cache Parameters function (CPUID executed with EAX=4), MONITOR/MWAIT function (CPUID instruction is executed with EAX=5), Extended L2 Cache Features function (CPUID executed with EAX=80000006), Extended Addresses Sizes function (CPUID is executed with EAX=80000008). | 01/04 |  |
|          | Modified Table 1 and Table 5 to reinforce no PSN on Pentium® 4 family processors.                                                                                                                                                                                                                                |       |  |
|          | Modified, added the Intel® Pentium® 4 processor and Intel® Celeron® processor on 90nm process.                                                                                                                                                                                                                   |       |  |
|          | Modified Table 6 to add new feature flags returned in ECX.                                                                                                                                                                                                                                                       |       |  |
|          | Modified Table 9 to include new Cache Descriptor values supported by various Intel processors.                                                                                                                                                                                                                   |       |  |
|          | Modified Table 11 to include new Brand ID values supported by the Intel processors with Intel NetBurst microarchitecture.                                                                                                                                                                                        |       |  |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for and identify<br>Intel processors based on the updated Brand ID values contained in Table 11.                                                                                                                                                         |       |  |
|          | Modified features.cpp, cpuid3.c, and cpuid3a.asm to check for and identify new feature flags based on the updated values in Table 6.                                                                                                                                                                             |       |  |
| -026     | Corrected the name of the feature flag returned in EDX[31] (PBE) when the CPUID instruction is executed with EAX set to a 1.                                                                                                                                                                                     |       |  |
|          | Modified Table 1 to indicate CPUID function 80000001h now returns extended feature flags in the EAX register.                                                                                                                                                                                                    |       |  |
|          | Added the Intel® Pentium® M processor (family 6, model D) to Table 4.                                                                                                                                                                                                                                            |       |  |
|          | Added section 3.4 Extended Feature Flags and Table 7.                                                                                                                                                                                                                                                            |       |  |
|          | Modified Table 6 to add new feature flags returned in ECX.                                                                                                                                                                                                                                                       |       |  |
|          | Modified Table 9 to include new Cache Descriptor values supported by various Intel processors.                                                                                                                                                                                                                   |       |  |
|          | Modified Table 11 to include new Brand ID values supported by the Intel processors with P6 family microarchitecture.                                                                                                                                                                                             |       |  |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for and identify<br>Intel processors based on the updated Brand ID values contained in Table 11.                                                                                                                                                         |       |  |
|          | Modified features.cpp, cpuid3.c, and cpuid3a.asm to check for and identify new feature flags based on the updated values in Table 6.                                                                                                                                                                             |       |  |
| -027     | Corrected the register used for Extended Feature Flags in Table 7                                                                                                                                                                                                                                                | 07/04 |  |

§

### 1 Introduction

As the Intel® Architecture evolves with the addition of new generations and models of processors (8086, 8088, Intel286, Intel386<sup>TM</sup>, Intel486<sup>TM</sup>, Pentium<sup>®</sup> processors, Pentium<sup>®</sup> OverDrive<sup>®</sup> processors with MMX<sup>TM</sup> technology, Pentium<sup>®</sup> OverDrive<sup>®</sup> processors with MMX<sup>TM</sup> technology, Pentium<sup>®</sup> II processors, Pentium<sup>®</sup> II Xeon<sup>TM</sup> processors, Pentium<sup>®</sup> II Overdrive<sup>®</sup> processors, Intel<sup>®</sup> Celeron<sup>®</sup> processors, Mobile Intel<sup>®</sup> Celeron<sup>®</sup> processors, Pentium<sup>®</sup> III Processor - M, Pentium<sup>®</sup> III Xeon<sup>TM</sup> processors, Pentium<sup>®</sup> 4 processors, Mobile Intel<sup>®</sup> Pentium<sup>®</sup> 4 processor - M, Intel<sup>®</sup> Pentium<sup>®</sup> M Processor, Intel<sup>®</sup> Xeon<sup>TM</sup> processors and Intel<sup>®</sup> Xeon<sup>TM</sup> processor MP), it is essential that Intel provide an increasingly sophisticated means with which software can identify the features available on each processor. This identification mechanism has evolved in conjunction with the Intel Architecture as follows:

- 1. Originally, Intel published code sequences that could detect minor implementation or architectural differences to identify processor generations.
- Later, with the advent of the Intel386 processor, Intel implemented processor signature identification that provided the processor family, model, and stepping numbers to software, but only upon reset.
- 3. As the Intel Architecture evolved, Intel extended the processor signature identification into the CPUID instruction. The CPUID instruction not only provides the processor signature, but also provides information about the features supported by and implemented on the Intel processor.

The evolution of processor identification was necessary because, as the Intel Architecture proliferates, the computing market must be able to tune processor functionality across processor generations and models that have differing sets of features. Anticipating that this trend will continue with future processor generations, the Intel Architecture implementation of the CPUID instruction is extensible.

This application note explains how to use the CPUID instruction in software applications, BIOS implementations, and various processor tools. By taking advantage of the CPUID instruction, software developers can create software applications and tools that can execute compatibly across the widest range of Intel processor generations and models, past, present, and future.

#### 1.1 Update Support

You can obtain new Intel processor signature and feature bits information from the developer's manual, programmer's reference manual or appropriate documentation for a processor. In addition, you can receive updated versions of the programming examples included in this application note; contact your Intel representative for more information, or visit Intel's website at <a href="http://developer.intel.com/">http://developer.intel.com/</a>.

Introduction

### int<sub>el</sub>.

### 2 **Detecting the CPUID Instruction**

The Intel486 family and subsequent Intel processors provide a straightforward method for determining whether the processor's internal architecture is able to execute the CPUID instruction. This method uses the ID flag in bit 21 of the EFLAGS register. If software can change the value of this flag, the CPUID instruction is executable<sup>1</sup> (see Figure 1).

#### **Figure 1. Flag Register Evolution**



The POPF, POPFD, PUSHF, and PUSHFD instructions are used to access the Flags in Eflags register. The program examples at the end of this application note show how you use the PUSHFD instruction to read and the POPFD instruction to change the value of the ID flag.

<sup>1</sup>Only in some Intel486<sup>TM</sup> and succeeding processors. Bit 21 in the Intel386<sup>TM</sup> processor's Eflag register cannot be changed by software, and the Intel386 processor cannot execute the CPUID instruction. Execution of CPUID on a processor that does not support this instruction will result in an invalid opcode exception.

<sup>§</sup> 

### **3 Output of the CPUID Instruction**

The CPUID instruction supports two sets of functions. The first set returns basic processor information. The second set returns extended processor information. Figure 2 summarizes the basic processor information output by the CPUID instruction. The output from the CPUID instruction is fully dependent upon the contents of the EAX register. This means, by placing different values in the EAX register and then executing CPUID, the CPUID instruction will perform a specific function dependent upon whatever value is resident in the EAX register (see Table 1). In order to determine the highest acceptable value for the EAX register input and CPUID functions that return the basic processor information, the program should set the EAX register parameter value to "0" and then execute the CPUID instruction as follows:

MOV EAX, 00H CPUID

After the execution of the CPUID instruction, a return value will be present in the EAX register. Always use an EAX parameter value that is equal to or greater than zero and less than or equal to this highest EAX "returned" value.

In order to determine the highest acceptable value for the EAX register input and CPUID functions that return the extended processor information, the program should set the EAX register parameter value to "80000000h" and then execute the CPUID instruction as follows:

MOV EAX, 8000000H CPUID

After the execution of the CPUID instruction, a return value will be present in the EAX register. Always use an EAX parameter value that is equal to or greater than 80000000h and less than or equal to this highest EAX "returned" value. On current and future IA-32 processors, bit 31 in the EAX register will be clear when CPUID is executed with an input parameter greater then highest value for either set of functions, and when the extended functions are not supported. All other bit values returned by the processor in response to a CPUID instruction with EAX set to a value higher than appropriate for that processor are model specific and should not be relied upon.

### 3.1 Vendor ID String

In addition to returning the highest value in the EAX register, the Intel Vendor-ID string can be simultaneously verified as well. If the EAX register contains an input value of 0, the CPUID instruction also returns the vendor identification string in the EBX, EDX, and ECX registers (see Figure 2). These registers contain the ASCII string:

#### GenuineIntel

While any imitator of the Intel Architecture can provide the CPUID instruction, no imitator can legitimately claim that its part is a genuine Intel part. So, the presence of the "GenuineIntel" string is an assurance that the CPUID instruction and the processor signature are implemented as described in this document. If the "GenuineIntel" string is not returned after execution of the CPUID instruction, do not rely upon the information described in this document to interpret the information returned by the CPUID instruction.

#### Figure 2. CPUID Instruction Outputs



#### Table 1. Information Returned by the CPUID Instruction

| Initial EAX Value | Information Provided about the Processor                                                                                                                                                                                                                                                                                                                                                             |  |  |  |  |
|-------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|--|--|
|                   | Basic CPUID Information                                                                                                                                                                                                                                                                                                                                                                              |  |  |  |  |
| 0H                | EAX: Maximum Input Value for Basic CPUID Information                                                                                                                                                                                                                                                                                                                                                 |  |  |  |  |
|                   | EBX: "Genu"                                                                                                                                                                                                                                                                                                                                                                                          |  |  |  |  |
|                   | ECX: "ntel"                                                                                                                                                                                                                                                                                                                                                                                          |  |  |  |  |
|                   | EDX "inel"                                                                                                                                                                                                                                                                                                                                                                                           |  |  |  |  |
| 1H                | EAX: <b>32-bit Processor Signature (Extended Family, Extended Model,</b><br><b>Type, Family, Model and Stepping ID</b> also bits 95-64 of the 96-bit<br>processor serial number when the PSN feature flag is set.                                                                                                                                                                                    |  |  |  |  |
|                   | EBX:       Bits 7-0:       Brand Index – Not supported if the value = 00h.         Bits 15-8:       CLFLUSH line size. (Value returned * 8 = cache line size)         Valid only if CLFSH feature flag is set.         Bits 23-16:       Count of logical processors.                                                                                                                                |  |  |  |  |
|                   | Valid only if Hyper-Threading Technology flag is set<br>Bits 31-24: Processor local APIC physical ID<br>Valid for Pentium 4 and subsequent processors                                                                                                                                                                                                                                                |  |  |  |  |
|                   | ECX: Feature Flags (see Table 6)                                                                                                                                                                                                                                                                                                                                                                     |  |  |  |  |
|                   | EDX: Feature Flags (see Table 5)                                                                                                                                                                                                                                                                                                                                                                     |  |  |  |  |
| 2H                | EAX, EBX, ECX, EDX Cache and TLB Descriptors                                                                                                                                                                                                                                                                                                                                                         |  |  |  |  |
| ЗH                | EAX: Reserved                                                                                                                                                                                                                                                                                                                                                                                        |  |  |  |  |
|                   | EBX: Reserved                                                                                                                                                                                                                                                                                                                                                                                        |  |  |  |  |
|                   | ECX: Bits 31-0 represent bits 31-0 of the 96-bit processor serial number.<br>(Available only in Pentium III processors when the PSN feature flag is set; otherwise, the value in this register is reserved.)                                                                                                                                                                                         |  |  |  |  |
|                   | EDX: Bits 31-0 represent bits 63-32 of the 96-bit processor serial number.<br>(Available only in Pentium III processors when the PSN feature flag is set; otherwise, the value in this register is reserved.)                                                                                                                                                                                        |  |  |  |  |
|                   | NOTE: Processor serial number (PSN) is not supported in the Pentium 4<br>processor or later. On all models, use the PSN flag (returned using<br>CPUID) to check for PSN support before accessing the feature. If<br>the PSN Feature Flags equals "0", the processor serial number<br>feature is either not supported, or disabled in the Pentium III<br>processor. Refer to Section 4.1 for details. |  |  |  |  |

| Initial EAX Value | Information Provided about the Processor                                                                                          |  |  |  |  |
|-------------------|-----------------------------------------------------------------------------------------------------------------------------------|--|--|--|--|
|                   | Basic CPUID Information                                                                                                           |  |  |  |  |
| 4H                | Deterministic Cache Parameters Function                                                                                           |  |  |  |  |
|                   | EAX: Bits 4-0: Cache Type**                                                                                                       |  |  |  |  |
|                   | Bits 7-5: Cache Level (starts at 1)                                                                                               |  |  |  |  |
|                   | Bits 8: Self Initializing cache level (does not need SW initialization)                                                           |  |  |  |  |
|                   | Bits 9: Fully Associative cache                                                                                                   |  |  |  |  |
|                   | Bits 13-10: Reserved                                                                                                              |  |  |  |  |
|                   | Bits 25-14: Number of threads sharing this cache*                                                                                 |  |  |  |  |
|                   | Bits 31-26: Number of processor cores on this die (Multicore)*                                                                    |  |  |  |  |
|                   | EBX: Bits 11-0: L = System Coherency Line Size*                                                                                   |  |  |  |  |
|                   | Bits 21-12: P = Physical Line partitions*                                                                                         |  |  |  |  |
|                   | Bits 31-22: W = Ways of associativity*                                                                                            |  |  |  |  |
|                   | ECX: Bits 31-0: S = Number of Sets*                                                                                               |  |  |  |  |
|                   | EDX: Reserved                                                                                                                     |  |  |  |  |
|                   |                                                                                                                                   |  |  |  |  |
|                   | * Add one to the value in the register file to get the number. For example, the number of processor cores is EAX[31:26]+1.        |  |  |  |  |
|                   | ** Cache Types fields                                                                                                             |  |  |  |  |
|                   | 0 = Null - No more caches                                                                                                         |  |  |  |  |
|                   | 1 = Data Cache                                                                                                                    |  |  |  |  |
|                   | 2 = Instruction Cache                                                                                                             |  |  |  |  |
|                   | 3 = Unified Cache                                                                                                                 |  |  |  |  |
|                   | 31-4 = Reserved                                                                                                                   |  |  |  |  |
|                   | NOTE: Deterministic Cache Parameters Function is available only when<br>IA32_CR_MISC_ENABLES.BOOT_NT4 (bit 22) is clear (Default) |  |  |  |  |
| 5H                | MONITOR/MWAIT Function                                                                                                            |  |  |  |  |
|                   | EAX Bits 15-0: Smallest monitor-line size in bytes (default is processor's monitor granularity)                                   |  |  |  |  |
|                   | Bits 31-16: Reserved                                                                                                              |  |  |  |  |
|                   | EBX Bits 15-0: Largest monitor-line size in bytes (default is processor's monitor granularity)                                    |  |  |  |  |
|                   | Bits 31-16: Reserved                                                                                                              |  |  |  |  |
|                   | ECX Reserved                                                                                                                      |  |  |  |  |
|                   | EDX Reserved                                                                                                                      |  |  |  |  |
|                   | NOTE: MONITOR/MWAIT Function is available only when<br>IA32_CR_MISC_ENABLES.BOOT_NT4 (bit 22) is clear (Default)                  |  |  |  |  |

| Initial EAX Value | Information Provided about the Processor                         |  |  |  |  |  |
|-------------------|------------------------------------------------------------------|--|--|--|--|--|
|                   | Extended Function CPUID Information                              |  |  |  |  |  |
| 8000000H          | EAX: Maximum Input Value for Extended Function CPUID Information |  |  |  |  |  |
|                   | EBX, ECX, EDX: Reserved                                          |  |  |  |  |  |
| 8000001H          | EAX: Extended Feature Flags (refer to Table 7)                   |  |  |  |  |  |
|                   | EBX, ECX, EDX: Reserved for additional extended feature flags.   |  |  |  |  |  |
| 8000002H          | EAX: Processor Brand String                                      |  |  |  |  |  |
|                   | EBX, ECX, EDX: Processor Brand String Continued                  |  |  |  |  |  |
| 8000003H          | EAX, EBX, ECX, EDX: Processor Brand String Continued             |  |  |  |  |  |
| 8000004H          | EAX, EBX, ECX, EDX: Processor Brand String Continued             |  |  |  |  |  |
| 80000005h         | EAX, EBX, ECX, EDX: Reserved                                     |  |  |  |  |  |
| 8000006h          | Extended L2 Cache Features Function                              |  |  |  |  |  |
|                   | EAX: Reserved                                                    |  |  |  |  |  |
|                   | EBX: Reserved                                                    |  |  |  |  |  |
|                   | ECX: Bits 7-0: L2 Cache Line Size                                |  |  |  |  |  |
|                   | Bits 15-8: L2 Cache Associativity                                |  |  |  |  |  |
|                   | Bits 31-16: L2 Cache Size in 1-K units                           |  |  |  |  |  |
|                   | EDX: Reserved                                                    |  |  |  |  |  |
| 8000007h          | EAX, EBX, ECX, EDX: Reserved                                     |  |  |  |  |  |
| 8000008h          | Extended Address Sizes Function                                  |  |  |  |  |  |
|                   | EAX: Bits 7-0 Physical Address Size (# of bits)                  |  |  |  |  |  |
|                   | Bits 15-8 Virtual Address Size (# of bits)                       |  |  |  |  |  |
|                   | Bits 31-16: Reserved                                             |  |  |  |  |  |
|                   | EBX, ECX, EDX: Reserved                                          |  |  |  |  |  |

#### 3.2 **Processor Signature**

Beginning with the Intel486 processor family, the EDX register contains the processor identification signature after reset (see Figure 3). **The processor identification signature is a 32-bit value.** The processor signature is composed from eight different bit fields. The fields in gray represent reserved bits, and should be masked out when utilizing the processor signature. The remaining six fields form the processor identification signature.

#### Figure 3. EDX Register after RESET



Processors that implement the CPUID instruction also return the 32-bit processor identification signature after reset; however, the CPUID instruction gives you the flexibility of checking the processor signature at any time. Figure 3 shows the format of the 32-bit processor signature for the Intel486, and subsequent Intel processors. Note that the EDX processor signature value after reset is equivalent to the processor signature output value in the EAX register in Figure 2. Table 4 shows the values returned in the EAX register currently defined for these processors.



The extended family, bit positions 20 through 27 are used in conjunction with the family code, specified in bit positions 8 through 11, to indicate whether the processor belongs to the Intel386, Intel486, Pentium, Pentium Pro or Pentium 4 family of processors. P6 family processors include all processors based on the Pentium Pro processor architecture and have an extended family equal to 00h and a family code equal to 6h. Pentium 4 family processors include all processors based on the Intel NetBurst<sup>®</sup> microarchitecture and have an extended family equal to 00h and a family code equal to 0Fh.

The extended model, bit positions 16 through 19 in conjunction with the model number, specified in bits 4 though 7, are used to identify the model of the processor within the processor's family. The stepping ID in bits 0 through 3 indicates the revision number of that model.

The processor type, specified in bit positions 12 and 13 of Table 2 indicates whether the processor is an original OEM processor, an OverDrive processor, or a dual processor (capable of being used in a dual processor system). Table 2 shows the processor type values returned in bits 12 and 13 of the EAX register.

| Value                          | Description                      |  |  |
|--------------------------------|----------------------------------|--|--|
| 00                             | Original OEM processor           |  |  |
| 01                             | OverDrive <sup>®</sup> processor |  |  |
| 10                             | Dual processor                   |  |  |
| 11 Intel reserved (Do Not Use) |                                  |  |  |

#### Table 2. Processor Type (Bit Positions 13 and 12)

The Pentium II processor, model 5, the Pentium II Xeon processor, model 5, and the Celeron processor, model 5 share the same extended family, family code, extended model and model number. To differentiate between the processors, software should check the cache descriptor values through executing CPUID instruction with EAX = 2. If no L2 cache is returned, the processor is identified as an Intel<sup>®</sup> Celeron<sup>®</sup> processor, model 5. If 1-MB or 2-MB L2 cache size is reported, the processor is the Pentium II Xeon processor otherwise it is a Pentium II processor, model 5 or a Pentium II Xeon processor with 512-KB L2 cache.

The Pentium III processor, model 7, and the Pentium III Xeon processor, model 7, share the same extended family, family code, extended model and model number. To differentiate between the processors, software should check the cache descriptor values through executing CPUID instruction with EAX = 2. If 1M or 2M L2 cache size is reported, the processor is the Pentium III Xeon processor otherwise it is a Pentium III processor or a Pentium III Xeon processor with 512-KB L2 cache.

The processor brand for the Pentium III processor, model 8, the Pentium III Xeon processor, model 8, and the Celeron processor, model 8, can be determined by using the Brand ID values returned by the CPUID instruction when executed with EAX equal to 1. Table 11 shows the processor brands defined by the Brand ID.

Older versions of Intel486 SX, Intel486 DX and IntelDX2<sup>TM</sup> processors do not support the CPUID instruction<sup>2</sup>, so they can only return the processor signature at reset. Refer to Table 4 to determine which processors support the CPUID instruction.

Figure 4 shows the format of the processor signature for Intel386 processors, which are different from other processors. Table 3 shows the values currently defined for these Intel386 processors.

#### Figure 4. Processor Signature Format on Intel386™ Processors



#### Table 3. Intel386<sup>™</sup> Processor Signatures

| Туре | Family | Major Stepping | Minor Stepping | Description            |
|------|--------|----------------|----------------|------------------------|
| 0000 | 0011   | 0000 xxxx      |                | Intel386™ DX processor |
| 0010 | 0011   | 0000           | хххх           | Intel386 SX processor  |
| 0010 | 0011   | 0000           | хххх           | Intel386 CX processor  |
| 0010 | 0011   | 0000           | хххх           | Intel386 EX processor  |
| 0100 | 0011   | 0000 and 0001  | хххх           | Intel386 SL processor  |
| 0000 | 0011   | 0100           | хххх           | RapidCAD* coprocessor  |

 $^{2}$ All Intel486 SL-enhanced and Write-Back enhanced processors are capable of executing the CPUID instruction. See Table 4.



| Extended<br>Family | Extended<br>Model | Туре   | Family<br>Code | Model<br>Number | Stepping<br>ID  | Description                                                                                                                             |
|--------------------|-------------------|--------|----------------|-----------------|-----------------|-----------------------------------------------------------------------------------------------------------------------------------------|
| 0000000            | 0000              | 00     | 0100           | 000x            | XXXX (1)        | Intel486™ DX processors                                                                                                                 |
| 0000000            | 0000              | 00     | 0100           | 0010            | XXXX (1)        | Intel486 SX processors                                                                                                                  |
| 0000000            | 0000              | 00     | 0100           | 0011            | XXXX (1)        | Intel487™ processors                                                                                                                    |
| 0000000            | 0000              | 00     | 0100           | 0011            | XXXX (1)        | IntelDX2 <sup>™</sup> processors                                                                                                        |
| 0000000            | 0000              | 00     | 0100           | 0011            | XXXX (1)        | IntelDX2 OverDrive® processors                                                                                                          |
| 00000000           | 0000              | 00     | 0100           | 0100            | XXXX (3)        | Intel486 SL processor                                                                                                                   |
| 00000000           | 0000              | 00     | 0100           | 0101            | XXXX (1)        | IntelSX2 <sup>™</sup> processors                                                                                                        |
| 0000000            | 0000              | 00     | 0100           | 0111            | XXXX (3)        | Write-Back Enhanced IntelDX2 processors                                                                                                 |
| 0000000            | 0000              | 00     | 0100           | 1000            | XXXX (3)        | IntelDX4 <sup>™</sup> processors                                                                                                        |
| 0000000            | 0000              | 0x     | 0100           | 1000            | XXXX (3)        | IntelDX4 OverDrive processors                                                                                                           |
| 00000000           | 0000              | 00     | 0101           | 0001            | XXXX (2)        | Pentium <sup>®</sup> processors (60, 66)                                                                                                |
| 00000000           | 0000              | 00     | 0101           | 0010            | XXXX (2)        | Pentium processors (75, 90, 100, 120, 133, 150, 166, 200)                                                                               |
| 00000000           | 0000              | 01 (4) | 0101           | 0001            | XXXX (2)        | Pentium OverDrive processor for Pentium processor (60, 66)                                                                              |
| 00000000           | 0000              | 01 (4) | 0101           | 0010            | XXXX (2)        | Pentium OverDrive processor for Pentium processor (75, 90, 100, 120, 133)                                                               |
| 00000000           | 0000              | 01     | 0101           | 0011            | XXXX (2)        | Pentium OverDrive processors for Intel486 processor-based systems                                                                       |
| 00000000           | 0000              | 00     | 0101           | 0100            | XXXX (2)        | Pentium processor with MMX <sup>™</sup> technology<br>(166, 200)                                                                        |
| 00000000           | 0000              | 01     | 0101           | 0100            | XXXX (2)        | Pentium OverDrive processor with MMX™<br>technology for Pentium processor (75, 90, 100,<br>120, 133)                                    |
| 00000000           | 0000              | 00     | 0110           | 0001            | XXXX (2)        | Pentium Pro processor                                                                                                                   |
| 00000000           | 0000              | 00     | 0110           | 0011            | XXXX (2)        | Pentium II processor, model 3                                                                                                           |
| 0000000            | 0000              | 00     | 0110           | 0101(5)         | <b>ХХХХ</b> (2) | Pentium II processor, model 5, Pentium II Xeon<br>processor, model 5, and Intel <sup>®</sup> Celeron <sup>®</sup><br>processor, model 5 |
| 0000000            | 0000              | 00     | 0110           | 0110            | XXXX (2)        | Celeron processor, model 6                                                                                                              |
| 0000000            | 0000              | 00     | 0110           | 0111(6)         | XXXX (2)        | Pentium III processor, model 7, and Pentium III<br>Xeon processor, model 7                                                              |
| 00000000           | 0000              | 00     | 0110           | 1000(7)         | XXXX (2)        | Pentium III processor, model 8, Pentium III Xeon<br>processor, model 8, and Celeron processor,<br>model 8                               |
| 00000000           | 0000              | 00     | 0110           | 1001            | XXXX (2)        | Intel Pentium M processor model 9.                                                                                                      |
| 00000000           | 0000              | 00     | 0110           | 1010            | XXXX (2)        | Pentium III Xeon processor, model A                                                                                                     |
| 00000000           | 0000              | 00     | 0110           | 1011            | XXXX (2)        | Pentium III processor, model B                                                                                                          |

#### Table 4. Intel486<sup>™</sup> and Subsequent Processor Signatures

| Extended<br>Family | Extended<br>Model | Туре | Family<br>Code | Model<br>Number | Stepping<br>ID  | Description                                                                                                                                                                                                                                                      |
|--------------------|-------------------|------|----------------|-----------------|-----------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 00000000           | 0000              | 00   | 0110           | 1101            | XXXX (2)        | Intel Pentium M processor model D. All processors are manufactured using the 90 nm process.                                                                                                                                                                      |
| 00000000           | 0000              | 01   | 0110           | 0011            | XXXX (2)        | Intel Pentium II OverDrive processor                                                                                                                                                                                                                             |
| 00000000           | 0000              | 00   | 1111           | 0000            | <b>XXXX</b> (2) | Pentium 4 processor, Intel Xeon processor. All processors are model 0 and manufactured using the 0.18 micron process.                                                                                                                                            |
| 00000000           | 0000              | 00   | 1111           | 0001            | XXXX (2)        | Pentium 4 processor, Intel Xeon processor, Intel Xeon processor MP, and Intel Celeron processor. All processors are model 1 and manufactured using the 0.18 micron process.                                                                                      |
| 0000000            | 0000              | 00   | 1111           | 0010            | XXXX (2)        | Pentium 4 processor, Mobile Intel Pentium 4<br>processor – M, Intel Xeon processor, Intel Xeon<br>processor MP, Intel Celeron processor, and<br>Mobile Intel Celeron processor. All processors<br>are model 2 and manufactured using the 0.13<br>micron process. |
| 00000000           | 0000              | 00   | 1111           | 0011            | XXXX (2)        | Pentium 4 processor, Mobile Intel Pentium 4<br>processor – M, Intel Xeon processor, Intel<br>Celeron processor. All processors are model 3<br>and manufactured using the 90 nm process.                                                                          |

#### NOTES:

- 1. This processor does not implement the CPUID instruction.
- 2. Refer to the Intel486<sup>™</sup> documentation, the Pentium® Processor Specification Update (Document Number 242480), the Pentium® Pro Processor Specification Update (Document Number 243337), the Pentium® II Xeon Processor Specification Update (Document Number 243776), the Intel<sup>®</sup> Celeron<sup>®</sup> Processor Specification Update (Document Number 243748), the Pentium® II Processor Specification Update (Document Number 243748), the Pentium® III Processor Specification Update (Document Number 243748), the Pentium® III Processor Specification Update (Document Number 243748), the Pentium® III Processor Specification Update (Document Number 24453), the Pentium® III Processor Specification Update (Document Number 244460), the Pentium® 4 Processor Specification Update (Document Number 249678) or the Intel® Xeon™ Processor Specification Update (Document Number 249741) for the latest list of stepping numbers.
- 3. Stepping 3 implements the CPUID instruction.
- 4. The definition of the type field for the OverDrive processor is 01h. An erratum on the Pentium OverDrive processor will always return 00h as the type.
- 5. To differentiate between the Pentium II processor, model 5, Pentium II Xeon processor and the Celeron processor, model 5, software should check the cache descriptor values through executing CPUID instruction with EAX = 2. If no L2 cache is returned, the processor is identified as an Celeron processor, model 5. If 1M or 2M L2 cache size is reported, the processor is the Pentium II Xeon processor otherwise it is a Pentium II processor, model 5 or a Pentium II Xeon processor with 512-KB L2 cache size.
- 6. To differentiate between the Pentium III processor, model 7 and the Pentium III Xeon processor, model 7, software should check the cache descriptor values through executing CPUID instruction with EAX = 2. If 1M or 2M L2 cache size is reported, the processor is the Pentium III Xeon processor otherwise it is a Pentium III processor or a Pentium III Xeon processor with 512-KB L2 cache size.
- To differentiate between the Pentium III processor, model 8 and the Pentium III Xeon processor, model 8, software should check the Brand ID values through executing CPUID instruction with EAX = 1.



### **3.3 Feature Flags**

When the EAX register contains a value of 1, the CPUID instruction (in addition to loading the processor signature in the EAX register) loads the EDX and ECX register with the feature flags. The feature flags (when a Flag = 1) indicate what features the processor supports. Table 5 and Table 6 list the currently defined feature flag values.

For future processors, refer to the programmer's reference manual, user's manual, or the appropriate documentation for the latest feature flag values.

*Note:* Use the feature flags in your applications to determine which processor features are supported. By using the CPUID feature flags to determine processor features, your software can detect and avoid incompatibilities introduced by the addition or removal of processor features.

| Bit | Name | Description when<br>Flag = 1       | Comments                                                                                                                                                                                                      |
|-----|------|------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 0   | FPU  | Floating-point unit on-Chip        | The processor contains an FPU that supports the Intel387 floating-<br>point instruction set.                                                                                                                  |
| 1   | VME  | Virtual Mode Extension             | The processor supports extensions to virtual-8086 mode.                                                                                                                                                       |
| 2   | DE   | Debugging Extension                | The processor supports I/O breakpoints, including the CR4.DE bit for enabling debug extensions and optional trapping of access to the DR4 and DR5 registers.                                                  |
| 3   | PSE  | Page Size Extension                | The processor supports 4-Mbyte pages.                                                                                                                                                                         |
| 4   | TSC  | Time-Stamp Counter                 | The RDTSC instruction is supported including the CR4.TSD bit for access/privilege control.                                                                                                                    |
| 5   | MSR  | Model Specific Registers           | Model Specific Registers are implemented with the RDMSR, WRMSR instructions                                                                                                                                   |
| 6   | PAE  | Physical Address Extension         | Physical addresses greater than 32 bits are supported.                                                                                                                                                        |
| 7   | MCE  | Machine Check Exception            | Machine Check Exception, Exception 18, and the CR4.MCE enable bit are supported                                                                                                                               |
| 8   | CX8  | CMPXCHG8 Instruction<br>Supported  | The compare and exchange 8 bytes instruction is supported.                                                                                                                                                    |
| 9   | APIC | On-chip APIC Hardware<br>Supported | The processor contains a software-accessible Local APIC.                                                                                                                                                      |
| 10  |      | Reserved                           | Do not count on their value.                                                                                                                                                                                  |
| 11  | SEP  | Fast System Call                   | Indicates whether the processor supports the Fast System Call instructions, SYSENTER and SYSEXIT. NOTE: Refer to Section 3.4 for further information regarding SYSENTER/ SYSEXIT feature and SEP feature bit. |
| 12  | MTRR | Memory Type Range<br>Registers     | The Processor supports the Memory Type Range Registers specifically the MTRR_CAP register.                                                                                                                    |

#### Table 5. Feature Flag Values Reported in the EDX Register

| Bit | Name   | Description when<br>Flag = 1                                             | Comments                                                                                                                                                                                                                                                                                    |
|-----|--------|--------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 13  | PGE    | Page Global Enable                                                       | The global bit in the page directory entries (PDEs) and page table<br>entries (PTEs) is supported, indicating TLB entries that are common<br>to different processes and need not be flushed. The CR4.PGE bit<br>controls this feature.                                                      |
| 14  | MCA    | Machine Check Architecture                                               | The Machine Check Architecture is supported, specifically the MCG_CAP register.                                                                                                                                                                                                             |
| 15  | CMOV   | Conditional Move Instruction<br>Supported                                | The processor supports CMOVcc, and if the FPU feature flag (bit 0) is also set, supports the FCMOVCC and FCOMI instructions.                                                                                                                                                                |
| 16  | PAT    | Page Attribute Table                                                     | Indicates whether the processor supports the Page Attribute Table.<br>This feature augments the Memory Type Range Registers<br>(MTRRs), allowing an operating system to specify attributes of<br>memory on 4K granularity through a linear address.                                         |
| 17  | PSE-36 | 36-bit Page Size Extension                                               | 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.                         |
| 18  | PSN    | Processor serial number is present and enabled                           | The processor supports the 96-bit processor serial number feature, and the feature is enabled.                                                                                                                                                                                              |
|     |        |                                                                          | Note: The Pentium 4 family of processors do not support this feature.                                                                                                                                                                                                                       |
| 19  | CLFSH  | CLFLUSH Instruction<br>supported                                         | Indicates that the processor supports the CLFLUSH instruction.                                                                                                                                                                                                                              |
| 20  |        | Reserved                                                                 | Do not count on their value.                                                                                                                                                                                                                                                                |
| 21  | DS     | Debug Store                                                              | Indicates that the processor has the ability to write a history of the branch to and from addresses into a memory buffer.                                                                                                                                                                   |
| 22  | ACPI   | Thermal Monitor and<br>Software Controlled Clock<br>Facilities supported | The processor implements internal MSRs that allow processor temperature to be monitored and processor performance to be modulated in predefined duty cycles under software control.                                                                                                         |
| 23  | MMX    | Intel Architecture MMX technology supported                              | The processor supports the MMX technology instruction set extensions to Intel Architecture.                                                                                                                                                                                                 |
| 24  | FXSR   | Fast floating point save and restore                                     | Indicates whether the processor supports the FXSAVE and FXRSTOR instructions for fast save and restore of the floating point context. Presence of this bit also indicates that CR4.OSFXSR is available for an operating system to indicate that it uses the fast save/restore instructions. |
| 25  | SSE    | Streaming SIMD Extensions supported                                      | The processor supports the Streaming SIMD Extensions to the Intel Architecture.                                                                                                                                                                                                             |
| 26  | SSE2   | Streaming SIMD Extensions 2                                              | Indicates the processor supports the Streaming SIMD Extensions -<br>2 Instructions.                                                                                                                                                                                                         |
| 27  | SS     | Self-Snoop                                                               | The processor supports the management of conflicting memory types by performing a snoop of its own cache structure for transactions issued to the bus.                                                                                                                                      |



| Bit | Name | Description when<br>Flag = 1  | Comments                                                                                                                                                                                                                                                                                                                                                |
|-----|------|-------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 28  | НТТ  | Hyper-Threading<br>Technology | This processor's microarchitecture has the capability to operate as multiple logical processors within the same physical package.                                                                                                                                                                                                                       |
|     |      |                               | This field does not indicate that Hyper-Threading Technology has<br>been enabled for this specific processor. To determine if Hyper-<br>Threading Technology is supported, check the value returned in<br>EBX[23:16] after executing CPUID with EAX=1. If EBX[23:16]<br>contains a value >1, then the processor supports Hyper-Threading<br>Technology. |
| 29  | ТМ   | Thermal Monitor supported     | The processor implements the Thermal Monitor automatic thermal control circuit (TCC).                                                                                                                                                                                                                                                                   |
| 30  |      | Reserved                      | Do not count on their value.                                                                                                                                                                                                                                                                                                                            |
| 31  | PBE  | Pending Break Enable          | The processor supports the use of the FERR#/PBE# pin when the processor is in the stop-clock state (STPCLK# is asserted) to signal the processor that an interrupt is pending and that the processor should return to normal operation to handle the interrupt. Bit 10 (PBE enable) in the IA32_MISC_ENABLE MSR enables this capability.                |

#### Table 6. Feature Flag Values Reported in the ECX Register

| Bit   | Name    | Description when<br>Flag = 1          | Comments                                                                                                                                                                                                                                                               |  |
|-------|---------|---------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
| 0     | SSE3    | Streaming SIMD Extensions 3           | The processor supports the Streaming SIMD Extensions 3 instructions.                                                                                                                                                                                                   |  |
| 2:1   |         | Reserved                              | Do not count on their value.                                                                                                                                                                                                                                           |  |
| 3     | MONITOR | MONITOR/MWAIT                         | The processor supports the MONITOR and MWAIT instructions.                                                                                                                                                                                                             |  |
| 4     | DS-CPL  | CPL Qualified Debug Store             | The processor supports the extensions to the Debug Store feature to allow for branch message storage qualified by CPL.                                                                                                                                                 |  |
| 6:5   |         | Reserved                              | Do not count on their value.                                                                                                                                                                                                                                           |  |
| 7     | EIST    | Enhanced Intel SpeedStep® technology  | The processor implements the second-generation Intel SpeedStep technology feature.                                                                                                                                                                                     |  |
| 8     | TM2     | Thermal Monitor 2                     | The processor implements the Thermal Monitor 2 thermal control circuit (TCC).                                                                                                                                                                                          |  |
| 9     |         | Reserved Do not count on their value. |                                                                                                                                                                                                                                                                        |  |
| 10    | CID     | Context ID                            | The L1 data cache mode can be set to either adaptive mode or shared mode by the BIOS.                                                                                                                                                                                  |  |
| 13:11 |         | Reserved                              | Do not count on their value.                                                                                                                                                                                                                                           |  |
| 14    | xTPR    | Send Task Priority<br>Messages        | The processor supports the ability to disable sending Task Priority<br>messages. When this feature flag is set, Task Priority messages<br>may be disabled. Bit 23 (Echo TPR disable) in the<br>IA32_MISC_ENABLE MSR controls the sending of Task Priority<br>messages. |  |
| 31:15 |         | Reserved                              | Do not count on their value.                                                                                                                                                                                                                                           |  |

### 3.4 Extended Feature Flags

When the EAX register contains a value of 80000001h, the CPUID instruction loads the EDX register with the extended feature flags. The feature flags (when a Flag = 1) indicate what extended features the processor supports. Table 7 lists the currently defined extended feature flag values.

For future processors, refer to the programmer's reference manual, user's manual, or the appropriate documentation for the latest extended feature flag values.

*Note:* Use the extended feature flags in your applications to determine which processor features are supported. By using the CPUID feature flags to determine processor features, your software can detect and avoid incompatibilities introduced by the addition or removal of processor features.

| Bit    | Name         | Description when<br>Flag = 1            | Comments                                                                                                                                                                                                                                                                                          |
|--------|--------------|-----------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 28 - 0 |              | Reserved                                | Do not count on their value.                                                                                                                                                                                                                                                                      |
| 29     | Intel® EM64T | Intel® Extended Memory 64<br>Technology | The processor supports 64-bit extensions to the IA-32 Architecture.<br>For additional information refer to the "64-bit Extensions Technology<br>Software Developers Guide" (document numbers 300834 and<br>300835) available at:<br><u>http://developer.intel.com/technology/64bitextensions/</u> |
| 31-30  |              | Reserved                                | Do not count on their value.                                                                                                                                                                                                                                                                      |

#### Table 7. Extended Feature Flag Values Reported in the EDX Register



#### 3.5 SYSENTER/SYSEXIT – SEP Features Bit

The SYSENTER Present (SEP) bit 11 of CPUID indicates the presence of this facility. An operating system that detects the presence of the SEP bit must also qualify the processor family and model to ensure that the SYSENTER/SYSEXIT instructions are actually present:

```
IF (CPUID SEP bit is set)
{
    IF ((Processor Signature & 0x0FFF3FFF) < 0x00000633)
        Fast System Call is NOT supported
    ELSE
        Fast System Call is supported
}</pre>
```

The Pentium Pro processor (Model = 1) returns a set SEP CPUID feature bit, but should not be used by software.

#### 3.6 Cache Size, Format and TLB Information

When the EAX register contains a value of 2, the CPUID instruction loads the EAX, EBX, ECX and EDX registers with descriptors that indicate the processors cache and TLB characteristics. The lower 8 bits of the EAX register (AL) contain a value that identifies the number of times the CPUID has to be executed to obtain a complete image of the processor's caching systems. For example, the Pentium 4 processor returns a value of 1 in the lower 8 bits of the EAX register to indicate that the CPUID instruction need only be executed once (with EAX = 2) to obtain a complete image of the processor configuration.

The remainder of the EAX register, the EBX, ECX and EDX registers contain the cache and TLB descriptors. Table 8 shows that when bit 31 in a given register is zero, that register contains valid 8-bit descriptors. To decode descriptors, move sequentially from the most significant byte of the register down through the least significant byte of the register. Assuming bit 31 is 0, then that register contains valid cache or TLB descriptors in bits 24 through 31, bits 16 through 23, bits 8 through 15 and bits 0 through 7. Software must compare the value contained in each of the descriptor bit fields with the values found in Table 9 to determine the cache and TLB features of a processor

Table 9 lists the current cache and TLB descriptor values and their respective characteristics. This list will be extended in the future as necessary. Between models and steppings of processors the cache and TLB information may change bit field locations, therefore it is important that software not assume fixed locations when parsing the cache and TLB descriptors.

| Register bit 31 | Descriptor Type   | Description                                                                                                                    |
|-----------------|-------------------|--------------------------------------------------------------------------------------------------------------------------------|
| 1               | Reserved          | Reserved for future use.                                                                                                       |
| 0               | 8-bit descriptors | Descriptors point to a parameter<br>table to identify cache<br>characteristics. The descriptor is<br>null if it has a 0 value. |

#### Table 8. Descriptor Formats

#### Table 9. Descriptor Decode Values

| Value | Cache or TLB Description                                                                                                        |
|-------|---------------------------------------------------------------------------------------------------------------------------------|
| 00h   | Null                                                                                                                            |
| 01h   | Instruction TLB: 4-KBPages, 4-way set associative, 32 entries                                                                   |
| 02h   | Instruction TLB: 4-MB Pages, fully associative, 2 entries                                                                       |
| 03h   | Data TLB: 4-KB Pages, 4-way set associative, 64 entries                                                                         |
| 04h   | Data TLB: 4-MB Pages, 4-way set associative, 8 entries                                                                          |
| 06h   | 1 <sup>st</sup> -level instruction cache: 8-KB, 4-way set associative, 32-byte line size                                        |
| 08h   | 1 <sup>st</sup> -level instruction cache: 16-KB, 4-way set associative, 32-byte line size                                       |
| 0Ah   | 1 <sup>st</sup> -level data cache: 8-KB, 2-way set associative, 32-byte line size                                               |
| 0Ch   | 1 <sup>st</sup> -level data cache: 16-KB, 4-way set associative, 32-byte line size                                              |
| 22h   | 3 <sup>rd</sup> -level cache: 512 KB, 4-way set associative, sectored cache, 64-byte line size                                  |
| 23h   | 3 <sup>rd</sup> -level cache: 1-MB, 8-way set associative, sectored cache, 64-byte line size                                    |
| 25h   | 3 <sup>rd</sup> -level cache: 2-MB, 8-way set associative, sectored cache, 64-byte line size                                    |
| 29h   | 3 <sup>rd</sup> -level cache: 4-MB, 8-way set associative, sectored cache, 64-byte line size                                    |
| 2Ch   | 1 <sup>st</sup> -level data cache: 32-KB, 8-way set associative, 64-byte line size                                              |
| 30h   | 1 <sup>st</sup> -level instruction cache: 32-KB, 8-way set associative, 64-byte line size                                       |
| 39h   | 2 <sup>nd</sup> -level cache: 128-KB, 4-way set associative, sectored cache, 64-byte line size                                  |
| 3Bh   | 2 <sup>nd</sup> -level cache: 128-KB, 2-way set associative, sectored cache, 64-byte line size                                  |
| 3Ch   | 2 <sup>nd</sup> -level cache: 256-KB, 4-way set associative, sectored cache, 64-byte line size                                  |
| 40h   | No 2 <sup>nd</sup> -level cache or, if processor contains a valid 2 <sup>nd</sup> -level cache, no 3 <sup>rd</sup> -level cache |
| 41h   | 2 <sup>nd</sup> -level cache: 128-KB, 4-way set associative, 32-byte line size                                                  |
| 42h   | 2 <sup>nd</sup> -level cache: 256-KB, 4-way set associative, 32-byte line size                                                  |
| 43h   | 2 <sup>nd</sup> -level cache: 512-KB, 4-way set associative, 32-byte line size                                                  |
| 44h   | 2 <sup>nd</sup> -level cache: 1-MB, 4-way set associative, 32-byte line size                                                    |
| 45h   | 2 <sup>nd</sup> -level cache: 2-MB, 4-way set associative, 32-byte line size                                                    |
| 50h   | Instruction TLB: 4-KB, 2-MB or 4-MB pages, fully associative, 64 entries                                                        |
| 51h   | Instruction TLB: 4-KB, 2-MB or 4-MB pages, fully associative, 128 entries                                                       |
| 52h   | Instruction TLB: 4-KB, 2-MB or 4-MB pages, fully associative, 256 entries                                                       |
| 5Bh   | Data TLB: 4-KB or 4-MB pages, fully associative, 64 entries                                                                     |
| 5Ch   | Data TLB: 4-KB or 4-MB pages, fully associative, 128 entries                                                                    |
| 5Dh   | Data TLB: 4-KB or 4-MB pages, fully associative, 256 entries                                                                    |
| 60h   | 1 <sup>st</sup> -level data cache: 16-KB, 8-way set associative, sectored cache, 64-byte line size                              |
| 66h   | 1 <sup>st</sup> -level data cache: 8-KB, 4-way set associative, sectored cache, 64-byte line size                               |
| 67h   | 1 <sup>st</sup> -level data cache: 16-KB, 4-way set associative, sectored cache, 64-byte line size                              |
| 68h   | 1 <sup>st</sup> -level data cache: 32-KB, 4 way set associative, sectored cache, 64-byte line size                              |

| Value | Cache or TLB Description                                                                       |
|-------|------------------------------------------------------------------------------------------------|
| 70h   | Trace cache: 12K-uops, 8-way set associative                                                   |
| 71h   | Trace cache: 16K-uops, 8-way set associative                                                   |
| 72h   | Trace cache: 32K-uops, 8-way set associative                                                   |
| 78h   | 2 <sup>nd</sup> -level cache: 1-MB, 4-way set associative, 64-byte line size                   |
| 79h   | 2 <sup>nd</sup> -level cache: 128-KB, 8-way set associative, sectored cache, 64-byte line size |
| 7Ah   | 2 <sup>nd</sup> -level cache: 256-KB, 8-way set associative, sectored cache, 64-byte line size |
| 7Bh   | 2 <sup>nd</sup> -level cache: 512-KB, 8-way set associative, sectored cache, 64-byte line size |
| 7Ch   | 2 <sup>nd</sup> -level cache: 1-MB, 8-way set associative, sectored cache, 64-byte line size   |
| 7Dh   | 2 <sup>nd</sup> -level cache: 2-MB, 8-way set associative, 64-byte line size                   |
| 7Fh   | 2 <sup>nd</sup> -level cache: 512-KB, 2-way set associative, 64-byte line size                 |
| 82h   | 2 <sup>nd</sup> -level cache: 256-KB, 8-way set associative, 32-byte line size                 |
| 83h   | 2 <sup>nd</sup> -level cache: 512-KB, 8-way set associative, 32-byte line size                 |
| 84h   | 2 <sup>nd</sup> -level cache: 1-MB, 8-way set associative, 32-byte line size                   |
| 85h   | 2 <sup>nd</sup> -level cache: 2-MB, 8-way set associative, 32-byte line size                   |
| 86h   | 2 <sup>nd</sup> -level cache: 512-KB, 4-way set associative, 64-byte line size                 |
| 87h   | 2 <sup>nd</sup> -level cache: 1-MB, 8-way set associative, 64-byte line size                   |
| B0h   | Instruction TLB: 4-KB Pages, 4-way set associative, 128 entries                                |
| B3h   | Data TLB: 4-KB Pages, 4-way set associative, 128 entries                                       |
| F0h   | 64-byte Prefetching                                                                            |
| F1h   | 128-byte Prefetching                                                                           |

### 3.7 Pentium<sup>®</sup> 4 Processor, Model 0 Output Example

The Pentium 4 processor, model 0 returns the values shown in Table 8. Since the value of AL=1, it is valid to interpret the remainder of the registers. Table 8 also shows the MSB (bit 31) of all the registers are 0 which indicates that each register contains valid 8-bit descriptor. The register values in Table 8 show that this Pentium 4 processor has the following cache and TLB characteristics:

- (66h) A 1<sup>st</sup>-level data cache that is 8 KB, 4-way set associative, dual-sectored line, with 64-byte sector size.
- (5Bh) A data TLB that maps 4-KB or 4-MB pages, is fully associative, and has 64 entries.
- (50h) An instruction TLB that maps 4-KB, 2-MB or 4-MB pages, is fully associative, and has 64 entries.
- (7Ah) A 2<sup>nd</sup>-level cache that is 256-KB, 8-way set associative, dual-sectored line, with 64byte sector size.
- (70h) A trace cache that can store up to 12K-uops, and is 8-way set associative.
- (40h) No 3<sup>rd</sup>-level cache

### Table 10. Pentium® 4 Processor, Model 0 with 256-KB L2 Cache CPUID (EAX=2) Example Return Values

|     | 31  | 23  | 15  | 7   | 0 |
|-----|-----|-----|-----|-----|---|
| EAX | 66h | 5Bh | 50h | 01h |   |
| EBX | 00h | 00h | 00h | 00h |   |
| ECX | 00h | 00h | 00h | 00h |   |
| EDX | 00h | 7Ah | 70h | 40h |   |

§

### 4 Processor Serial Number

The processor serial number extends the concept of processor identification. Processor serial number is a 96-bit number accessible through the CPUID instruction. Processor serial number can be used by applications to identify a processor, and by extension, its system.

The processor serial number creates a software accessible identity for an individual processor. The processor serial number, combined with other qualifiers, could be applied to user identification. Applications include membership authentication, data backup/restore protection, removable storage data protection, managed access to files, or to confirm document exchange between appropriate users.

Processor serial number is another tool for use in asset management, product tracking, remote systems load and configuration, or to aid in boot-up configuration. In the case of system service, processor serial number could be used to differentiate users during help desk access, or track error reporting. Processor serial number provides an identifier for the processor, but should not be assumed to be unique in itself. There are potential modes in which erroneous processor serial numbers may be reported. For example, in the event a processor is operated outside its recommended operating specifications, (e.g., voltage, frequency, etc.) the processor serial number may not be correctly read from the processor. Improper BIOS or software operations could yield an inaccurate processor serial number. These events could lead to possible erroneous or duplicate processor serial numbers being reported. System manufacturers can strengthen the robustness of the feature by including redundancy features, or other fault tolerant methods.

Processor serial number used as a qualifier for another independent number could be used to create an electrically accessible number that is likely to be distinct. Processor serial number is one building block useful for the purpose of enabling the trusted, connected PC.

#### 4.1 Presence of Processor Serial Number

To determine if the processor serial number feature is supported, the program should set the EAX register parameter value to "1" and then execute the CPUID instruction as follows:

```
MOV EAX, 01H
CPUID
```

After execution of the CPUID instruction, the ECX and EDX register contains the Feature Flags. If the PSN Feature Flags, (EDX register, bit 18) equals "1", the processor serial number feature is supported, and enabled. If the PSN Feature Flags equals "0", the processor serial number feature is either not supported, or disabled in a Pentium III processor.



### 4.2 Forming the 96-bit Processor Serial Number

The 96-bit processor serial number is the concatenation of three 32-bit entities.

To access the most significant 32-bits of the processor serial number the program should set the EAX register parameter value to "1" and then execute the CPUID instruction as follows:

MOV EAX, 01H CPUID

After execution of the CPUID instruction, the EAX register contains the Processor Signature. The Processor Signature comprises the most significant 32-bits of the processor serial number. The value in EAX should be saved prior to gathering the remaining 64-bits of the processor serial number.

To access the remaining 64-bits of the processor serial number the program should set the EAX register parameter value to "3" and then execute the CPUID instruction as follows:

MOV EAX, 03H CPUID

After execution of the CPUID instruction, the EDX register contains the middle 32-bits, and the ECX register contains the least significant 32-bits of the processor serial number. Software may then concatenate the saved Processor Signature, EDX, and ECX before returning the complete 96-bit processor serial number.

Processor serial number should be displayed as 6 groups of 4 hex nibbles (Ex. XXXX-XXXX-XXXX-XXXX-XXXX-XXXX) where X represents a hex digit). Alpha hex characters should be displayed as capital letters.

§

### 5 Brand ID and Brand String

### 5.1 Brand ID

Beginning with the Pentium III processors, model 8, the Pentium III Xeon processors, model 8, and Celeron processor, model 8, the concept of processor identification is further extended with the addition of Brand ID. Brand ID is an 8-bit number accessible through the CPUID instruction. Brand ID may be used by applications to assist in identifying the processor.

Processors that implement the Brand ID feature return the Brand ID in bits 7 through 0 of the EBX register when the CPUID instruction is executed with EAX=1 (see Table 11). Processors that do not support the feature return a value of 0 in EBX bits 7 through 0.

To differentiate previous models of the Pentium II processor, Pentium II Xeon processor, Celeron processor, Pentium III processor and Pentium III Xeon processor, application software relied on the L2 cache descriptors. In a few cases, the results were ambiguous; for example, software could not accurately differentiate a Pentium II processor from a Pentium II Xeon processor with a 512-KB L2 cache. Brand ID eliminates this ambiguity by providing a software accessible value unique to each processor brand. Table 11 shows the values defined for each processor.

### 5.2 Brand String

The Brand string is a new extension to the CPUID instruction implemented in some Intel IA-32 processors, including the Pentium 4 processor. Using the brand string feature, future IA-32 architecture based processors will return their ASCII brand identification string and maximum operating frequency via an extended CPUID instruction. Note that the frequency returned is the maximum operating frequency that the processor has been qualified for and not the current operating frequency of the processor.

When CPUID is executed with EAX set to the values listed in Table 1, the processor will return an ASCII brand string in the general-purpose registers as detailed in Table 1.

The brand/frequency string is defined to be 48 characters long, 47 bytes will contain characters and the  $48^{\text{th}}$  byte is defined to be NULL (0). A processor may return less than the 47 ASCII characters as long as the string is null terminated and the processor returns valid data when CPUID is executed with EAX = 80000002h, 8000003h and 80000004h.

The cpuid3a.asm program shows how software forms the brand string (see Example 1). To determine if the brand string is supported on a processor, software must follow the step below:

- 1. Execute the CPUID instruction with EAX=8000000h
- 2. If ((returned value in EAX) > 8000000h) then the processor supports the extended CPUID functions and EAX contains the largest extended function supported.
- 3. The processor brand string feature is supported if  $EAX \ge 8000004h$



| Value            | Description                                                                                                 |
|------------------|-------------------------------------------------------------------------------------------------------------|
| 00h              | Unsupported                                                                                                 |
| 01h              | Intel® Celeron® processor                                                                                   |
| 02h              | Intel® Pentium® III processor                                                                               |
| 03h              | Intel® Pentium® III Xeon™ processor<br>If processor signature = 000006B1h, then "Intel® Celeron® processor" |
| 04h              | Intel® Pentium® III processor                                                                               |
| 06h              | Mobile Intel® Pentium® III Processor-M                                                                      |
| 07h              | Mobile Intel® Celeron® processor                                                                            |
| 08h              | Intel® Pentium® 4 processor<br>If processor signature is >=00000F13h, then "Intel® Genuine processor"       |
| 09h              | Intel® Pentium® 4 processor                                                                                 |
| 0Ah              | Intel® Celeron® Processor                                                                                   |
| 0Bh              | Intel® Xeon™ processor<br>If processor signature is <00000F13h, then "Intel® Xeon™ processor MP"            |
| 0Ch              | Intel® Xeon™ processor MP                                                                                   |
| 0Eh              | Mobile Intel® Pentium® 4 processor–M<br>If processor signature is <00000F13h, then "Intel® Xeon™ processor" |
| 0Fh              | Mobile Intel® Celeron® processor                                                                            |
| 11h              | Mobile Genuine Intel® processor                                                                             |
| 12h              | Intel® Celeron® M processor                                                                                 |
| 13h              | Mobile Intel® Celeron® processor                                                                            |
| 14h              | Intel® Celeron® Processor                                                                                   |
| 15h              | Mobile Genuine Intel® processor                                                                             |
| 16h              | Intel® Pentium® M processor                                                                                 |
| 17h              | Mobile Intel® Celeron® processor                                                                            |
| All other values | Reserved                                                                                                    |

#### Table 11. Brand ID, CPUID (EAX=1) Return Values in EBX (bits 7 through 0)

#### Table 12. Processor Brand String Feature

| EAX Input<br>Value | Function                                                  | Return Value                                                                  |
|--------------------|-----------------------------------------------------------|-------------------------------------------------------------------------------|
| 80000000h          | Largest Extended Function<br>Supported                    | EAX=Largest supported extended function number,<br>EBX = ECX = EDX = Reserved |
| 80000001h          | Extended Processor Signature<br>and Extended Feature Bits | EAX = Extended Feature Flags<br>EBX = ECX = EDX = Reserved                    |
| 8000002h           | Processor Brand String                                    | EAX, EBX, ECX, EDX contain ASCII brand string                                 |
| 8000003h           | Processor Brand String                                    | EAX, EBX, ECX, EDX contain ASCII brand string                                 |
| 80000004h          | Processor Brand String                                    | EAX, EBX, ECX, EDX contain ASCII brand string                                 |

### 6 Usage Guidelines

This document presents Intel-recommended feature-detection methods. Software should not try to identify features by exploiting programming tricks, undocumented features, or otherwise deviating from the guidelines presented in this application note.

The following guidelines are intended to help programmers maintain the widest range of compatibility for their software.

- Do not depend on the absence of an invalid opcode trap on the CPUID opcode to detect the CPUID instruction. Do not depend on the absence of an invalid opcode trap on the PUSHFD opcode to detect a 32-bit processor. Test the ID flag, as described in Section 2 and shown in Section 7.
- Do not assume that a given family or model has any specific feature. For example, do not assume the family value 5 (Pentium processor) means there is a floating-point unit on-chip. Use the feature flags for this determination.
- Do not assume processors with higher family or model numbers have all the features of a processor with a lower family or model number. For example, a processor with a family value of 6 (P6 family processor) may not necessarily have all the features of a processor with a family value of 5.
- Do not assume that the features in the OverDrive processors are the same as those in the OEM version of the processor. Internal caches and instruction execution might vary.
- Do not use undocumented features of a processor to identify steppings or features. For example, the Intel386 processor A-step had bit instructions that were withdrawn with the B-step. Some software attempted to execute these instructions and depended on the invalid-opcode exception as a signal that it was not running on the A-step part. The software failed to work correctly when the Intel486 processor used the same opcodes for different instructions. The software should have used the stepping information in the processor signature.
- Test feature flags individually and do not make assumptions about undefined bits. For example, it would be a mistake to test the FPU bit by comparing the feature register to a binary 1 with a compare instruction.
- Do not assume the clock of a given family or model runs at a specific frequency, and do not write processor speed-dependent code, such as timing loops. For instance, an OverDrive Processor could operate at a higher internal frequency and still report the same family and/or model. Instead, use a combination of the system's timers to measure elapsed time and the TSC (Time-Stamp Counter) to measure processor core clocks to allow direct calibration of the processor core. See Section 11 and Example 6 for details.
- Processor model-specific registers may differ among processors, including in various models
  of the Pentium processor. Do not use these registers unless identified for the installed
  processor. This is particularly important for systems upgradeable with an OverDrive
  processor. Only use Model Specific registers that are defined in the BIOS writers guide for
  that processor.
- Do not rely on the result of the CPUID algorithm when executed in virtual 8086 mode.



- Do not assume any ordering of model and/or stepping numbers. They are assigned arbitrarily.
- Do not assume processor serial number is a unique number without further qualifiers.
- Display alpha hex characters as capital letters.
- A zero in the lower 64 bits of the processor serial number indicate the processor serial number is invalid, not supported, or disabled on this processor.

§

### 7 Proper Identification Sequence

To identify the processor using the CPUID instructions, software should follow the following steps.

- 1. Determine if the CPUID instruction is supported by modifying the ID flag in the EFLAGS register. If the ID flag cannot be modified, the processor cannot be identified using the CPUID instruction.
- 2. Execute the CPUID instruction with EAX equal to 80000000h. CPUID function 80000000h is used to determine if Brand String is supported. If the CPUID function 80000000h returns a value in EAX greater than 80000004h the Brand String feature is supported and software should use CPUID functions 80000002h through 80000004h to identify the processor.
- 3. If the Brand String feature is not supported, execute CPUID with EAX equal to 1. CPUID function 1 returns the processor signature in the EAX register, and the Brand ID in the EBX register bits 0 through 7. If the EBX register bits 0 through 7 contain a non-zero value, the Brand ID is supported. Software should scan the list of Brand IDs (see Table 11) to identify the processor.
- 4. If the Brand ID feature is not supported, software should use the processor signature (see Figure 2) in conjunction with the cache descriptors (see Table 9) to identify the processor.

The cpuid3a.asm program example demonstrates the correct use of the CPUID instruction (see Example 1). It also shows how to identify earlier processor generations that do not implement the Brand String, Brand ID, processor signature or CPUID instruction (see Figure 5). This program example contains the following two procedures:

- get\_cpu\_type identifies the processor type. Figure 5 illustrates the flow of this procedure.
- get\_fpu\_type determines the type of floating-point unit (FPU) or math coprocessor (MCP).

This procedure has been tested with 8086, 80286, Intel386, Intel486, Pentium processor, Pentium processor with MMX technology, OverDrive processor with MMX technology, Pentium Pro processors, Pentium II processors, Pentium II Xeon processors, Pentium II Overdrive processors, Celeron processors, Pentium III processors, Pentium III Xeon processors and Pentium 4 processors. This program example is written in assembly language and is suitable for inclusion in a run-time library, or as system calls in operating systems.



### Figure 5. Flow of Processor get\_cpu\_type Procedure

§

### 8 Usage Program Examples

The cpuid3b.asm or cpuid3.c program examples demonstrate applications that call get\_cpu\_type and get\_fpu\_type procedures and interpret the returned information. This code is shown in Example 2 and Example 3. The results, which are displayed on the monitor, identify the installed processor and features. The cpuid3b.asm example is written in assembly language and demonstrates an application that displays the returned information in the DOS environment. The cpuid3.c example is written in the C language (see Example 2 and Example 3). Figure 6 presents an overview of the relationship between the three program examples.

### Figure 6. Flow of Processor Identification Extraction Procedure



§

### 9 Alternate Method of Detecting Features

Some feature flags indicate support of instruction set extensions (i.e. MMX, SSE and SSE2). The preferred mechanism for determining support of instruction extensions is through the use of the CPUID instruction, and testing the feature flags. However, an alternate method for determining processor support of instruction extensions is to install an exception handler and execute one of the instructions. If the instruction extensions. If an exception is raised, and the exception handler is executed, then those instruction extensions are not supported by the processor. Before installing the exception handler, the software should execute the CPUID instruction with EAX = 0. If the CPUID instruction returns the Intel vendor-ID string "GenuineIntel", then software knows that it can test for the Intel instruction extensions. As long as the CPUID instruction returns the Intel vendor-ID, this method can be used to support future Intel processors. This method does not require software to check the family and model.

The features.cpp program is written using the C++ language (see Example 4) and demonstrates the use of exceptions to determine support of SSE3, SSE2, SSE, and MMX instruction extensions. Features.cpp performs the following steps:

- 1. Check that the vendor-ID == "GenuineIntel"
- 2. Install exception handler for SSE3 test
- 3. Attempt to execute a SSE3 instruction (haddpd xmm1, xmm2)
- 4. Install exception handler for SSE2 test
- 5. Attempt to execute a SSE2 instruction (paddq xmm1, xmm2)
- 6. Install exception handler for SSE test
- 7. Attempt to execute a SSE instruction (orps xmm1, xmm2)
- 8. Install exception handler for MMX test
- 9. Attempt to execute a MMX instruction (emms)
- 10. Print supported instruction set extensions.

### §

### 10 Denormals Are Zero

With the introduction of the SSE2 extensions, some Intel Architecture processors have the ability to convert SSE and SSE2 source operand denormal numbers to zero. This feature is referred to as Denormals-Are-Zero (DAZ). The DAZ mode is not compatible with IEEE Standard 754. The DAZ mode is provided to improve processor performance for applications such as streaming media processing, where rounding a denormal operand to zero does not appreciably affect the quality of the processed data.

Some processor steppings support SSE2 but do not support the DAZ mode. To determine if a processor supports the DAZ mode, software must perform the following steps.

- 1. Execute the CPUID instruction with an input value of EAX=0 and ensure the vendor-ID string returned is "GenuineIntel".
- 2. Execute the CPUID instruction with EAX=1. This will load the EDX register with the feature flags.
- 3. Ensure that the FXSR feature flag (EDX bit 24) is set. This indicates the processor supports the FXSAVE and FXRSTOR instructions.
- Ensure that the XMM feature flag (EDX bit 25) or the EMM feature flag (EDX bit 26) is set. This indicates that the processor supports at least one of the SSE/SSE2 instruction sets and its MXCSR control register.
- 5. Zero a 16-byte aligned, 512-byte area of memory. This is necessary since some implementations of FXSAVE do not modify reserved areas within the image.
- 6. Execute an FXSAVE into the cleared area.
- Bytes 28-31 of the FXSAVE image are defined to contain the MXCSR\_MASK. If this value is 0, then the processor's MXCSR\_MASK is 0xFFBF, otherwise MXCSR\_MASK is the value of this dword.
- 8. If bit 6 of the MXCSR\_MASK is set, then DAZ is supported.

After completing this algorithm, if DAZ is supported, software can enable DAZ mode by setting bit 6 in the MXCSR register save area and executing the FXRSTOR instruction. Alternately software can enable DAZ mode by setting bit 6 in the MXCSR by executing the LDMXCSR instruction. Refer to the chapter titled "Programming with the Streaming SIMD Extensions (SSE)" in the Intel Architecture Software Developer's Manual volume 1: Basic Architecture.

The assembly language program dazdtect.asm (see Example 5) demonstrates this DAZ detection algorithm.

§

### **11 Operating Frequency**

With the introduction of the Time-Stamp Counter, it is possible for software operating in real mode or protected mode with ring 0 privilege to calculate the actual operating frequency of the processor. To calculate the operating frequency, the software needs a reference period. The reference period can be a periodic interrupt, or another timer that is based on time, and not based on a system clock. Software needs to read the Time-Stamp Counter (TSC) at the beginning and ending of the reference period. Software can read the TSC by executing the RDTSC instruction, or by setting the ECX register to 10h and executing the RDMSR instruction. Both instructions copy the current 64-bit TSC into the EDX:EAX register pair.

To determine the operating frequency of the processor, software performs the following steps. The assembly language program frequenc.asm (see Example 6) demonstrates the frequency detection algorithm.

- 1. Execute the CPUID instruction with an input value of EAX=0 and ensure the vendor-ID string returned is "GenuineIntel".
- 2. Execute the CPUID instruction with EAX=1 to load the EDX register with the feature flags.
- 3. Ensure that the TSC feature flag (EDX bit 4) is set. This indicates the processor supports the Time-Stamp Counter and RDTSC instruction.
- 4. Read the TSC at the beginning of the reference period
- 5. Read the TSC at the end of the reference period.
- 6. Compute the TSC delta from the beginning and ending of the reference period.
- 7. Compute the actual frequency by dividing the TSC delta by the reference period.

Actual frequency = (Ending TSC value – Beginning TSC value) / reference period

*Note:* The measured accuracy is dependent on the accuracy of the reference period. A longer reference period produces a more accurate result. In addition, repeating the calculation multiple times may also improve accuracy.

### Example 1. Processor Identification Extraction Procedure

| ;    | Filename: cpuid3a.asm                                                    |
|------|--------------------------------------------------------------------------|
| :    | Copyright (c) Intel Corporation 1993-2004                                |
| •    |                                                                          |
| ·    | This preserve has been developed by Intel Corresponding Intel            |
| ;    | This program has been developed by Intel Corporation. Intel              |
| ;    | has various intellectual property rights which it may assert             |
| ;    | under certain circumstances, such as if another                          |
|      | manufacturer's processor mis-identifies itself as being                  |
| ,    | "GenuineIntel" when the CPUID instruction is executed.                   |
| ;    | Genumentier when the CPUID instruction is executed.                      |
| ;    |                                                                          |
| ;    | Intel specifically disclaims all warranties, express or                  |
| ;    | implied, and all liability, including consequential and other            |
| •    | indirect damages, for the use of this program, including                 |
| ,    |                                                                          |
| ;    | liability for infringement of any proprietary rights,                    |
| ;    | and including the warranties of merchantability and fitness              |
| ;    | for a particular purpose. Intel does not assume any                      |
| ;    | responsibility for any errors which may appear in this program           |
|      | nor any responsibility to update it.                                     |
| ,    | nor any responsionity to update it.                                      |
| ;;;; |                                                                          |
| ;    | This code contains two procedures:                                       |
| ;    | _get_cpu_type: Identifies processor type in _cpu_type:                   |
| ;    | 0=8086/8088 processor                                                    |
|      | •                                                                        |
| ;    | 2=Intel 286 processor                                                    |
| ;    | 3=Intel386(TM) family processor                                          |
| ;    | 4=Intel486(TM) family processor                                          |
| ;    | 5=Pentium(R) family processor                                            |
| ;    | 6=P6 family of processors                                                |
|      | F=Pentium 4 family of processors                                         |
| ;    | r=rentium 4 ranning or processors                                        |
| ;    |                                                                          |
| ;    | _get_fpu_type: Identifies FPU type in _fpu_type:                         |
| ;    | 0=FPU not present                                                        |
| ;    | 1=FPU present                                                            |
|      | 2=287 present (only if cpu_type=3)                                       |
| ,    |                                                                          |
| ;    | 3=387 present (only if cpu_type=3)                                       |
| ;    |                                                                          |
| ;    | This program has been tested with the Microsoft Developer Studio.        |
| :    | This code correctly detects the current Intel 8086/8088,                 |
|      | 80286, 80386, 80486, Pentium(R) processor, Pentium(R) Pro                |
| ,    |                                                                          |
| ;    | processor, Pentium(R) II processor, Pentium II Xeon(TM) processor,       |
| ;    | Pentium II Overdrive(R), Intel Celeron processor, Pentium III processor, |
| ;    | Pentium III Xeon processor, Pentium 4 processors and                     |
| ;    | Intel(R) Xeon(TM) processors.                                            |
| '    | · · · · · · · · · · · · · · · · · · ·                                    |
|      | NOTE: When using this code with C program enuid? a 22 bit                |
| ,    | NOTE: When using this code with C program cpuid3.c, 32-bit               |
| ;    | segments are recommended.                                                |
|      |                                                                          |
| ;    | To assemble this code with TASM, add the JUMPS directive.                |
| ;    | jumps ; Uncomment this line for TASM                                     |

| TITLE | cpuid3a |
|-------|---------|
|-------|---------|

comment this line for 32-bit segments

### ; ; co ; DOSSEG

| ;;          | uncomment the following 2 lines for 32-bit segments |
|-------------|-----------------------------------------------------|
| ;<br>;<br>; | .386<br>.model flat                                 |
| ;           | comment this line for 32-bit segments               |
| ,           | .model small                                        |

; Hardcoded CPUID instruction

CPU\_IDMACRO

| db | 0fh  |  |
|----|------|--|
| db | 0a2h |  |
|    |      |  |

### ENDM

### .data

| public _cpu_ty<br>public _fpu_ty<br>public _fpu_ty<br>public _v86_f<br>public _cpuid<br>public _intel_0<br>public _cpu_s<br>public _featur<br>public _featur<br>public _featur<br>public _cache<br>public _cache<br>public _cache<br>public _cache<br>public _sep_fl<br>public _brand | /pe<br>lag<br>_flag<br>CPU<br>r_id<br>ignature<br>es_ecx<br>es_edx<br>es_ebx<br>_eax<br>_ebx<br>_ecx<br>_edx<br>ag |                |
|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------|----------------|
| _cpu_type                                                                                                                                                                                                                                                                             | db                                                                                                                 | 0              |
| _fpu_type                                                                                                                                                                                                                                                                             | db                                                                                                                 | 0              |
| _v86_flag                                                                                                                                                                                                                                                                             | db                                                                                                                 | 0              |
| _cpuid_flag                                                                                                                                                                                                                                                                           | db                                                                                                                 | 0              |
| _intel_CPU                                                                                                                                                                                                                                                                            | db                                                                                                                 | 0              |
| _sep_flag                                                                                                                                                                                                                                                                             | db                                                                                                                 | 0              |
| _vendor_id                                                                                                                                                                                                                                                                            | db                                                                                                                 | ""             |
| intel_id                                                                                                                                                                                                                                                                              | db                                                                                                                 | "GenuineIntel" |
| _cpu_signature                                                                                                                                                                                                                                                                        | dd                                                                                                                 | 0              |
| _features_ecx                                                                                                                                                                                                                                                                         | dd                                                                                                                 | 0              |
| _features_edx                                                                                                                                                                                                                                                                         | dd                                                                                                                 | 0              |
| _features_ebx                                                                                                                                                                                                                                                                         | dd                                                                                                                 | 0              |
| _cache_eax                                                                                                                                                                                                                                                                            | dd                                                                                                                 | 0              |
| _cache_ebx                                                                                                                                                                                                                                                                            | dd                                                                                                                 | 0              |
|                                                                                                                                                                                                                                                                                       |                                                                                                                    |                |

```
0
        _cache_ecx
                         dd
        _cache_edx
                         dd
                                 0
                                 0
        fp_status
                         dw
        _brand_string
                         db
                                 48 dup (0)
.code
        comment this line for 32-bit segments
:
.8086
        uncomment this line for 32-bit segments
        .386
:
*************************
        public _get_cpu_type
        _get_cpu_type proc
        This procedure determines the type of processor in a system
;
        and sets the _cpu_type variable with the appropriate
;
        value. If the CPUID instruction is available, it is used
;
        to determine more specific details about the processor.
;
        All registers are used by this procedure, none are preserved.
        To avoid AC faults, the AM bit in CR0 must not be set.
        Intel 8086 processor check
;
        Bits 12-15 of the FLAGS register are always set on the
;
;
        8086 processor.
        For 32-bit segments comment the following lines down to the next
        comment line that says "STOP"
check_8086:
        pushf
                                         ; push original FLAGS
                                         ; get original FLAGS
        pop
                ax
        mov
                cx, ax
                                         ; save original FLAGS
        and
                ax, Offfh
                                         ; clear bits 12-15 in FLAGS
                                         ; save new FLAGS value on stack
        push
                ax
        popf
                                         ; replace current FLAGS value
        pushf
                                         ; get new FLAGS
                                         ; store new FLAGS in AX
        pop
                ax
                ax, 0f000h
                                         ; if bits 12-15 are set, then
        and
        cmp
                ax, 0f000h
                                         ; processor is an 8086/8088
                                         ; turn on 8086/8088 flag
        mov
                _cpu_type, 0
                check_80286
                                         ; go check for 80286
        ine
                                         ; double check with push sp
        push
                sp
                dx
                                         ; if value pushed was different
        pop
                dx, sp
                                         ; means it's really an 8086
        cmp
                                         ; jump if processor is 8086/8088
        jne
                end_cpu_type
                                         ; indicate unknown processor
        mov
                _cpu_type, 10h
```

| jmp | end_ | cpu | type |
|-----|------|-----|------|
| Jr  |      |     |      |

Intel 286 processor check ;

- Bits 12-15 of the FLAGS register are always clear on the ;
- Intel 286 processor in real-address mode. ;

### .286

check 80286:

| check_8      | 80286:                                            |                                                                 |                                        |  |  |  |  |
|--------------|---------------------------------------------------|-----------------------------------------------------------------|----------------------------------------|--|--|--|--|
|              | smsw                                              | ax                                                              | ; save machine status word             |  |  |  |  |
|              | and                                               | ax, 1                                                           | ; isolate PE bit of MSW                |  |  |  |  |
|              | mov                                               | _v86_flag, al                                                   | ; save PE bit to indicate V86          |  |  |  |  |
|              |                                                   |                                                                 |                                        |  |  |  |  |
|              | or                                                | cx, 0f000h                                                      | ; try to set bits 12-15                |  |  |  |  |
|              | push                                              | сх                                                              | ; save new FLAGS value on stack        |  |  |  |  |
|              | popf                                              |                                                                 | ; replace current FLAGS value          |  |  |  |  |
|              | pushf                                             |                                                                 | ; get new FLAGS                        |  |  |  |  |
|              | рор                                               | ax                                                              | ; store new FLAGS in AX                |  |  |  |  |
|              | and                                               | ax, 0f000h                                                      | ; if bits 12-15 are clear              |  |  |  |  |
|              | mov                                               | _cpu_type, 2                                                    | ; processor=80286, turn on 80286 flag  |  |  |  |  |
|              | jz                                                | end_cpu_type                                                    | ; jump if processor is 80286           |  |  |  |  |
|              | 0                                                 |                                                                 |                                        |  |  |  |  |
| :            | Intel386                                          | o processor check                                               |                                        |  |  |  |  |
| :            |                                                   | bit, bit #18, is a new bit in                                   | troduced in the EFLAGS                 |  |  |  |  |
| :            |                                                   |                                                                 |                                        |  |  |  |  |
| :            | faults.                                           | register on the Intel486 processor to generate alignment faults |                                        |  |  |  |  |
| :            | This bit cannot be set on the Intel386 processor. |                                                                 |                                        |  |  |  |  |
| ,            |                                                   |                                                                 | 1                                      |  |  |  |  |
| .386         |                                                   |                                                                 |                                        |  |  |  |  |
| :            |                                                   |                                                                 |                                        |  |  |  |  |
| :            | "STOP"                                            | ,                                                               |                                        |  |  |  |  |
| :            |                                                   |                                                                 |                                        |  |  |  |  |
| :            |                                                   |                                                                 | ; it is safe to use 386 instructions   |  |  |  |  |
| ,<br>check_8 | 30386:                                            |                                                                 | , 10 10 5410 10 400 2000 1104 40410115 |  |  |  |  |
| •••••••      | pushfd                                            |                                                                 | ; push original EFLAGS                 |  |  |  |  |
|              | pop                                               | eax                                                             | ; get original EFLAGS                  |  |  |  |  |
|              | mov                                               | ecx, eax                                                        | ; save original EFLAGS                 |  |  |  |  |
|              | xor                                               | eax, 40000h                                                     | ; flip AC bit in EFLAGS                |  |  |  |  |
|              | push                                              | eax                                                             | ; save new EFLAGS value on stack       |  |  |  |  |
|              | popfd                                             | Curr                                                            | ; replace current EFLAGS value         |  |  |  |  |
|              | pushfd                                            |                                                                 | ; get new EFLAGS                       |  |  |  |  |
|              | pop                                               | eax                                                             | ; store new EFLAGS in EAX              |  |  |  |  |
|              | xor                                               | eax, ecx                                                        | ; can't toggle AC bit, processor=80386 |  |  |  |  |
|              | mov                                               | _cpu_type, 3                                                    | ; turn on 80386 processor flag         |  |  |  |  |
|              |                                                   |                                                                 |                                        |  |  |  |  |
|              | jz                                                | end_cpu_type                                                    | ; jump if 80386 processor              |  |  |  |  |

Intel486 processor check ;

ecx

; Checking for ability to set/clear ID flag (Bit 21) in EFLAGS

; restore AC bit in EFLAGS first

which indicates the presence of a processor with the CPUID ;

; instruction.

push

popfd

.486 check\_80486: \_cpu\_type, 4 ; turn on 80486 processor flag mov ; get original EFLAGS mov eax, ecx ; flip ID bit in EFLAGS xor eax, 200000h push eax ; save new EFLAGS value on stack ; replace current EFLAGS value popfd pushfd ; get new EFLAGS eax pop ; store new EFLAGS in EAX ; can't toggle ID bit, xor eax, ecx ; processor=80486 je end\_cpu\_type ; Execute CPUID instruction to determine vendor, family, ; model, stepping and features. For the purpose of this code, only the initial set of CPUID information is saved. : \_cpuid\_flag, 1 ; flag indicating use of CPUID inst. mov push ebx ; save registers esi push edi push eax, 0 ; set up for CPUID instruction mov CPU ID ; get and save vendor ID dword ptr \_vendor\_id, ebx mov dword ptr \_vendor\_id[+4], edx mov dword ptr \_vendor\_id[+8], ecx mov dword ptr intel\_id, ebx cmp jne end\_cpuid\_type cmp dword ptr intel\_id[+4], edx end\_cpuid\_type jne dword ptr intel\_id[+8], ecx cmp end\_cpuid\_type ; if not equal, not an Intel processor jne \_intel\_CPU, 1 mov ; indicate an Intel processor cmp eax, 1 ; make sure 1 is valid input for CPUID jl end\_cpuid\_type ; if not, jump to end mov eax, 1 CPU ID ; get family/model/stepping/features \_cpu\_signature, eax mov \_features\_ebx, ebx mov \_features\_edx, edx mov \_features\_ecx, ecx mov shr eax, 8 ; isolate family and eax, 0fh ; set \_cpu\_type with family mov \_cpu\_type, al ; Execute CPUID instruction to determine the cache descriptor : information.

| mov eax, 0 | ; set up to check the EAX value |
|------------|---------------------------------|
|------------|---------------------------------|

CPU\_ID ax, 2 ; Are cache descriptors supported? cmp end\_cpuid\_type il eax, 2 mov ; set up to read cache descriptor CPU\_ID cmp al, 1 ; Is one iteration enough to obtain jne end\_cpuid\_type ; cache information? ; This code supports one iteration ; only. mov \_cache\_eax, eax ; store cache information ; NOTE: for future processors, CPUID \_cache\_ebx, ebx mov ; instruction may need to be run more mov \_cache\_ecx, ecx mov \_cache\_edx, edx ; than once to get complete cache ; information eax, 8000000h ; check if brand string is supported mov CPU ID eax, 8000000h cmp ibe end\_cpuid\_type ; take jump if not supported mov di, offset \_brand\_string eax, 8000002h ; get first 16 bytes of brand string mov CPU\_ID mov dword ptr [di], eax ; save bytes 0..15 mov dword ptr [di+4], ebx dword ptr [di+8], ecx mov dword ptr [di+12], edx mov add di, 16 eax, 8000003h mov CPU\_ID mov dword ptr [di], eax ; save bytes 16 .. 31 dword ptr [di+4], ebx mov dword ptr [di+8], ecx mov mov dword ptr [di+12], edx add di, 16 eax, 8000004h mov CPU\_ID mov dword ptr [di], eax ; save bytes 32 .. 47 dword ptr [di+4], ebx mov dword ptr [di+8], ecx mov mov dword ptr [di+12], edx end\_cpuid\_type: edi ; restore registers pop pop esi pop ebx comment this line for 32-bit segments

;

;

; .8086 end\_cpu\_type: ret \_get\_cpu\_type endp public \_get\_fpu\_type \_get\_fpu\_type proc ; This procedure determines the type of FPU in a system ; and sets the \_fpu\_type variable with the appropriate value. All registers are used by this procedure, none are preserved. ; ; Coprocessor check The algorithm is to determine whether the floating-point status and control words are present. If not, no coprocessor exists. If the status and control words can be saved, the correct coprocessor is then determined ; depending on the processor type. The Intel386 processor can ; work with either an Intel287 NDP or an Intel387 NDP. ; The infinity of the coprocessor must be checked to determine ; the correct coprocessor type. : fninit ; reset FP status word mov fp\_status, 5a5ah ; initialize temp word to non-zero ; save FP status word fnstsw fp\_status mov ax, fp\_status ; check FP status word cmp al, 0 : was correct status written \_fpu\_type, 0 ; no FPU present mov ine end\_fpu\_type check\_control\_word: ; save FP control word fnstcw fp\_status ; check FP control word mov ax, fp\_status ax, 103fh and ; selected parts to examine cmp ax, 3fh ; was control word correct mov \_fpu\_type, 0 ine end\_fpu\_type ; incorrect control word, no FPU mov \_fpu\_type, 1 80287/80387 check for the Intel386 processor ; check infinity: cmp \_cpu\_type, 3 ine end\_fpu\_type fld1 ; must use default control from FNINIT fldz ; form infinity ; 8087/Intel287 NDP say +inf = -inf fdiv fld ; form negative infinity st fchs ; Intel387 NDP says +inf <> -inf

| fcompp        |               | ; see if they are the same            |
|---------------|---------------|---------------------------------------|
| fstsw         | fp_status     | ; look at status from FCOMPP          |
| mov           | ax, fp_status |                                       |
| mov           | _fpu_type, 2  | ; store Intel287 NDP for FPU type     |
| sahf          |               | ; see if infinities matched           |
| jz            | end_fpu_type  | ; jump if 8087 or Intel287 is present |
| mov           | _fpu_type, 3  | ; store Intel387 NDP for FPU type     |
| end_fpu_type: |               |                                       |
| ret           |               |                                       |
| _get_fpu_type | endp          |                                       |
|               |               |                                       |

end

### Example 2. Processor Identification Procedure in Assembly Language

| ; File<br>;                                                                                 | ename: cp<br>Copyrigh                                                                                                                                                                                                                                                                                                                                                                                                                                                         |                                                                                          | 1<br>Corporation 199                                                                                                                     | 93-2004                                                                                                                                                                                                                                                                                           |  |
|---------------------------------------------------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
| ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;                                                     | This program has been developed by Intel Corporation. Intel<br>has various intellectual property rights which it may assert<br>under certain circumstances, such as if another<br>manufacturer's processor mis-identifies itself as being<br>"GenuineIntel" when the CPUID instruction is executed.                                                                                                                                                                           |                                                                                          |                                                                                                                                          |                                                                                                                                                                                                                                                                                                   |  |
| ·<br>·<br>·<br>·<br>·<br>·<br>·<br>·                                                        | Intel specifically disclaims all warranties, express or<br>implied, and all liability, including consequential and<br>other indirect damages, for the use of this program,<br>including liability for infringement of any proprietary<br>rights, and including the warranties of merchantability and<br>fitness for a particular purpose. Intel does not assume any<br>responsibility for any errors which may appear in this<br>program nor any responsibility to update it. |                                                                                          |                                                                                                                                          |                                                                                                                                                                                                                                                                                                   |  |
| ,<br>,<br>,<br>,<br>,                                                                       | This prog<br>Part 1:                                                                                                                                                                                                                                                                                                                                                                                                                                                          |                                                                                          | ains three parts:<br>s processor type<br>pe:                                                                                             | in the variable                                                                                                                                                                                                                                                                                   |  |
| ;                                                                                           | Part 2:                                                                                                                                                                                                                                                                                                                                                                                                                                                                       | Identifie                                                                                | s FPU type in the                                                                                                                        | e variable _fpu_type:                                                                                                                                                                                                                                                                             |  |
| ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;                                                     | Part 3:                                                                                                                                                                                                                                                                                                                                                                                                                                                                       | specific                                                                                 |                                                                                                                                          | e message. This part is ronment and uses the DOS the messages.                                                                                                                                                                                                                                    |  |
| ·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>·<br>· | this code<br>with the o<br>Intel 808<br>Pentium(<br>Overdriv<br>processor                                                                                                                                                                                                                                                                                                                                                                                                     | is assemi<br>cpuid3a n<br>6/8088, 8<br>R) II proc<br>e(R) proc<br>rs, Pentiu<br>(R) Xeon | bled with no opti<br>nodule, it correct<br>0286, 80386, 80<br>cessors, Pentium<br>essors, Intel(R)<br>m(R) III Xeon(T<br>(TM) processors | the Microsoft Developer Studio. If<br>ons specified and linked<br>ly identifies the current<br>486, Pentium(R), Pentium(R) Pro,<br>(R) II Xeon(TM) processors, Pentium(R) II<br>Celeron(R) processors, Pentium(R) III<br>M) processors, Pentium(R) 4 processors<br>DP and MP when executed in the |  |
| ; NOTE                                                                                      | E: This cod                                                                                                                                                                                                                                                                                                                                                                                                                                                                   | le is writte                                                                             | en using 16-bit S                                                                                                                        | egments                                                                                                                                                                                                                                                                                           |  |
| ;;;                                                                                         | To assem<br>jumps                                                                                                                                                                                                                                                                                                                                                                                                                                                             | ble this c                                                                               | ode with TASM                                                                                                                            | , add the JUMPS directive.<br>; Uncomment this line for TASM                                                                                                                                                                                                                                      |  |
|                                                                                             | TITLE                                                                                                                                                                                                                                                                                                                                                                                                                                                                         | cpuid3b                                                                                  |                                                                                                                                          |                                                                                                                                                                                                                                                                                                   |  |
| DOSSEC<br>.model                                                                            | G<br>small                                                                                                                                                                                                                                                                                                                                                                                                                                                                    |                                                                                          |                                                                                                                                          |                                                                                                                                                                                                                                                                                                   |  |
| .stack                                                                                      | 100h                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |                                                                                          |                                                                                                                                          |                                                                                                                                                                                                                                                                                                   |  |
| OP_O<br>ENDM                                                                                | MACRO<br>db                                                                                                                                                                                                                                                                                                                                                                                                                                                                   | 66h                                                                                      |                                                                                                                                          | ; hardcoded operand override                                                                                                                                                                                                                                                                      |  |
| .data                                                                                       | extrn                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |                                                                                          | _cpu_type:                                                                                                                               | byte                                                                                                                                                                                                                                                                                              |  |

|        | extrn   | _fpu_type:              | byte        |                                 |  |
|--------|---------|-------------------------|-------------|---------------------------------|--|
|        | extrn   | _cpuid_flag:            | byte        |                                 |  |
|        | extrn   | _intel_CPU:             | byte        |                                 |  |
|        | extrn   | _vendor_id:             | byte        |                                 |  |
|        | extrn   | _cpu_signature:         | dword       |                                 |  |
|        | extrn   | _features_ecx:          | dword       |                                 |  |
|        | extrn   | _features_edx:          | dword       |                                 |  |
|        | extrn   | _features_ebx:          | dword       |                                 |  |
|        | extrn   | _cache_eax:             | dword       |                                 |  |
|        | extrn   | _cache_ebx:             | dword       |                                 |  |
|        | extrn   | _cache_ecx:             | dword       |                                 |  |
|        | extrn   | _cache_edx:             | dword       |                                 |  |
|        | extrn   | _brand_string:          | byte        |                                 |  |
|        |         |                         |             |                                 |  |
| ;      | The pu  | pose of this code is    | to identify | the processor and               |  |
| ;      |         | essor that is currently |             |                                 |  |
| ;      |         | termines the processo   |             |                                 |  |
| ;      |         | r a coprocessor exist   |             |                                 |  |
| ;      | coproce | essor or integrated co  | processor e | exists, the program             |  |
| ;      |         | es the coprocessor ty   |             |                                 |  |
| ;      |         |                         |             | sors present and type.          |  |
|        |         |                         | •           |                                 |  |
| .code  |         |                         |             |                                 |  |
| .8086  |         |                         |             |                                 |  |
| start: |         |                         |             |                                 |  |
|        | mov     | ax, @data               |             |                                 |  |
|        | mov     | ds, ax                  |             | ; set segment register          |  |
|        | mov     | es, ax                  |             | ; set segment register          |  |
|        | and     | sp, not 3               |             | ; align stack to avoid AC fault |  |
|        | call    | _get_cpu_type           |             | ; determine processor type      |  |
|        | call    | _get_fpu_type           |             |                                 |  |
|        | call    | print                   |             |                                 |  |
|        |         | I ·                     |             |                                 |  |
|        | mov     | ax, 4c00h               |             |                                 |  |
|        | int     | 21h                     |             |                                 |  |
|        |         |                         |             |                                 |  |
| ·***** | ******* | *****                   | *******     | **********                      |  |
|        |         |                         |             |                                 |  |
|        | extrn   | _get_cpu_type: pr       | roc         |                                 |  |
|        |         | -6 - 1 - 71 1           |             |                                 |  |
| *****  | ******  | *****                   | *******     | *******                         |  |
| ,      |         |                         |             |                                 |  |
|        | extrn   | _get_fpu_type: pr       | oc          |                                 |  |
|        |         |                         |             |                                 |  |
| *****  | ******  | *****                   | *******     | *******                         |  |
| ,      |         |                         |             |                                 |  |
| FPU_F  | LAG     | equ 0001h               |             |                                 |  |
| VME_I  |         | equ 0002h               |             |                                 |  |
| DE_FL  |         | equ 0004h               | -           |                                 |  |
| PSE_FI |         | equ 0008h               | -           |                                 |  |
| TSC_F  |         | equ 0010h               | -           |                                 |  |
| MSR_F  |         | equ 0020h               |             |                                 |  |
| PAE_F  |         | equ 0020h               |             |                                 |  |
| MCE I  |         | equ 0040h               |             |                                 |  |
| CX8_F  |         | equ 0100h               |             |                                 |  |
| APIC_I |         | -                       |             |                                 |  |
|        |         | equ 0200h               |             |                                 |  |
| SEP_FI |         | equ 0800h               |             |                                 |  |
| MTRR   | I'LAU   | equ 1000h               |             |                                 |  |

# int<sub>e</sub>l

| PGE_FLAG         | equ 2000 |                                                           |
|------------------|----------|-----------------------------------------------------------|
| MCA_FLAG         | equ 4000 | )h                                                        |
| CMOV_FLAG        | equ 8000 | )h                                                        |
| PAT_FLAG         | equ 1000 | OOh                                                       |
| PSE36_FLAG       | equ 2000 | OOh                                                       |
| PSNUM_FLAG       | equ 4000 | OOh                                                       |
| CLFLUSH_FLAG     | equ 8000 | DOh                                                       |
| DTS_FLAG         | equ 2000 | 000h                                                      |
| ACPI_FLAG        | equ 4000 |                                                           |
| MMX_FLAG         | equ 8000 |                                                           |
| FXSR_FLAG        | equ 1000 |                                                           |
| SSE_FLAG         | equ 2000 |                                                           |
| SSE2_FLAG        | equ 4000 |                                                           |
| SS_FLAG          | equ 8000 |                                                           |
| HTT_FLAG         | equ 1000 |                                                           |
| TM_FLAG          | equ 2000 |                                                           |
| PBE_FLAG         | equ 2000 |                                                           |
| SSE3_FLAG        | equ 000  |                                                           |
| MONITOR_FLAC     | -        |                                                           |
| DS_CPL_FLAG      | equ 0000 |                                                           |
|                  | -        |                                                           |
| EIST_FLAG        | equ 0080 |                                                           |
| TM2_FLAG         | equ 0100 |                                                           |
| CID_FLAG         | equ 0400 |                                                           |
| XTPR_FLAG        | equ 0400 | JUN                                                       |
| •                |          |                                                           |
| .data            |          |                                                           |
| id_msg           | db       | "This system has a\$"                                     |
| cp_error         | db       | "n unknown processor\$"                                   |
| cp_8086          | db       | "n 8086/8088 processor\$"                                 |
| cp_286           | db       | "n 80286 processor\$"                                     |
| cp_386           | db       | "n 80386 processor\$"                                     |
|                  |          |                                                           |
| cp_486           | db       | "n 80486DX, 80486DX2 processor or"                        |
|                  | db       | " 80487SX math coprocessor\$"                             |
| cp_486sx         | db       | "n 80486SX processor\$"                                   |
|                  |          |                                                           |
| fp_8087          | db       | " and an 8087 math coprocessor\$"                         |
| fp_287           | db       | " and an 80287 math coprocessor\$"                        |
| fp_387           | db       | " and an 80387 math coprocessor\$"                        |
|                  |          |                                                           |
| intel486_msg     | db       | " Genuine Intel486(TM) processor\$"                       |
| intel486dx_msg   | db       | " Genuine Intel486(TM) DX processor\$"                    |
| intel486sx_msg   | db       | " Genuine Intel486(TM) SX processor\$"                    |
| inteldx2_msg     | db       | " Genuine IntelDX2(TM) processor\$"                       |
| intelsx2_msg     | db       | " Genuine IntelSX2(TM) processor\$"                       |
| inteldx4_msg     | db       | " Genuine IntelDX4(TM) processor\$"                       |
| inteldx2wb_msg   | db       | " Genuine Write-Back Enhanced"                            |
| 0                | db       | " IntelDX2(TM) processor\$"                               |
| pentium_msg      | db       | " Genuine Intel(R) Pentium(R) processor\$"                |
| pentiumpro_msg   | db       | " Genuine Intel Pentium(R) Pro processor\$"               |
| 1 1 - 0          |          |                                                           |
| pentiumiimodel3_ | msg      | db "Genuine Intel(R) Pentium(R) II processor, model 3\$"  |
| pentiumiixeon_m5 |          | db "Genuine Intel(R) Pentium(R) II processor, model 5 or" |
| • = •            | 0        | db "Intel(R) Pentium(R) II Xeon(TM) processor\$"          |
| pentiumiixeon_ms | g        | db "Genuine Intel(R) Pentium(R) II Xeon(TM) processor\$"  |
| celeron_msg      | -        | db "Genuine Intel(R) Celeron(R) processor, model 5\$"     |
| celeronmodel6_ms | sg       | db "Genuine Intel(R) Celeron(R) processor, model 6\$"     |
| celeron_brand    | 0        | db "Genuine Intel(R) Celeron(R) processor\$"              |
|                  |          |                                                           |

| pentiumiii_msg<br>pentiumiiixeon_m<br>pentiumiiixeon_br<br>pentiumiii_brand<br>mobile_piii_brand<br>mobile_icp_brand | and       | db<br>db<br>db<br>db<br>db<br>db<br>db | <ul> <li>" Genuine Intel(R) Pentium(R) III processor, model 7 or"</li> <li>" Intel Pentium(R) III Xeon(TM) processor, model 7\$"</li> <li>" Genuine Intel(R) Pentium(R) III Xeon(TM) processor, model 7\$"</li> <li>" Genuine Intel(R) Pentium(R) III Xeon(TM) processor\$"</li> <li>" Genuine Intel(R) Pentium(R) III processor\$"</li> <li>" Genuine Mobile Intel(R) Pentium(R) III Processor-M\$"</li> <li>" Genuine Mobile Intel(R) Celeron(R) processor\$"</li> </ul> |
|----------------------------------------------------------------------------------------------------------------------|-----------|----------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| mobile_P4_brand                                                                                                      |           | db                                     | " Genuine Mobile Intel(R) Pentium(R) 4 processor - M\$"                                                                                                                                                                                                                                                                                                                                                                                                                    |
| pentium4_brand                                                                                                       |           | db                                     | "Genuine Intel(R) Pentium(R) 4 processor \$"                                                                                                                                                                                                                                                                                                                                                                                                                               |
| xeon_brand                                                                                                           |           | db                                     | " Genuine Intel(R) Xeon(TM) processor\$"                                                                                                                                                                                                                                                                                                                                                                                                                                   |
| xeon_mp_brand                                                                                                        |           | db                                     | " Genuine Intel(R) Xeon(TM) processor MP\$"                                                                                                                                                                                                                                                                                                                                                                                                                                |
| mobile_icp_brand                                                                                                     | _2        | db                                     | " Genuine Mobile Intel(R) Celeron(R) processor\$"                                                                                                                                                                                                                                                                                                                                                                                                                          |
| mobile_pentium_r                                                                                                     | n_brand   | db                                     | " Genuine Intel(R) Pentium(R) M processor\$"                                                                                                                                                                                                                                                                                                                                                                                                                               |
| mobile_genuine_b                                                                                                     |           | db                                     | " Mobile Genuine Intel(R) processor\$"                                                                                                                                                                                                                                                                                                                                                                                                                                     |
| mobile_icp_m_bra                                                                                                     | and       | db                                     | " Genuine Intel(R) Celeron(R) M processor\$"                                                                                                                                                                                                                                                                                                                                                                                                                               |
| unknown_msg                                                                                                          |           | db                                     | "n unknown Genuine Intel(R) processor\$"                                                                                                                                                                                                                                                                                                                                                                                                                                   |
|                                                                                                                      |           |                                        |                                                                                                                                                                                                                                                                                                                                                                                                                                                                            |
| brand_entry                                                                                                          | struct    | 11                                     | 0                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |
| brand_v                                                                                                              |           | db                                     | ?                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |
| brand_st                                                                                                             | <u> </u>  | dw                                     | ?                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |
| brand_entry                                                                                                          | ends      |                                        |                                                                                                                                                                                                                                                                                                                                                                                                                                                                            |
| brand_table                                                                                                          | brand_e   | ntrv                                   | <01h, offset celeron_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                                |
|                                                                                                                      | brand_e   | •                                      | <02h, offset pentiumiii_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                             |
|                                                                                                                      | brand_e   | -                                      | <03h, offset pentiumiiixeon_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                         |
|                                                                                                                      | brand_er  | ntry                                   | <04h, offset pentiumiii_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                             |
|                                                                                                                      | brand_er  | ntry                                   | <06h, offset mobile_piii_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                            |
|                                                                                                                      | brand_er  | ntry                                   | <07h, offset mobile_icp_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                             |
|                                                                                                                      | brand_e   | ntry                                   | <08h, offset pentium4_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                               |
|                                                                                                                      | brand_er  | ntry                                   | <09h, offset pentium4_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                               |
|                                                                                                                      | brand_e   | •                                      | <0Ah, offset celeron_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                                |
|                                                                                                                      | brand_e   | -                                      | <0Bh, offset xeon_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                                   |
|                                                                                                                      | brand_e   | -                                      | <0Ch, offset xeon_mp_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                                |
|                                                                                                                      | brand_e   | -                                      | <0Eh, offset mobile_p4_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                              |
|                                                                                                                      | brand_e   | -                                      | <pre>&lt;0Fh, offset mobile_icp_brand&gt;</pre>                                                                                                                                                                                                                                                                                                                                                                                                                            |
|                                                                                                                      | brand_er  | -                                      | <11h, offset mobile_genuine_brand><br><12h, offset mobile_icp_m_brand>                                                                                                                                                                                                                                                                                                                                                                                                     |
|                                                                                                                      | brand_er  | -                                      | <13h, offset mobile_icp_brand_2>                                                                                                                                                                                                                                                                                                                                                                                                                                           |
|                                                                                                                      | brand_er  |                                        | <14h, offset celeron_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                                |
|                                                                                                                      | brand_e   | -                                      | <15h, offset mobile_genuine_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                         |
|                                                                                                                      | brand_e   | -                                      | <16h, offset mobile_pentium_m_brand>                                                                                                                                                                                                                                                                                                                                                                                                                                       |
|                                                                                                                      | brand en  | -                                      | <17h, offset mobile_icp_brand_2>                                                                                                                                                                                                                                                                                                                                                                                                                                           |
| brand_table_size                                                                                                     | equ       | (\$ - offse                            | et brand_table) / (sizeof brand_entry)                                                                                                                                                                                                                                                                                                                                                                                                                                     |
| ; The following 16                                                                                                   | entries m | ust stav ir                            | ntact as an array                                                                                                                                                                                                                                                                                                                                                                                                                                                          |
| intel_486_0                                                                                                          | dw        |                                        | tel486dx_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                               |
| intel_486_1                                                                                                          | dw        |                                        | tel486dx_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                               |
| intel_486_2                                                                                                          | dw        |                                        | tel486sx_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                               |
| intel_486_3                                                                                                          | dw        |                                        | teldx2_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| intel_486_4                                                                                                          | dw        | offset in                              | tel486_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| intel_486_5                                                                                                          | dw        | offset in                              | telsx2_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| intel_486_6                                                                                                          | dw        | offset in                              | tel486_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| intel_486_7                                                                                                          | dw        |                                        | teldx2wb_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                               |
| intel_486_8                                                                                                          | dw        |                                        | teldx4_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| intel_486_9                                                                                                          | dw        |                                        | tel486_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| intel_486_a                                                                                                          | dw        |                                        | tel486_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| intel_486_b                                                                                                          | dw        | offset in                              | tel486_msg                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |

| intel_486_c          | dw             | offset intel486_msg                                                  |
|----------------------|----------------|----------------------------------------------------------------------|
| intel_486_d          | dw             | offset intel486_msg                                                  |
| intel_486_e          | dw             | offset intel486_msg                                                  |
| intel_486_f          | dw             | offset intel486_msg                                                  |
| ; end of array       |                |                                                                      |
|                      |                |                                                                      |
| family_msg           | db             | 13,10,"Processor Family: \$"                                         |
| model_msg            | db             | 13,10,"Model: \$"                                                    |
| stepping_msg         | db             | 13,10,"Stepping: \$"                                                 |
| ext_fam_msg          | db             | 13,10," Extended Family: \$"                                         |
| ext_mod_msg<br>cr lf | db<br>db       | 13,10," Extended Model: \$"<br>13,10,"\$"                            |
| turbo_msg            | db             | 13,10, 5<br>13,10, "The processor is an OverDrive(R)"                |
| turbo_msg            | db             | " processor\$"                                                       |
| dp_msg               | db             | 13,10,"The processor is the upgrade"                                 |
| up_msg               | db             | " processor in a dual processor system\$"                            |
| fpu_msg              | db             | 13,10,"The processor contains an on-chip"                            |
| 1 = 0                | db             | " FPU\$"                                                             |
| vme_msg              | db             | 13,10,"The processor supports Virtual"                               |
| - 0                  | db             | " Mode Extensions\$"                                                 |
| de_msg               | db             | 13,10,"The processor supports Debugging"                             |
| -                    | db             | "Extensions\$"                                                       |
| pse_msg              | db             | 13,10,"The processor supports Page Size"                             |
|                      | db             | " Extensions\$"                                                      |
| tsc_msg              | db             | 13,10,"The processor supports Time Stamp"                            |
|                      | db             | " Counter\$"                                                         |
| msr_msg              | db             | 13,10,"The processor supports Model"                                 |
|                      | db             | " Specific Registers\$"                                              |
| pae_msg              | db             | 13,10,"The processor supports Physical"                              |
|                      | db             | " Address Extensions\$"                                              |
| mce_msg              | db             | 13,10,"The processor supports Machine"                               |
| 0                    | db             | " Check Exceptions\$"                                                |
| cx8_msg              | db             | 13,10,"The processor supports the"                                   |
| apic_msg db          | db<br>13 10 "T | " CMPXCHG8B instruction\$"<br>he processor contains an on-chip"      |
| apic_msg ub          | db             | " APIC\$"                                                            |
| sep_msg              | db             | 13,10,"The processor supports Fast System"                           |
| sep_msg              | db             | " Call\$"                                                            |
| no_sep_msg           | db             | 13,10,"The processor does not support Fast"                          |
| no_oop_mog           | db             | " System Call\$"                                                     |
| mtrr_msgdb           |                | he processor supports Memory Type"                                   |
| _ 8                  | db             | "Range Registers\$"                                                  |
| pge_msg              | db             | 13,10,"The processor supports Page Global"                           |
|                      | db             | "Enable\$"                                                           |
| mca_msg              | db             | 13,10,"The processor supports Machine"                               |
|                      | db             | " Check Architecture\$"                                              |
| cmov_msg             | db             | 13,10,"The processor supports Conditional"                           |
|                      | db             | " Move Instruction\$"                                                |
| pat_msg              | db             | 13,10,"The processor supports Page Attribute"                        |
|                      | db             | " Table\$"                                                           |
| pse36_msg            | db             | 13,10,"The processor supports 36-bit Page"                           |
|                      | db             | " Size Extension\$"                                                  |
| psnum_msg            | db             | 13,10,"The processor supports the"                                   |
| -1611-               | db             | " processor serial number\$"                                         |
| clflush_msg          | db             | 13,10,"The processor supports the"                                   |
| dte mer              | db<br>db       | " CLFLUSH instruction\$"                                             |
| dts_msg              | db<br>db       | 13,10,"The processor supports the"<br>" Debug Trace Store feature\$" |
|                      | ab             | Debug Hate Stole reduies                                             |
|                      |                |                                                                      |

| acpi_msg       | g db         | 13,10,"Th<br>db | e processor supports the"<br>" ACPI registers in MSR space\$"        |
|----------------|--------------|-----------------|----------------------------------------------------------------------|
| mmx_ms         | sg           | db              | 13,10,"The processor supports Intel Architecture"                    |
| fxsr_msg       | r dh         | db<br>13 10 "Th | " MMX(TM) Technology\$"<br>e processor supports Fast floating point" |
| 17.01_11156    | , <b>u</b> o | db              | " save and restore\$"                                                |
| sse_msg        |              | db              | 13,10,"The processor supports the Streaming"                         |
| g              |              | db              | " SIMD extensions\$"                                                 |
| sse2_msg       | g            | db              | 13,10,"The processor supports the Streaming"                         |
|                |              | db              | " SIMD extensions 2 instructions\$"                                  |
| ss_msg         |              | db              | 13,10,"The processor supports Self-Snoop\$"                          |
| htt_msg        |              | db              | 13,10,"The processor supports Hyper-Threading Technology\$"          |
| tm_msg         |              | db              | 13,10,"The processor supports the"                                   |
|                |              | db              | " Thermal Monitor\$"                                                 |
| pbe_msg        |              | db              | 13,10,"The processor supports the"                                   |
|                |              | db              | " Pending Break Event\$"                                             |
| sse3_msg       | 5            | db              | 13,10,"The processor supports the Streaming SIMD"                    |
|                |              | db              | " Extensions 3 instructions\$"                                       |
| monitor_       | msg          | db              | 13,10,"The processor supports the MONITOR and MWAIT"                 |
|                |              | db              | " instructions\$"                                                    |
| ds_cpl_n       | nsg          | db              | 13,10,"The processor supports Debug Store extensions for"            |
| •              |              | db              | " branch message storage by CPL\$"                                   |
| eist_msg       |              | db              | 13,10,"The processor supports"                                       |
|                |              | db              | "Enhanced SpeedStep(TM) Technology\$"                                |
| tm2_msg        | 5            | db              | 13,10,"The processor supports the"                                   |
| • •            |              | db              | "Thermal Monitor 2\$"                                                |
| cid_msg        |              | db              | 13,10,"The processor supports L1 Data Cache Context ID\$"            |
| xtpr_msg       | 5            | db              | 13,10,"The processor supports transmitting TPR messages\$"           |
| not_intel      |              | db              | "t least an 80486 processor."                                        |
|                |              | db              | 13,10,"It does not contain a Genuine"                                |
|                |              | db              | "Intel part and as a result,"                                        |
|                |              | db              | "the",13,10,"CPUID"                                                  |
|                |              | db              | " detection information cannot be"                                   |
|                |              | db              | " determined at this time.\$"                                        |
| ASC_MS         | SG           | MACRO           | msg                                                                  |
|                | LOCAL        | ascii_don       | e ; local label                                                      |
|                | add          | al, 30h         |                                                                      |
|                | cmp          | al, 39h         | ; is it 0-9?                                                         |
|                | jle          | ascii_don       | e                                                                    |
|                | add          | al, 07h         |                                                                      |
| ascii_dor      |              |                 |                                                                      |
|                | mov          |                 | nsg[20], al                                                          |
|                | mov          | dx, offset      | msg                                                                  |
|                | mov          | ah, 9h          |                                                                      |
| ENDV           | int          | 21h             |                                                                      |
| ENDM           |              |                 |                                                                      |
| .code<br>.8086 |              |                 |                                                                      |
| .8080          |              |                 |                                                                      |
| print          | proc         |                 |                                                                      |
| ;              | This prod    | cedure prin     | ts the appropriate cpuid string and                                  |
| ;              |              |                 | presence status. If the CPUID instruction                            |
| ;              |              |                 | edure prints out the CPUID info.                                     |
| ;              |              | -               | d by this procedure, none are                                        |
|                | 0            |                 | - · ·                                                                |

| •         | preserved.        |                                    |                                                          |  |
|-----------|-------------------|------------------------------------|----------------------------------------------------------|--|
|           | mov<br>mov<br>int | dx, offset id_msg<br>ah, 9h<br>21h | ; print initial message                                  |  |
|           | cmp               | _cpuid_flag, 1                     | ; if set to 1, processor<br>; supports CPUID instruction |  |
|           | je                | print_cpuid_data                   | ; print detailed CPUID info                              |  |
| print_86: | :                 |                                    |                                                          |  |
|           | cmp               | _cpu_type, 0                       |                                                          |  |
|           | jne               | print_286                          |                                                          |  |
|           | mov               | dx, offset cp_8086                 |                                                          |  |
|           | mov               | ah, 9h                             |                                                          |  |
|           | int               | 21h                                |                                                          |  |
|           | cmp               | _fpu_type, 0                       |                                                          |  |
|           | je                | end_print                          |                                                          |  |
|           | mov               | dx, offset fp_8087                 |                                                          |  |
|           | mov               | ah, 9h                             |                                                          |  |
|           | int               | 21h                                |                                                          |  |
|           | jmp               | end_print                          |                                                          |  |
| print_280 | ۶.                |                                    |                                                          |  |
| print_200 | cmp               | _cpu_type, 2                       |                                                          |  |
|           | jne               | print_386                          |                                                          |  |
|           | mov               | dx, offset cp_286                  |                                                          |  |
|           | mov               | ah, 9h                             |                                                          |  |
|           | int               | 21h                                |                                                          |  |
|           | cmp               | _fpu_type, 0                       |                                                          |  |
|           | je                | end_print                          |                                                          |  |
| nrint 20  | 7.                |                                    |                                                          |  |
| print_287 |                   | dy offset fp 287                   |                                                          |  |
|           | mov               | dx, offset fp_287<br>ah, 9h        |                                                          |  |
|           | mov<br>int        | 21h                                |                                                          |  |
|           | jmp               | end_print                          |                                                          |  |
|           | Jinp              | ena_print                          |                                                          |  |
| print_380 | 5:                |                                    |                                                          |  |
|           | cmp               | _cpu_type, 3                       |                                                          |  |
|           | jne               | print_486                          |                                                          |  |
|           | mov               | dx, offset cp_386                  |                                                          |  |
|           | mov               | ah, 9h                             |                                                          |  |
|           | int               | 21h                                |                                                          |  |
|           | cmp               | _fpu_type, 0                       |                                                          |  |
|           | je                | end_print                          |                                                          |  |
|           | cmp               | _fpu_type, 2                       |                                                          |  |
|           | je                | print_287                          |                                                          |  |
|           | mov               | dx, offset fp_387                  |                                                          |  |
|           | mov               | ah, 9h<br>21h                      |                                                          |  |
|           | int<br>imp        |                                    |                                                          |  |
|           | jmp               | end_print                          |                                                          |  |
| print_480 | 5:                |                                    |                                                          |  |
|           | cmp               | _cpu_type, 4                       |                                                          |  |
|           | jne               | print_unknown                      | ; Intel processors will have                             |  |
|           | mov               | dx, offset cp_486sx                | ; CPUID instruction                                      |  |
|           | cmp               | _fpu_type, 0                       |                                                          |  |
|           |                   |                                    |                                                          |  |

|                  | je<br>mov                       | print_486sx<br>dx, offset cp_486                                                         |                                          |
|------------------|---------------------------------|------------------------------------------------------------------------------------------|------------------------------------------|
| print_48         | 6sx:<br>mov<br>int<br>jmp       | ah, 9h<br>21h<br>end_print                                                               |                                          |
| print_un         | known:<br>mov<br>jmp            | dx, offset cp_error<br>print_486sx                                                       |                                          |
| print_cp<br>.486 | uid_data:                       |                                                                                          |                                          |
| .480             | cmp<br>jne                      | _intel_CPU, 1<br>not_GenuineIntel                                                        | ; check for genuine Intel<br>; processor |
|                  | mov<br>cmp<br>je                | di, offset _brand_string<br>byte ptr [di], 0<br>print_brand_id                           | ; brand string supported?                |
|                  | mov                             | cx, 47                                                                                   | ; max brand string length                |
| skip_spa         | ices:<br>cmp<br>jne             | byte ptr [di], ' '<br>print_brand_string                                                 | ; skip leading space chars               |
|                  | inc<br>loop                     | di<br>skip_spaces                                                                        |                                          |
| print_bra        | and_string                      |                                                                                          | N. 4                                     |
|                  | cmp<br>je<br>cmp<br>je          | cx, 0<br>print_brand_id<br>byte ptr [di], 0<br>print_brand_id                            | ; Nothing to print                       |
| print_bra        | and_char:                       |                                                                                          |                                          |
|                  | mov<br>mov<br>int               | dl, [di]<br>ah, 2<br>21h                                                                 | ; print upto the max chars               |
|                  | inc<br>cmp<br>je<br>loop<br>jmp | di<br>byte ptr [di], 0<br>print_family<br>print_brand_char<br>print_family               |                                          |
| print_bra        | and_id:<br>cmp<br>jb<br>ja      | _cpu_type, 6<br>print_486_type<br>print_pentiumiiimodel8_type                            | e                                        |
|                  | mov<br>shr<br>and<br>cmp<br>jae | eax, dword ptr _cpu_signatu<br>eax, 4<br>al, 0fh<br>al, 8<br>print_pentiumiiimodel8_type |                                          |

| print_486   | 5_type:           |                                |                                                |  |  |
|-------------|-------------------|--------------------------------|------------------------------------------------|--|--|
|             | cmp               | _cpu_type, 4                   | ; if 4, print 80486 processor                  |  |  |
|             | jne               | print_pentium_type             |                                                |  |  |
|             | mov               | eax, dword ptr _cpu_signature  |                                                |  |  |
|             | shr               | eax, 4                         |                                                |  |  |
|             | and               | eax, 0fh                       | ; isolate model                                |  |  |
|             | mov               | dx, intel_486_0[eax*2]         |                                                |  |  |
|             | jmp               | print_common                   |                                                |  |  |
|             | 5 1               | r <u> </u>                     |                                                |  |  |
| print_pen   | tium_type         | 2:                             |                                                |  |  |
|             | cmp               | _cpu_type, 5                   | ; if 5, print Pentium processor                |  |  |
|             | jne               | print_pentiumpro_type          |                                                |  |  |
|             | mov               | dx, offset pentium_msg         |                                                |  |  |
|             | jmp               | print_common                   |                                                |  |  |
|             | 5 1               |                                |                                                |  |  |
| print_pen   | tiumpro_t         | type:                          |                                                |  |  |
|             | cmp               | _cpu_type, 6                   | ; if 6 & model 1, print Pentium                |  |  |
|             | 1                 |                                | ; Pro processor                                |  |  |
|             | jne               | print_unknown_type             |                                                |  |  |
|             | mov               | eax, dword ptr _cpu_signatur   | e                                              |  |  |
|             | shr               | eax, 4                         |                                                |  |  |
|             | and               | eax, 0fh                       | ; isolate model                                |  |  |
|             | cmp               | eax, 3                         |                                                |  |  |
|             | jge               | print_pentiumiimodel3_type     |                                                |  |  |
|             | cmp               | eax, 1                         |                                                |  |  |
|             | jne               | print_unknown_type             | ; incorrect model number = 2                   |  |  |
|             | mov               | dx, offset pentiumpro_msg      | , meorreet moder number – 2                    |  |  |
|             | jmp               | print_common                   |                                                |  |  |
|             | Jinp              | print_common                   |                                                |  |  |
| print pen   | tiumiimoo         | del3_type:                     |                                                |  |  |
| r           | cmp               | eax, 3                         | ; if 6 & model 3, print Pentium                |  |  |
|             | ····P             |                                | ; II processor, model 3                        |  |  |
|             | jne               | print_pentiumiimodel5_type     | , in processor, moder c                        |  |  |
|             | mov               | dx, offset pentiumiimodel3_r   | nsa                                            |  |  |
|             | jmp               | print_common                   | lisg                                           |  |  |
|             | Juib              | print_common                   |                                                |  |  |
| print per   | tiumiimo          | del5_type:                     |                                                |  |  |
| print_pen   | cmp               | eax, 5                         | ; if 6 & model 5, either Pentium               |  |  |
|             | emp               | oux, 5                         | ; II processor, model 5, Pentium II            |  |  |
|             |                   |                                | ; Xeon processor or Intel Celeron              |  |  |
|             |                   |                                | •                                              |  |  |
|             | ia                | coloron voon dataat            | ; processor, model 5                           |  |  |
|             | je                | celeron_xeon_detect            |                                                |  |  |
|             | cmp               | eax, 7                         | ; If model 7 check cache descriptors           |  |  |
|             | unp               | oun, /                         | ; to determine Pentium III or Pentium III Xeon |  |  |
|             | jne               | print_celeronmodel6_type       | , to determine rentum in or rentum in Acon     |  |  |
| celeron s   | sile<br>keon_dete |                                |                                                |  |  |
|             |                   | ct.                            |                                                |  |  |
| ; Is it Pen | itium II pr       | ocessor, model 5, Pentium II X | Keon processor, Intel Celeron processor,       |  |  |
|             |                   |                                |                                                |  |  |

; Pentium III processor or Pentium III Xeon processor.

mov eax, dword ptr \_cache\_eax rol eax, 8 mov cx, 3 celeron\_detect\_eax: cmp al, 40h ; Is it no L2 je print\_celeron\_type

|                 | cmp<br>jae        | al, 44h<br>print_pentiumiixeon_type                        | ; Is L2 >= 1M   |
|-----------------|-------------------|------------------------------------------------------------|-----------------|
|                 | rol<br>loop       | eax, 8<br>celeron_detect_eax                               |                 |
|                 | mov<br>mov        | eax, dword ptr _cache_ebx cx, 4                            |                 |
| celeron         | detect_eb         | x:                                                         |                 |
|                 | cmp               | al, 40h                                                    | ; Is it no L2   |
|                 | je                | print_celeron_type                                         |                 |
|                 | cmp               | al, 44h                                                    | ; Is L2 >= 1M   |
|                 | jae               | print_pentiumiixeon_type                                   |                 |
|                 | rol               | eax, 8                                                     |                 |
|                 | loop              | celeron_detect_ebx                                         |                 |
|                 | •                 |                                                            |                 |
|                 | mov               | eax, dword ptr _cache_ecx                                  |                 |
|                 | mov               | cx, 4                                                      |                 |
| celeron         | _detect_ec        | x.                                                         |                 |
| eeleron <u></u> | _uctect_cc        | al, 40h                                                    | ; Is it no L2   |
|                 | je                | print_celeron_type                                         | ,               |
|                 | cmp               | al, 44h                                                    | ; Is L2 >= 1M   |
|                 | jae               | print_pentiumiixeon_type                                   |                 |
|                 |                   |                                                            |                 |
|                 | rol               | eax, 8                                                     |                 |
|                 | loop              | celeron_detect_ecx                                         |                 |
|                 | mov               | eax, dword ptr _cache_edx                                  |                 |
|                 | mov               | cx, 4                                                      |                 |
|                 |                   | - 7                                                        |                 |
| celeron         | _detect_ed        | x:                                                         |                 |
|                 | cmp               | al, 40h                                                    | ; Is it no L2   |
|                 | je                | print_celeron_type                                         |                 |
|                 | cmp               | al, 44h                                                    | ; Is L2 >= 1M   |
|                 | jae               | print_pentiumiixeon_type                                   |                 |
|                 | rol               | eax, 8                                                     |                 |
|                 | loop              | celeron_detect_edx                                         |                 |
|                 | •                 |                                                            |                 |
|                 | mov               | dx, offset pentiumiixeon_m                                 |                 |
|                 | mov               | eax, dword ptr _cpu_signatu                                | ire             |
|                 | shr               | eax, 4                                                     |                 |
|                 | and               | eax, 0fh                                                   | ; isolate model |
|                 | cmp               | eax, 5                                                     |                 |
|                 | je<br>mov         | print_common<br>dx, offset pentiumiii_msg                  |                 |
|                 | jmp               | print_common                                               |                 |
|                 | J-**P             | r                                                          |                 |
| print_ce        | eleron_typ        |                                                            |                 |
|                 | mov               | dx, offset celeron_msg                                     |                 |
|                 | jmp               | print_common                                               |                 |
| nrint -         | ntiumiira         | on tuno:                                                   |                 |
| print_pe        | entiumiixe<br>mov |                                                            | a               |
|                 | mov<br>mov        | dx, offset pentiumiixeon_ms<br>ax, word ptr _cpu_signature |                 |
|                 | 1110 V            | un, word pu _opu_signature                                 |                 |

|           | shr          | ax, 4                        |                                                   |
|-----------|--------------|------------------------------|---------------------------------------------------|
|           | and          | eax, 0fh                     | ; isolate model                                   |
|           | cmp          | eax, 5                       |                                                   |
|           | je           | print_common                 |                                                   |
|           | mov          | dx, offset pentiumiiixeon_ms | sg                                                |
|           | jmp          | print_common                 |                                                   |
| print_cel | leronmode    | el6_type:                    |                                                   |
|           | cmp          | eax, 6                       | ; if 6 & model 6, print Intel Celeron             |
|           |              |                              | ; processor, model 6                              |
|           | jne          | print_pentiumiiimodel8_type  |                                                   |
|           | mov          | dx, offset celeronmodel6_ms  | sg                                                |
|           | jmp          | print_common                 |                                                   |
| print_pe  | ntiumiiim    | odel8_type:                  |                                                   |
|           | cmp          | eax, 8                       | ; Pentium III processor, model 8, or              |
|           | jb           | print_unknown_type           | ; Pentium III Xeon processor, model 8             |
|           | mou          | any dword ptr factures aby   |                                                   |
|           | mov          | eax, dword ptr _features_ebx |                                                   |
|           | cmp          | al, 0<br>print_unknown_type  | ; Is brand_id supported?                          |
|           | je           | print_unknown_type           |                                                   |
|           | mov          | di, offset brand_table       | ; Setup pointer to brand_id table                 |
|           | mov          | cx, brand_table_size         | ; Get maximum entry count                         |
| next_bra  | ınd:         |                              |                                                   |
| -         | cmp          | al, byte ptr [di]            | ; Is this the brand reported by the processor     |
|           | je           | brand_found                  |                                                   |
|           | add          | di, sizeof brand_entry       | ; Point to next Brand Defined                     |
|           | loop         | next_brand                   | ; Check next brand if the table is not exhausted  |
|           | jmp          | print_unknown_type           |                                                   |
|           | 5 1          |                              |                                                   |
| brand_fo  |              | 1 1                          |                                                   |
|           | mov          | eax, dword ptr _cpu_signatur |                                                   |
|           | cmp          | eax, 06B1h                   | ; Check for Pentium III, model B, stepping 1      |
|           | jne          | not_b1_celeron               |                                                   |
|           | mov          | dx, offset celeron_brand     | ; Assume this is a the special case (see Table 9) |
|           | cmp          | byte ptr[di], 3              | ; Is this a B1 Celeron?                           |
|           | je           | print_common                 |                                                   |
| not_b1_   | celeron:     |                              |                                                   |
| _         | cmp          | eax, 0F13h                   |                                                   |
|           | jae          | not_xeon_mp                  |                                                   |
|           | mov          | dx, offset xeon_mp_brand     | ; Early "Intel(R) Xeon(TM) processor MP"?         |
|           | cmp          | byte ptr [di], 0Bh           | · · · · · · · · · · · · · · · · · · ·             |
|           | je           | print_common                 |                                                   |
|           | mov          | dx, offset xeon_brand        | ; Early "Intel(R) Xeon(TM) processor"?            |
|           | cmp          | byte ptr[di], 0Eh            | , Early month room (10) processor :               |
|           | je           | print_common                 |                                                   |
| not       | a            |                              |                                                   |
| not_xeo   | n_mp:<br>mov | dx, word ptr [di+1]          | ; Load DX with the offset of the brand string     |
|           | jmp          | print_common                 | , 2000 Dre mar are oriset of the orang suning     |
|           | JF           | 1                            |                                                   |

# intəl

print\_unknown\_type: dx, offset unknown\_msg ; if neither, print unknown mov print\_common: ah, 9h mov int 21h ; print family, model, and stepping print\_family: mov al, \_cpu\_type ASC\_MSG family\_msg ; print family msg eax, dword ptr \_cpu\_signature mov ah, Ofh ; Check for Extended Family and cmp ah, Ofh print\_model jne dx, offset ext\_fam\_msg mov ah, 9h mov 21h int eax, 20 shr ah, al ; Copy extended family into ah mov shr al, 4 ax, 0f0fh and ah, '0' add ; Convert upper nibble to ascii al, '0' ; Convert lower nibble to ascii add ax push dl, al mov mov ah, 2 21h ; print upper nibble of ext family int pop ax dl, ah mov mov ah. 2 int 21h ; print lower nibble of ext family print\_model: mov eax, dword ptr \_cpu\_signature shr ax, 4 al, Ofh and ASC\_MSG ; print model msg model\_msg mov eax, dword ptr \_cpu\_signature al, OfOh ; Check for Extended Model and ah, OfOh cmp jne print\_stepping dx, offset ext\_mod\_msg mov ah, 9h mov 21h int shr eax, 16 al, 0fh and al, '0' add ; Convert extended model to ascii dl, al mov ah, 2 mov int 21h ; print lower nibble of ext family print\_stepping: eax, dword ptr \_cpu\_signature mov and al. 0fh ASC\_MSG ; print stepping msg stepping\_msg

| print_upg | grade: |                              |                            |
|-----------|--------|------------------------------|----------------------------|
|           | mov    | eax, dword ptr _cpu_signatur |                            |
|           | test   | ax, 1000h                    | ; check for turbo upgrade  |
|           | JZ     | check_dp                     |                            |
|           | mov    | dx, offset turbo_msg         |                            |
|           | mov    | ah, 9h                       |                            |
|           | int    | 21h                          |                            |
|           | Jmp    | print_features               |                            |
| check_dp  | ):     |                              |                            |
|           | test   | ax, 2000h                    | ; check for dual processor |
|           | jz     | print_features               |                            |
|           | mov    | dx, offset dp_msg            |                            |
|           | mov    | ah, 9h                       |                            |
|           | int    | 21h                          |                            |
| print_fea | tures: |                              |                            |
|           | mov    | eax, dword ptr _features_edx |                            |
|           | and    | eax, FPU_FLAG                | ; check for FPU            |
|           | jz     | check_VME                    |                            |
|           | mov    | dx, offset fpu_msg           |                            |
|           | mov    | ah, 9h                       |                            |
|           | int    | 21h                          |                            |
| check_V   | ME:    |                              |                            |
|           | mov    | eax, dword ptr _features_edx |                            |
|           | and    | eax, VME_FLAG                | ; check for VME            |
|           | jz     | check_DE                     |                            |
|           | mov    | dx, offset vme_msg           |                            |
|           | mov    | ah, 9h                       |                            |
|           | int    | 21h                          |                            |
| check_D   | E:     |                              |                            |
|           | mov    | eax, dword ptr _features_edx |                            |
|           | and    | eax, DE_FLAG                 | ; check for DE             |
|           | jz     | check_PSE                    | ,                          |
|           | mov    | dx, offset de_msg            |                            |
|           | mov    | ah, 9h                       |                            |
|           | int    | 21h                          |                            |
| check PS  | SE:    |                              |                            |
|           | mov    | eax, dword ptr _features_edx |                            |
|           | and    | eax, PSE_FLAG                | ; check for PSE            |
|           | jz     | check_TSC                    | ,                          |
|           | mov    | dx, offset pse_msg           |                            |
|           | mov    | ah, 9h                       |                            |
|           | int    | 21h                          |                            |
| check_TS  | SC:    |                              |                            |
|           | mov    | eax, dword ptr _features_edx |                            |
|           | and    | eax, TSC FLAG                | ; check for TSC            |
|           | jz     | check_MSR                    |                            |
|           | mov    | dx, offset tsc_msg           |                            |
|           | mov    | ah, 9h                       |                            |
|           | int    | 21h                          |                            |
|           |        |                              |                            |

check\_MSR:

| mov<br>and  | eax, dword ptr _features_edx<br>eax, MSR_FLAG | ; check for MSR              |
|-------------|-----------------------------------------------|------------------------------|
| jz          | check_PAE                                     |                              |
| mov         | dx, offset msr_msg                            |                              |
| mov         | ah, 9h                                        |                              |
| int         | 21h                                           |                              |
| check_PAE:  |                                               |                              |
| mov         | eax, dword ptr _features_edx                  |                              |
| and         | eax, PAE_FLAG                                 | ; check for PAE              |
| jz          | check_MCE                                     |                              |
| mov         | dx, offset pae_msg                            |                              |
| mov         | ah, 9h<br>21h                                 |                              |
| int         | 2111                                          |                              |
| check_MCE:  |                                               |                              |
| mov         | eax, dword ptr _features_edx                  |                              |
| and         | eax, MCE_FLAG                                 | ; check for MCE              |
| jz          | check_CX8                                     |                              |
| mov         | dx, offset mce_msg                            |                              |
| mov         | ah, 9h                                        |                              |
| int         | 21h                                           |                              |
| check_CX8:  |                                               |                              |
| mov         | eax, dword ptr _features_edx                  |                              |
| and         | eax, CX8_FLAG                                 | ; check for CMPXCHG8B        |
| jz          | check_APIC                                    |                              |
| mov         | dx, offset cx8_msg                            |                              |
| mov         | ah, 9h                                        |                              |
| int         | 21h                                           |                              |
| check_APIC: |                                               |                              |
| mov         | eax, dword ptr _features_edx                  |                              |
| and         | eax, APIC_FLAG                                | ; check for APIC             |
| jz          | check_SEP                                     | ,                            |
| mov         | dx, offset apic_msg                           |                              |
| mov         | ah, 9h                                        |                              |
| int         | 21h                                           |                              |
| check_SEP:  |                                               |                              |
| mov         | eax, dword ptr _features_edx                  |                              |
| and         | eax, SEP_FLAG                                 | ; Check for Fast System Call |
| jz          | check_MTRR                                    | , check for f ast system can |
| cmp         | _cpu_type, 6                                  | ; Determine if Fast System   |
| -           | _cpu_type, o<br>print_sep                     | ; Calls are supported.       |
| jne         | print_och                                     | , cans are supported.        |
| mov         | eax, dword ptr _cpu_signatur                  | re                           |
| cmp         | al, 33h                                       |                              |
| jb          | print_no_sep                                  |                              |
| print_sep:  |                                               |                              |
| mov         | dx, offset sep_msg                            |                              |
| mov         | ah, 9h                                        |                              |
| int         | 21h                                           |                              |
| jmp         | check_MTRR                                    |                              |
|             |                                               |                              |

|          | mov              | dx, offset no_sep_msg                           |                                     |
|----------|------------------|-------------------------------------------------|-------------------------------------|
|          | mov              | ah, 9h                                          |                                     |
|          | int              | 21h                                             |                                     |
|          |                  |                                                 |                                     |
| check_M  | FRR:             |                                                 |                                     |
|          | mov              | eax, dword ptr _features_edx                    | K                                   |
|          | and              | eax, MTRR_FLAG                                  | ; check for MTRR                    |
|          | jz               | check_PGE                                       |                                     |
|          | mov              | dx, offset mtrr_msg                             |                                     |
|          | mov              | ah, 9h                                          |                                     |
|          | int              | 21h                                             |                                     |
|          |                  |                                                 |                                     |
| check_PC | E:               |                                                 |                                     |
|          | mov              | eax, dword ptr _features_edx                    |                                     |
|          | and              | eax, PGE_FLAG                                   | ; check for PGE                     |
|          | jz               | check_MCA                                       |                                     |
|          | mov              | dx, offset pge_msg                              |                                     |
|          | mov              | ah, 9h                                          |                                     |
|          | int              | 21h                                             |                                     |
|          |                  |                                                 |                                     |
| check_M0 | CA:              |                                                 |                                     |
|          | mov              | eax, dword ptr _features_edx                    |                                     |
|          | and              | eax, MCA_FLAG                                   | ; check for MCA                     |
|          | jz               | check_CMOV                                      | ,                                   |
|          | mov              | dx, offset mca_msg                              |                                     |
|          | mov              | ah, 9h                                          |                                     |
|          | int              | 21h                                             |                                     |
|          | int              | 2111                                            |                                     |
| check_CM | AOV:             |                                                 |                                     |
|          | mov              | eax, dword ptr _features_edx                    |                                     |
|          | and              | eax, CMOV_FLAG                                  | ; check for CMOV                    |
|          | jz               | check_PAT                                       | , check for child v                 |
|          | mov              | dx, offset cmov_msg                             |                                     |
|          | mov              | ah, 9h                                          |                                     |
|          | int              | 21h                                             |                                     |
|          | int              | 2111                                            |                                     |
| check_PA | T:               |                                                 |                                     |
|          | mov              | eax, dword ptr _features_edx                    |                                     |
|          | and              | eax, PAT_FLAG                                   |                                     |
|          | jz               | check_PSE36                                     |                                     |
|          | JZ<br>mov        | dx, offset pat_msg                              |                                     |
|          | mov              | ah, 9h                                          |                                     |
|          | int              | 21h                                             |                                     |
|          |                  | 21                                              |                                     |
| check_PS | E36 <sup>.</sup> |                                                 |                                     |
| _        | mov              | eax, dword ptr _features_edx                    |                                     |
|          | and              | eax, dword pit _leatures_edx<br>eax, PSE36_FLAG |                                     |
|          |                  | check_PSNUM                                     |                                     |
|          | jz               | dx, offset pse36_msg                            |                                     |
|          | mov              |                                                 |                                     |
|          | mov              | ah, 9h                                          |                                     |
|          | int              | 21h                                             |                                     |
| akl- DO  | NTLIN (          |                                                 |                                     |
| check_PS |                  | and descend and the first of the later          |                                     |
|          | mov              | eax, dword ptr _features_edx                    |                                     |
|          | and              | eax, PSNUM_FLAG                                 | ; check for processor serial number |
|          | jz               | check_CLFLUSH                                   |                                     |
|          | mov              | dx, offset psnum_msg                            |                                     |
|          | mov              | ah, 9h                                          |                                     |

|          | int              | 21h                                                            |                                     |
|----------|------------------|----------------------------------------------------------------|-------------------------------------|
| check_CI | LFLUSH:          |                                                                |                                     |
|          | mov<br>and<br>jz | eax, dword ptr _features_edx<br>eax, CLFLUSH_FLAG<br>check_DTS | ; check for Cache Line Flush        |
|          | mov              | dx, offset clflush_msg                                         |                                     |
|          | mov              | ah, 9h                                                         |                                     |
|          | int              | 21h                                                            |                                     |
| check_D  | ГS:              |                                                                |                                     |
|          | mov              | $eax,  dword  ptr \_features\_edx$                             |                                     |
|          | and              | eax, DTS_FLAG                                                  | ; check for Debug Trace Store       |
|          | jz               | check_ACPI                                                     |                                     |
|          | mov              | dx, offset dts_msg                                             |                                     |
|          | mov<br>int       | ah, 9h<br>21h                                                  |                                     |
|          | IIIt             | 2111                                                           |                                     |
| check_A  |                  | any dword ptr. factures adv                                    |                                     |
|          | mov<br>and       | eax, dword ptr _features_edx<br>eax, ACPI_FLAG                 | ; check for processor serial number |
|          | jz               | check_MMX                                                      | , check for processor serial number |
|          | mov              | dx, offset acpi_msg                                            |                                     |
|          | mov              | ah, 9h                                                         |                                     |
|          | int              | 21h                                                            |                                     |
| check_M  | MX:              |                                                                |                                     |
| encen_n  | mov              | eax, dword ptr _features_edx                                   |                                     |
|          | and              | eax, MMX_FLAG                                                  | ; check for MMX technology          |
|          | jz               | check_FXSR                                                     |                                     |
|          | mov              | dx, offset mmx_msg                                             |                                     |
|          | mov              | ah, 9h                                                         |                                     |
|          | ınt              | 21h                                                            |                                     |
| check_FX | KSR:             |                                                                |                                     |
|          | mov              | eax, dword ptr _features_edx                                   |                                     |
|          | and              | eax, FXSR_FLAG                                                 | ; check for FXSR                    |
|          | JZ               | check_SSE<br>dv. offset fysr. msg.                             |                                     |
|          | mov<br>mov       | dx, offset fxsr_msg<br>ah, 9h                                  |                                     |
|          | int              | 21h                                                            |                                     |
| 1 1 00   | F                |                                                                |                                     |
| check_SS | mov              | eax, dword ptr _features_edx                                   |                                     |
|          | and              | eax, SSE_FLAG                                                  | ; check for Streaming SIMD          |
|          | jz               | check_SSE2                                                     | ; Extensions                        |
|          | mov              | dx, offset sse_msg                                             | ,                                   |
|          | mov              | ah, 9h                                                         |                                     |
|          | int              | 21h                                                            |                                     |
| check_SS | E2:              |                                                                |                                     |
|          | mov              | eax, dword ptr _features_edx                                   |                                     |
|          | and              | eax, SSE2_FLAG                                                 | ; check for Streaming SIMD          |
|          | jz               | check_SS                                                       | ; Extensions 2                      |
|          | mov              | dx, offset sse2_msg                                            |                                     |
|          | mov              | ah, 9h                                                         |                                     |
|          | int              | 21h                                                            |                                     |

| check_SS | S:                                               |                                                                                                              |                                                                           |
|----------|--------------------------------------------------|--------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------|
|          | mov<br>and<br>jz<br>mov<br>mov<br>int            | eax, dword ptr _features_edx<br>eax, SS_FLAG<br>check_HTT<br>dx, offset ss_msg<br>ah, 9h<br>21h              | ; check for Self Snoop                                                    |
| check_H  | ΓT:<br>mov<br>and<br>jz                          | eax, dword ptr _features_edx<br>eax, HTT_FLAG<br>check_TM                                                    | ; check for Hyper-Thread Technology                                       |
|          | mov<br>bswap<br>cmp<br>je                        | eax, dword ptr _features_ebx<br>eax<br>ah, 1<br>check_TM                                                     | ; Put Logical processor count in reg AH<br>; Logical processor count > 1? |
|          | mov<br>mov<br>int                                | dx, offset htt_msg<br>ah, 9h<br>21h                                                                          | ; Supports HTT                                                            |
| check_TI | M:<br>mov<br>and<br>jz<br>mov<br>mov<br>int      | eax, dword ptr _features_edx<br>eax, TM_FLAG<br>check_PBE<br>dx, offset tm_msg<br>ah, 9h<br>21h              | ; check for Thermal Monitor                                               |
| check_PI | BE:<br>mov<br>and<br>jz<br>mov<br>mov<br>int     | eax, dword ptr _features_edx<br>eax, PBE_FLAG<br>check_sse3<br>dx, offset pbe_msg<br>ah, 9h<br>21h           | ; check for Pending Break Event                                           |
| check_ss | e3:<br>mov<br>and<br>jz<br>mov<br>mov<br>int     | eax, dword ptr _features_ecx<br>eax, SSE3_FLAG<br>check_monitor<br>dx, offset sse3_msg<br>ah, 9h<br>21h      | ; check for SSE3 instructions                                             |
| check_m  | onitor:<br>mov<br>and<br>jz<br>mov<br>mov<br>int | eax, dword ptr _features_ecx<br>eax, MONITOR_FLAG<br>check_ds_cpl<br>dx, offset monitor_msg<br>ah, 9h<br>21h | ; check for monitor/mwait instructions                                    |
| check_ds | _cpl:<br>mov<br>and<br>jz                        | eax, dword ptr _features_ecx<br>eax, DS_CPL_FLAG<br>check_EIST                                               | ; check for Debug Store extensions qualified by CPL                       |

|           | mov<br>mov<br>int                | dx, offset ds_cpl_msg<br>ah, 9h<br>21h                                                              |                                           |
|-----------|----------------------------------|-----------------------------------------------------------------------------------------------------|-------------------------------------------|
| check_EI  | mov<br>and<br>jz                 | eax, dword ptr _features_ecx<br>eax, EIST_FLAG<br>check_TM2<br>dx, offset eist_msg<br>ah, 9h<br>21h | ; check for Enhanced SpeedStep Technology |
| check_TM  | mov<br>and<br>jz                 | eax, dword ptr _features_ecx<br>eax, TM2_FLAG<br>check_CID<br>dx, offset tm2_msg<br>ah, 9h<br>21h   | ; check for Thermal Monitor 2             |
| check_CI  | mov<br>and<br>jz                 | eax, dword ptr _features_ecx<br>eax, CID_FLAG<br>check_XTPR<br>dx, offset cid_msg<br>ah, 9h<br>21h  | ; check for L1 Context ID                 |
| check_X   | mov<br>and<br>jz                 | eax, dword ptr _features_ecx<br>eax, XTPR_FLAG<br>end_print<br>dx, offset xtpr_msg<br>ah, 9h<br>21h | ; check for echo Task Priority            |
|           | jmp                              | end_print                                                                                           |                                           |
| not_Genu  |                                  | dx, offset not_intel<br>ah, 9h<br>21h                                                               |                                           |
| end_print | mov<br>mov<br>int<br>ret<br>endp | dx, offset cr_lf<br>ah, 9h<br>21h                                                                   |                                           |
| print     | end start                        |                                                                                                     |                                           |

### Example 3. Processor Identification Procedure in the C Language

|                                                                                                                                  | /* FILENAME: CPUID3.C                                                                                                                                 |                                         | */       |  |  |
|----------------------------------------------------------------------------------------------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------|----------|--|--|
|                                                                                                                                  | /* Copyright (c) Intel Corpor                                                                                                                         | ation 1994-2004                         | */<br>*/ |  |  |
|                                                                                                                                  | ,                                                                                                                                                     | veloped by Intel Corporation. Intel has | */       |  |  |
|                                                                                                                                  |                                                                                                                                                       | ty rights which it may assert under     | */       |  |  |
|                                                                                                                                  |                                                                                                                                                       | h as if another manufacturer's          | */       |  |  |
| /* processor mis-identifies itself as being "GenuineIntel" when                                                                  |                                                                                                                                                       |                                         |          |  |  |
|                                                                                                                                  | /* the CPUID instruction is e                                                                                                                         | executed.                               | */       |  |  |
|                                                                                                                                  | /*                                                                                                                                                    |                                         | */       |  |  |
|                                                                                                                                  |                                                                                                                                                       | s all warranties, express or implied,   | */<br>*/ |  |  |
|                                                                                                                                  |                                                                                                                                                       | consequential and other indirect        | */       |  |  |
| /* damages, for the use of this program, including liability for<br>/* infringement of any proprietary rights, and including the |                                                                                                                                                       |                                         |          |  |  |
| /* warranties of merchantability and fitness for a particular                                                                    |                                                                                                                                                       |                                         |          |  |  |
| /* purpose. Intel does not assume any responsibility for any                                                                     |                                                                                                                                                       |                                         |          |  |  |
| /* errors which may appear in this program nor any responsibility                                                                |                                                                                                                                                       |                                         |          |  |  |
|                                                                                                                                  | /* to update it.                                                                                                                                      |                                         | */       |  |  |
|                                                                                                                                  | /*                                                                                                                                                    |                                         | */       |  |  |
|                                                                                                                                  | /*                                                                                                                                                    |                                         | */       |  |  |
|                                                                                                                                  | /* This program contains thre                                                                                                                         |                                         | */       |  |  |
|                                                                                                                                  | /* Part 1: Identifies CPU type<br>/*                                                                                                                  | e in the variable _cpu_type:            | */<br>*/ |  |  |
|                                                                                                                                  | /* /* Part 2: Identifies FPU type in the variable _fpu_type:                                                                                          |                                         |          |  |  |
|                                                                                                                                  | /* Part 2: identifies FPO type in the variable _ipu_type:<br>/*                                                                                       |                                         |          |  |  |
|                                                                                                                                  | /* Part 3: Prints out the appropriate message.                                                                                                        |                                         |          |  |  |
|                                                                                                                                  | /*<br>/*                                                                                                                                              |                                         |          |  |  |
|                                                                                                                                  | /* This program has been tested with the Microsoft Developer Studio.                                                                                  |                                         |          |  |  |
|                                                                                                                                  | /* If this code is compiled with no options specified and linked                                                                                      |                                         |          |  |  |
|                                                                                                                                  | /* with the cpuid3a module, it correctly identifies the current                                                                                       |                                         |          |  |  |
|                                                                                                                                  | /* Intel 8086/8088, 80286, 80386, 80486, Pentium(R), Pentium(R) Pro,                                                                                  |                                         |          |  |  |
|                                                                                                                                  | /* Pentium(R) II, Pentium(R) II Xeon(TM), Pentium(R) II OverDrive(R),<br>/* Intol(P) Calaron(P), Pantium(P) III processor, Pentium(P) III Xeon(TM)    |                                         |          |  |  |
|                                                                                                                                  | /* Intel(R) Celeron(R), Pentium(R) III processors, Pentium(R) III Xeon(TM)<br>/* processors, Pentium(R) 4 processors and Intel(R) Xeon(TM) processors |                                         |          |  |  |
|                                                                                                                                  | , processors, rentram(rt) r                                                                                                                           |                                         | */       |  |  |
|                                                                                                                                  | #define FPU_FLAG                                                                                                                                      | 0x0001                                  |          |  |  |
|                                                                                                                                  | #define VME_FLAG                                                                                                                                      | 0x0002                                  |          |  |  |
|                                                                                                                                  | #define DE_FLAG                                                                                                                                       | 0x0004                                  |          |  |  |
|                                                                                                                                  | #define PSE_FLAG                                                                                                                                      | 0x0008                                  |          |  |  |
|                                                                                                                                  | #define TSC_FLAG<br>#define MSR_FLAG                                                                                                                  | 0x0010<br>0x0020                        |          |  |  |
|                                                                                                                                  | #define PAE_FLAG                                                                                                                                      | 0x0020<br>0x0040                        |          |  |  |
|                                                                                                                                  | #define MCE FLAG                                                                                                                                      | 0x0040                                  |          |  |  |
|                                                                                                                                  | #define CX8_FLAG                                                                                                                                      | 0x0100                                  |          |  |  |
|                                                                                                                                  | #define APIC FLAG                                                                                                                                     | 0x0200                                  |          |  |  |
|                                                                                                                                  | #define SEP FLAG                                                                                                                                      | 0x0800                                  |          |  |  |
|                                                                                                                                  | #define MTRR_FLAG                                                                                                                                     | 0x1000                                  |          |  |  |
|                                                                                                                                  | #define PGE_FLAG                                                                                                                                      | 0x2000                                  |          |  |  |
|                                                                                                                                  | #define MCA_FLAG                                                                                                                                      | 0x4000                                  |          |  |  |
|                                                                                                                                  | #define CMOV_FLAG                                                                                                                                     | 0x8000                                  |          |  |  |
|                                                                                                                                  | #define PAT_FLAG                                                                                                                                      | 0x10000                                 |          |  |  |
|                                                                                                                                  | #define PSE36_FLAG                                                                                                                                    | 0x20000<br>0x40000                      |          |  |  |
|                                                                                                                                  | #define PSNUM_FLAG<br>#define CLFLUSH FLAG                                                                                                            | 0x40000<br>0x80000                      |          |  |  |
|                                                                                                                                  | #define DTS FLAG                                                                                                                                      | 0x200000                                |          |  |  |
|                                                                                                                                  | #define ACPI FLAG                                                                                                                                     | 0x400000                                |          |  |  |
|                                                                                                                                  | #define MMX_FLAG                                                                                                                                      | 0x800000                                |          |  |  |
|                                                                                                                                  |                                                                                                                                                       |                                         |          |  |  |

0x1000000 0x2000000 0x4000000

0x8000000 0x10000000 0x20000000 0x8000000 0x0001

#define FXSR\_FLAG #define SSE\_FLAG #define SSE2\_FLAG

#define SSE2\_FLAG #define SS\_FLAG #define HTT\_FLAG #define TM\_FLAG #define PBE\_FLAG #define SSE3\_FLAG

72

```
#define MONITOR_FLAG
                                                      0x0008
#define DS_CPL_FLAG
#define EIST_FLAG
                                                        0x0010
                                                        0x0080
#define TM2 FLAG
                                                        0x0100
#define CID_FLAG
                                                        0x0400
#define XTPR_FLAG
                                                        0x4000
extern char cpu_type;
extern char fpu_type;
extern char cpuid_flag;
extern char intel_CPU;
extern char vendor_id[12];
extern long cpu_signature;
extern long features_ecx;
extern long features_edx;
extern long features_ebx;
extern long cache_eax;
extern long cache_ebx;
extern long cache_ecx;
extern long cache edx;
extern char brand_string[48];
extern int brand_id;
long cache_temp;
long celeron_flag;
long pentiumxeon_flag;
struct brand_entry {
    long
                brand_value;
                 *brand_string;
    char
};
#define brand_table_size 15
struct brand_entry brand_table[brand_table_size] = {
    0x01, " Genuine Intel(R) Celeron(R) processor",
0x02, " Genuine Intel(R) Pentium(R) III processor",
   0x02, "Genuine Intel(R) Pentium(R) III processor,

0x03, "Genuine Intel(R) Pentium(R) III Xeon(TM) processor",

0x04, "Genuine Intel(R) Pentium(R) III processor",

0x06, "Genuine Mobile Intel(R) Pentium(R) III Processor - M",

0x07, "Genuine Mobile Intel(R) Celeron(R) processor",

0x08, "Genuine Intel(R) Pentium(R) 4 processor",

0x08, "Genuine Intel(R) Pentium(R) 4 processor",

0x0A, "Genuine Intel(R) Celeron(R) processor",

0x0A, "Genuine Intel(R) Xeon(TM) processor",

0x0B, "Genuine Intel(R) Xeon(TM) processor",
   0x0R, "Genuine Intel(R) Celefon(R) processor",
0x0B, "Genuine Intel(R) Xeon(TM) processor",
0x0C, "Genuine Intel(R) Xeon(TM) Processor MP",
0x0E, "Genuine Mobile Intel(R) Pentium(R) 4 Processor - M",

OxOE, "Genuine Mobile Intel(R) Pentium(R) 4 Processor - 1
OxOF, "Genuine Mobile Intel(R) Celeron(R) processor",
Ox11, "Mobile Genuine Intel(R) processor",
Ox12, "Genuine Mobile Intel(R) Celeron(R) M processor",
Ox13, "Genuine Intel(R) Celeron(R) processor",
Ox14, "Genuine Intel(R) Celeron(R) processor",
Ox15, "Mobile Genuine Intel(R) processor",
Ox15, "Genuine Intel(R) Pentium(R) M processor"
Ox16, "Genuine Intel(R) Pentium(R) M processor"

    0x17, "Genuine Mobile Intel(R) Celeron(R) processor",
};
int main() {
    get_cpu_type();
    get_fpu_type();
    print();
    return(0);
int print() {
    int brand_index = 0;
```

```
printf("This system has a");
if (cpuid_flag == 0) {
  switch (cpu_type) {
  case 0:
     printf("n 8086/8088 processor");
     if (fpu_type) printf(" and an 8087 math coprocessor");
     break;
  case 2:
     printf("n 80286 processor");
     if (fpu_type) printf(" and an 80287 math coprocessor");
     break;
  case 3:
     printf("n 80386 processor");
     if (fpu_type == 2)
printf(" and an 80287 math coprocessor");
     else if (fpu_type)
       printf(" and an 80387 math coprocessor");
     break;
  case 4:
     if (fpu_type)
       printf("n 80486DX, 80486DX2 processor or 80487SX math coprocessor");
     else
       printf("n 80486SX processor");
     break;
  default:
     printf("n unknown processor");
else {
  using cpuid instruction */
  if (intel_CPU) {
     if (brand_string[0]) {
       brand_index = 0;
       while ((brand_string[brand_index] == ' ') && (brand_index < 48))
          brand_index++;
       if (brand_index != 48)
          printf(" %s", &brand_string[brand_index]);
     else if (cpu_type == 4) {
       switch ((cpu_signature>>4) & 0xf) {
       case 0:
       case 1:
          printf(" Genuine Intel486(TM) DX processor");
          break;
       case 2:
          printf(" Genuine Intel486(TM) SX processor");
          break;
       case 3:
          printf(" Genuine IntelDX2(TM) processor");
          break;
       case 4:
          printf(" Genuine Intel486(TM) processor");
          break;
       case 5:
          printf(" Genuine IntelSX2(TM) processor");
          break;
       case 7:
          printf(" Genuine Write-Back Enhanced \
            IntelDX2(TM) processor");
          break;
       case 8:
          printf(" Genuine IntelDX4(TM) processor");
          break;
       default:
          printf(" Genuine Intel486(TM) processor");
     }
```

```
else if (cpu_type == 5)
  printf(" Genuine Intel Pentium(R) processor");
else if ((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 1))
printf(" Genuine Intel Pentium(R) Pro processor");
else if ((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 3))
printf(" Genuine Intel Pentium(R) II processor, model 3");
else if (((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 5)) ||
       ((cpu_type == 6) \&\& (((cpu_signature >> 4) \& 0xf) == 7)))
{
  celeron_flag = 0;
  pentiumxeon_flag = 0;
  cache_temp = cache_eax & 0xFF000000;
  if (cache_temp == 0x4000000)
     celeron_flag = 1;
  if ((cache_temp >= 0x4400000) && (cache_temp <= 0x4500000))
     pentiumxeon_flag = 1;
  cache_temp = cache_eax & 0xFF0000;
  if (cache_temp == 0x400000)
     celeron_flag = 1;
  if ((cache temp \geq 0x440000) && (cache temp \leq 0x450000))
     pentiumxeon_flag = 1;
  cache_temp = cache_eax & 0xFF00;
  if (cache_temp == 0x4000)
     celeron_flag = 1;
  if ((cache_temp >= 0x4400) && (cache_temp <= 0x4500))
     pentiumxeon_flag = 1;
  cache_temp = cache_ebx & 0xFF000000;
  if (cache_temp == 0x4000000)
     celeron_flag = 1;
  if ((cache_temp >= 0x44000000) && (cache_temp <= 0x45000000))
     pentiumxeon_flag = 1;
  cache_temp = cache_ebx & 0xFF0000;
  if (cache_temp == 0x400000)
     celeron_flag = 1;
  if ((cache_temp >= 0x440000) && (cache_temp <= 0x450000))
     pentiumxeon_flag = 1;
  cache_temp = cache_ebx & 0xFF00;
  if (cache temp == 0x4000)
     celeron_flag = 1;
  if ((cache_temp >= 0x4400) && (cache_temp <= 0x4500))
     pentiumxeon_flag = 1;
  cache_temp = cache_ebx & 0xFF;
  if (cache_temp == 0x40)
     celeron_flag = 1;
  if ((cache_temp \geq 0x44) && (cache_temp \leq 0x45))
     pentiumxeon_flag = 1;
  cache_temp = cache_ecx & 0xFF000000;
  if (cache_temp == 0x4000000)
     celeron_flag = 1;
  if ((cache_temp >= 0x44000000) && (cache_temp <= 0x45000000))
     pentiumxeon_flag = 1;
  cache_temp = cache_ecx & 0xFF0000;
  if (cache_temp == 0x400000)
     celeron_flag = 1;
  if ((cache_temp >= 0x440000) && (cache_temp <= 0x450000))
     pentiumxeon_flag = 1;
  cache_temp = cache_ecx & 0xFF00;
  if (cache_temp == 0x4000)
     celeron_flag = 1;
```

```
if ((cache_temp >= 0x4400) && (cache_temp <= 0x4500))
     pentiumxeon_flag = 1;
   cache_temp = cache_ecx & 0xFF;
   if (cache_temp == 0x40)
     celeron flag = 1;
   if ((cache_temp \geq 0x44) && (cache_temp \leq 0x45))
     pentiumxeon_flag = 1;
   cache_temp = cache_edx & 0xFF000000;
   if (cache_temp == 0x4000000)
     celeron_flag = 1;
   if ((cache_temp >= 0x44000000) && (cache_temp <= 0x45000000))
     pentiumxeon_flag = 1;
   cache_temp = cache_edx & 0xFF0000;
   if (cache_temp == 0x400000)
     celeron_flag = 1;
   if ((cache_temp >= 0x440000) && (cache_temp <= 0x450000))
     pentiumxeon_flag = 1;
   cache_temp = cache_edx & 0xFF00;
   if (cache_temp == 0x4000)
     celeron_flag = 1;
   if ((cache_temp >= 0x4400) && (cache_temp <= 0x4500))
     pentiumxeon_flag = 1;
   cache_temp = cache_edx & 0xFF;
   if (cache_temp == 0x40)
     celeron_flag = 1;
   if ((cache_temp \geq 0x44) && (cache_temp \leq 0x45))
     pentiumxeon_flag = 1;
  if (celeron_flag == 1)
     printf(" Genuine Intel Celeron(R) processor, model 5");
   else
   {
     if (pentiumxeon_flag == 1) {
        if (((cpu_signature >> 4) & 0x0f) == 5)
          printf(" Genuine Intel Pentium(R) II Xeon(TM) processor");
        else
          printf(" Genuine Intel Pentium(R) III Xeon(TM) processor,");
printf(" model 7");
     else {
        if (((cpu_signature >> 4) & 0x0f) == 5) {
          printf(" Genuine Intel Pentium(R) II processor, model 5 ");
          printf("or Intel Pentium(R) II Xeon(TM) processor");
        else {
          printf(" Genuine Intel Pentium(R) III processor, model 7");
printf(" or Intel Pentium(R) III Xeon(TM) processor,");
printf(" model 7");
        }
     }
  }
else if ((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 6))
printf(" Genuine Intel Celeron(R) processor, model 6");
else if ((features_ebx & 0xff) != 0) {
   while ((brand_index < brand_table_size) &&
     ((features_ebx & 0xff) != brand_table[brand_index].brand_value))
     brand_index++;
   if (brand_index < brand_table_size) {
     if ((cpu_signature == 0x6B1) \&\&
        (brand\_table[brand\_index].brand\_value == 0x3))
       printf(" Genuine Intel(R) Celeron(R) processor");
     else if ((cpu_signature < 0xF13) &&
```

 $(brand_table[brand_index].brand_value == 0x0B))$ printf(" Genuine Intel(R) Xeon(TM) processor MP"); else if ((cpu\_signature < 0xF13) &&  $(brand_table[brand_index].brand_value == 0x0E))$ printf(" Genuine Intel(R) Xeon(TM) processor"); else printf("%s", brand\_table[brand\_index].brand\_string); else printf("n unknown Genuine Intel processor"); else printf("n unknown Genuine Intel processor"); printf("\nProcessor Family: %X", cpu\_type); if (cpu\_type == 0xf) printf("\n Extended Family: %x",(cpu\_signature>>20)&0xff); printf("\nModel: %X", (cpu\_signature>>4)&0xf); %X", (cpu\_signature>>4)&0xf); if (((cpu\_signature>>4) & 0xf) == 0xf) printf("\n Extended Model: %x",(cpu\_signature>>16)&0xf); intf("\nStepping: %X\n", cpu\_signature&0xf); printf("\nStepping: if (cpu\_signature & 0x1000) printf("\nThe processor is an OverDrive(R) processor"); else if (cpu\_signature & 0x2000) printf("\nThe processor is the upgrade processor in a dual processor system"); if (features\_edx & FPU\_FLAG) printf("\nThe processor contains an on-chip FPU"); if (features\_edx & VME\_FLAG) printf("\nThe processor supports Virtual Mode Extensions"); if (features\_edx & DE\_FLAG) printf("\nThe processor supports the Debugging Extensions"); if (features\_edx & PSE\_FLAG) printf("\nThe processor supports Page Size Extensions"); if (features\_edx & TSC\_FLAG) printf("\nThe processor supports Time Stamp Counter"); if (features\_edx & MSR\_FLAG) printf("\nThe processor supports Model Specific Registers"); if (features\_edx & PAE\_FLAG) printf("\nThe processor supports Physical Address Extension"); if (features\_edx & MCE\_FLAG) printf("\nThe processor supports Machine Check Exceptions"); if (features\_edx & CX8\_FLAG) printf("\nThe processor supports the CMPXCHG8B instruction"); if (features edx & APIC FLAG) printf("\nThe processor contains an on-chip APIC"); if (features\_edx & SEP\_FLAG) { if  $((cpu\_type\_== 6) \&\& ((cpu\_signature \& 0xff) < 0x33))$ printf("\nThe processor does not support the Fast System Call"); else printf("\nThe processor supports the Fast System Call"); if (features edx & MTRR FLAG) printf("\nThe processor supports the Memory Type Range Registers"); if (features\_edx & PGE\_FLAG) printf("\nThe processor supports Page Global Enable"); if (features\_edx & MCA\_FLAG) printf("\nThe processor supports the Machine Check Architecture"); if (features\_edx & CMOV\_FLAG) printf("\nThe processor supports the Conditional Move Instruction"); if (features\_edx & PAT\_FLAG) printf("\nThe processor supports the Page Attribute Table"); if (features\_edx & PSE36\_FLAG) printf("\nThe processor supports 36-bit Page Size Extension"); if (features\_edx & PSNUM\_FLAG) printf("\nThe processor supports the processor serial number"); if (features edx & CLFLUSH FLAG) printf("\nThe processor supports the CLFLUSH instruction");

if (features\_edx & DTS\_FLAG) printf("\nThe processor supports the Debug Trace Store feature");



```
if (features_edx & ACPI_FLAG)
       printf("\nThe processor supports ACPI registers in MSR space");
     if (features_edx & MMX_FLÂG)
       printf("\nThe processor supports Intel Architecture MMX(TM) technology");
     if (features_edx & FXSR_FLAG)
       printf("\nThe processor supports the Fast floating point save and restore");
     if (features_edx & SSE_FLAG)
     printf("\nThe processor supports the Streaming SIMD extensions to the Intel Architecture"); if (features_edx & SSE2_FLAG)
       printf("\nThe processor supports the Streaming SIMD extensions 2 instructions");
     if (features_edx & SS_FLAG)
       printf("\nThe processor supports Self-Snoop");
     if ((features_edx & HTT_FLAG) &&
       (((features_ebx >> 16) \& 0x0FF) > 1))
       printf("\nThe processor supports Hyper-Threading Technology");
     if (features_edx & TM_FLAG)
       printf("\nThe processor supports the Thermal Monitor");
     if (features_edx & PBE_FLAG)
       printf("\nThe processor supports Pending Break Event signaling");
     if (features_ecx & SSE3_FLAG)
     printf("\nThe processor supports the Streaming SIMD extensions 3 instructions");
if (features_ecx & MONITOR_FLAG)
     printf("\nThe processor supports the MONITOR and MWAIT instructions");
if (features_ecx & DS_CPL_FLAG)
       printf("\nThe processor supports Debug Store extensions for branch message storage by CPL");
     if (features_ecx & EIST_FLAG)
       printf("\nThe processor supports Enhanced SpeedStep(TM) Technology");
     if (features_ecx & TM2_FLAG)
       printf("\nThe processor supports the Thermal Monitor 2");
     if (features_ecx & CID_FLAG)
       printf("\nThe processor supports L1 Data Cache Context ID");
     if (features_ecx & XTPR_FLAG)
       printf("\nThe processor supports transmitting TPR messages");
  else {
     printf("t least an 80486 processor. ");
     printf("\nIt does not contain a Genuine Intel part and as a result, the ");
     printf("\nCPUID detection information cannot be determined at this time.");
printf("\n");
return(0);
```

#### **Example 4. Instruction Extension Detection Using Exception Handlers**

// FILENAME: FEATURES.CPP

// Copyright (c) Intel Corporation 2000-2004
//

// This program has been developed by Intel Corporation. Intel has

// various intellectual property rights which it may assert under

// certain circumstances, such as if another manufacturer's

 $/\!/$  processor mis-identifies itself as being "GenuineIntel" when

 $\ensuremath{\textit{//}}\xspace$  the CPUID instruction is executed.

//

// Intel specifically disclaims all warranties, express or implied,

 $\ensuremath{\textit{//}}\xspace$  and all liability, including consequential and other indirect

// damages, for the use of this program, including liability for

// infringement of any proprietary rights, and including the

// warranties of merchantability and fitness for a particular

// purpose. Intel does not assume any responsibility for any

// errors which may appear in this program nor any responsibility

// to update it.

#include "stdio.h"
#include "string.h"
#include "excpt.h"

// The follow code sample demonstrate using exception handlers to identify available IA-32 features,

 $/\!/$  The sample code Identifies IA-32 features such as support for Streaming SIMD Extensions 3,

// Streaming SIMD Extensions 2 (SSE2), support for Streaming SIMD Extensions (SSE),

// support for MMX (TM) instructions.

// This technique can be used safely to determined IA-32 features and provide

// forward compatibility to run optimally on future IA-32 processors.

// Please note that the technique of trapping invalid opcodes is not suitable

// for identifying the processor family and model.

int main(int argc, char\* argv[])

{

char sSupportSSE3[80]="Don't know";

char sSupportSSE2[80]="Don't know";

char sSupportSSE[80]="Don't know";

char sSupportMMX[80]="Don't know";

// To identify whether SSE3, SSE2, SSE, or MMX instructions are supported on an x86 compatible

// processor in a fashion that will be compatible to future IA-32 processors,

// The following tests are performed in sequence: (This sample code will assume cpuid

// instruction is supported by the target processor.)

// 1. Test whether target processor is a Genuine Intel processor, if yes

// 2. Test if executing an SSE3 instruction would cause an exception, if no exception occurs,

// SSE3 is supported; if exception occurs,

 $\prime\prime$  3. Test if executing an SSE2 instruction would cause an exception, if no exception occurs,

// SSE2 is supported; if exception occurs,

// 4. Test if executing an SSE instruction would cause an exception, if no exception occurs,

// SSE is supported; if exception occurs,

// 5. Test if executing an MMX instruction would cause an exception, if no exception occurs,

- // MMX instruction is supported,
- // if exception occurs, MMX instruction is not supported by this processor.

// For clarity, the following stub function "IsGenuineIntelProcessor()" is not shown in this example,



```
// The function "IsGenuineIntelProcessor()" can be adapted from the sample code implementation
// of the assembly procedure "_get_cpu_type". The purpose of this stub function is to examine
// whether the Vendor ID string, which is returned when executing
// cpuid instruction with EAX = 0, indicates the processor is a genuine Intel processor.
if (IsGenuineIntelProcessor())
{
    // First, execute an SSE3 instruction to see whether an exception occurs
      _try
     {
           asm {
              haddpd xmm1, xmm2
                                          // this is an instruction available in SSE3
              // _emit 0x66 __asm _emit 0x0F __asm _emit 0x7C __asm _emit 0xCA
         strcpy(&sSupportSSE3[0], "Yes");
                                                    // No exception executing an SSE3 instruction
     }
      _except( EXCEPTION_EXECUTE_HANDLER ) // SSE3 exception handler
     {
         // exception occurred when executing an SSE3 instruction
         strcpy(&sSupportSSE3[0], "No");
     }
    // Second, execute an SSE2 instruction to see whether an exception occurs
      _try
     {
           _asm {
              paddq xmm1, xmm2
                                          // this is an instruction available in SSE2
         }
         strcpy(&sSupportSSE2[0], "Yes");
                                                    // No exception executing an SSE2 instruction
     }
       except( EXCEPTION_EXECUTE_HANDLER ) // SSE2 exception handler
     {
         // exception occurred when executing an SSE2 instruction
         strcpy(&sSupportSSE2[0], "No");
     }
    // Third, execute an SSE instruction to see whether an exception occurs
     __try
    {
         __asm {
              orps xmm1, xmm2 // this is an instruction available in SSE
              //_asm _emit 0x66 __asm _emit 0x0f __asm _emit 0x57 __asm _emit 0xc0
         }
         strcpy(&sSupportSSE[0], "Yes"); // no exception executing an SSE instruction
     }
       _except( EXCEPTION_EXECUTE_HANDLER )
                                                             // SSE exception handler
     {
         // exception occurred when executing an SSE instruction
         strcpy(&sSupportSSE[0], "No");
     }
```

 $/\!/$  Fourth, execute an MMX instruction to see whether an exception occurs

# inte

}

}

```
_try
     {
            _asm {
                        // this is an instruction available in MMX
              emms
                                                                                    }
                                                     /\!/ no exception executing an MMX instruction
         strcpy(&sSupportMMX[0], "Yes");
     }
      _except( EXCEPTION_EXECUTE_HANDLER )
                                                               // MMX exception handler
     {
         \ensuremath{\textit{//}}\xspace exception occurred when executing an MMX instruction
         strcpy(&sSupportMMX[0], "No");
     }
printf("This Processor supports the following instruction extensions: \n");
printf("SSE3 instruction: \t\t%s \n", &sSupportSSE3[0]);
printf("SSE2 instruction: \t\t%s \n", &sSupportSSE2[0]);
printf("SSE instruction: \t\t%s \n", &sSupportSSE[0]);
printf("MMX instruction: \t\t%s \n", &sSupportMMX[0]);
return 0;
```

#### Example 5. Detecting Denormals-Are-Zero Support

Filename: DAZDTECT.ASM Copyright (c) Intel Corporation 2001-2004 This program has been developed by Intel Corporation. Intel has various intellectual property rights which it may assert under certain circumstances, such as if another manufacturer's processor mis-identifies itself as being "GenuineIntel" when the CPUID instruction is executed. Intel specifically disclaims all warranties, express or implied, and all liability, including consequential and other indirect damages, for the use of this program, including liability for infringement of any proprietary rights, and including the warranties of merchantability and fitness for a particular purpose. Intel does not assume any responsibility for any errors which may appear in this program nor any responsibility to update it. This example assumes the system has booted DOS. This program runs in Real mode. This program was assembled using MASM 6.14.8444. This program performs the following 8 steps to determine if the processor supports the SSE/SSE2 DAZ mode. Step 1. Execute the CPUID instruction with an input value of EAX=0 and ensure the vendor-ID string returned is "GenuineIntel". Step 2. Execute the CPUID instruction with EAX=1. This will load the EDX register with the feature flags. Ensure that the FXSR feature flag (EDX bit 24) is set. Step 3. This indicates the processor supports the FXSAVE and FXRSTOR instructions. Step 4. Ensure that the XMM feature flag (EDX bit 25) or the EMM feature flag (EDX bit 26) is set. This indicates that the processor supports at least one of the SSE/SSE2 instruction sets and its MXCSR control register. Step 5. Zero a 16-byte aligned, 512-byte area of memory. This is necessary since some implementations of FXSAVE do not modify reserved areas within the image. Step 6. Execute an FXSAVE into the cleared area. Step 7. Bytes 28-31 of the FXSAVE image are defined to contain the MXCSR\_MASK. If this value is 0, then the processor's MXCSR\_MASK is 0xFFBF, otherwise MXCSR\_MASK is the value of this dword. Step 8. If bit 6 of the MXCSR\_MASK is set, then DAZ is supported. DOSSEG .MODEL small, c .STACK

; Data segment

#### .DATA

| buffer                                                                            | DB                               | 512+16 DUP (0)                                                                                                                                                                                                                                                                                                            |
|-----------------------------------------------------------------------------------|----------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| not_intel<br>noSSEorSSE2<br>no_FXSAVE<br>daz_mask_clear<br>no_daz<br>supports_daz | DB<br>DB<br>DB<br>DB<br>DB<br>DB | "This is not an Genuine Intel processor.", 0Dh, 0Ah, "\$"<br>"Neither SSE or SSE2 extensions are supported.", 0Dh, 0Ah, "\$"<br>"FXSAVE not supported.", 0Dh, 0Ah, "\$"<br>"DAZ bit in MXCSR_MASK is zero (clear).", 0Dh, 0Ah, "\$"<br>"DAZ mode not supported.", 0Dh, 0Ah, "\$"<br>"DAZ mode supported.", 0Dh, 0Ah, "\$" |
|                                                                                   |                                  |                                                                                                                                                                                                                                                                                                                           |

; Code segment

| .CODE |
|-------|
| .686p |
| .XMM  |

dazdtect PROC NEAR

.startup

; Allow assembler to create code that

; initializes stack and data segment

; registers

#### ; Step 1.

;Verify Genuine Intel processor by checking CPUID generated vendor ID

| mov<br>cpuid | eax, 0            |                                        |
|--------------|-------------------|----------------------------------------|
| cmp          | ebx, 'uneG'       | ; Compare first 4 letters of Vendor ID |
| jne          | notIntelprocessor | ; Jump if not Genuine Intel processor  |
| cmp          | edx, 'Ieni'       | ; Compare next 4 letters of Vendor ID  |
| jne          | notIntelprocessor | ; Jump if not Genuine Intel processor  |
| cmp          | ecx, 'letn'       | ; Compare last 4 letters of Vendor ID  |
| jne          | notIntelprocessor | ; Jump if not Genuine Intel processor  |

; Step 2, 3, and 4

; Get CPU feature flags ; Verify FXSAVE and either SSE or ; SSE2 are supported

| mov                | eax, 1                |                                                                            |  |  |  |  |
|--------------------|-----------------------|----------------------------------------------------------------------------|--|--|--|--|
| cpuid<br>bt<br>jnc | edx, 24t<br>noFxsave  | ; Feature Flags Bit 24 is FXSAVE support<br>; jump if FXSAVE not supported |  |  |  |  |
| bt                 | edx, 25t              | ; Feature Flags Bit 25 is SSE support                                      |  |  |  |  |
| jc                 | sse_or_sse2_supported | ; jump if SSE is not supported                                             |  |  |  |  |
| bt                 | edx, 26t              | ; Feature Flags Bit 26 is SSE2 support                                     |  |  |  |  |
| jnc                | no_sse_sse2           | ; jump if SSE2 is not supported                                            |  |  |  |  |

sse\_or\_sse2\_supported:

|            | VE requires a 16-byte aligned so get offset into buffer |                                     |
|------------|---------------------------------------------------------|-------------------------------------|
| mov<br>and | bx, OFFSET buffer<br>bx, 0FFF0h                         | ; Get offset of the buffer into bx  |
| add        | bx, 16t                                                 | ; DI is aligned at 16-byte boundary |

; Step 5.

; Clear the buffer that will be ; used for FXSAVE data

|           | push<br>pop<br>mov<br>xor<br>mov<br>cld<br>rep | ds<br>es<br>di, bx<br>ax, ax<br>cx, 512/2<br>stosw     | ; Fill at FXSAVE buffer with zeroes                          |
|-----------|------------------------------------------------|--------------------------------------------------------|--------------------------------------------------------------|
| ; Step 6. |                                                |                                                        |                                                              |
|           | fxsave                                         | [bx]                                                   |                                                              |
| ; Step 7. |                                                |                                                        |                                                              |
|           | mov<br>cmp<br>jne                              | eax, DWORD PTR [bx][28t]<br>eax, 0<br>check_mxcsr_mask | ; Get MXCSR_MASK<br>; Check for valid mask                   |
|           | mov                                            | eax, 0FFBFh                                            | ; Force use of default MXCSR_MASK                            |
|           | xcsr_masl<br>ontains M2                        | c:<br>KCSR_MASK from FXSAVE                            | buffer or default mask                                       |
| ; Step 8. |                                                |                                                        |                                                              |
|           | bt<br>jc                                       | eax, 6t<br>supported                                   | ; MXCSR_MASK Bit 6 is DAZ support<br>; Jump if DAZ supported |
|           | mov<br>jmp                                     | dx, OFFSET daz_mask_clear<br>notSupported              |                                                              |
| supported | d:<br>mov<br>jmp                               | dx, OFFSET supports_daz<br>print                       | ; Indicate DAZ is supported.                                 |
| notIntelP | rocessor:<br>mov<br>jmp                        | dx, OFFSET not_intel print                             | ; Assume not an Intel processor                              |
| no_sse_s  | se2:<br>mov<br>jmp                             | dx, OFFSET noSSEorSSE2<br>notSupported                 | ; Setup error message assuming no SSE/SSE2                   |
| noFxsave  | e:<br>mov                                      | dx, OFFSET no_FXSAVE                                   |                                                              |
| notSuppo  | orted:<br>mov<br>int                           | ah, 09h<br>21h                                         | ; Execute DOS print string function                          |
|           | mov                                            | dx, OFFSET no_daz                                      |                                                              |
| print:    | mov<br>int                                     | ah, 09h<br>21h                                         | ; Execute DOS print string function                          |
| exit:     | .exit                                          |                                                        | ; Allow assembler to generate code                           |
|           | ret                                            |                                                        | ; that returns control to DOS                                |
| dazdtect  | ENDP                                           |                                                        |                                                              |
|           | END                                            |                                                        |                                                              |
|           |                                                |                                                        |                                                              |

#### **Example 6. Frequency Calculation**

| ;                          | Filename: FREQUENC.ASM<br>Copyright (c) Intel Corporation 2001-2004                                                                                                                                                                                                                                                                                                                                                                                                           |
|----------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|                            | This program has been developed by Intel Corporation. Intel<br>has various intellectual property rights which it may assert<br>under certain circumstances, such as if another<br>manufacturer's processor mis-identifies itself as being<br>"GenuineIntel" when the CPUID instruction is executed.                                                                                                                                                                           |
| ,<br>,<br>,<br>,<br>,      | Intel specifically disclaims all warranties, express or<br>implied, and all liability, including consequential and other<br>indirect damages, for the use of this program, including<br>liability for infringement of any proprietary rights,<br>and including the warranties of merchantability and fitness<br>for a particular purpose. Intel does not assume any<br>responsibility for any errors which may appear in this program<br>nor any responsibility to update it. |
|                            | This example assumes the system has booted DOS.<br>This program runs in Real mode.                                                                                                                                                                                                                                                                                                                                                                                            |
| ,*******<br>,              | ***********************                                                                                                                                                                                                                                                                                                                                                                                                                                                       |
| ,<br>,<br>,<br>,<br>,<br>, | This program was assembled using MASM 6.14.8444 and tested on a system with a Pentium(r) II processor, a system with a Pentium(r) 4 processor, B2 stepping, and a system with a Pentium(r) 4 processor, C1 stepping.                                                                                                                                                                                                                                                          |
| ;                          | This program performs the following 8 steps to determine the actual processor frequency.                                                                                                                                                                                                                                                                                                                                                                                      |
| ;                          | Execute the CPUID instruction with an input value of EAX=0<br>and ensure the vendor-ID string returned is "GenuineIntel".<br>Execute the CPUID instruction with EAX=1 to load the EDX<br>register with the feature flags.                                                                                                                                                                                                                                                     |
| ;                          | Ensure that the TSC feature flag (EDX bit 4) is set. This indicates the processor supports the Time-Stamp Counter and RDTSC instruction.                                                                                                                                                                                                                                                                                                                                      |
|                            | Read the TSC at the beginning of the reference period                                                                                                                                                                                                                                                                                                                                                                                                                         |
|                            | Read the TSC at the end of the reference period.<br>Compute the TSC delta from the beginning and ending of the                                                                                                                                                                                                                                                                                                                                                                |
| , step 6.                  | reference period.                                                                                                                                                                                                                                                                                                                                                                                                                                                             |
| Step 7.                    | Compute the actual frequency by dividing the TSC delta by<br>the reference period.                                                                                                                                                                                                                                                                                                                                                                                            |
| ,<br>,*******<br>,         | **********************                                                                                                                                                                                                                                                                                                                                                                                                                                                        |
|                            | .DOSSEG<br>.MODEL small, pascal<br>.STACK ;4096                                                                                                                                                                                                                                                                                                                                                                                                                               |

wordToDec PROTO NEAR PASCAL decAddr:WORD, hexData:WORD

| ; Macro<br>;<br>;<br>;<br>; | printst<br>This macro is used to print a string passed as an input<br>parameter and a word value immediately after the string.<br>The string is delared in the data segment routine during<br>assembly time. The word is converted to dec ascii and<br>printed after the string. |                            |                                                       |                                                              |
|-----------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------|-------------------------------------------------------|--------------------------------------------------------------|
| ;<br>; Input:<br>;          |                                                                                                                                                                                                                                                                                  |                            | g to be printed.<br>to be converted to dec            | ascii and printed                                            |
| ;<br>; Destroy              | ys: None                                                                                                                                                                                                                                                                         |                            |                                                       |                                                              |
| ;<br>; Output               | : None                                                                                                                                                                                                                                                                           |                            |                                                       |                                                              |
| ;<br>; Assum(<br>;          | es: Stack                                                                                                                                                                                                                                                                        | is availab                 | le                                                    |                                                              |
| ,                           |                                                                                                                                                                                                                                                                                  | stringda                   | ta, hexWord<br>bel, decData                           |                                                              |
|                             | .data                                                                                                                                                                                                                                                                            |                            |                                                       |                                                              |
| stringlab<br>decData        |                                                                                                                                                                                                                                                                                  | DB<br>DB<br>DB             | stringdata<br>5 dup (0)<br>0dh, 0ah, '\$'             |                                                              |
|                             | .code                                                                                                                                                                                                                                                                            |                            |                                                       |                                                              |
|                             | pushf<br>pusha                                                                                                                                                                                                                                                                   |                            |                                                       |                                                              |
|                             |                                                                                                                                                                                                                                                                                  |                            | d ino hex ascii and store<br>ec, offset decData, hex` |                                                              |
|                             | mov<br>mov<br>int                                                                                                                                                                                                                                                                | dx, offs<br>ah, 09h<br>21h | et stringlabel                                        | ; Setup string to be printed<br>; Execute DOS print function |
|                             | popa<br>popf                                                                                                                                                                                                                                                                     |                            |                                                       |                                                              |
| ENDM                        |                                                                                                                                                                                                                                                                                  |                            |                                                       |                                                              |
| OFFSET                      | OS_DAT<br>[_TICK_(<br>/AL_IN_′                                                                                                                                                                                                                                                   |                            | EQU 40h<br>EQU 6ch<br>EQU 91                          |                                                              |
| ; Data se                   | egment                                                                                                                                                                                                                                                                           |                            |                                                       |                                                              |
|                             | .DATA                                                                                                                                                                                                                                                                            |                            |                                                       |                                                              |
| ; Code s                    | egment                                                                                                                                                                                                                                                                           |                            |                                                       |                                                              |
|                             | .CODE                                                                                                                                                                                                                                                                            |                            |                                                       |                                                              |
|                             |                                                                                                                                                                                                                                                                                  |                            |                                                       |                                                              |

.686p

cpufreq PROC NEAR local tscLoDword:DWORD, \ tscHiDword:DWORD, \ mhz:WORD, \ Nearest66Mhz:WORD, \ Nearest50Mhz:WORD, \ delta66Mhz:WORD

.startup

; Allow assembler to create code that ; initializes stack and data segment ; registers

#### ; Step 1.

;Verify Genuine Intel processor by checking CPUID generated vendor ID

|          | mov<br>cpuid                           | eax, 0                                                            |                                                                          |
|----------|----------------------------------------|-------------------------------------------------------------------|--------------------------------------------------------------------------|
|          | cmp<br>jne<br>cmp<br>jne<br>cmp<br>jne | ebx, 'uneG'<br>exit<br>edx, 'Ieni'<br>exit<br>ecx, 'letn'<br>exit | ; Check VendorID = GenuineIntel<br>; Jump if not Genuine Intel processor |
| ; Step 2 | and 3                                  |                                                                   |                                                                          |
|          |                                        | PU feature flags<br>TSC is supported                              |                                                                          |
|          | mov<br>cpuid                           | eax, 1                                                            |                                                                          |
|          | bt<br>jnc                              | edx, 4t<br>exit                                                   | ; Flags Bit 4 is TSC support<br>; jump if TSC not supported              |
|          | push<br>pop                            | SEG_BIOS_DATA_AREA<br>es                                          |                                                                          |
|          | mov<br>mov                             | si, OFFSET_TICK_COUNT<br>ebx, DWORD PTR es:[si]                   | ; The BIOS tick count updateds<br>; ~ 18.2 times per second.             |
| wait_for | _new_ticl                              | k:                                                                |                                                                          |
|          | cmp<br>je                              | ebx, DWORD PTR es:[si]<br>wait_for_new_tick                       | ; Wait for tick count change                                             |
| ; Step 4 | ;**Tim                                 | ed interval starts**                                              |                                                                          |
|          | ; Read CPU time-stamp                  |                                                                   |                                                                          |
|          | rdtsc<br>mov<br>mov                    | tscLoDword, eax<br>tscHiDword, edx                                | ; Read and save TSC immediately<br>; after a tick                        |
|          | add                                    | ebx, INTERVAL_IN_TICKS + 1                                        | ; Set time delay value ticks.                                            |

| wait_for | _elapsed_<br>cmp<br>jne | _ticks:<br>ebx, DWORD PTR es:[si]<br>wait_for_elapsed_ticks                   | ; Have we hit the delay?                                     |
|----------|-------------------------|-------------------------------------------------------------------------------|--------------------------------------------------------------|
| ; Step 5 | · **Tim                 | e interval ends**                                                             |                                                              |
|          | , · · i iii             | le interval ends.                                                             |                                                              |
|          | ; Read (<br>rdtsc       | CPU time-stamp immediatly after tick de                                       | elay reached.                                                |
| ; Step 6 |                         |                                                                               |                                                              |
|          | sub<br>sbb              | eax, tscLoDword<br>edx, tscHiDword                                            | ; Calculate TSC delta from<br>; beginning to end of interval |
| ; Step 7 |                         |                                                                               |                                                              |
|          |                         | = (1 / 18.2) * 1,000,000 This adjusts fo<br>*INTERVAL_IN_TICKS adjusts for nu |                                                              |
|          | mov<br>div              | ebx, 54945*INTERVAL_IN_TICKS<br>ebx                                           |                                                              |
|          | · ax con                | tains measured speed in MHz                                                   |                                                              |
|          | mov                     | mhz, ax                                                                       |                                                              |
|          | ; Find n                | earest full/half multiple of 66/133 MHz                                       |                                                              |
|          | xor                     | dx, dx                                                                        |                                                              |
|          | mov                     | ax, mhz                                                                       |                                                              |
|          | -                       | bx, 3t                                                                        |                                                              |
|          | mul<br>add              | bx<br>ax, 100t                                                                |                                                              |
|          | mov                     | bx, 200t                                                                      |                                                              |
|          | div                     | bx                                                                            |                                                              |
|          | mul                     | bx                                                                            |                                                              |
|          | xor                     | dx, dx                                                                        |                                                              |
|          | mov                     | bx, 3                                                                         |                                                              |
|          | div                     | bx                                                                            |                                                              |
|          | ; ax con                | tains nearest full/half multiple of 66/100                                    | ) MHz                                                        |
|          | mov                     | Nearest66Mhz, ax                                                              |                                                              |
|          | sub                     | ax, mhz                                                                       |                                                              |
|          | jge                     | delta66                                                                       |                                                              |
|          | neg                     | ax                                                                            | ; $ax = abs(ax)$                                             |
| delta66: |                         |                                                                               |                                                              |
|          | ; ax con                | tains delta between actual and nearest 66                                     | 5/133 multiple                                               |
|          | mov                     | Delta66Mhz, ax                                                                |                                                              |
|          | ; Find n                | earest full/half multiple of 100 MHz                                          |                                                              |
|          | xor                     | dx, dx                                                                        |                                                              |
|          | mov                     | ax, mhz                                                                       |                                                              |
|          | add                     | ax, 25t                                                                       |                                                              |
|          | mov<br>div              | bx, 50t                                                                       |                                                              |
|          | div<br>mul              | bx<br>bx                                                                      |                                                              |
|          | mui                     |                                                                               |                                                              |

; ax contains nearest full/half multiple of 100 MHz

|           | , un com        | and hearest fail, had maniple of 100 hi         |                                                     |
|-----------|-----------------|-------------------------------------------------|-----------------------------------------------------|
|           | mov             | Nearest50Mhz, ax                                |                                                     |
|           | sub             | ax, mhz                                         |                                                     |
|           | jge             | delta50                                         |                                                     |
|           |                 |                                                 | $\cdot a \mathbf{x} = a \mathbf{b} c(a \mathbf{x})$ |
|           | neg             | ax                                              | ; $ax = abs(ax)$                                    |
| delta50:  |                 |                                                 |                                                     |
| denuso.   | ; ax cont       | tains delta between actual and nearest 50       | 0/100 MHz multiple                                  |
|           | mov             | bx, Nearest50Mhz                                |                                                     |
|           | cmp             | ax, Delta66Mhz                                  |                                                     |
|           | jb              | useNearest50Mhz                                 |                                                     |
|           | mov             | bx, Nearest66Mhz                                |                                                     |
|           | ; Correc<br>cmp | tion for 666 MHz (should be reported as bx, 666 | 667 MHZ)                                            |
|           | jne             | correct666                                      |                                                     |
|           | inc             | bx                                              |                                                     |
| correct6  | 66:             |                                                 |                                                     |
|           |                 |                                                 |                                                     |
| useNear   | est50MHz        |                                                 |                                                     |
|           | ; bx con        | tains nearest full/half multiple of 66/100      | /133 MHz                                            |
|           | • • • •         |                                                 |                                                     |
|           |                 | Reported MHz = $\sim$ ", bx                     |                                                     |
|           | printst         | Measured $MHz = ", mhz$                         | ; print decimal value                               |
| exit:     |                 |                                                 |                                                     |
| CAIL.     |                 |                                                 |                                                     |
|           | .exit           |                                                 | ; returns control to DOS                            |
|           |                 |                                                 |                                                     |
|           | ret             |                                                 |                                                     |
| c         |                 |                                                 |                                                     |
| cpufreq   | ENDP            |                                                 |                                                     |
|           |                 |                                                 |                                                     |
| ; Proced  |                 | wordToDec                                       | -                                                   |
| , 110ccu  |                 | tine will convert a word value into a 5 b       | vte decimal                                         |
| ,         | ascii stri      |                                                 | yte deelmai                                         |
| ;         | usen sui        |                                                 |                                                     |
| ; Input:  | decAdd          | r = address to 5 byte location for conve        | rted string                                         |
| ;         |                 | (near address assumes DS as segn                |                                                     |
| ;         | hexData         | = word value to be converted to hex as          |                                                     |
| ;         |                 |                                                 |                                                     |
| ; Destroy | ys: ax, bx,     | cx                                              |                                                     |
| ;         |                 |                                                 |                                                     |
| ; Output  | :               | 5 byte converted hex string                     |                                                     |
| ;         |                 |                                                 |                                                     |
| ; Assum   | es:             | Stack is available                              |                                                     |
| ;         |                 |                                                 |                                                     |
| ;         |                 |                                                 | -                                                   |
| wordTo    |                 | C NEAR PUBLIC uses es,                          |                                                     |
| woru101   |                 | decAddr:WORD, hexData:WORD                      |                                                     |
|           |                 |                                                 |                                                     |

pusha mov di, decAddr



|         | push<br>pop                              | @data<br>es                                          | ; ES:DI -> 5-byte converted string |
|---------|------------------------------------------|------------------------------------------------------|------------------------------------|
|         | mov<br>xor<br>mov<br>div<br>add<br>stosb | ax, hexData<br>dx, dx<br>bx, 10000t<br>bx<br>ax, 30h |                                    |
|         | mov<br>xor<br>mov<br>div<br>add<br>stosb | ax, dx<br>dx, dx<br>bx, 1000t<br>bx<br>ax, 30h       |                                    |
|         | mov<br>xor<br>mov<br>div<br>add<br>stosb | ax, dx<br>dx, dx<br>bx, 100t<br>bx<br>ax, 30h        |                                    |
|         | mov<br>xor<br>mov<br>div<br>add<br>stosb | ax, dx<br>dx, dx<br>bx, 10t<br>bx<br>ax, 30h         |                                    |
|         | mov<br>add<br>stosb                      | ax, dx<br>ax, 30h                                    |                                    |
| wordToI | popa<br>ret<br>Dec                       | ENDP                                                 |                                    |
|         | END                                      |                                                      |                                    |

§