## intel

# AP-485 APPLICATION NOTE

### Intel® Processor Identification and the CPUID Instruction

November 2003

Order Number: 241618-024

INFORMATION IN THIS DOCUMENT IS PROVIDED IN CONNECTION WITH INTEL 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.

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.

Copies of documents which have an ordering number and are referenced in this document, or other Intel literature, may be obtained from:

Intel Corporation P.O. Box 7641 Mt. Prospect IL 60056-7641

or call 1-800-879-4683 or visit Intel's website at <a href="http://www.intel.com/">http://www.intel.com/</a>

Copyright © Intel Corporation 1993-2003.

Intel, Pentium, MMX, Intel Xeon, Pentium II Xeon, Pentium III Xeon, Intel NetBurst, and Celeron 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.



#### **CONTENTS**

|    |                                                              | PAGE |
|----|--------------------------------------------------------------|------|
| 1  | INTRODUCTION                                                 |      |
|    | 1.1 Update Support                                           | 7    |
| 2  | DETECTING THE CPUID INSTRUCTION                              | 7    |
| 3  | OUTPUT OF THE CPUID INSTRUCTION                              | 8    |
|    | 3.1 Vendor ID String                                         | 8    |
|    | 3.2 Processor Signature                                      | 10   |
|    | 3.3 Feature Flags                                            |      |
|    | 3.4 SYSENTER/SYSEXIT – SEP Features Bit                      |      |
|    | 3.5 Cache Size, Format and TLB Information                   |      |
|    | 3.6 Pentium <sup>®</sup> 4 Processor, Model 0 Output Example | 22   |
| 4  | PROCESSOR SERIAL NUMBER                                      | 22   |
|    | 4.1 Presence of Processor Serial Number                      | 22   |
|    | 4.2 Forming the 96-bit Processor Serial Number               | 23   |
| 5  | BRAND ID                                                     | 24   |
| 6  | BRAND STRING                                                 | 24   |
| 7  | USAGE GUIDELINES                                             | 25   |
| 8  | PROPER IDENTIFICATION SEQUENCE                               | 26   |
| 9  | USAGE PROGRAM EXAMPLES                                       | 27   |
| 10 | ALTERNATE METHOD OF DETECTING FEATURES                       | 28   |
| 11 | DENORMALS ARE ZERO                                           | 29   |
| 12 | OPERATING FREQUENCY                                          | 29   |



#### **REVISION HISTORY**

| Revision | Revision History                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      | Date  |
|----------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|
| -001     | Original Issue.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       | 05/93 |
| -002     | Modified Table 2, Intel486™ and Pentium® Processor Signatures.                                                                                                                                                                                                                                                                                                                                                                                                                                                                        | 10/93 |
| -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 Tables1, 3 and 5. Inserted Tables 6, 7 and 8. 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™ technology. Modified Figure 6.0.                                                                                                                                                                                                                                       | 11/96 |
| -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. | 03/97 |
| - 007    | Modified Table 2. Added Pentium II processor, model 5 entry. Modified existing Pentium II processor entry to read "Pentium II processor, model 3". Modified Table 5. Added additional feature bits, PAT and FXSR. Modified 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"                                                               | 01/98 |
|          | 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 Celeron® processor, model 5 in section 3.2. Modified Table 2. Added Intel Celeron processor & Pentium® OverDrive® processor with MMX™ technology entry. Modified Table 5. Added additional feature bit, PSE-36. Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Intel Celeron processor.                                                                                                                                                                                  | 04/98 |
| -009     | Added note to identify Pentium II Xeon™ 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.                                                                                                                                                                                                                                                                                                  | 06/98 |
| -010     | No Changes                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                            |       |
| -011     | Modified Table 2. Added Intel Celeron processor, model 6 entry.  Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Intel Celeron processor, model 6.                                                                                                                                                                                                                                                                                                                                                     | 12/98 |



#### **REVISION HISTORY**

| Revision | Revision History                                                                                                                                                                                                                                                                                                                                                                                                                                            | Date  |
|----------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|
| -012     | Modified Figure 1 to add the reserved information for the Intel386 processors. Modified Figure 2. Added the Processor serial number information returned when the CPUID instruction is executed with EAX=3. Modified Table 1. Added the Processor serial number parameter. Modified Table 2. Added the Pentium III processor and Pentium III Xeon processor. Added Section 4 "Processor serial number".                                                     | 12/98 |
|          | 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.  Modified cpuid3a.asm, cpuid3b.asm and cpuid3.c example code to check for and identify the Pentium III processor, model 8 and the Pentium III Xeon processor, model 8.                                                                                     | 10/99 |
| -014     | Modified Table 4. Added Intel Celeron processor, model 8                                                                                                                                                                                                                                                                                                                                                                                                    | 03/00 |
| -015     | Modified Table 4. Added Pentium III Xeon processor, model A. Modified Table 7, Added the 8-way set associative 1M, and 8-way set associative 2M cache descriptor entries.                                                                                                                                                                                                                                                                                   | 05/00 |
| -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 Example 4.  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 7.  Removed Pentium Pro cache descriptor example.            | 11/00 |
| -017     | Modified Figure 2 to include additional features reported by the Pentium 4 processors.  Modified Table 7 to include additional Cache and TLB descriptors defined by the Intel® NetBurst™ microarchitecture.  Added Section 11 and program Example 5 which describes how to detect if a processor supports the DAZ feature.  Added Section 12 and program Example 6 which describes a method of calculating the actual operating frequency of the processor. | 02/01 |
| -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.                                                      | 06/01 |



#### **REVISION HISTORY**

| Revision | Revision History                                                                                                                                       | Date   |
|----------|--------------------------------------------------------------------------------------------------------------------------------------------------------|--------|
| -019     | Changed to use registered trademark for Intel® Celeron® throughout entire document.                                                                    | 1/2    |
|          | Modified Table 10 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 Intel® processors based on the updated Brand ID values contained in Table 10. |        |
| -020     | Modified Table 8 to include new Cache Descriptor values supported by the Intel® processors with Intel® NetBurst™ microarchitecture.                    | 03/02  |
|          | Modified Table 10 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 Intel® processors based on the updated Brand ID values contained in Table 10. |        |
| -021     | Modified Table 4 to include additional processors that return a processor signature with a value in the family code equal to 0Fh.                      | 05/02  |
|          | Modified Table 8 to include new Cache Descriptor values supported by various Intel processors.                                                         |        |
|          | Modified Table 10 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 Intel processors based on the updated Brand ID values contained in Table 10.  |        |
| -022     | Modified Table 8 with correct Cache Descriptor descriptions.                                                                                           | 11/02  |
|          | 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.                                                                                                    |        |
| -024     | Modified Table 8 with correct Cache Descriptor descriptions.  Corrected feature flag definitions in Table 6 for bits 7 and 8.                          | 11/03  |
| -024     | Corrected realtire may deminitions in Table 0 for bits 7 and 0.                                                                                        | 1 1/03 |



#### 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® processors, Pentium® OverDrive® processors, Pentium® processors with MMX<sup>TM</sup> technology, Pentium® Pro processors, Pentium® II processors, Pentium® II Xeon<sup>TM</sup> processors, Pentium® II Overdrive® processors, Intel® Celeron® processors, Mobile Intel® Celeron® processors, Pentium® III processors, Mobile Intel® Pentium® III Processor - M, Pentium® III Xeon<sup>TM</sup> processors, Pentium® 4 processors, Mobile Intel® Pentium® 4 processor - M, Intel® Pentium® M Processor, Intel® Xeon<sup>TM</sup> processors and Intel® 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:

- Originally, Intel published code sequences that could detect minor implementation or architectural differences to identify processor generations.
- 2. 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>.

#### 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 (see Figure 1).



Figure 1. Flag Register Evolution

7

<sup>&</sup>lt;sup>1</sup> Only in some Intel486<sup>™</sup> and succeeding processors. Bit 21 in the Intel386<sup>™</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.



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.

#### 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, 80000000H
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 | able 1. Information Neturned by the Of Old Instruction                                                                                                                                       |  |  |  |  |  |
|-------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|--|--|--|
| 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 32-bit Processor Signature (Extended Family, Extended Model, Type, Family, Model and Stepping ID also bits 95-64 of the 96-bit processor serial number when the PSN feature flag is set. |  |  |  |  |  |
|             | 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                                                                                                                                         |  |  |  |  |  |
|             | Bits 31-24: Processor local APIC physical ID  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                                                                                                                                                 |  |  |  |  |  |
| 3H          | EAX Reserved                                                                                                                                                                                 |  |  |  |  |  |
|             | EBX Reserved                                                                                                                                                                                 |  |  |  |  |  |
|             | ECX Bits 31-0 of 96-bit processor serial number. (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 the 96-bit processor serial number. (Available only in Pentiur III processors when the PSN feature flag is set; otherwise, the value in this register is reserved.)            |  |  |  |  |  |
|             | Extended Function CPUID Information                                                                                                                                                          |  |  |  |  |  |
| 80000000H   | EAX Maximum Input Value for Extended Function CPUID Information EBX, ECX, EDX Reserved                                                                                                       |  |  |  |  |  |
| 80000001H   | EAX Extended Processor Signature and Extended Feature Bits (Currently Reserved.)                                                                                                             |  |  |  |  |  |
|             | EBX, ECX, EDX Reserved                                                                                                                                                                       |  |  |  |  |  |
| 80000002H   | EAX Processor Brand String                                                                                                                                                                   |  |  |  |  |  |
|             | EBX, ECX, EDX Processor Brand String Continued                                                                                                                                               |  |  |  |  |  |
| 80000003H   | EAX, EBX, ECX, EDX Processor Brand String Continued                                                                                                                                          |  |  |  |  |  |
| 80000004H   | EAX, EBX, ECX, EDX Processor Brand String Continued                                                                                                                                          |  |  |  |  |  |

#### 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 8 different bit fields. The fields in gray represent reserved bits, and should be masked out when utilizing the processor signature. The remaining 6 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>TM</sup> micro-architecture 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.

The Pentium II processor, model 5, the Pentium II Xeon processor, model 5, and the Intel 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

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

| Value | Description                  |
|-------|------------------------------|
| 00    | Original OEM processor       |
| 01    | OverDrive® processor         |
| 10    | Dual processor               |
| 11    | Intel reserved (Do not use.) |

cache descriptor values through executing CPUID instruction with EAX = 2. If no L2 cache is returned, the processor is identified as an Intel 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 512K 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 512K L2 cache.

The processor brand for the Pentium III processor, model 8, the Pentium III Xeon processor, model 8, and the Intel 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 9 shows the processor brands defined by the Brand ID.

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

<sup>&</sup>lt;sup>2</sup> All Intel486 SL-enhanced and Write-Back enhanced processors are capable of executing the CPUID instruction. See Table 4.



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™ Processor Signatures

| Type | Family | Major Stepping | Minor Stepping | Description            |
|------|--------|----------------|----------------|------------------------|
| 0000 | 0011   | 0000           | XXXX           | Intel386™ DX processor |
| 0010 | 0011   | 0000           | XXXX           | Intel386 SX processor  |
| 0010 | 0011   | 0000           | XXXX           | Intel386 CX processor  |
| 0010 | 0011   | 0000           | xxxx           | Intel386 EX processor  |
| 0100 | 0011   | 0000 and 0001  | XXXX           | Intel386 SL processor  |
| 0000 | 0011   | 0100           | xxxx           | RapidCAD® coprocessor  |



Table 4. Intel486™, and Subsequent Processor Signatures

| Table 4. Intel486™, and Subsequent Processor Signatures |                |        |                |                 |             |                                                                                                                 |
|---------------------------------------------------------|----------------|--------|----------------|-----------------|-------------|-----------------------------------------------------------------------------------------------------------------|
| Extended Family                                         | Extended Model | Type   | Family<br>Code | Model<br>Number | Stepping ID | Description                                                                                                     |
| 00000000                                                | 0000           | 00     | 0100           | 000x            | XXXX (1)    | Intel486™ DX processors                                                                                         |
| 00000000                                                | 0000           | 00     | 0100           | 0010            | XXXX (1)    | Intel486 SX processors                                                                                          |
| 00000000                                                | 0000           | 00     | 0100           | 0011            | XXXX (1)    | Intel487™ processors                                                                                            |
| 00000000                                                | 0000           | 00     | 0100           | 0011            | XXXX (1)    | IntelDX2™ processors                                                                                            |
| 00000000                                                | 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™ processors                                                                                            |
| 00000000                                                | 0000           | 00     | 0100           | 0111            | XXXX (3)    | Write-Back Enhanced IntelDX2 processors                                                                         |
| 00000000                                                | 0000           | 00     | 0100           | 1000            | XXXX (3)    | IntelDX4™ processors                                                                                            |
| 00000000                                                | 0000           | 0x     | 0100           | 1000            | XXXX (3)    | IntelDX4 OverDrive processors                                                                                   |
| 00000000                                                | 0000           | 00     | 0101           | 0001            | XXXX (2)    | Pentium® 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 (166, 200)                                                   |
| 00000000                                                | 0000           | 01     | 0101           | 0100            | XXXX (2)    | Pentium OverDrive processor with MMX <sup>™</sup> technology for Pentium processor (75, 90, 100, 120, 133)      |
| 00000000                                                | 0000           | 00     | 0110           | 0001            | XXXX (2)    | Pentium Pro processor                                                                                           |
| 00000000                                                | 0000           | 00     | 0110           | 0011            | XXXX (2)    | Pentium II processor, model 3                                                                                   |
| 00000000                                                | 0000           | 00     | 0110           | 0101(5)         | XXXX (2)    | Pentium II processor, model 5, Pentium II<br>Xeon processor, model 5, and Intel<br>Celeron processor, model 5   |
| 00000000                                                | 0000           | 00     | 0110           | 0110            | XXXX (2)    | Intel Celeron processor, model 6                                                                                |
| 00000000                                                | 0000           | 00     | 0110           | 0111(6)         | XXXX (2)    | Pentium III processor, model 7, and Pentium III Xeon processor, model 7                                         |
| 00000000                                                | 0000           | 00     | 0110           | 1000(7)         | XXXX (2)    | Pentium III processor, model 8, Pentium III<br>Xeon processor, model 8, and Intel<br>Celeron processor, model 8 |
| 00000000                                                | 0000           | 00     | 0110           | 1001            | XXXX (2)    | Intel Pentium M processor                                                                                       |
| 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                                                                                  |
| 00000000                                                | 0000           | 01     | 0110           | 0011            | XXXX (2)    | Intel Pentium II OverDrive processor                                                                            |
| 00000000                                                | 0000           | 00     | 1111           | 0000            | XXXX (2)    | Pentium 4 processor, Intel Xeon processor. All processors are manufactured using the 0.18 micron process.       |



Table 4. Intel486™, and Subsequent Processor Signatures

| Extended Family | Extended Model | Type | Family<br>Code | Model<br>Number | Stepping ID | Description                                                                                                                                                                                                                           |
|-----------------|----------------|------|----------------|-----------------|-------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| 00000000        | 0000           | 00   | 1111           | 0001            | XXXX (2)    | Pentium 4 processor, Intel Xeon processor, Intel Xeon processor MP, and Intel Celeron processor. All processors are manufactured using the 0.18 micron process.                                                                       |
| 00000000        | 0000           | 00   | 1111           | 0010            | XXXX (2)    | Pentium 4 processor, Mobile Intel Pentium 4 processor – M, Intel Xeon processor, Intel Xeon processor MP, Intel Celeron processor, and Mobile Intel Celeron processor. All processors are manufactured using the 0.13 micron process. |



#### NOTES:

- 1. This processor does not implement the CPUID instruction.
- 2. Refer to the Intel486™ documentation, the Pentium® Processor Specification Update (Order Number 242480), the Pentium® Pro Processor Specification Update (Order Number 242689), the Pentium® II Processor Specification Update (Order Number 243337), the Pentium® II Xeon Processor Specification Update (Order Number 243748), the Pentium® III Processor Specification Update (Order Number 243748), the Pentium® III Processor Specification Update (Order Number 244453), the Pentium® III Xeon™ Processor Specification Update (Order Number 244460), the Pentium® 4 Processor Specification Update (Order Number 249199), the Intel® Xeon™ Processor Specification Update (Order Number 249678) or the Intel® Xeon™ Processor MP Specification Update (Order Number 290741) 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 Intel 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 Intel 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 512K 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 512K L2 cache size.
- 7. 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 lists 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.

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.

Table 5. Feature Flag Values Reported in the EDX Register

| Table 5. Feature Flag Values Reported III the EDA Register |      |                                           |                                                                                                                                                                                                                               |  |  |  |  |  |
|------------------------------------------------------------|------|-------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|--|--|--|
| Bit                                                        | Name | Description when Flag = 1                 | Comments                                                                                                                                                                                                                      |  |  |  |  |  |
| 0                                                          | FPU  | Floating-point unit on-<br>Chip           | The processor contains an FPU that supports the Intel387 floating-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<br>Extension             | Physical addresses greater than 32 bits are supported.                                                                                                                                                                        |  |  |  |  |  |
| 7                                                          | MCE  | Machine Check<br>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 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.                                                                                                                                    |  |  |  |  |  |
| 13                                                         | PGE  | Page Global Enable                        | The global bit in the page directory entries (PDEs) and page table entries (PTEs) is supported, indicating TLB entries that are common to different processes and need not be flushed. The CR4.PGE bit controls this feature. |  |  |  |  |  |
| 14                                                         | MCA  | Machine Check<br>Architecture             | The Machine Check Architecture is supported, specifically the MCG_CAP register.                                                                                                                                               |  |  |  |  |  |
| 15                                                         | CMOV | Conditional Move<br>Instruction Supported | The processor supports CMOVcc, and if the FPU feature flag (bit 0) is also set, supports the FCMOVCC and FCOMI instructions.                                                                                                  |  |  |  |  |  |



Table 5. Feature Flag Values Reported in the EDX Register

|     |        | Description when                                                            | liues Reported in the EDX Register                                                                                                                                                                                                                                                                                                                                                                                                                                         |  |  |
|-----|--------|-----------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|
| Bit | Name   | Flag = 1                                                                    | Comments                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |  |  |
| 16  | PAT    | Page Attribute Table                                                        | Indicates whether the processor supports the Page Attribute Table. This feature augments the Memory Type Range Registers (MTRRs), allowing an operating system to specify attributes of 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.                                                                                                                                                                                                                                                                                                                                                                             |  |  |
| 19  | CLFSH  | CLFLUSH Instruction 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<br>Clock Facilities<br>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<br>Extensions supported                                      | The processor supports the Streaming SIMD Extensions to the Intel Architecture.                                                                                                                                                                                                                                                                                                                                                                                            |  |  |
| 26  | SSE2   | Streaming SIMD<br>Extensions 2                                              | Indicates the processor supports the Streaming SIMD Extensions - 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.                                                                                                                                                                                                                                                                                                                     |  |  |
| 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 been enabled for this specific processor. To determine if Hyper-Threading Technology is supported, check the value returned in EBX[23:16] after executing CPUID with EAX=1. If EBX[23:16] contains a value >1, then the processor supports Hyper-Threading Technology. |  |  |
| 29  | TM     | Thermal Monitor supported                                                   | The processor implements the Thermal Monitor automatic thermal control circuit (TCC).                                                                                                                                                                                                                                                                                                                                                                                      |  |  |
| 30  |        | Reserved                                                                    | Do not count on their value.                                                                                                                                                                                                                                                                                                                                                                                                                                               |  |  |
| 31  | SBF    | Signal Break on FERR                                                        | The processor supports the Signal Break on FERR feature. The FERR signal is asserted if an interrupt is pending and STPCLK is asserted.                                                                                                                                                                                                                                                                                                                                    |  |  |



Table 6. Feature Flag Values Reported in the ECX Register

| Bit   | Name | Description when Flag = 1               | Comments                                                                              |
|-------|------|-----------------------------------------|---------------------------------------------------------------------------------------|
| 6:0   |      | Reserved                                | Do not count on their value.                                                          |
| 7     | EST  | Enhanced Intel<br>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. |
| 31:11 |      | Reserved                                | Do not count on their value.                                                          |



#### 3.4 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.5 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 6 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 7 to determine the cache and TLB features of a processor.

**Table 7. Descriptor Formats** 

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

Table 7 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.



**Table 8. Descriptor Decode Values** 

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



#### **Table 8. Descriptor Decode Values**

| Value | Cache or TLB Description                                              |  |
|-------|-----------------------------------------------------------------------|--|
| 84h   | 2nd-level cache: 1M-bytes, 8-way set associative, 32 byte line size   |  |
| 85h   | 2nd-level cache: 2M-bytes, 8-way set associative, 32 byte line size   |  |
| B0h   | Instruction TLB: 4K-byte Pages, 4-way set associative, 128 entries    |  |
| B3h   | Data TLB: 4K-byte Pages, 4-way set associative, 128 entries           |  |
| 86h   | 2nd-level cache: 512K-bytes, 4-way set associative, 64 byte line size |  |
| 87h   | 2nd-level cache: 1M-bytes, 8-way set associative, 64 byte line size   |  |



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

Table 9. Pentium® 4 Processor, model 0 with 256K 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.

#### 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.



#### 5 BRAND ID

Beginning with the Pentium III processors, model 8, the Pentium III Xeon processors, model 8, and Intel 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 9). 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, Intel 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 512K L2 cache. Brand ID eliminates this ambiguity by providing a software accessible value unique to each processor brand. Table 9 shows the values defined for each processor.

#### 6 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.

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

| Value            | Description                                                            |
|------------------|------------------------------------------------------------------------|
| 00h              | Unsupported                                                            |
| 01h              | Intel® Celeron® processor                                              |
| 02h              | Intel® Pentium® III processor                                          |
| 03h              | Intel® Pentium® III Xeon™ processor                                    |
|                  | 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                                            |
|                  | If processor signature is >=00000F13h, then "Intel® Genuine processor" |
| 09h              | Intel® Pentium® 4 processor                                            |
| 0Ah              | Intel® Celeron® Processor                                              |
| 0Bh              | Intel® Xeon™ processor                                                 |
|                  | If processor signature is <00000F13h, then "Intel® Xeon™ processor MP" |
| 0Ch              | Intel® Xeon™ processor MP                                              |
| 0Eh              | Mobile Intel® Pentium® 4 processor–M                                   |
|                  | If processor signature is <00000F13h, then "Intel® Xeon™ processor"    |
| 0Fh              | Mobile Intel® Celeron® Processor                                       |
| All other values | Reserved                                                               |

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

The brand/frequency string is defined to be 48 characters long, 47 bytes will contain characters and the 48<sup>th</sup> 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, 80000003h 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=80000000h
- 2. If ((returned value in EAX) > 80000000h) 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 80000004h$



| Table III I 1000000 Braile Stillig I Catalo |                                                        |                                               |  |  |
|---------------------------------------------|--------------------------------------------------------|-----------------------------------------------|--|--|
| EAX input value                             | Function                                               | Return value                                  |  |  |
| 80000000h                                   | Largest Extended Function Supported                    | EAX=80000004, EBX = ECX = EDX = Reserved      |  |  |
| 80000001h                                   | Extended Processor Signature and Extended Feature Bits | EAX = EBX = ECX = EDX = Reserved              |  |  |
| 80000002h                                   | Processor Brand String                                 | EAX, EBX, ECX, EDX contain ASCII brand string |  |  |
| 80000003h                                   | Processor Brand String                                 | EAX, EBX, ECX, EDX contain ASCII brand string |  |  |
| 80000004h                                   | Processor Brand String                                 | EAX, EBX, ECX, EDX contain ASCII brand string |  |  |

**Table 11. Processor Brand String Feature** 

#### 7 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.0, and shown in Section 8.
- 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 12 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 processor serial number as 6 groups of 4 hex nibbles (Ex. XXXX-XXXX-XXXX-XXXX-XXXX where X represents a hex digit).
- 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.

#### 8 PROPER IDENTIFICATION SEQUENCE

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

- 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.
- 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 9) 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 7) 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 Overdrive processors, Intel 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

#### 9 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

#### 10 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 executes without generating an exception, then the processor supports that set of 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 SSE2, SSE, and MMX instruction extensions. Features.cpp performs the following steps:

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



#### 11 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.
- 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.
- 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.
- 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.
- 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.

#### 12 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.

#### **AP-485**



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-2003 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 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 \_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 cpuid3.c, 32-bit segments are recommended. To assemble this code with TASM, add the JUMPS directive. ; Uncomment this line for TASM jumps TITLE cpuid3a comment this line for 32-bit segments DOSSEG

uncomment the following 2 lines for 32-bit segments

.386

31

#### **AP-485**



```
.model
                  flat
         comment this line for 32-bit segments
         .model
                  small
CPU ID MACRO
                  0fh
                                              ; Hardcoded CPUID instruction
         db
         db
                  0a2h
ENDM
.data
         public
                  _cpu_type
                  _fpu_type
         public
                  _v86_flag
         public
                  _cpuid_flag
         public
                  _intel_CPU
         public
                  _vendor_id
         public
                  _cpu_signature
         public
         public
                  _features_ecx
         public
                   features edx
         public
                  _features_ebx
         public
                   cache_eax
         public
                   cache ebx
         public
                   cache ecx
                   cache edx
         public
         public
                  _sep_flag
         public
                  _brand_string
                                     0
         _cpu_type
                            db
         _fpu_type
                                     0
                            db
         _v86_flag
                                     0
                            db
         _cpuid_flag
                            db
                                     0
         _intel_CPU
                                     0
                            db
         _sep_flagdb
                            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
         fp status
                            dw
                                     0
                                     48 dup (0)
         _brand_string
                            db
.code
         comment this line for 32-bit segments
.8086
         uncomment this line for 32-bit segments
         .386
         public
                  _get_cpu_type
         _get_cpu_type
```



```
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
                                                ; save original FLAGS
         mov
                   cx, ax
         and
                   ax, 0fffh
                                                ; clear bits 12-15 in FLAGS
                                                ; save new FLAGS value on stack
         push
                                                ; replace current FLAGS value
         popf
                                                ; get new FLAGS
         pushf
                                                ; store new FLAGS in AX
         pop
                   ax
                   ax, 0f000h
                                                ; if bits 12-15 are set, then
         and
                                                ; processor is an 8086/8088
                   ax, 0f000h
         cmp
                                                ; turn on 8086/8088 flag
                   _cpu_type, 0
         mov
         jne
                   check_80286
                                                ; go check for 80286
         push
                   sp
                                                ; double check with push sp
                   dx
                                                ; if value pushed was different
         pop
                                                ; means it's really an 8086
                   dx, sp
         cmp
                                                ; jump if processor is 8086/8088
                   end cpu type
         jne
         mov
                   cpu type, 10h
                                                ; indicate unknown processor
                   end_cpu_type
         jmp
         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:
                                                ; save machine status word
         smsw
                   ax
                                                ; isolate PE bit of MSW
         and
                   ax, 1
                   v86 flag, al
                                                ; save PE bit to indicate V86
         mov
                                                ; try to set bits 12-15
         or
                   cx, 0f000h
                                                ; save new FLAGS value on stack
         push
                                                ; replace current FLAGS value
         popf
         pushf
                                                ; get new FLAGS
                                                ; store new FLAGS in AX
         pop
                   ax
                   ax, 0f000h
                                                ; if bits 12-15 are clear
         and
                                                ; processor=80286, turn on 80286 flag
         mov
                   _cpu_type, 2
         jz
                   end cpu type
                                                ; jump if processor is 80286
         Intel386 processor check
         The AC bit, bit #18, is a new bit introduced in the EFLAGS
         register on the Intel486 processor to generate alignment
         faults.
         This bit cannot be set on the Intel386 processor.
```



```
.386
         "STOP"
                                               ; it is safe to use 386 instructions
check 80386:
                                               ; push original EFLAGS
         pushfd
                                               ; get original EFLAGS
         pop
                                               ; save original EFLAGS
         mov
                  ecx, eax
                                                ; flip AC bit in EFLAGS
                  eax, 40000h
         xor
                                               ; save new EFLAGS value on stack
         push
                  eax
         popfd
                                               ; replace current EFLAGS value
         pushfd
                                               ; get new EFLAGS
                                               ; store new EFLAGS in EAX
                  eax
         pop
                                                ; can't toggle AC bit, processor=80386
         xor
                  eax. ecx
                                                ; turn on 80386 processor flag
         mov
                   cpu type, 3
         įΖ
                   end_cpu_type
                                               ; jump if 80386 processor
         push
                  ecx
                                               ; restore AC bit in EFLAGS first
         popfd
         Intel486 processor check
         Checking for ability to set/clear ID flag (Bit 21) in EFLAGS
         which indicates the presence of a processor with the CPUID
         instruction.
.486
check_80486:
                                               ; turn on 80486 processor flag
                   _cpu_type, 4
         mov
         mov
                  eax, ecx
                                               ; get original EFLAGS
         xor
                  eax, 200000h
                                               ; flip ID bit in EFLAGS
                                               ; save new EFLAGS value on stack
         push
         popfd
                                               ; replace current EFLAGS value
         pushfd
                                               ; get new EFLAGS
                                               ; store new EFLAGS in EAX
         pop
                  eax
                                               ; can't toggle ID bit,
         xor
                  eax, ecx
                                               ; processor=80486
                  end_cpu_type
         je
         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
         mov
                                               ; flag indicating use of CPUID inst.
         push
                  ebx
                                                ; save registers
         push
                  esi
         push
                  edi
         mov
                   eax, 0
                                               ; set up for CPUID instruction
         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
         ine
                  end cpuid type
         cmp
                   dword ptr intel id[+4], edx
                  end cpuid type
         ine
                  dword ptr intel id[+8], ecx
         cmp
         jne
                  end cpuid type
                                               ; if not equal, not an Intel processor
                   _intel_CPU, 1
                                               ; indicate an Intel processor
         mov
                                               ; make sure 1 is valid input for CPUID
                  eax, 1
         cmp
```



```
end_cpuid_type
                                      ; if not, jump to end
         eax, 1
mov
CPU ID
                                      ; get family/model/stepping/features
          _cpu_signature, eax
mov
mov
          features ebx, ebx
          features edx, edx
mov
mov
          features ecx, ecx
shr
          eax, 8
                                      ; isolate family
         eax, 0fh
and
          _cpu_type, al
                                      ; set _cpu_type with family
mov
Execute CPUID instruction to determine the cache descriptor
information.
mov
         eax, 0
                                      ; set up to check the EAX value
CPU ID
cmp
         ax, 2
                                      ; Are cache descriptors supported?
jl
         end_cpuid_type
mov
         eax, 2
                                      ; set up to read cache descriptor
CPU ID
                                       ; Is one iteration enough to obtain
          al, 1
cmp
         end cpuid type
                                       ; cache information?
jne
                                      ; This code supports one iteration
                                       ; only.
                                       ; store cache information
          _cache_eax, eax
mov
                                       ; NOTE: for future processors, CPUID
          _cache_ebx, ebx
mov
          _cache_ecx, ecx
                                       ; instruction may need to be run more
mov
mov
          cache edx, edx
                                       ; than once to get complete cache
                                       ; information
          eax, 80000000h
mov
                                      ; check if brand string is supported
CPU ID
          eax, 80000000h
cmp
jbe
          end_cpuid_type
                                      ; take jump if not supported
mov
         di, offset _brand_string
         eax, 80000002h
                                      ; get first 16 bytes of brand string
mov
CPU ID
mov
          dword ptr [di], eax
                                      ; save bytes 0 .. 15
          dword ptr [di+4], ebx
mov
mov
          dword ptr [di+8], ecx
mov
          dword ptr [di+12], edx
add
         di, 16
          eax, 80000003h
mov
CPU ID
         dword ptr [di], eax
                                      ; save bytes 16 .. 31
mov
         dword ptr [di+4], ebx
mov
         dword ptr [di+8], ecx
mov
         dword ptr [di+12], edx
mov
add
         di, 16
          eax, 80000004h
mov
CPU ID
         dword ptr [di], eax
mov
                                      ; save bytes 32 .. 47
mov
         dword ptr [di+4], ebx
         dword ptr [di+8], ecx
mov
```

dword ptr [di+12], edx

mov



```
end_cpuid_type:
         pop
                  edi
                                              ; restore registers
         pop
                  esi
         pop
                  ebx
         comment this line for 32-bit segments
.8086
end_cpu_type:
         ret
                  endp
_get_cpu_type
public
                   _get_fpu_type
         _get_fpu_type
         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
                                              ; initialize temp word to non-zero
         mov
                  fp status, 5a5ah
                                              ; save FP status word
         fnstsw
                  fp_status
                  ax, fp_status
                                              ; check FP status word
         mov
                  al, 0
                                              ; was correct status written
         cmp
         mov
                  _fpu_type, 0
                                              ; no FPU present
                  end_fpu_type
        jne
check control word:
                  fp_status
         fnstcw
                                              ; save FP control word
         mov
                  ax, fp_status
                                              ; check FP control word
                                              ; selected parts to examine
         and
                  ax, 103fh
                                              ; was control word correct
         cmp
                  ax, 3fh
         mov
                  fpu type, 0
                                              ; incorrect control word, no FPU
        jne
                  end_fpu_type
                  _fpu_type, 1
         mov
         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
         fdiv
                                              ; 8087/Intel287 NDP say +inf = -inf
         fld
                                              ; form negative infinity
                                              ; Intel387 NDP says +inf <> -inf
         fchs
         fcompp
                                              ; see if they are the same
                                              ; look at status from FCOMPP
         fstsw
                  fp_status
```



mov ax, fp\_status
mov \_fpu\_type, 2 ; store Intel287 NDP for FPU type
sahf
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

Filename: cpuid3b.asm

Copyright (c) Intel Corporation 1993-2003

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 program contains three parts:

Part 1: Identifies processor type in the variable \_cpu\_type:

Part 2: Identifies FPU type in the variable fpu type:

Part 3: Prints out the appropriate message. This part is specific to the DOS environment and uses the DOS system calls to print out the messages.

This program has been tested with the Microsoft Developer Studio. If this code is assembled 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 processors, Pentium(R) II Xeon(TM) processors, Pentium(R) II Overdrive(R) processors, Intel(R) Celeron(R) processors, Pentium(R) III processors, Pentium(R) III Xeon(TM) processors, Pentium(R) 4 processors and Intel(R) Xeon(TM) processors DP and MP when executed in the real-address mode.

NOTE: This code is written using 16-bit Segments

To assemble this code with TASM, add the JUMPS directive.

; jumps ; Uncomment this line for TASM

TITLE cpuid3b

DOSSEG

.model small

.stack 100h

OP O MACRO

db 66h ; hardcoded operand override

**ENDM** 

.data

extrn \_cpu\_type: byte extrn \_fpu\_type: byte extrn \_cpuid\_flag: byte extrn \_intel\_CPU: byte



```
_vendor_id:
                                      byte
       extrn
                                      dword
                       _cpu_signature:
       extrn
                                      dword
       extrn
                       features ecx:
       extrn
                       features edx:
                                      dword
       extrn
                       features ebx:
                                      dword
       extrn
                       cache eax:
                                      dword
       extrn
                       cache ebx:
                                      dword
                                      dword
       extrn
                       cache ecx:
       extrn
                       _cache_edx:
                                      dword
                       _brand_string:
                                      byte
       extrn
       The purpose of this code is to identify the processor and
       coprocessor that is currently in the system. The program
       first determines the processor type. Then it determines
       whether a coprocessor exists in the system. If a
       coprocessor or integrated coprocessor exists, the program
       identifies the coprocessor type. The program then prints
       the processor and floating point processors present and type.
.code
.8086
start:
               ax, @data
       mov
               ds, ax
                                      ; set segment register
       mov
               es, ax
                                      ; set segment register
       mov
               sp, not 3
                                      ; align stack to avoid AC fault
       and
       call
                                      ; determine processor type
               _get_cpu_type
       call
               _get_fpu_type
       call
               print
       mov
               ax, 4c00h
               21h
extrn
               _get_cpu_type: proc
extrn
               get fpu type: proc
FPU FLAG
                       equ 0001h
VME FLAG
                       equ 0002h
DE FLAG
                       equ 0004h
PSE_FLAG
                       equ 0008h
TSC FLAG
                       equ 0010h
MSR FLAG
                       equ 0020h
PAE FLAG
                       equ 0040h
MCE_FLAG
                       equ 0080h
CX8 FLAG
                       equ 0100h
APIC FLAG
                       equ 0200h
SEP FLAG
                       equ 0800h
MTRR FLAG
                       equ 1000h
PGE FLAG
                       equ 2000h
MCA FLAG
                       equ 4000h
CMOV FLAG
                       equ 8000h
                      equ 10000h
PAT FLAG
                       equ 20000h
PSE36 FLAG
PSNUM FLAG
                       equ 40000h
```



```
equ 80000h
CLFLUSH FLAG
DTS FLAG
                          equ 200000h
ACPI FLAG
                          egu 400000h
MMX FLAG
                          equ 800000h
                          equ 1000000h
FXSR FLAG
                          equ 2000000h
SSE FLAG
                          equ 4000000h
SSE2 FLAG
                          equ 8000000h
SS FLAG
                          equ 10000000h
HTT_FLAG
TM FLAG
                          equ 20000000h
PBE FLAG
                          equ 80000000h
EST_FLAG
                          egu 80h
TM2 FLAG
                          equ 100h
CID FLAG
                          equ 400h
.data
                           "This system has a$"
id msg
                 db
                           "n unknown processor$"
cp error
                 db
cp_8086
                          "n 8086/8088 processor$"
                 db
                          "n 80286 processor$"
cp 286
                 db
                          "n 80386 processor$"
cp_386
                 db
                 db
                          "n 80486DX, 80486DX2 processor or"
cp 486
                          " 80487SX math coprocessor$"
                 db
                          "n 80486SX processor$"
cp 486sx
                 db
fp 8087
                 db
                          " and an 8087 math coprocessor$"
                          " and an 80287 math coprocessor$"
fp_287
                 db
                          " and an 80387 math coprocessor$"
fp 387
                 db
                 db
                          "Genuine Intel486(TM) processor$"
intel486 msg
intel486dx msg
                          " Genuine Intel486(TM) DX processor$"
                 db
                          "Genuine Intel486(TM) SX processor$"
intel486sx msg
                 db
                          " Genuine IntelDX2(TM) processor$"
inteldx2 msg
                 db
                          " Genuine IntelSX2(TM) processor$"
intelsx2 msg
                 db
inteldx4_msg
                 db
                           " Genuine IntelDX4(TM) processor$"
                          " Genuine Write-Back Enhanced"
inteldx2wb msg
                 db
                           " IntelDX2(TM) processor$"
                 db
pentium msg
                           "Genuine Intel(R) Pentium(R) processor$"
                 db
                           "Genuine Intel Pentium(R) Pro processor$"
pentiumpro msg
                                    "Genuine Intel(R) Pentium(R) II processor, model 3$"
pentiumiimodel3 msg
                          db
pentiumiixeon m5 msg
                                    "Genuine Intel(R) Pentium(R) II processor, model 5 or Intel(R) Pentium(R) II
                          db
Xeon(TM) processor$"
pentiumiixeon msg
                          db
                                    "Genuine Intel(R) Pentium(R) II Xeon(TM) processor$"
                                    "Genuine Intel(R) Celeron(R) processor, model 5$"
celeron msg
                          db
celeronmodel6 msg
                                    "Genuine Intel(R) Celeron(R) processor, model 6$"
                          db
celeron brand
                                    "Genuine Intel(R) Celeron(R) processor$"
                          db
                                    "Genuine Intel(R) Pentium(R) III processor, model 7 or Intel Pentium(R) III Xeon(TM)
pentiumiii msg
                          db
processor, model 7$"
pentiumiiixeon msg
                                   "Genuine Intel(R) Pentium(R) III Xeon(TM) processor, model 7$"
                          db
pentiumiiixeon brand
                                    "Genuine Intel(R) Pentium(R) III Xeon(TM) processor$"
                          db
pentiumiii brand
                          db
                                    "Genuine Intel(R) Pentium(R) III processor$"
mobile piii brand
                          db
                                    "Genuine Mobile Intel(R) Pentium(R) III Processor-M$"
mobile icp brand
                                    "Genuine Mobile Intel(R) Celeron(R) processor$"
                          db
                                    "Genuine Mobile Intel(R) Pentium(R) 4 processor - M$"
mobile p4 brand
                          db
pentium4 brand
                                    "Genuine Intel(R) Pentium(R) 4 processor$"
                          db
                                    "Genuine Intel(R) Xeon(TM) processor$"
xeon brand
                          db
xeon_mp_brand
                          db
                                    "Genuine Intel(R) Xeon(TM) processor MP$"
unknown msg
                          db
                                    "n unknown Genuine Intel(R) processor$"
```



```
brand entry
                   struct
                            db
         brand value
                                      ?
         brand string
                            dw
brand entry
brand table
                   brand entry
                                      <01h, offset celeron brand>
                   brand entry
                                      <02h, offset pentiumiii brand>
                  brand_entry
                                      <03h, offset pentiumiiixeon_brand>
                                      <04h, offset pentiumiii brand>
                  brand entry
                  brand entry
                                      <06h, offset mobile piii brand>
                   brand_entry
                                      <07h, offset mobile_icp_brand>
                   brand_entry
                                      <08h, offset pentium4 brand>
                                      <09h, offset pentium4_brand>
                   brand entry
                                      <0Ah, offset celeron brand>
                   brand entry
                   brand entry
                                      <0Bh, offset xeon brand>
                   brand_entry
                                      <0Ch, offset xeon mp brand>
                   brand entry
                                      <0Eh, offset mobile p4 brand>
                   brand entry
                                      <0Fh, offset mobile icp brand>
brand table size
                            ($ - offset brand table) / (sizeof brand entry)
The following 16 entries must stay intact as an array
intel 486 0
                            offset intel486dx msg
                   dw
intel 486 1
                  dw
                            offset intel486dx msg
intel 486 2
                            offset intel486sx msg
                  dw
intel 486 3
                  dw
                            offset inteldx2 msg
intel 486 4
                            offset intel486 msg
                  dw
intel 486 5
                  dw
                            offset intelsx2 msg
intel 486 6
                   dw
                            offset intel486 msg
intel 486 7
                   dw
                            offset inteldx2wb msg
intel 486 8
                   dw
                            offset inteldx4 msg
intel 486 9
                            offset intel486 msg
                   dw
intel 486 a
                            offset intel486 msg
                  dw
intel 486 b
                            offset intel486 msg
                  dw
intel 486 c
                            offset intel486_msg
                  dw
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:
                            13,10,"Stepping:
stepping msg
                   db
                            13,10," Extended Family: $"
ext fam msg
                  db
                            13,10," Extended Model: $"
ext mod msg
                   db
                            13,10,"$"
cr 1f
                   db
                            13,10,"The processor is an OverDrive(R)"
turbo_msg
                   db
                            " processor$"
                   db
                   db
                            13,10,"The processor is the upgrade"
dp msg
                   db
                            " processor in a dual processor system$"
                   db
                            13,10,"The processor contains an on-chip"
fpu_msg
                            " FPU$"
                   db
vme msg
                   db
                            13,10,"The processor supports Virtual"
                   db
                            " Mode Extensions$"
                   db
                            13,10,"The processor supports Debugging"
de msg
                            " Extensions$"
                   db
                            13,10,"The processor supports Page Size"
pse msg
                   db
                            " Extensions$"
                   db
                  db
                            13,10,"The processor supports Time-Stamp"
tsc_msg
                  db
                            " Counter$"
```



| msr_msg     | db       | 13,10,"The processor supports Model" "Specific Registers\$"               |
|-------------|----------|---------------------------------------------------------------------------|
| nae mca     | db<br>db | 13,10,"The processor supports Physical"                                   |
| pae_msg     | db       | " Address Extensions\$"                                                   |
| mce_msg     | db       | 13,10,"The processor supports Machine"                                    |
|             | db       | "Check Exceptions\$"                                                      |
| cx8_msg     | db       | 13,10,"The processor supports the"                                        |
|             | db       | " CMPXCHG8B instruction\$"                                                |
| apic_msg    | db       | 13,10,"The processor contains an on-chip"                                 |
|             | db       | " APIC\$"                                                                 |
| sep_msg     | db       | 13,10,"The processor supports Fast System"                                |
| no san msa  | db<br>db | " Call\$" 13,10,"The processor does not support Fast"                     |
| no_sep_msg  | db       | " System Call\$"                                                          |
| mtrr msg    | db       | 13,10,"The processor supports Memory Type"                                |
|             | 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<br>db | 13,10,"The processor supports Page Attribute" " Table\$"                  |
| pse36_msg   | db       | 13,10,"The processor supports 36-bit Page"                                |
| psc30_msg   | db       | "Size Extension\$"                                                        |
| psnum msg   | db       | 13,10,"The processor supports the"                                        |
| 1 _ 0       | db       | " processor serial number\$"                                              |
| clflush_msg | db       | 13,10,"The processor supports the"                                        |
|             | db       | " CLFLUSH instruction\$"                                                  |
| dts_msg     | db       | 13,10,"The processor supports the"                                        |
|             | db       | "Debug Trace Store feature\$"                                             |
| acpi_msg    | db       | 13,10,"The processor supports the"                                        |
| mmy mag     | db<br>db | " ACPI registers in MSR space\$"                                          |
| mmx_msg     | db       | 13,10,"The processor supports Intel Architecture" " MMX(TM) Technology\$" |
| fxsr msg    | db       | 13,10,"The processor supports Fast floating point"                        |
|             | db       | " save and restore\$"                                                     |
| sse msg     | db       | 13,10,"The processor supports the Streaming"                              |
| _ ~         | db       | " SIMD extensions\$"                                                      |
| sse2_msg    | 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<br>db | 13,10,"The processor supports the" "Thermal Monitor\$"                    |
| pbe_msg     | db       | 13,10,"The processor supports the"                                        |
| poc_msg     | db       | " Pending Break Event\$"                                                  |
| est msg     | db       | 13,10,"The processor supports"                                            |
|             | db       | "Enhanced SpeedStep(R) Technology\$"                                      |
| tm2_msg     | db       | 13,10,"The processor supports the"                                        |
|             | db       | " Thermal Monitor 2\$"                                                    |
| cid_msg     | db       | 13,10,"The processor supports L1 Data Cache Context ID"                   |
|             |          |                                                                           |
|             | .11.     | "t 1 t                                                                    |
| not_intel   | db<br>db | "t least an 80486 processor."                                             |
|             | db       | 13,10,"It does not contain a Genuine" "Intel part and as a result,"       |
|             | db       | "the",13,10,"CPUID"                                                       |
|             | db       | " detection information cannot be"                                        |
|             | db       | " determined at this time.\$"                                             |
|             |          |                                                                           |



```
ASC_MSG
                   MACRO msg
         LOCAL ascii done
                                               ; local label
         add
                   al, 30h
                   al, 39h
                                               ; is it 0-9?
         cmp
                   ascii done
         ile
                   al, 07h
         add
ascii done:
                   byte ptr msg[20], al
                   dx, offset msg
         mov
                   ah, 9h
         mov
         int
                   21h
ENDM
.code
.8086
print
         proc
         This procedure prints the appropriate cpuid string and
         numeric processor presence status. If the CPUID instruction
         was used, this procedure prints out the CPUID info.
         All registers are used by this procedure, none are
         preserved.
         mov
                   dx, offset id msg
                                               ; print initial message
                   ah, 9h
         mov
                   21h
         int
         cmp
                   _cpuid_flag, 1
                                               ; if set to 1, processor
                                               ; supports CPUID instruction
                   print_cpuid_data
                                               ; print detailed CPUID info
         je
print 86:
         cmp
                   _cpu_type, 0
                   print_286
         jne
                   dx, offset cp_8086
         mov
                   ah, 9h
         mov
                   21h
         int
                   _fpu_type, 0
         cmp
                   end print
         je
                   dx, offset fp 8087
         mov
                   ah, 9h
         mov
                   21h
         int
                   end print
         jmp
print_286:
                   _cpu_type, 2
         cmp
                   print 386
         jne
                   dx, offset cp 286
         mov
                   ah, 9h
         mov
                   21h
         int
                   _fpu_type, 0
         cmp
         je
                   end print
print 287:
                   dx, offset fp_287
         mov
                   ah, 9h
         mov
                   21h
         int
                   end print
         jmp
```

# **AP-485**



```
print_386:
                   _cpu_type, 3
         cmp
                   print 486
         jne
         mov
                   dx, offset cp 386
                   ah, 9h
         mov
                   21h
         int
                   _fpu_type, 0
         cmp
                   end_print
         je
                   _fpu_type, 2
         cmp
                   print_287
         je
                   dx, offset fp_387
         mov
         mov
                   ah, 9h
         int
                   21h
                   end print
         jmp
print_486:
                   _cpu_type, 4
         cmp
                                               ; Intel processors will have
         jne
                   print_unknown
                   dx, offset cp_486sx
                                               ; CPUID instruction
         mov
                   fpu type, 0
         cmp
                   print 486sx
         je
                   dx, offset cp_486
         mov
print_486sx:
                   ah, 9h
         mov
                   21h
         int
                   end_print
         jmp
print_unknown:
                   dx, offset cp error
         mov
         jmp
                   print 486sx
print_cpuid_data:
.486
                   _intel_CPU, 1
                                               ; check for genuine Intel
         cmp
                   not_GenuineIntel
                                               ; processor
         jne
                   di, offset _brand_string
         mov
                                               ; brand string supported?
                   byte ptr [di], 0
         cmp
         je
                   print_brand_id
                                               ; max brand string length
         mov
                   cx, 47
skip_spaces:
                   byte ptr [di], ' '
                                               ; skip leading space chars
                   print brand string
         jne
         inc
                   skip_spaces
         loop
print brand string:
                   cx, 0
                                               ; Nothing to print
         cmp
                   print_brand_id
         je
         cmp
                   byte ptr [di], 0
         je
                   print brand id
print brand char:
                   dl, [di]
                                               ; print upto the max chars
         mov
                   ah, 2
         mov
                   21h
         int
```



```
inc
                  byte ptr [di], 0
         cmp
                  print family
         je
         loop
                  print brand char
         jmp
                  print family
print brand id:
         cmp
                   cpu type, 6
         jb
                  print_486_type
                  print_pentiumiiimodel8_type
         ja
                  eax, dword ptr _cpu_signature
         mov
         shr
                  eax, 4
                  al. 0fh
         and
                  al. 8
         cmp
                  print pentiumiiimodel8 type
         jae
print_486_type:
                   _cpu_type, 4
                                               ; if 4, print 80486 processor
         cmp
         jne
                  print pentium type
         mov
                  eax, dword ptr cpu signature
         shr
                  eax, 4
                  eax, 0fh
                                               ; isolate model
         and
                  dx, intel 486 0[eax*2]
         mov
                  print common
         jmp
print_pentium_type:
                                               ; if 5, print Pentium processor
                  _cpu_type, 5
         cmp
         jne
                  print_pentiumpro_type
         mov
                  dx, offset pentium msg
                  print common
         jmp
print pentiumpro type:
                                               ; if 6 & model 1, print Pentium
         cmp
                  cpu type, 6
                                               ; Pro processor
                  print_unknown_type
         jne
                  eax, dword ptr _cpu_signature
         mov
         shr
                  eax, 4
                  eax, 0fh
                                               ; isolate model
         and
                  eax, 3
         cmp
                  print pentiumiimodel3 type
         jge
         cmp
                  eax, 1
         jne
                  print_unknown_type
                                               ; incorrect model number = 2
                  dx, offset pentiumpro msg
         mov
         jmp
                  print common
print_pentiumiimodel3_type:
                  eax, 3
                                               ; if 6 & model 3, print Pentium
         cmp
                                               ; II processor, model 3
         jne
                  print pentiumiimodel5 type
                  dx, offset pentiumiimodel3 msg
         mov
                  print_common
         jmp
print pentiumiimodel5 type:
         cmp
                  eax, 5
                                               ; if 6 & model 5, either Pentium
                                               ; II processor, model 5, Pentium II
                                               ; Xeon processor or Intel Celeron
                                               ; processor, model 5
         je
                  celeron xeon detect
                                               ; If model 7 check cache descriptors
                  eax, 7
         cmp
```



; to determine Pentium III or Pentium III Xeon

```
print_celeronmodel6_type
         jne
celeron_xeon_detect:
; Is it Pentium II processor, model 5, Pentium II Xeon processor, Intel Celeron processor,
; Pentium III processor or Pentium III Xeon processor.
                  eax, dword ptr cache eax
         mov
         rol
                  eax, 8
                  cx, 3
         mov
celeron detect eax:
                  al, 40h
                                               ; Is it no L2
         cmp
                  print_celeron_type
         je
                  al, 44h
                                               ; Is L2 >= 1M
         cmp
                  print pentiumiixeon type
         jae
         rol
                  eax, 8
                  celeron_detect_eax
         loop
         mov
                  eax, dword ptr cache ebx
         mov
                  cx, 4
celeron_detect_ebx:
                  al, 40h
                                               ; Is it no L2
         cmp
                  print celeron type
         je
                  al, 44h
                                               ; Is L2 >= 1M
         cmp
                  print_pentiumiixeon_type
         jae
         rol
         loop
                  celeron detect ebx
                  eax, dword ptr cache ecx
         mov
         mov
                  cx, 4
celeron detect ecx:
         cmp
                  al, 40h
                                               ; Is it no L2
                  print_celeron_type
         je
                                               ; Is L2 >= 1M
                  al, 44h
         cmp
                  print pentiumiixeon type
         jae
         rol
         loop
                  celeron_detect_ecx
                  eax, dword ptr cache edx
         mov
         mov
                  cx, 4
celeron_detect_edx:
                                               ; Is it no L2
                  al, 40h
         cmp
         je
                  print celeron type
                                               ; Is L2 >= 1M
                  al, 44h
         cmp
                  print_pentiumiixeon_type
         jae
         rol
         loop
                  celeron detect edx
                  dx, offset pentiumiixeon m5 msg
         mov
                  eax, dword ptr cpu signature
         mov
         shr
                  eax, 4
                  eax, 0fh
                                               ; isolate model
         and
                  eax, 5
         cmp
```



```
print_common
                  dx, offset pentiumiii_msg
         mov
         jmp
                  print common
print celeron type:
         mov
                   dx, offset celeron msg
         jmp
                  print common
print_pentiumiixeon_type:
                  dx, offset pentiumiixeon_msg
         mov
                  ax, word ptr _cpu_signature
         mov
         shr
                  ax, 4
         and
                  eax, 0fh
                                               ; isolate model
                  eax, 5
         cmp
         je
                  print common
                  dx, offset pentiumiiixeon msg
         mov
         jmp
                  print common
print_celeronmodel6_type:
                                               ; if 6 & model 6, print Intel Celeron
         cmp
                   eax, 6
                                               ; processor, model 6
         jne
                   print_pentiumiiimodel8_type
                  dx, offset celeronmodel6 msg
         mov
                  print common
         jmp
print pentiumiiimodel8 type:
                                               ; Pentium III processor, model 8, or
         cmp
                  eax, 8
                                               ; Pentium III Xeon processor, model 8
         jb
                  print_unknown_type
         mov
                  eax, dword ptr features ebx
                                               ; Is brand id supported?
         cmp
         je
                  print unknown type
                  di, offset brand table
                                               ; Setup pointer to brand id table
         mov
                  cx, brand_table_size
                                               ; Get maximum entry count
         mov
next_brand:
                  al, byte ptr [di]
                                               ; Is this the brand reported by the processor
         cmp
                   brand found
         je
         add
                  di, sizeof brand entry
                                               ; Point to next Brand Defined
                  next brand
                                               ; Check next brand if the table is not exhausted
         loop
                   print_unknown_type
         jmp
brand found:
                   eax, dword ptr cpu signature
                                               ; Check for Pentium III, model B, stepping 1
                  eax, 06B1h
         cmp
                  not_b1_celeron
         jne
                  dx, offset celeron brand
                                               ; Assume this is a the special case (see Table 10)
         mov
                                               ; Is this a B1 Celeron?
                  byte ptr[di], 3
         cmp
                  print_common
         je
not b1 celeron:
                  eax, 0F13h
         cmp
         jae
                  not xeon mp
                                              ; Early "Intel(R) Xeon(TM) processor MP"?
         mov
                  dx, offset xeon mp brand
                  byte ptr [di], 0Bh
         cmp
                  print_common
         je
```



```
dx, offset xeon brand
                                               ; Early "Intel(R) Xeon(TM) processor"?
         mov
                  byte ptr[di], 0Eh
         cmp
         je
                  print common
not xeon mp:
                  dx, word ptr [di+1]
                                               ; Load DX with the offset of the brand string
         mov
         jmp
                  print common
print_unknown_type:
                  dx, offset \ unknown\_msg
                                               ; if neither, print unknown
         mov
print_common:
         mov
                  ah, 9h
                  21h
         int
; print family, model, and stepping
print family:
                  al, _cpu_type
         mov
         ASC_MSG
                            family_msg
                                               ; print family msg
                  eax, dword ptr cpu signature
                  ah, 0fh
                                               ; Check for Extended Family
         and
                  ah, 0fh
         cmp
                  print model
         jne
                  dx, offset ext fam msg
         mov
                  ah, 9h
         mov
                  21h
         int
                  eax, 20
         shr
         mov
                  ah, al
                                               ; Copy extended family into ah
         shr
                  al, 4
                  ax, 0f0fh
         and
                  ah, '0'
                                               ; Convert upper nibble to ascii
         add
                  al, '0'
         add
                                               ; Convert lower nibble to ascii
         push
                  ax
                  dl, al
         mov
                  ah, 2
         mov
                  21h
                                               ; print upper nibble of ext family
         int
                  ax
         pop
                  dl. ah
         mov
                  ah. 2
         mov
         int
                  21h
                                               ; print lower nibble of ext family
print_model:
                  eax, dword ptr cpu signature
         mov
                  ax, 4
         shr
         and
                  al, 0fh
         ASC MSG
                                               ; print model msg
                            model_msg
                  eax, dword ptr _cpu_signature
         mov
         and
                  al, 0f0h
                                               ; Check for Extended Model
                  ah, 0f0h
         cmp
                  print_stepping
         jne
                  dx, offset ext_mod_msg
         mov
         mov
                  ah, 9h
         int
                  21h
                  eax, 16
                  al, 0fh
         and
                  al, '0'
                                               ; Convert extended model to ascii
         add
                  dl, al
         mov
                  ah, 2
         mov
                  21h
                                               ; print lower nibble of ext family
         int
```



```
print_stepping:
                  eax, dword ptr _cpu_signature
         mov
         and
                  al, 0fh
         ASC_MSG
                           stepping msg
                                             ; print stepping msg
print upgrade:
                  eax, dword ptr _cpu_signature
         test
                  ax, 1000h
                                             ; check for turbo upgrade
                  check_dp
        jz
                  dx, offset turbo_msg
         mov
                  ah, 9h
         mov
         int
                  21h
                  print features
        jmp
check_dp:
         test
                  ax, 2000h
                                             ; check for dual processor
                  print_features
        jz
         mov
                  dx, offset dp_msg
                  ah, 9h
         mov
         int
                  21h
print_features:
                  eax, dword ptr _features_edx
         mov
                  eax, FPU FLAG
         and
                                             ; check for FPU
                  check VME
         jΖ
                  dx, offset fpu_msg
         mov
                  ah, 9h
         mov
         int
                  21h
check VME:
                  eax, dword ptr features edx
                  eax, VME FLAG
                                             ; check for VME
         and
                  check DE
        jz
                  dx, offset vme_msg
         mov
                  ah, 9h
         mov
         int
                  21h
check_DE:
                  eax, dword ptr features edx
         mov
         and
                  eax, DE FLAG
                                             ; check for DE
                  check PSE
        įΖ
                  dx, offset de_msg
         mov
                  ah, 9h
         mov
                  21h
         int
check_PSE:
                  eax, dword ptr _features_edx
         mov
         and
                  eax, PSE FLAG
                                             ; check for PSE
                  check TSC
         jz
                  dx, offset pse msg
         mov
                  ah, 9h
         mov
                  21h
         int
check TSC:
         mov
                  eax, dword ptr features edx
                  eax, TSC FLAG
                                             ; check for TSC
         and
                  check MSR
        įΖ
                  dx, offset tsc msg
         mov
                  ah, 9h
         mov
                  21h
         int
```



```
check_MSR:
                 eax, dword ptr _features_edx
         mov
         and
                 eax, MSR FLAG
                                            ; check for MSR
                 check PAE
        jz
                 dx, offset msr msg
        mov
                 ah, 9h
         mov
                 21h
         int
check\_PAE:
                 eax, dword ptr _features_edx
         mov
         and
                 eax, PAE_FLAG
                                            ; check for PAE
                 check\_MCE
        jz
                 dx, offset pae msg
         mov
                 ah, 9h
         mov
         int
                 21h
check_MCE:
                 eax, dword ptr _features_edx
         mov
                 eax, MCE FLAG
                                            ; check for MCE
         and
         jz
                 check CX8
                 dx, offset mce_msg
         mov
                 ah, 9h
         mov
                 21h
         int
check CX8:
                 eax, dword ptr _features_edx
         mov
                 eax, CX8 FLAG
                                            ; check for CMPXCHG8B
         and
        jz
                 check APIC
                  dx, offset cx8 msg
        mov
                  ah, 9h
         mov
                 21h
         int
check APIC:
                 eax, dword ptr _features_edx
         mov
                 eax, APIC_FLAG
                                            ; check for APIC
         and
                 check_SEP
        jz
                 dx, offset apic_msg
         mov
                 ah, 9h
         mov
                  21h
         int
check SEP:
                  eax, dword ptr _features_edx
         mov
                 eax, SEP FLAG
                                            ; Check for Fast System Call
         and
                 check MTRR
         jz
                                            ; Determine if Fast System
         cmp
                  _cpu_type, 6
                                            ; Calls are supported.
                 print_sep
        jne
         mov
                 eax, dword ptr cpu signature
        cmp
                 al, 33h
        jb
                 print_no_sep
print_sep:
         mov
                 dx, offset sep msg
                 ah, 9h
         mov
                 21h
         int
                 check MTRR
        jmp
print no sep:
                 dx, offset no_sep_msg
         mov
```



```
ah, 9h
        mov
                 21h
        int
check MTRR:
         mov
                 eax, dword ptr features edx
        and
                 eax, MTRR FLAG
                                           ; check for MTRR
                 check PGE
        įΖ
                 dx, offset mtrr msg
        mov
                 ah, 9h
        mov
                 21h
        int
check_PGE:
        mov
                 eax, dword ptr _features_edx
                 eax, PGE FLAG
                                           ; check for PGE
        and
                 check MCA
        jz
                 dx, offset pge msg
        mov
        mov
                 ah, 9h
        int
                 21h
check MCA:
        mov
                 eax, dword ptr features edx
                 eax, MCA_FLAG
                                           ; check for MCA
        and
                 check_CMOV
        jΖ
                 dx, offset mca msg
        mov
                 ah, 9h
        mov
                 21h
        int
check_CMOV:
        mov
                 eax, dword ptr _features_edx
                                           ; check for CMOV
        and
                 eax, CMOV FLAG
        jΖ
                 check PAT
                 dx, offset cmov_msg
        mov
                 ah, 9h
        mov
                 21h
        int
check_PAT:
                 eax, dword ptr _features_edx
        mov
                 eax, PAT_FLAG
        and
                 check PSE36
        jΖ
                 dx, offset pat msg
        mov
                 ah, 9h
        mov
        int
                 21h
check PSE36:
                 eax, dword ptr features edx
        mov
                 eax, PSE36 FLAG
        and
                 check_PSNUM
        jz
                 dx, offset pse36_msg
        mov
                 ah, 9h
        mov
        int
                 21h
check_PSNUM:
                 eax, dword ptr _features_edx
        mov
        and
                 eax, PSNUM FLAG
                                           ; check for processor serial number
        jz
                 check CLFLUSH
        mov
                 dx, offset psnum msg
                 ah, 9h
        mov
                 21h
        int
check_CLFLUSH:
                 eax, dword ptr _features_edx
        mov
```



```
and
                 eax, CLFLUSH_FLAG
                                            ; check for Cache Line Flush
                 check DTS
        jz
                 dx, offset clflush_msg
        mov
        mov
                 ah, 9h
        int
                 21h
check DTS:
                 eax, dword ptr features edx
        and
                 eax, DTS_FLAG
                                            ; check for Debug Trace Store
                 check_ACPI
        jΖ
                 dx, offset dts msg
        mov
                 ah, 9h
        mov
        int
                 21h
check_ACPI:
                 eax, dword ptr features edx
        mov
        and
                 eax, ACPI_FLAG
                                            ; check for processor serial number
                 check\_MMX
        jΖ
                 dx, offset acpi_msg
        mov
                 ah, 9h
        mov
        int
                 21h
check_MMX:
                 eax, dword ptr features edx
        mov
                 eax, MMX FLAG
        and
                                            ; check for MMX technology
                 check FXSR
        jz
                 dx, offset mmx_msg
        mov
                 ah, 9h
        mov
        int
                 21h
check FXSR:
                 eax, dword ptr _features_edx
                 eax, FXSR FLAG
                                            ; check for FXSR
        and
                 check SSE
        jΖ
                 dx, offset fxsr_msg
        mov
                 ah, 9h
        mov
        int
                 21h
check_SSE:
                 eax, dword ptr features edx
        mov
        and
                 eax, SSE FLAG
                                            ; check for Streaming SIMD
         įΖ
                 check SSE2
                                            ; Extensions
        mov
                 dx, offset sse_msg
                 ah, 9h
        mov
                 21h
        int
check_SSE2:
                 eax, dword ptr _features_edx
        mov
        and
                 eax, SSE2 FLAG
                                            ; check for Streaming SIMD
                 check SS
        jz
                                            ; Extensions 2
                 dx, offset sse2 msg
        mov
                 ah, 9h
        mov
                 21h
        int
check SS:
                 eax, dword ptr features edx
        mov
                 eax, SS FLAG
                                            ; check for Self Snoop
        and
                 check HTT
        įΖ
                 dx, offset ss msg
        mov
                 ah, 9h
        mov
                 21h
        int
```



```
check_HTT:
                  eax, dword ptr _features_edx
         mov
         and
                  eax, HTT FLAG
                                             ; check for Hyper-Thread Technology
        jz
                  check TM
                  eax, dword ptr _features_ebx
         mov
                                             ; Put Logical processor count in reg AH
         bswap
         cmp
                  ah, 1
                                             ; Logical processor count > 1?
                  check_TM
        je
                  dx, offset htt_msg
                                             ; Supports HTT
         mov
         mov
                  ah, 9h
                  21h
         int
check TM:
         mov
                  eax, dword ptr _features_edx
                  eax, TM_FLAG
         and
                                             ; check for Thermal Monitor
                  check PBE
         įΖ
                  dx, offset tm msg
         mov
         mov
                  ah, 9h
         int
                  21h
check PBE:
                  eax, dword ptr features edx
         mov
                  eax, PBE FLAG
         and
                                             ; check for Pending Break Event
        jz
                  check_EST
                  dx, offset pbe_msg
         mov
         mov
                  ah, 9h
         int
                  21h
check EST:
                  eax, dword ptr features ecx
                  eax, EST_FLAG
                                             ; check for Enhanced SpeedStep Technology
         and
                  check_TM2
        jz
                  dx, offset est msg
         mov
                  ah, 9h
         mov
         int
                  21h
check_TM2
                  eax, dword ptr features ecx
         mov
         and
                  eax, TM2 FLAG
                                             ; check for Thermal Monitor 2
                  check CID
        įΖ
                  dx, offset tm2 msg
         mov
                  ah, 9h
         mov
         int
                  21h
check_CID:
                  eax, dword ptr _features_ecx
         mov
                  eax, CID FLAG
         and
                                             ; check for L1 Context ID
                  end print
         jΖ
                  dx, offset cid_msg
         mov
                  ah, 9h
         mov
         int
                  21h
        jmp
                  end print
not GenuineIntel:
         mov
                  dx, offset not intel
                  ah, 9h
         mov
                  21h
         int
```

# **AP-485**



end start



#define FXSR FLAG

#define SSE FLAG

Example 3. Processor Identification Procedure in the C Language

```
/* FILENAME: CPUID3.C
/* Copyright (c) Intel Corporation 1994-2003
                                                                           */
                                                                           */
/* 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 program contains three parts:
/* Part 1: Identifies CPU type in the variable _cpu_type:
/* Part 2: Identifies FPU type in the variable fpu type:
/* Part 3: Prints out the appropriate message.
/* 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),
                                                                           */
/* Intel(R) Celeron(R), Pentium(R) III processors, Pentium(R) III Xeon(TM)
/* processors, Pentium(R) 4 processors and Intel(R) Xeon(TM) processors
#define FPU FLAG
                            0x0001
#define VME FLAG
                            0x0002
#define DE FLAG
                            0x0004
#define PSE FLAG
                            0x0008
#define TSC FLAG
                            0x0010
#define MSR FLAG
                            0x0020
#define PAE FLAG
                            0x0040
#define MCE FLAG
                            0x0080
#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
#define PSNUM_FLAG
                            0x40000
#define CLFLUSH FLAG
                           0x80000
#define DTS FLAG
                            0x200000
#define ACPI FLAG
                            0x400000
#define MMX FLAG
                            0x800000
```

0x1000000

0x2000000



```
#define SSE2 FLAG
                                0x4000000
#define SS FLAG
                                0x8000000
#define HTT FLAG
                                0x10000000
#define TM FLAG
                                0x20000000
#define PBE FLAG
                                0x80000000
#define EST FLAG
                                0x80
#define TM2 FLAG
                                0x100
#define CID FLAG
                                0x400
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;
          char
                     *brand string;
};
#define brand_table_size 13
struct brand_entry brand_table[brand_table_size] = {
          0x01, "Genuine Intel(R) Celeron(R) 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",
          0x09, "Genuine Intel(R) Pentium(R) 4 processor", 0x0A, "Genuine Intel(R) Celeron(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",
          0x0F, "Genuine Mobile Intel(R) Celeron(R) processor"
};
int main() {
          get cpu type();
          get fpu type();
          print();
          return(0);
}
```



```
int print() {
                   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");
                   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");
                                       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 == 0x40000000)
                  celeron flag = 1;
         if ((cache temp \geq 0x44000000) && (cache temp \leq 0x45000000))
                  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 \ge 0x4400) && (cache_temp \le 0x4500))
                  pentiumxeon_flag = 1;
         cache temp = cache ebx & 0xFF000000;
         if (cache temp == 0x40000000)
                  celeron flag = 1;
         if ((cache temp \ge 0x44000000) && (cache temp \le 0x45000000))
                  pentiumxeon flag = 1;
         cache temp = cache ebx & 0xFF0000;
         if (cache temp == 0x400000)
                  celeron flag = 1;
         if ((cache temp \ge 0x440000) && (cache temp \le 0x450000))
                  pentiumxeon flag = 1;
         cache_temp = cache_ebx & 0xFF00;
         if (cache temp == 0x4000)
                  celeron flag = 1;
         if ((cache temp \ge 0x4400) && (cache temp \le 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))
```



```
cache temp = cache ecx & 0xFF000000;
                                     if (cache temp == 0x40000000)
                                              celeron flag = 1;
                                     if ((cache temp \geq 0x44000000) && (cache temp \leq 0x45000000))
                                              pentiumxeon flag = 1;
                                     cache_temp = cache_ecx & 0xFF0000;
                                     if (cache temp == 0x400000)
                                              celeron flag = 1;
                                     if ((cache_temp \ge 0x440000) && (cache_temp \le 0x450000))
                                              pentiumxeon_flag = 1;
                                     cache temp = cache ecx & 0xFF00;
                                     if (cache temp == 0x4000)
                                              celeron flag = 1;
                                     if ((cache_temp \ge 0x4400) && (cache_temp \le 0x4500))
                                              pentiumxeon flag = 1;
                                     cache temp = cache ecx & 0xFF;
                                     if (cache_temp == 0x40)
                                              celeron_flag = 1;
                                     if ((cache temp \ge 0x44) && (cache temp \le 0x45))
                                              pentiumxeon flag = 1;
                                     cache_temp = cache_edx & 0xFF000000;
                                     if (cache_temp == 0x40000000)
                                              celeron flag = 1;
                                     if ((cache temp \geq 0x44000000) && (cache temp \leq 0x45000000))
                                              pentiumxeon flag = 1;
                                     cache temp = cache edx & 0xFF0000;
                                     if (cache temp == 0x400000)
                                              celeron_flag = 1;
                                     if ((cache temp \ge 0x440000) && (cache temp \le 0x450000))
                                              pentiumxeon_flag = 1;
                                     cache temp = cache_edx & 0xFF00;
                                     if (cache temp == 0x4000)
                                              celeron flag = 1;
                                     if ((cache temp \ge 0x4400) && (cache temp \le 0x4500))
                                              pentiumxeon flag = 1;
                                     cache temp = cache edx & 0xFF;
                                     if (cache temp == 0x40)
                                              celeron_flag = 1;
                                     if ((cache_temp >= 0x44) && (cache_temp <= 0x45))
                                              pentiumxeon flag = 1;
                                     if (celeron flag == 1)
                                              printf(" Genuine Intel Celeron(R) processor, model 5");
                                     else
                                              if (pentium xeon 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,");
```

pentiumxeon\_flag = 1;



```
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);
if (((cpu signature>>4) & 0xf) == 0xf)
         printf("\n Extended Model: %x",(cpu signature>>16)&0xf);
printf("\nStepping:
                       %X\n", cpu signature&0xf);
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 FLAG)
                                     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("\n The processor supports Pending Break Event signaling");
                            if (features ecx & EST FLAG)
                                     printf("\n The processor supports Enhanced SpeedStep(R) 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");
                  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-2003
// 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.
#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 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 sSupportSSE2[80]="Don't know";
          char sSupportSSE[80]="Don't know";
          char sSupportMMX[80]="Don't know";
          // To identify whether 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 SSE2 instruction would cause an exception, if no exception occurs,
                                         SSE2 is supported; if exception occurs,
          // 3. Test if executing an SSE instruction would cause an exception, if no exception occurs,
          //
                                         SSE is supported; if exception occurs,
          // 4. 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 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");
                  // Second, execute an SSE instruction to see whether an exception occurs
                              asm {
                                     orps xmm1, xmm2
                                                                          // this is an instruction available in SSE
                            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");
                  // Third, execute an MMX instruction to see whether an exception occurs
                  {
                              asm {
                                                                          // this is an instruction available in MMX
                                     emms
technology
                                                                          // no exception executing an MMX instruction
                            strcpy(&sSupportMMX[0], "Yes");
                    except( EXCEPTION_EXECUTE_HANDLER )
                                                                          // MMX exception handler
                            // exception occurred when executing an MMX instruction
                            strcpy(&sSupportMMX[0], "No");
         }
         printf("This Processor supports the following instruction extensions: \n");
         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-2003

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.
- Step 3. Ensure that the FXSR feature flag (EDX bit 24) is set.

  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 DB DB DB DB DB | "This is not an Genuine Intel processor.", 0Dh, 0Ah, "\$" "Neither SSE or SSE2 extensions are supported.", 0Dh, 0Ah, "\$" "FXSAVE not supported.", 0Dh, 0Ah, "\$" "DAZ bit in MXCSR_MASK is zero (clear).", 0Dh, 0Ah, "\$" "DAZ mode not supported.", 0Dh, 0Ah, "\$" "DAZ mode supported.", 0Dh, 0Ah, "\$" |
|                                                                                   |                   |                                                                                                                                                                                                                                                                                                            |

# ; Code segment

.CODE .686p .XMM

# dazdtect PROC NEAR

mov

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

```
cpuid
                                        ; Compare first 4 letters of Vendor ID
cmp
          ebx, 'uneG'
          notIntelprocessor
                                        ; Jump if not Genuine Intel processor
jne
          edx, 'Ieni'
                                        ; Compare next 4 letters of Vendor ID
cmp
          notIntelprocessor
                                        ; Jump if not Genuine Intel processor
jne
                                        ; Compare last 4 letters of Vendor ID
cmp
          ecx, 'letn'
          not Intel processor \\
                                        ; Jump if not Genuine Intel processor
jne
```

### ; Step 2, 3, and 4

; Get CPU feature flags : Verify EXSAVE and eitl

eax, 0

; Verify FXSAVE and either SSE or

; SSE2 are supported

mov eax, 1 cpuid edx, 24t ; Feature Flags Bit 24 is FXSAVE support bt inc noFxsave ; jump if FXSAVE not supported ; Feature Flags Bit 25 is SSE support bt edx, 25t sse or sse2 supported ; jump if SSE is not supported jc ; Feature Flags Bit 26 is SSE2 support bt edx, 26t ; jump if SSE2 is not supported  $no\_sse\_sse2$ jnc



```
sse_or_sse2_supported:
         ; FXSAVE requires a 16-byte aligned
         ; buffer so get offset into buffer
                  bx, OFFSET buffer
                                             ; Get offset of the buffer into bx
         mov
                  bx, 0FFF0h
         and
                  bx, 16t
                                             ; DI is aligned at 16-byte boundary
         add
; Step 5.
         ; Clear the buffer that will be
         ; used for FXSAVE data
                  ds
         push
                  es
         pop
         mov
                  di, bx
                  ax, ax
         xor
                  cx, 512/2
         mov
         cld
         rep
                  stosw
                                             ; Fill at FXSAVE buffer with zeroes
; Step 6.
         fxsave
                  [bx]
; Step 7.
         mov
                  eax, DWORD PTR [bx][28t]; Get MXCSR_MASK
         cmp
                                             ; Check for valid mask
         jne
                  check mxcsr mask
                  eax, 0FFBFh
                                             ; Force use of default MXCSR MASK
         mov
check mxcsr mask:
; EAX contains MXCSR_MASK from FXSAVE buffer or default mask
; Step 8.
         bt
                  eax, 6t
                                             ; MXCSR MASK Bit 6 is DAZ support
                  supported
                                             ; Jump if DAZ supported
         jc
         mov
                  dx, OFFSET daz mask clear
                  notSupported
        jmp
supported:
         mov
                  dx, OFFSET supports daz
                                             ; Indicate DAZ is supported.
        jmp
                  print
notIntelProcessor:
                  dx, OFFSET not intel
                                             ; Assume not an Intel processor
         mov
                  print
        jmp
no_sse_sse2:
         mov
                  dx, OFFSET noSSEorSSE2 ; Setup error message assuming no SSE/SSE2
        jmp
                  notSupported
noFxsave:
                  dx, OFFSET no FXSAVE
         mov
notSupported:
```

ah, 09h

mov

; Execute DOS print string function

# **AP-485**



21h int

mov  $dx, OFFSET \ no\_daz$ 

print:

ah, 09h mov int

21h

; Execute DOS print string function

exit:

.exit

; Allow assembler to generate code ; that returns control to DOS

ret

dazdtect ENDP

**END** 



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

Filename: FREQUENC.ASM

Copyright (c) Intel Corporation 2001-2003

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 and tested on a system with a Pentium(r) II processor, a system with a Pentium(r) III 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.

- 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 to load the EDX register with the feature flags.
- Step 3. Ensure that the TSC feature flag (EDX bit 4) is set. This indicates the processor supports the Time-Stamp Counter and RDTSC instruction.
- Step 4. Read the TSC at the beginning of the reference period
- ; Step 5. Read the TSC at the end of the reference period.
- Step 6. Compute the TSC delta from the beginning and ending of the reference period.
- Step 7. Compute the actual frequency by dividing the TSC delta by the reference period.

.DOSSEG .MODEL small, pascal .STACK ;4096

wordToDec PROTO NEAR PASCAL decAddr:WORD, hexData:WORD

·-----

; Macro printst

This macro is used to print a string passed as an input parameter and a word value immediately after the string.



```
The string is delared in the data segment routine during
         assembly time. The word is converted to dec ascii and
         printed after the string.
         stringData = string to be printed.
         wordData = word to be converted to dec ascii and printed
; Destroys: None
Output: None
; Assumes: Stack is available
printst MACRO
                  stringdata, hexWord
         local
                  stringlabel, decData
         .data
stringlabel
                  DB
                            stringdata
decData
                  DB
                            5 dup (0)
                  DB
                            0dh, 0ah, '$'
         .code
         pushf
         pusha
         ; Convert the word ino hex ascii and store in the string
         invoke wordToDec, offset decData, hexWord
                  dx, offset stringlabel
                                                        ; Setup string to be printed
         mov
                  ah, 09h
                                                        ; Execute DOS print function
         mov
                  21h
         int
         popa
         popf
ENDM
SEG BIOS DATA AREA EQU
                                     40h
OFFSET_TICK_COUNT INTERVAL_IN_TICKS
                            EQU
                                     6ch
                            EQU
                                     91
; Data segment
         .DATA
; Code segment
         .CODE
         .686p
cpufreq PROC NEAR
                  tscLoDword:DWORD, \
         local
                  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
                  eax, 0
         cpuid
                  ebx, 'uneG'
                                                         ; Check VendorID = GenuineIntel
         cmp
         jne
                  exit
                                                         ; Jump if not Genuine Intel processor
         cmp
                  edx, 'Ieni'
         jne
                  exit
                  ecx, 'letn'
         cmp
         jne
                  exit
; Step 2 and 3
         ; Get CPU feature flags
         ; Verify TSC is supported
         mov
                  eax, 1
         cpuid
         bt
                  edx, 4t
                                                         ; Flags Bit 4 is TSC support
         inc
                  exit
                                                         ; jump if TSC not supported
                  SEG_BIOS_DATA_AREA
         push
         pop
                  si, OFFSET_TICK_COUNT
                                                         ; The BIOS tick count updateds
         mov
                  ebx, DWORD PTR es:[si]
                                                         ; \sim 18.2 times per second.
         mov
wait_for_new_tick:
                   ebx, DWORD PTR es:[si]
                                                         ; Wait for tick count change
         cmp
                   wait for new tick
         je
; Step 4
         ; **Timed interval starts**
         ; Read CPU time-stamp
         rdtsc
                                                         ; Read and save TSC immediately
                   tscLoDword, eax
         mov
                                                         ; after a tick
                  tscHiDword, edx
         mov
         add
                  ebx, INTERVAL IN TICKS + 1
                                                         ; Set time delay value ticks.
wait_for_elapsed_ticks:
                  ebx, DWORD PTR es:[si]
                                                         ; Have we hit the delay?
         cmp
         jne
                   wait_for_elapsed_ticks
; Step 5
         ; **Time interval ends**
         ; Read CPU time-stamp immediatly after tick delay reached.
         rdtsc
```



```
; Step 6
                  eax, tscLoDword
                                                        ; Calculate TSC delta from
         sub
         sbb
                  edx, tscHiDword
                                                        ; beginning to end of interval
; Step 7
         54945 = (1/18.2) * 1,000,000 This adjusts for MHz.
         ; 54945*INTERVAL_IN_TICKS adjusts for number of ticks in interval
                  ebx, 54945*INTERVAL_IN_TICKS
         mov
         div
         ; ax contains measured speed in MHz
                  mhz, ax
         mov
         ; Find nearest full/half multiple of 66/133 MHz
                  dx, dx
         xor
         mov
                  ax, mhz
         mov
                  bx, 3t
         mul
                  bx
                  ax, 100t
         add
                  bx, 200t
         mov
         div
                  bx
         mul
                  bx
                  dx, dx
         xor
                  bx, 3
         mov
         div
                  bx
         ; ax contains nearest full/half multiple of 66/100 MHz
                  Nearest66Mhz, ax
         mov
                  ax, mhz
         sub
                  delta66
         jge
                                                        ; ax = abs(ax)
                  ax
         neg
delta66:
         ; ax contains delta between actual and nearest 66/133 multiple
                   Delta66Mhz, ax
         ; Find nearest full/half multiple of 100 MHz
         xor
                  dx, dx
         mov
                  ax, mhz
                  ax, 25t
         add
                  bx, 50t
         mov
         div
                  bx
         mul
         ; ax contains nearest full/half multiple of 100 MHz
                  Nearest50Mhz, ax
         mov
                  ax, mhz
         sub
         jge
                  delta50
         neg
                  ax
                                                        ; ax = abs(ax)
delta50:
         ; ax contains delta between actual and nearest 50/100 MHz multiple
                  bx, Nearest50Mhz
         mov
                  ax, Delta66Mhz
         cmp
```



```
useNearest50Mhz
                  bx, Nearest66Mhz
         mov
         ; Correction for 666 MHz (should be reported as 667 MHZ)
         ine
                  correct666
         inc
                  bx
correct666:
useNearest50MHz:
         ; bx contains nearest full/half multiple of 66/100/133~\text{MHz}
         printst "Reported MHz = \sim", bx
         printst "Measured MHz = ", mhz
                                                        ; print decimal value
exit:
         .exit
                                                        ; returns control to DOS
         ret
cpufreq ENDP
; Procedure
                  wordToDec
         This routine will convert a word value into a 5 byte decimal
         ascii string.
; Input: decAddr = address to 5 byte location for converted string
                       (near address assumes DS as segment)
         hexData = word value to be converted to hex ascii
; Destroys: ax, bx, cx
                  5 byte converted hex string
Output:
; Assumes:
                  Stack is available
wordToDec PROC NEAR PUBLIC uses es,
                  decAddr:WORD, hexData:WORD
         pusha
                   di, decAddr
         mov
         push
                   @data
                                                        ; ES:DI -> 5-byte converted string
         pop
                  es
                  ax, hexData
         mov
                  dx, dx
         xor
                  bx, 10000t
         mov
         div
                  bx
         add
                  ax, 30h
         stosb
         mov
                  ax, dx
         xor
                  dx, dx
                  bx, 1000t
         mov
         div
                  bx
                  ax, 30h
         add
         stosb
```



ax, dx dx, dx bx, 100t mov xor mov div bx ax, 30h add stosb ax, dx dx, dx bx, 10t mov xor mov div bxax, 30h add stosb ax, dx ax, 30h mov add stosb popa ret word To Dec**ENDP** END