# **AP-485**

# APPLICATION NOTE



# Intel Processor Identification and the CPUID Instruction

December 1998

Order Number: 241618-011

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's Intel Architecture processors (e.g., Pentium® processor, Pentium processor with MMX™ technology, Pentium Pro processor, Pentium II processor, Pentium II Xeon™ processor and Intel® Celeron™ processor) 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 by calling 1-800-548-4725 or by visiting Intel's website at http://www.intel.com

Copyright © Intel Corporation 1998.

\* Third-party brands and names are the property of their respective owners.



# **CONTENTS**

|                                                                | PAGE |
|----------------------------------------------------------------|------|
| 1.0. INTRODUCTION                                              |      |
| 1.1. Update Support                                            | 6    |
| 2.0. DETECTING THE CPUID INSTRUCTION                           | 6    |
| 3.0. OUTPUT OF THE CPUID INSTRUCTION                           | 7    |
| 3.1. Vendor ID String                                          | 7    |
| 3.2. Processor Signature                                       | 8    |
| 3.3. Feature Flags                                             | 12   |
| 3.4. Cache Size and Format Information                         |      |
| 3.5. SYSENTER/SYSEXIT – SEP Features Bit                       | 14   |
| 3.6. Pentium® Pro Processor Output Example                     |      |
| 3.7. Pentium <sup>®</sup> II Processor, Model 3 Output Example | 15   |
| 4.0. USAGE GUIDELINES                                          | 16   |
| 5.0. PROPER IDENTIFICATION SEQUENCE                            | 16   |
| 6.0. USAGE PROGRAM EXAMPLES                                    | 19   |



# **REVISION HISTORY**

| Revision | Revision History                                                                                                                                                                                                                                                                                             | Date  |  |  |
|----------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------|--|--|
| -001     | Original Issue.                                                                                                                                                                                                                                                                                              | 05/93 |  |  |
| -002     | Modified Table 2, Intel486™ and Pentium® Processor Signatures.                                                                                                                                                                                                                                               |       |  |  |
| 003      | Updated to accommodate new processor versions. Program examples modified for ease of use, section added discussing BIOS recognition for OverDrive® processors, and feature flag information updated.                                                                                                         |       |  |  |
| -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 <sup>™</sup> technology. Modified Figure 6.0.  |       |  |  |
| -006     | Modified Table 3. Added reserved for future member of P6 family of processors entry. Modified table header to reflect Pentium II processor family. Modified Table 5. Added SEP bit definition. Added Section 3.5. Added Section 3.7 and Table 9. Corrected references of P6 family to reflect correct usage. | 3/97  |  |  |
|          | Modified cpuid3a.asm, cpuid3b.asm and cpuid3.c example code sections to check for SEP feature bit and to check for, and identify, the Pentium II processor. Added additional disclaimer related to designers and errata.                                                                                     |       |  |  |
| - 007    | Modified Table 2. Added Pentium II processor, model 5 entry. Modified 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.                                               | 1/98  |  |  |
|          | Removed the note "Do not assume a value of 1 in a feature flag indicates that a given feature is present. For future feature flags, a value of 1 may indicate that the specific feature is not present" in section 4.0.                                                                                      |       |  |  |
|          | Modified cpuid3b.asm and cpuid3.c example code section to check for, and identify, the Pentium II processor, model 5. Modified existing Pentium II processor code to print Pentium II processor, model 3.                                                                                                    |       |  |  |
| - 008    | Added note to identify Intel 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.                                                                 | 4/98  |  |  |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Intel Celeron processor.                                                                                                                                                                                                      |       |  |  |
| -009     | Added note to identify Pentium II Xeon™ processor in section 3.2. Modified Table 2. Added Pentium II Xeon processor entry.                                                                                                                                                                                   | 6/98  |  |  |
|          | Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Pentium II Xeon processor.                                                                                                                                                                                                    |       |  |  |
| -010     |                                                                                                                                                                                                                                                                                                              |       |  |  |



| -011 | Modified Table 2. Added Intel Celeron processor, model 6 entry.                                                  | 12/98 |
|------|------------------------------------------------------------------------------------------------------------------|-------|
|      | Modified cpuid3b.asm and cpuid3.c example code to check for, and identify, the Intel Celeron processor, model 6. |       |



### 1.0. 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 OverDrive processors with MMX technology, Pentium Pro processors, Pentium II processors, Pentium II processors, Pentium II processors, Pentium II processors, it is essential that Intel provide an increasingly sophisticated means with which software can identify the features available on each processor. This identification mechanism has evolved in conjunction with the Intel Architecture as follows:

- 1 Originally, Intel published code sequences that could detect minor implementation or architectural differences to identify processor generations.
- 2 Later, with the advent of the Intel386 processor, Intel implemented processor signature identification which 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.

# 2.0. DETECTING THE CPUID INSTRUCTION

Starting with the Intel486 family and subsequent Intel processors, Intel provides 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

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.

## **Footnotes**

Only in some Intel486™ and succeeding processors. Bit 21 in the Intel386™ 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.



# 3.0. OUTPUT OF THE CPUID INSTRUCTION

Figure 2 summarizes the outputs of the CPUID instruction. The function of 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 operation, the program should set the EAX register parameter value to "0" and then execute the CPUID instruction as follows

MOV EAX, 00H CPUID

Table 1. Effects of EAX Contents on CPUID Instruction Output

| Parameter               | Outputs of CPUID                                                                                                           |  |  |
|-------------------------|----------------------------------------------------------------------------------------------------------------------------|--|--|
| EAX = 0                 | $\label{eq:expectation} \begin{split} EAX \leftarrow Highest \ value \ recognized \\ by \ CPUID \ instruction \end{split}$ |  |  |
|                         | EBX:EDX:ECX ← Vendor identification string                                                                                 |  |  |
| EAX = 1                 | $EAX \leftarrow Processor \ signature$                                                                                     |  |  |
|                         | EDX ← Feature flags                                                                                                        |  |  |
|                         | $\begin{array}{l} EBX : ECX \leftarrow Intel \ reserved \\ (Do \ not \ use.) \end{array}$                                  |  |  |
| EAX = 2                 | $\label{eq:expectation} \begin{split} EAX.EBX.ECX.EDX \leftarrow Processor \\ configuration \ parameters \end{split}$      |  |  |
| 3 ≤ EAX ≤ highest value | Intel reserved                                                                                                             |  |  |
| EAX > highest value     | $\begin{array}{l} EAX.EBX.ECX.EDX \leftarrow Undefined \\ (Do \ not \ use.) \end{array}$                                   |  |  |

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. On current and future IA-32 processors, bit 31 in the EAX register will be clear when CPUID is called with an input parameter greater then highest value. 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

# 3.2. Processor Signature

Beginning with the Intel486 processor family, the processor will return a processor identification signature value after reset in the EDX register (see Figure 3).



Figure 3. EDX Register Value after RESET



Processors that implement the CPUID instruction also return the 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 signature for the Intel486, Pentium, Pentium Pro, Pentium II processors, Pentium II Xeon processors and Intel Celeron 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 3 shows the values returned in the EAX register currently defined for these processors. (The high-order 18 bits are undefined and reserved.)

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 family values, specified in bit positions 8 through 11, indicates whether the processor belongs to the Intel386, Intel486, Pentium or P6 family of processors.

The model number, specified in bits 4 though 7, indicates the processor's family model number, while the stepping number in bits 0 through 3 indicates the revision number of that model.

The Pentium II processor, model 5, the Pentium II Xeon processor and the Intel Celeron processor, model 5 share the same family and model number. To differentiate between the processors, software should check the cache descriptor values through executing CPUID instruction with EAX = 2. If no L2 cache is returned, the processor

is identified as an Intel 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.

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

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

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

### **Footnotes**

All Intel486 SL-enhanced and Write-Back enhanced processors are capable of executing the CPUID instruction. See Table 3.



Table 3. Intel486™, Pentium® Processor Family, OverDrive®, Pentium Pro Processor, Pentium II Xeon Processor and Intel Celeron Processor Signatures

| Туре   | Family | Model         | Stepping | Description                                                                                                |  |
|--------|--------|---------------|----------|------------------------------------------------------------------------------------------------------------|--|
| 00     | 0100   | 0000 and 0001 | xxxx (1) | Intel486™ DX processors                                                                                    |  |
| 00     | 0100   | 0010          | xxxx (1) | Intel486 SX processors                                                                                     |  |
| 00     | 0100   | 0011          | xxxx (1) | Intel487™ processors                                                                                       |  |
| 00     | 0100   | 0011          | xxxx (1) | IntelDX2™ processors                                                                                       |  |
| 00     | 0100   | 0011          | xxxx (1) | IntelDX2 OverDrive® processors                                                                             |  |
| 00     | 0100   | 0100          | XXXX (3) | Intel486 SL processor                                                                                      |  |
| 00     | 0100   | 0101          | xxxx (1) | IntelSX2™ processors                                                                                       |  |
| 00     | 0100   | 0111          | XXXX (3) | Write-Back Enhanced IntelDX2 processors                                                                    |  |
| 00     | 0100   | 1000          | XXXX (3) | IntelDX4™ processors                                                                                       |  |
| 00, 01 | 0100   | 1000          | XXXX (3) | IntelDX4 OverDrive processors                                                                              |  |
| 00     | 0101   | 0001          | XXXX (2) | Pentium® processors (60, 66)                                                                               |  |
| 00     | 0101   | 0010          | XXXX (2) | Pentium processors (75, 90, 100, 120, 133, 150, 166, 200)                                                  |  |
| 01 (4) | 0101   | 0001          | XXXX (2) | Pentium OverDrive processor for Pentium processor (60, 66)                                                 |  |
| 01 (4) | 0101   | 0010          | XXXX (2) | Pentium OverDrive processor for Pentium processor (75, 90, 100, 120, 133)                                  |  |
| 01     | 0101   | 0011          | xxxx (2) | Pentium OverDrive processors for Intel486 processor-<br>based systems                                      |  |
| 00     | 0101   | 0100          | XXXX (2) | Pentium processor with MMX™ technology (166, 200)                                                          |  |
| 01     | 0101   | 0100          | XXXX (2) | Pentium OverDrive processor with MMX <sup>™</sup> technology for Pentium processor (75, 90, 100, 120, 133) |  |
| 00     | 0110   | 0001          | XXXX (2) | Pentium Pro processor                                                                                      |  |
| 00     | 0110   | 0011          | xxxx (2) | Pentium II processor, model 3                                                                              |  |
| 00     | 0110   | 0101(5)       | xxxx (2) | Pentium II processor, model 5, Pentium II Xeon processor and Intel Celeron processor, model 5              |  |
| 00     | 0110   | 0110          | XXXX (2) | Intel Celeron processor, model 6                                                                           |  |
| 01     | 0110   | 0011          | xxxx (2) | Reserved for a future OverDrive processor for Pentium Pro processor                                        |  |



#### NOTES:

- 1. This processor does not implement the CPUID instruction.
- 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 243776) or the Intel Celeron Processor Specification Update (Order Number 243748) 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 errata 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.



# 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 register with the feature flags. The current feature flags (when 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 predetermine processor features, your software can detect and avoid incompatibilities.

# 3.4. Cache Size and Format 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 processor's cache 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 Pro 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.



Figure 4. Processor Signature Format on Intel386™ Processors

| Туре | 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 <sup>®</sup> coprocessor |

Table 4. Intel386™ Processor Signatures



Table 5. Feature Flag Values

|         |                                | 1                                           | reature riag values                                                                                                                                                                                                                                                                         |  |
|---------|--------------------------------|---------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
| Bit     | Name                           | Description when<br>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 Supported              | The compare and exchange 8 bytes instruction is supported.                                                                                                                                                                                                                                  |  |
| 9       | APIC                           | On-chip APIC Hardware Supported (1)         | The processor contains a 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.5 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 PDEs and PTEs and the CR4.PGE enable bit are supported.                                                                                                                                                                                                               |  |
| 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.                                                                                                                                                                |  |
| 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 a 4K granularity through a linear address.                                                |  |
| 17      | PSE-36                         | 36-bit Page Size<br>Extension               | Indicates whether the processor supports 4-Mbyte pages the 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 – 22 |                                | Reserved                                    | Do not count on their value.                                                                                                                                                                                                                                                                |  |
| 23      | MMX <sup>™</sup><br>Technology | 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. |  |



Table 5. Feature Flag Values

| Bit              | Bit Name Description when Flag = 1 |          | Comments                     |
|------------------|------------------------------------|----------|------------------------------|
| 25 – 31 Reserved |                                    | Reserved | Do not count on their value. |

#### NOTE:

1. The processor contains a software-accessible Local APIC.

The remainder of the EAX register, and the EBX, ECX, and EDX registers, contain valid 8 bit descriptors. Table 6 shows that a most significant bit of zero indicates a valid 8-bit descriptor. To decode descriptors, move sequentially from the most significant byte of the register down through the least significant byte of the register. Table 7 lists the current descriptor values and their respective cache characteristics. This list will be extended in the future as necessary.

Table 6. Descriptor Formats

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

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

The presence of this facility is indicated by the SYSENTER Present (SEP) bit 11 of CPUID. 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 (Family == 6) AND (Model < 3) AND

(Stepping < 3) {

THEN

Fast System Call is NOT supported

}

ELSE Fast System Call is supported
```

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

# 3.6. Pentium® Pro Processor Output Example

The Pentium Pro processor returns the values shown in Table 8. As the value of AL=1, it is valid to interpret the remainder of the registers according to Table 7. Table 8 also shows that the MSB of the EAX register is 0. This indicates that the upper 8 bits constitute an 8 bit descriptor. The remaining register values in Table 8 show that the Pentium Pro processor has the following cache characteristics:

- A data TLB that maps 4K pages, is 4 way set associative, and has 64 entries.
- An instruction TLB that maps 4M pages, is fully associative, and has 2 entries.
- An instruction TLB that maps 4K pages, is 4 way set associative, and has 32 entries.
- An instruction cache that is 8K, is 4 way set associative, and has a 32 byte line size.
- A data TLB that maps 4M pages, is 4 way set associative, and has 8 entries.
- A data cache that is 8K, is 2 way set associative, and has a 32 byte line size.

A unified cache that is 256K, is 4 way set associative, and has a 32 byte line size.



| Descriptor<br>Value | Cache Description                                                |
|---------------------|------------------------------------------------------------------|
| 00h                 | Null                                                             |
| 01h                 | Instruction TLB, 4K pages, 4-way set associative, 32 entries     |
| 02h                 | Instruction TLB, 4M pages, fully associative, 2 entries          |
| 03h                 | Data TLB, 4K pages, 4-way set associative, 64 entries            |
| 04h                 | Data TLB, 4M pages, 4-way set associative, 8 entries             |
| 06h                 | Instruction cache, 8K, 4-way set associative, 32 byte line size  |
| 08h                 | 16KB instruction cache, 4-way set associative, 32 byte line size |
| 0Ah                 | Data cache, 8K, 2-way set associative, 32 byte line size         |
| 0Ch                 | 16KB data cache, 4-way set associative, 32 byte line size        |
| 40h                 | No L2 cache                                                      |
| 41h                 | Unified cache, 32 byte cache line,4-way set associative, 128K    |
| 42h                 | Unified cache, 32 byte cache line, 4-way set associative, 256K   |
| 43h                 | Unified cache, 32 byte cache line, 4-way set associative, 512K   |
| 44h                 | Unified cache, 32 byte cache line, 4-way set associative, 1M     |
| 45h                 | Unified cache, 32 byte cache line, 4-way set associative, 2M     |

**Table 7. Descriptor Decode Values** 

Table 8. Pentium® Pro Processor, with 256K L2 Cache, CPUID (EAX=2) Example Return Values

|     | 31  | 23  | 15  | 7 0 |
|-----|-----|-----|-----|-----|
| EAX | 03h | 02h | 01h | 01h |
| EBX | 0   | 0   | 0   | 0   |
| ECX | 0   | 0   | 0   | 0   |
| EDX | 06h | 04h | 0Ah | 42h |

# 3.7. Pentium® II Processor, model 3 Output Example

The Pentium II processor, model 3 returns the values shown in Table 9. If the value of AL=1, it is valid to interpret the remainder of the registers according to Table 7. Table 9 also shows the MSB of EAX register is 0. As with the Pentium Pro processor this indicates the upper 8 bits constitute an 8 bit descriptor. The remaining register values in Table 9 shows the Pentium II processor has the following cache characteristics:

 A data TLB that maps 4K pages, is 4 way set associative, and has 64 entries.

- An instruction TLB that maps 4M pages, is fully associative, and has 2 entries.
- An instruction TLB that maps 4K pages, is 4 way set associative, and has 32 entries.
- A data cache that is 16K, is 4 way set associative, and has a 32 byte line size.
- A data TLB that maps 4M pages, is 4 way set associative, and has 8 entries.
- An instruction cache that is 16K, is 4 way set associative, and has a 32 byte line size.
- A unified cache that is 512K, is 4 way set associative, and has a 32 byte line size.



|     | 31  | 23  | 15  | 7 0 |
|-----|-----|-----|-----|-----|
| EAX | 03h | 02h | 01h | 01h |
| EBX | 0   | 0   | 0   | 0   |
| ECX | 0   | 0   | 0   | 0   |
| EDX | 0Ch | 04h | 08h | 43h |

### 4.0. USAGE GUIDELINES

This document presents Intel-recommended featuredetection 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 5.0.
- 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.
- 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.

# 5.0. PROPER IDENTIFICATION SEQUENCE

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 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 floatingpoint unit (FPU) or math coprocessor (MCP).

This procedure has been tested with 8086, 80286, Intel386, Intel486, Pentium processor, Pentium processor

with MMX technology, OverDrive processor with MMX technology, Pentium Pro processors, Pentium II processors, Pentium II Xeon processors and Intel Celeron 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





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



#### Example 1. Processor Identification Extraction Procedure

Filename: cpuid3a.asm Copyright 1993, 1994, 1995, 1996, 1997, 1998 by Intel Corp. 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 \_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 and Intel Celeron processor in the real-address mode only. 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

20

.386



.386

```
.model flat
        comment this line for 32-bit segments
        .model small
CPU_IDMACRO
        db
                                        ; Hardcoded CPUID instruction
        db
                0a2h
ENDM
        .data
        public
                _cpu_type
        public
               _fpu_type
        public
               _v86_flag
        public
               _cpuid_flag
        public
               _intel_CPU
        public
               _vendor_id
               _cpu_signature
        public
               _features_ecx
        public
        public _features_edx
               _features_ebx
        public
        public _cache_eax
        public _cache_ebx
        public _cache_ecx
        public _cache_edx
        public _sep_flag
                                        0
        _cpu_type
                                db
                                        0
        _fpu_type
                                db
        _v86_flag
                                db
                                        0
        _cpuid_flag
                                        0
                                db
        _intel_CPU
                                db
                                        0
        _sep_flag
                                        0
                                db
                                db
        _vendor_id
                                db
                                         "GenuineIntel"
        intel_id
        _cpu_signature
                                dd
                                        0
        _features_ecx
                                dd
                                        0
                                dd
                                        0
        _features_edx
                                        0
        _features_ebx
                                dd
                                        0
        _cache_eax
                                dd
                                        0
        _cache_ebx
                                dd
        _cache_ecx
                                dd
                                        0
        _cache_edx
                                        0
                                dd
                                        0
        fp_status
                                dw
        .code
        comment this line for 32-bit segments
        .8086
        uncomment this line for 32-bit segments
```



```
public _get_cpu_type
        _get_cpu_type proc
        This procedure determines the type of processor in a system
        and sets the _cpu_type variable with the appropriate
        value. If the CPUID instruction is available, it is used
        to determine more specific details about the processor.
        All registers are used by this procedure, none are preserved.
        To avoid AC faults, the AM bit in CR0 must not be set.
        Intel 8086 processor check
        Bits 12-15 of the FLAGS register are always set on the
        8086 processor.
        For 32-bit segments comment the following lines down to the next
        comment line that says "STOP"
check 8086:
                                                 ; push original FLAGS
        pushf
                                                 ; get original FLAGS
        pop
                ax
                cx. ax
                                                 ; save original FLAGS
        mov
                ax, 0fffh
                                                 ; clear bits 12-15 in FLAGS
        and
        push
                                                 ; save new FLAGS value on stack
        popf
                                                 ; replace current FLAGS value
        pushf
                                                 ; get new FLAGS
        pop
                ax
                                                 ; store new FLAGS in AX
                ax. 0f000h
                                                 ; if bits 12-15 are set, then
        and
                                                 ; processor is an 8086/8088
        cmp
                ax, 0f000h
                _cpu_type, 0
                                                 ; turn on 8086/8088 flag
        mov
        ine
                check 80286
                                                 ; go check for 80286
                                                 ; double check with push sp
        push
                sp
                dx
                                                 ; if value pushed was different
        pop
        cmp
                dx, sp
                                                 ; means it's really not an 8086
                                                 ; jump if processor is 8086/8088
        jne
                end_cpu_type
                _cpu_type, 10h
                                                 ; indicate unknown processor
        mov
        imp
                end_cpu_type
        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:
        smsw
                                                 ; save machine status word
                ax
                                                 : isolate PE bit of MSW
        and
                ax. 1
                _v86_flag, al
                                                 ; save PE bit to indicate V86
        mov
        or
                cx, 0f000h
                                                 ; try to set bits 12-15
                                                 ; save new FLAGS value on stack
        push
                CX
        popf
                                                 ; replace current FLAGS value
        pushf
                                                 ; get new FLAGS
                                                 ; store new FLAGS in AX
        pop
        and
                ax, 0f000h
                                                 ; if bits 12-15 are clear
                _cpu_type, 2
        mov
                                                 ; processor=80286, turn on 80286 flag
                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
        This bit cannot be set on the Intel386 processor.
        "STOP"
                                                   ; it is safe to use 386 instructions
check_80386:
        pushfd
                                                   ; push original EFLAGS
        pop
                 eax
                                                   ; get original EFLAGS
        mov
                 ecx, eax
                                                   ; save original EFLAGS
                 eax, 40000h
                                                   ; flip AC bit in EFLAGS
        xor
                                                   ; save new EFLAGS value on stack
        push
                 eax
        popfd
                                                   ; replace current EFLAGS value
        pushfd
                                                   ; get new EFLAGS
                                                   ; store new EFLAGS in EAX
        pop
                 eax
                 eax, ecx
                                                   ; can't toggle AC bit, processor=80386
        xor
                                                   ; turn on 80386 processor flag
        mov
                 _cpu_type, 3
                                                   ; jump if 80386 processor
        jΖ
                 end_cpu_type
        push
        popfd
                                                   ; restore AC bit in EFLAGS first
        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:
        mov
                 _cpu_type, 4
                                                   ; turn on 80486 processor flag
                                                   ; get original EFLAGS
        mov
                 eax, ecx
                 eax, 200000h
                                                   ; flip ID bit in EFLAGS
        xor
        push
                                                   ; save new EFLAGS value on stack
                                                   ; replace current EFLAGS value
        popfd
        pushfd
                                                   ; get new EFLAGS
                                                   ; store new EFLAGS in EAX
        pop
                 eax
        xor
                 eax, ecx
                                                   ; can't toggle ID bit,
                                                   ; processor=80486
        je
                 end_cpu_type
        Execute CPUID instruction to not determine vendor, family,
        model, stepping and features. For the purpose of this
;
        code, only the initial set of CPUID information is saved.
        mov
                 _cpuid_flag, 1
                                                   ; flag indicating use of CPUID inst.
        push
                 ebx
                                                   ; save registers
        push
                 esi
        push
                 edi
                 eax, 0
                                                   ; set up for CPUID instruction
        mov
        CPU_ID
                                                   ; get and save vendor ID
        mov
                 dword ptr _vendor_id, ebx
        mov
                 dword ptr _vendor_id[+4], edx
```



```
mov
                dword ptr _vendor_id[+8], ecx
       cmp
                dword ptr intel_id, ebx
       jne
                end_cpuid_type
                dword ptr intel_id[+4], edx
       cmp
                end_cpuid_type
       jne
       cmp
                dword ptr intel_id[+8], ecx
       ine
                end_cpuid_type
                                                ; if not equal, not an Intel processor
       mov
                _intel_CPU, 1
                                                ; indicate an Intel processor
       cmp
                eax, 1
                                                ; make sure 1 is valid input for CPUID
                                                ; if not, jump to end
       jl
                end_cpuid_type
       mov
                eax, 1
       CPU ID
                                                ; get family/model/stepping/features
       mov
                _cpu_signature, eax
       mov
                _features_ebx, ebx
       mov
                _features_edx, edx
                _features_ecx, ecx
       mov
       shr
                eax, 8
                                                ; isolate family
                eax. 0fh
       and
       mov
                _cpu_type, al
                                                ; set _cpu_type with family
; Execute CPUID instruction to determine the cache descriptor
; information.
  mov eax. 0
                                                ; set up to check the EAX value
  CPU ID
                                                ; Are cache descriptors supported?
  cmp ax, 2
 jl end_cpuid_type
  mov eax, 2
                                                ; set up to read cache descriptor
  CPU ID
                                                ; Is one iteration enough to obtain
  cmp al, 1
 jne end_cpuid_type
                                                ; cache information?
                                                ; This code supports one iteration
                                                ; only.
  mov _cache_eax, eax
                                                ; store cache information
  mov _cache_ebx, ebx
                                                ; NOTE: for future processors, CPUID
                                                ; instruction may need to be run more
  mov _cache_ecx, ecx
  mov _cache_edx, edx
                                                ; than once to get complete cache
                                                ; information
end_cpuid_type:
                edi
                                                ; restore registers
       pop
                esi
        pop
                ebx
       pop
       comment this line for 32-bit segments
        .8086
end_cpu_type:
_get_cpu_type
                        endp
```



```
public _get_fpu_type
        _get_fpu_type proc
        This procedure determines the type of FPU in a system
        and sets the _fpu_type variable with the appropriate value.
        All registers are used by this procedure, none are preserved.
        Coprocessor check
        The algorithm is to determine whether the floating-point
        status and control words are present. If not, no
        coprocessor exists. If the status and control words can
        be saved, the correct coprocessor is then determined
        depending on the processor type. The Intel386 processor can
        work with either an Intel287 NDP or an Intel387 NDP.
        The infinity of the coprocessor must be checked to determine
        the correct coprocessor type.
        fninit
                                                    ; reset FP status word
        mov
                 fp_status, 5a5ah
                                                    ; initialize temp word to non-zero
                                                    ; save FP status word
        fnstsw
                 fp_status
        mov
                 ax, fp_status
                                                    ; check FP status word
        cmp
                 al. 0
                                                    ; was correct status written
                                                    ; no FPU present
        mov
                 _fpu_type, 0
        jne
                 end_fpu_type
check_control_word:
        fnstcw fp_status
                                                    ; save FP control word
                                                    ; check FP control word
        mov
                 ax, fp_status
                 ax, 103fh
        and
                                                    ; selected parts to examine
        cmp
                 ax, 3fh
                                                    ; was control word correct
        mov
                 _fpu_type, 0
                 end_fpu_type
                                                    ; incorrect control word, no FPU
        ine
        mov
                 _fpu_type, 1
        80287/80387 check for the Intel386 processor
check_infinity:
        cmp
                 _cpu_type, 3
        jne
                 end_fpu_type
        fld1
                                                    ; must use default control from FNINIT
        fldz
                                                    ; form infinity
        fdiv
                                                    ; 8087/Intel287 NDP say +inf = -inf
        fld
                                                    ; form negative infinity
        fchs
                                                    ; Intel387 NDP says +inf <> -inf
        fcompp
                                                    ; see if they are the same
        fstsw
                 fp_status
                                                    ; look at status from FCOMPP
        mov
                 ax, fp_status
        mov
                                                    ; store Intel287 NDP for FPU type
                 _fpu_type, 2
        sahf
                                                    ; see if infinities matched
                                                    ; jump if 8087 or Intel287 is present
        jΖ
                 end_fpu_type
        mov
                 _fpu_type, 3
                                                    ; store Intel387 NDP for FPU type
end_fpu_type:
_get_fpu_type
                          endp
        end
```



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

Filename: cpuid3b.asm Copyright 1993, 1994, 1995, 1996, 1997, 1998 by Intel Corp. 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 processors and Intel Celeron(TM) processors 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 100h .stack OP\_O MACRO db 66h ; hardcoded operand override **ENDM** data extrn \_cpu\_type: byte



start:

TSC FLAG

MSR\_FLAG

```
extrn
                           _fpu_type:
                                             byte
         extrn
                           _cpuid_flag:
                                             byte
                           _intel_CPU:
                                             byte
         extrn
                           _vendor_id:
         extrn
                                             byte
         extrn
                           _sep_flag:
                                             byte
                           _cpu_signature:
                                             dword
         extrn
                           _features_ecx:
                                             dword
         extrn
                                             dword
         extrn
                           _features_edx:
         extrn
                           _features_ebx:
                                             dword
                           _cache_eax:
                                             dword
         extern
                                             dword
         extern
                           _cache_ebx:
         extern
                           _cache_ecx:
                                             dword
         extern
                           _cache_edx:
                                             dword
        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
         mov
                  ax, @data
                  ds, ax
         mov
                                             ; set segment register
         mov
                  es, ax
                                             ; set segment register
         and
                                             ; align stack to avoid AC fault
                  sp, not 3
        call
                  _get_cpu_type
                                             ; determine processor type
        call
                  _get_fpu_type
        call
                  print
         mov ax, 4c0h
        int
                           21h
_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
```

equ 0010h

equ 0020h



```
PAE_FLAG
                              equ 0040h
MCE FLAG
                              egu 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
PAT FLAG
                              egu 10000h
PSE36_FLAG
                              equ 20000h
MMX_FLAG
                              equ 800000h
FXSR_FLAG
                              equ 1000000h
          .data
                    db
                              "This system has a$"
id_msg
                    db
                              "n unknown processor$"
cp error
cp_8086
                    db
                              "n 8086/8088 processor$"
                    db
                              "n 80286 processor$"
cp_286
cp_386
                    db
                              "n 80386 processor$"
cp_486
                    db
                              "n 80486DX, 80486DX2 processor or"
                    db
                              " 80487SX math coprocessor$"
cp_486sx
                    db
                              "n 80486SX processor$"
                    db
                              " and an 8087 math coprocessor$"
fp_8087
fp_287
                    db
                              " and an 80287 math coprocessor$"
fp_387
                    db
                              " and an 80387 math coprocessor$"
intel486_msg
                    db
                              "Genuine Intel486(TM) processor$"
                    db
                              "Genuine Intel486(TM) DX processor$"
intel486dx_msg
intel486sx msg
                    db
                              "Genuine Intel486(TM) SX processor$"
                    db
                              "Genuine IntelDX2(TM) processor$"
inteldx2 msg
intelsx2_msg
                    db
                              " Genuine IntelSX2(TM) processor$"
                              "Genuine IntelDX4(TM) processor$"
inteldx4_msg
                    db
inteldx2wb_msg
                    db
                              " Genuine Write-Back Enhanced"
                              "IntelDX2(TM) processor$"
                    db
                    db
                              "Genuine Intel Pentium(R) processor$"
pentium_msg
                    db
                              "Genuine Intel Pentium(R) Pro processor$"
pentiumpro_msg
pentiumiimodel3_msg db
                              "Genuine Intel Pentium(R) II processor, model 3$"
pentiumiixeon_m5_msg db
                              "Genuine Intel Pentium(R) II processor, model 5 or Intel Pentium(R) II Xeon(TM)
processor$"
pentiumiixeon_msg
                      db
                              "Genuine Intel Pentium(R) II Xeon(TM) processor$"
celeron_msg
                      db
                              "Genuine Intel Celeron(TM) processor, model 5$"
celeronmodel6_msg
                      db
                               "Genuine Intel Celeron(TM) processor, model 6$"
unknown msg
                      db
                              "n unknown Genuine Intel processor$"
; The following 16 entries must stay intact as an array
intel_486_0
                    dw
                              offset intel486dx_msg
intel 486 1
                    dw
                              offset intel486dx msg
intel_486_2
                                        offset intel486sx_msg
                              dw
                              dw
intel_486_3
                                        offset inteldx2_msg
intel_486_4
                              dw
                                        offset intel486_msg
intel 486 5
                              dw
                                        offset intelsx2 msg
intel_486_6
                              dw
                                        offset intel486_msg
```

28



| intel_486_7    | dw | offset inteldx2wb_msg                             |
|----------------|----|---------------------------------------------------|
| intel_486_8    | dw | offset inteldx4_msg                               |
| intel_486_9    | dw | offset intel486_msg                               |
| intel_486_a    | dw | offset intel486_msg                               |
| intel_486_b    | dw | offset intel486_msg                               |
| intel_486_c    | dw | offset intel486_msg                               |
| intel_486_d    | dw | offset intel486_msg                               |
| intel_486_e    | dw | offset intel486_msg                               |
| intel_486_f    | dw | offset intel486_msg                               |
| ; end of array |    |                                                   |
| ,,             |    |                                                   |
| family_msg     | db | 13,10,"Processor Family: \$"                      |
| model_msg      | db | 13,10,"Model: \$"                                 |
| stepping_msg   | db | 13,10,"Stepping: "                                |
| cr_lf          | db | 13,10, Stepping.<br>13,10,"\$"                    |
| turbo_msg      | db | 13,10, "The processor is an OverDrive(R)"         |
| turbo_msg      | db | " processor\$"                                    |
| dn mag         | db | 13,10,"The processor is the upgrade"              |
| dp_msg         | db |                                                   |
| fan maa        |    | "processor in a dual processor system\$"          |
| fpu_msg        | db | 13,10,"The processor contains an on-chip"         |
|                | db | " FPU\$"                                          |
| vme_msg        | db | 13,10,"The processor supports Virtual"            |
|                | db | " Mode Extensions\$"                              |
| de_msg         | db | 13,10,"The processor supports Debugging"          |
|                | db | "Extensions\$"                                    |
| pse_msg        | db | 13,10,"The processor supports Page Size"          |
|                | db | "Extensions\$"                                    |
| tsc_msg        | db | 13,10,"The processor supports Time Stamp"         |
|                | db | " Counter\$"                                      |
| msr_msg        | db | 13,10,"The processor supports Model"              |
|                | db | "Specific Registers\$"                            |
| pae_msg        | db | 13,10,"The processor supports Physical"           |
|                | db | " Address Extensions\$"                           |
| mce_msg        | db | 13,10,"The processor supports Machine"            |
|                | 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"        |
|                | db | "Call\$"                                          |
| no_sep_msg     | db | 13,10,"The processor does not support Fast"       |
| _ 1_ 0         | db | " System Call\$"                                  |
| mtrr_msg       | db | 13,10,"The processor supports Memory Type"        |
| = 8            | db | "Range Registers\$"                               |
| pge_msg        | db | 13,10,"The processor supports Page Global"        |
| 18 = 8         | db | "Enable\$"                                        |
| mca_msg        | db | 13,10,"The processor supports Machine"            |
| 8              | db | "Check Architecture\$"                            |
| cmov msg       | db | 13,10,"The processor supports Conditional"        |
| cmov_msg       | db | " Move Instruction\$"                             |
| pat_msg        | db | 13,10,"The processor supports Page Attribute"     |
| Par_1110B      | db | "Table\$"                                         |
| pse36_msg      | db | 13,10,"The processor supports 36-bit Page"        |
| pseso_msg      | db | "Size Extension\$"                                |
| mmy mea        | db | 13,10,"The processor supports Intel Architecture" |
| mmx_msg        | uυ | 15,16, The processor supports filter Architecture |

### **AP-485**



```
" MMX(TM) Technology$"
                                db
fxsr_msg
                                db
                                            13,10,"The processor supports Fast floating point"
                                db
                                            "save and restore$"
not_intel
                                db
                                           "t least an 80486 processor."
                                db
                                            13,10,"It does not contain a Genuine"
                                db
                                           "Intel part and as a result,"
                                           "the",13,10,"CPUID"
                                db
                                           " detection information cannot be"
                                db
                                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
          jle
          add
                     al, 07h
ascii_done:
                     byte ptr msg[20], al
          mov
          mov
                     dx, offset msg
          mov
                     ah, 9h
  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.
                     dx, offset id_msg
                                                                 ; print initial message
          mov
          mov
                     ah, 9h
                     21h
           int
                     _cpuid_flag, 1
                                                                 ; if set to 1, processor
          cmp
                                                                 ; supports CPUID instruction
          je
                     print_cpuid_data
                                                                 ; print detailed CPUID info
print_86:
                      _cpu_type, 0
          cmp
          jne
                     print_286
                     dx, offset cp_8086
          mov
                     ah, 9h
          mov
                     21h
          int
          cmp
                     _fpu_type, 0
                     end_print
          je
          mov
                     dx, offset fp_8087
          mov
                     ah, 9h
                     21h
          int
                     end_print
          jmp
```

print\_286:

30



cmp

\_cpu\_type, 2

```
jne
                      print_386
                      dx, offset cp_286
          mov
                      ah, 9h
          mov
          int
                      21h
                      _fpu_type, 0
          cmp
          je
                      end_print
print_287:
           mov
                      dx, offset fp_287
                      ah, 9h
          mov
                      21h
          int
          jmp
                      end_print
print_386:
          cmp
                      _cpu_type, 3
                      print_486
          jne
          mov
                      dx, offset cp_386
                      ah, 9h
          mov
          int
                      21h
                      _fpu_type, 0
           cmp
                      end_print
          je
                      _fpu_type, 2
           cmp
          je
                      print_287
                      dx, offset fp_387
          mov
                      ah, 9h
          mov
                      21h
          int
          jmp
                      end_print
print_486:
           cmp
                      _cpu_type, 4
                      print_unknown
                                                      ; Intel processors will have
          jne
                                                      ; CPUID instruction
          mov
                      dx, offset cp_486sx
                      _fpu_type, 0
          cmp
                      print_486sx
          je
                      dx, offset cp_486
          mov
print_486sx:
                      ah, 9h
          mov
          int
                      21h
          jmp
                      end_print
print_unknown:
          mov
                      dx, offset cp_error
          jmp
                      print_486sx
print_cpuid_data:
           .486
           cmp
                      _intel_CPU, 1
                                                                 ; check for genuine Intel
          jne
                      not_GenuineIntel
                                                                 ; processor
print_486_type:
                                                                 ; if 4, print 80486 processor
          cmp
                      _cpu_type, 4
          jne
                      print_pentium_type
          mov
                      ax, word ptr _cpu_signature
           shr
                      ax, 4
```



and eax. 0fh ; isolate model mov dx, intel\_486\_0[eax\*2] jmp print\_common print\_pentium\_type: \_cpu\_type, 5 ; if 5, print Pentium processor cmp jne print\_pentiumpro\_type dx, offset pentium\_msg mov jmp print\_common print\_pentiumpro\_type: ; if 6 & model 1, print Pentium cmp \_cpu\_type, 6 ; Pro processor jne print\_unknown\_type mov ax, word ptr \_cpu\_signature ax. 4 shr and eax, 0fh ; isolate model cmp eax, 3 jge print\_pentiumiimodel3\_type cmp eax. 1 print\_unknown\_type ; incorrect model number = 2jne \_sep\_flag, 0 ; does not support Fast System mov ; Call mov dx, offset pentiumpro\_msg jmp print\_common print\_pentiumiimodel3\_type: cmp eax, 3 ; if 6 & model 3, print Pentium ; II processor, model 3 jne print\_pentiumiimodel5\_type mov ax, word ptr \_cpu\_signature and al, 0fh ; isolate stepping al, 3 cmp jl no\_sep \_sep\_flag, 1 mov mov dx, offset pentiumiimodel3\_msg jmp print\_common no\_sep: mov \_sep\_flag, 0 ; stepping does not support ; Fast System Call mov dx, offset pentiumiimodel3\_msg jmp print\_common print\_pentiumiimodel5\_type: eax, 5 ; if 6 & model 5, either Pentium cmp ; II processor, model 5, Pentium II ; Xeon processor or Intel Celeron ; processor, model 5 ine print\_celeronmodel6\_type mov \_sep\_flag, 1 ; Pentium II processor, model 5, Pentium II ; Xeon processor and ; & Intel Celeron processor support sep flag

; Is it Pentium II processor, model 5, Pentium II Xeon processor or Intel Celeron processor?



OP\_O mov ax, word ptr \_cache\_eax OP O rol ax, 8 cmp al, 40h ; Is it no L2 je print\_celeron\_type cmp al, 44h ; Is it 1M L2 cache je print\_pentiumiixeon\_type cmp al, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type OP\_O rol ax, 8 cmp al, 40h je print\_celeron\_type : Is it 1M L2 cache cmp al, 44h je print\_pentiumiixeon\_type cmp al, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type OP O rol ax, 8 cmp al, 40h je print\_celeron\_type cmp al, 44h ; Is it 1M L2 cache je print\_pentiumiixeon\_type cmp al, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type OP O mov bx, word ptr \_cache\_ebx cmp bl, 40h je print\_celeron\_type cmp bl, 44h ; Is it 1M L2 cache je print\_pentiumiixeon\_type cmp bl, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type OP\_O rol bx, 8 cmp bl, 40h : Is it no L2 je print\_celeron\_type ; Is it 1M L2 cache cmp bl, 44h je print\_pentiumiixeon\_type ; Is it 2M L2 cache cmp bl, 45h je print\_pentiumiixeon\_type OP O rol bx, 8 cmp bl, 40h je print\_celeron\_type ; Is it 1M L2 cache cmp bl, 44h je print\_pentiumiixeon\_type cmp bl, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type

### **AP-485**



 $OP_O$ rol bx, 8 cmp bl, 40h je print\_celeron\_type cmp bl, 44h ; Is it 1M L2 cache je print\_pentiumiixeon\_type ; Is it 2M L2 cache cmp bl, 45h je print\_pentiumiixeon\_type OP O mov cx, word ptr \_cache\_ecx cmp cl, 40h je print\_celeron\_type cmp cl, 44h ; Is it 1M L2 cache je print\_pentiumiixeon\_type cmp cl, 45h : Is it 2M L2 cache je print\_pentiumiixeon\_type OP O rol cx. 8 cmp cl, 40h ; Is it no L2 je print\_celeron\_type cmp cl, 44h ; Is it 1M L2 cache je print\_pentiumiixeon\_type cmp cl, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type OP O rol cx, 8 cmp cl, 40h je print\_celeron\_type cmp cl, 44h ; Is it 1M L2 cache je print\_pentiumiixeon\_type cmp cl, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type OP O rol cx. 8 cmp cl, 40h je print\_celeron\_type : Is it 1M L2 cache cmp cl, 44h je print\_pentiumiixeon\_type cmp cl, 45h ; Is it 2M L2 cache je print\_pentiumiixeon\_type  $OP_O$ mov dx, word ptr \_cache\_edx cmp dl, 40h je print\_celeron\_type : Is it 1M L2 cache cmp dl, 44h je print\_pentiumiixeon\_type ; Is it 2M L2 cache cmp dl, 45h je print\_pentiumiixeon\_type OP O rol dx, 8



```
cmp dl, 40h
                                                                ; Is it no L2
  je print_celeron_type
                                                                ; Is it 1M L2 cache
  cmp dl, 44h
  je print_pentiumiixeon_type
  cmp dl, 45h
                                                                ; Is it 2M L2 cache
  je print_pentiumiixeon_type
  OP_O
  rol dx, 8
  cmp dl, 40h
  je print_celeron_type
                                                                ; Is it 1M L2 cache
  cmp dl, 44h
  je print_pentiumiixeon_type
  cmp dl, 45h
                                                                ; Is it 2M L2 cache
  je print_pentiumiixeon_type
  OP O
  rol dx, 8
  cmp dl, 40h
  je print_celeron_type
                                                                ; Is it 1M L2 cache
  cmp dl, 44h
  je print_pentiumiixeon_type
  cmp dl, 45h
                                                                ; Is it 2M L2 cache
  je print_pentiumiixeon_type
  mov dx, offset pentiumiixeon_m5_msg
  jmp print_common
print_celeron_type:
  mov dx, offset celeron_msg
  jmp print_common
print_pentiumiixeon_type:
 mov dx, offset pentiumiixeon_msg
 jmp print_common
print_celeronmodel6_type:
                     eax, 6
                                                                ; if 6 & model 6, print Intel Celeron
          cmp
                                                                ; processor, model 6
                     print_unknown_type
          jne
          mov
                     ax, word ptr _cpu_signature
          and
                     al, 0fh
                                                                ; isolate stepping
                                                                ; Intel Celeron processor, model 6 supports SEP
          mov
                     _sep_flag, 1
                     dx, offset celeronmodel6_msg
          mov
                     print_common
          jmp
print_unknown_type:
                     dx, offset unknown_msg
                                                                ; if neither, print unknown
          mov
print_common:
          mov
                     ah, 9h
          int
                     21h
; print family, model, and stepping
print_family:
```

36



```
mov
                     al, _cpu_type
          ASC_MSG
                                                              ; print family msg
                               family_msg
print_model:
          mov
                     ax, word ptr _cpu_signature
          shr
                     ax, 4
                     al, 0fh
          and
          ASC_MSG
                               model_msg
                                                              ; print model msg
print_stepping:
                     ax, word ptr _cpu_signature
          mov
          and
                     al, 0fh
          ASC_MSG
                                                              ; print stepping msg
                               stepping_msg
print_upgrade:
          mov
                     ax, word ptr _cpu_signature
          test
                     ax, 1000h
                                                              ; check for turbo upgrade
                     check_dp
          jz
          mov
                     dx, offset turbo_msg
          mov
                     ah, 9h
          int
                     21h
                     print_features
          jmp
check dp:
                     ax, 2000h
          test
                                                              ; check for dual processor
                     print_features
          jΖ
          mov
                     dx, offset dp_msg
          mov
                     ah, 9h
          int
                     21h
print_features:
                     ax, word ptr _features_edx
          mov
          and
                     ax, FPU_FLAG
                                                              ; check for FPU
                     check_VME
          įΖ
          mov
                     dx, offset fpu_msg
                     ah, 9h
          mov
          int
                     21h
check_VME:
          mov
                     ax, word ptr _features_edx
                     ax, VME_FLAG
                                                              ; check for VME
          and
                     check_DE
          įΖ
          mov
                     dx, offset vme_msg
                     ah. 9h
          mov
                     21h
          int
check_DE:
          mov
                     ax, word ptr _features_edx
                                                              ; check for DE
          and
                     ax, DE_FLAG
          jz
                     check_PSE
                     dx, offset de_msg
          mov
                     ah. 9h
          mov
                     21h
          int
check_PSE:
```



mov

ax, word ptr \_features\_edx

and ax, PSE\_FLAG ; check for PSE check\_TSC jΖ dx, offset pse\_msg mov mov ah, 9h int 21h check\_TSC: mov ax, word ptr \_features\_edx and ax, TSC\_FLAG ; check for TSC check\_MSR jΖ dx, offset tsc\_msg mov mov ah, 9h int 21h check\_MSR: ax, word ptr \_features\_edx mov and ax, MSR\_FLAG ; check for MSR check\_PAE jΖ dx, offset msr\_msg mov ah, 9h mov int 21h check\_PAE: ax, word ptr \_features\_edx mov ax, PAE\_FLAG ; check for PAE and check\_MCE jΖ mov dx, offset pae\_msg mov ah, 9h 21h int check\_MCE: ax, word ptr \_features\_edx mov ax, MCE\_FLAG ; check for MCE and check\_CX8 įΖ dx, offset mce\_msg mov ah, 9h mov 21h int check\_CX8: ax, word ptr \_features\_edx mov ax, CX8\_FLAG ; check for CMPXCHG8B and check\_APIC įΖ mov dx, offset cx8\_msg mov ah, 9h 21h int check\_APIC: mov ax, word ptr \_features\_edx ax, APIC\_FLAG ; check for APIC and jz check\_SEP dx, offset apic\_msg mov ah. 9h mov 21h int check\_SEP:



cmp \_sep\_flag, 1 jne print\_no\_sep dx, offset sep\_msg mov ah, 9h mov int 21h jmp check\_MTRR print\_no\_sep: dx, offset no\_sep\_msg mov ah, 9h 21h int check\_MTRR: mov ax, word ptr \_features\_edx ax, MTRR\_FLAG ; check for MTRR and check\_PGE įΖ dx, offset mtrr\_msg mov mov ah, 9h 21h int check PGE: ax, word ptr\_features\_edx mov ax, PGE\_FLAG ; check for PGE and įΖ check\_MCA dx, offset pge\_msg mov ah, 9h mov int 21h check MCA: ax, word ptr \_features\_edx mov and ax, MCA\_FLAG ; check for MCA jz check\_CMOV mov dx, offset mca\_msg mov ah, 9h 21h int check\_CMOV: mov ax, word ptr \_features\_edx and ax, CMOV\_FLAG ; check for CMOV 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Ζ mov dx, offset pat\_msg ah, 9h mov int 21h check\_PSE36: mov eax, dword ptr \_features\_edx and eax, PSE36 FLAG jz check\_mmx

; check for MMX technology

; check for FXSR



mov dx, offset pse36\_msg mov ah, 9h int 21h

Check\_MMX:

mov eax, dword ptr \_features\_edx

and eax, MMX\_FLAG

jz check\_fxsr

mov dx, offset mmx\_msg

mov ah, 9h int 21h

check\_FXSR:

mov eax, dword ptr \_features\_edx

and eax, FXSR\_FLAG

jz end\_print

mov dx, offset fxsr\_msg

mov ah, 9h int 21h

not\_GenuineIntel:

jmp

mov dx, offset not\_intel

end\_print

mov ah, 9h int 21h

end\_print:

mov dx, offset cr\_lf

mov ah, 9h int 21h

ret print endp

end start

39



Example 3. Processor Identification Procedure in the C Language

```
/* Filename: cpuid3.c
                                                                                */
/* Copyright 1994, 1995, 1996, 1997, 1998 by Intel Corp.
                                                                                */
/* 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 and Intel Celeron processors
/* in the real-address mode.
```

| 0x0001    |
|-----------|
| OAGGGI    |
| 0x0002    |
| 0x0004    |
| 0x0008    |
| 0x0010    |
| 0x0020    |
| 0x0040    |
| 0x0080    |
| 0x0100    |
| 0x0200    |
| 0x0800    |
| 0x1000    |
| 0x2000    |
| 0x4000    |
| 0x8000    |
| 0x10000   |
| 0x20000   |
| 0x800000  |
| 0x1000000 |
|           |

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;
long cache_temp;
long celeron_flag;
long pentiumiixeon_flag;
main() {
  get_cpu_type();
  get_fpu_type();
  print();
}
print() {
  printf("This system has a");
  if (cpuid_flag == 0) {
           switch (cpu_type) {
           case 0:
             printf("n 8086/8088 processor");
             if (fpu_type) printf(" and an 8087 math coprocessor");
             break;
           case 2:
             printf("n 80286 processor");
             if (fpu_type) printf(" and an 80287 math coprocessor");
             break;
           case 3:
             printf("n 80386 processor");
             if (fpu\_type == 2)
                      printf(" and an 80287 math coprocessor");
             else if (fpu_type)
                      printf(" and an 80387 math coprocessor");
             break:
           case 4:
             if (fpu_type) printf("n 80486DX, 80486DX2 processor or \
80487SX math coprocessor");
             else printf("n 80486SX processor");
             break;
           default:
             printf("n unknown processor");
  } else {
           /* using cpuid instruction */
           if (intel_CPU) {
             if (cpu\_type == 4) {
                      switch ((cpu_signature>>4)&0xf) {
                      case 0:
```



```
printf(" Genuine Intel486(TM) DX processor");
          break;
        case 2:
           printf(" Genuine Intel486(TM) SX processor");
           break;
        case 3:
          printf(" Genuine IntelDX2(TM) processor");
          break:
        case 4:
          printf(" Genuine Intel486(TM) processor");
          break:
        case 5:
          printf(" Genuine IntelSX2(TM) processor");
          break;
        case 7:
          printf(" Genuine Write-Back Enhanced \
                   IntelDX2(TM) processor");
          break;
        case 8:
          printf(" Genuine IntelDX4(TM) processor");
          break;
        default:
          printf(" Genuine Intel486(TM) processor");
} else if (cpu_type == 5)
        printf(" Genuine Intel Pentium(R) processor");
        else if ((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 1))
        printf(" Genuine Intel Pentium(R) Pro processor");
        else if ((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 3))
        printf(" Genuine Intel Pentium(R) II processor, model 3");
        else if ((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 5))
                   celeron_flag = 0;
                   pentiumiixeon_flag = 0;
                   cache_temp = cache_eax & 0xFF000000;
                   if (cache_temp == 0x40000000)
                             celeron_flag = 1;
                   if (cache_temp == 0x44000000)
                             pentiumiixeon_flag = 1;
                   if (cache_temp == 0x45000000)
                             pentiumiixeon_flag = 1;
                   cache_temp = cache_eax & 0xFF0000;
                   if (cache_temp == 0x400000)
                             celeron_flag = 1;
                   if (cache_temp == 0x440000)
                             pentiumiixeon flag = 1;
                   if (cache_temp == 0x450000)
                              pentiumiixeon_flag = 1;
                   cache_temp = cache_eax & 0xFF00;
                   if (cache_temp == 0x4000)
                              celeron_flag = 1;
                   if (cache\_temp == 0x4400)
                             pentiumiixeon_flag = 1;
                   if (cache_temp == 0x4500)
                             pentiumiixeon_flag = 1;
```



```
cache_temp = cache_ebx & 0xFF000000;
if (cache_temp == 0x40000000)
          celeron_flag = 1;
if (cache_temp == 0x44000000)
          pentiumiixeon_flag = 1;
if (cache_temp == 0x45000000)
          pentiumiixeon_flag = 1;
cache_temp = cache_ebx & 0xFF0000;
if (cache_temp == 0x400000)
          celeron_flag = 1;
if (cache_temp == 0x440000)
          pentiumiixeon_flag = 1;
if (cache_temp == 0x450000)
          pentiumiixeon_flag = 1;
cache_temp = cache_ebx & 0xFF00;
if (cache_temp == 0x4000)
          celeron_flag = 1;
if (cache\_temp == 0x4400)
          pentiumiixeon_flag = 1;
if (cache_temp == 0x4500)
          pentiumiixeon_flag = 1;
cache_temp = cache_ebx & 0xFF;
if (cache\_temp == 0x40)
          celeron_flag = 1;
if (cache_temp == 0x44)
          pentiumiixeon_flag = 1;
if (cache\_temp == 0x45)
          pentiumiixeon_flag = 1;
cache_temp = cache_ecx & 0xFF000000;
if (cache_temp == 0x40000000)
          celeron_flag = 1;
if (cache_temp == 0x44000000)
          pentiumiixeon_flag = 1;
if (cache_temp == 0x45000000)
          pentiumiixeon_flag = 1;
cache_temp = cache_ecx & 0xFF0000;
if (cache_temp == 0x400000)
          celeron_flag = 1;
if (cache_temp == 0x440000)
          pentiumiixeon_flag = 1;
if (cache_temp == 0x450000)
          pentiumiixeon_flag = 1;
cache_temp = cache_ecx & 0xFF00;
if (cache_temp == 0x4000)
          celeron_flag = 1;
if (cache\_temp == 0x4400)
          pentiumiixeon_flag = 1;
if (cache_temp == 0x4500)
          pentiumiixeon_flag = 1;
cache_temp = cache_ecx & 0xFF;
if (cache_temp == 0x40)
          celeron_flag = 1;
if (cache\_temp == 0x44)
          pentiumiixeon_flag = 1;
if (cache_temp == 0x45)
          pentiumiixeon_flag = 1;
```



```
cache_temp = cache_edx & 0xFF000000;
                                if (cache_temp == 0x40000000)
                                           celeron_flag = 1;
                                if (cache_temp == 0x44000000)
                                           pentiumiixeon_flag = 1;
                                if (cache_temp == 0x45000000)
                                          pentiumiixeon_flag = 1;
                                cache_temp = cache_edx & 0xFF0000;
                                if (cache_temp == 0x400000)
                                          celeron_flag = 1;
                                if (cache_temp == 0x440000)
                                          pentiumiixeon_flag = 1;
                                if (cache_temp == 0x450000)
                                          pentiumiixeon_flag = 1;
                                cache\_temp = cache\_edx \& 0xFF00;
                                if (cache temp == 0x4000)
                                           celeron_flag = 1;
                                if (cache_temp == 0x4400)
                                           pentiumiixeon_flag = 1;
                                if (cache\_temp == 0x4500)
                                          pentiumiixeon_flag = 1;
                                cache_temp = cache_edx & 0xFF;
                                if (cache\_temp == 0x40)
                                           celeron_flag = 1;
                                if (cache\_temp == 0x44)
                                           pentiumiixeon_flag = 1;
                                if (cache\_temp == 0x45)
                                          pentiumiixeon_flag = 1;
                                if (celeron_flag == 1)
                                           printf(" Genuine Intel Celeron(TM) processor, model 5");
                                else
                                           if (pentiumiixeon_flag == 1)
                                           printf(" Genuine Intel Pentium(R) II Xeon(TM) processor");
                                           else
                                           printf(" Genuine Intel Pentium(R) II processor, model 5 or Genuine Intel
                                           Pentium(R) II Xeon(TM) processor");
            else if ((cpu_type == 6) && (((cpu_signature >> 4) & 0xf) == 6))
                                printf("Genuine Intel Celeron(TM) processor, model 6");
             else
                     printf("n unknown Genuine Intel processor");
             printf("\nProcessor Family: %X", cpu type);
             printf("\nModel:
                                    %X", (cpu_signature>>4)&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 >> 4) &0xf) < 3)
         && ((cpu_signature & 0xf) < 3))
           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\ \ \backslash
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 & MMX_FLAG)
                     printf("\nThe processor supports Intel Architecture \
MMX technology");
```



```
if \ (features\_edx \ \& FXSR\_FLAG) \\ printf("\nThe processor supports the Fast floating \noint save and restore"); \\ else \ \{ \\ printf("t least an 80486 processor.\nIt does not \noint \noint
```

# intel<sub>®</sub>

UNITED STATES, Intel Corporation 2200 Mission College Blvd., P.O. Box 58119, Santa Clara, CA 95052-8119 Tel: +1 408 765-8080

> JAPAN, Intel Japan K.K. 5-6 Tokodai, Tsukuba-shi, Ibaraki-ken 300-26 Tel: + 81-29847-8522

> > FRANCE, Intel Corporation S.A.R.L. 1, Quai de Grenelle, 75015 Paris Tel: +33 1-45717171

UNITED KINGDOM, Intel Corporation (U.K.) Ltd. Pipers Way, Swindon, Wiltshire, England SN3 1RJ Tel: +44 1-793-641440

GERMANY, Intel GmbH Dornacher Strasse 1 85622 Feldkirchen/ Muenchen Tel: +49 89/99143-0

HONG KONG, Intel Semiconductor Ltd. 32/F Two Pacific Place, 88 Queensway, Central Tel: +852 2844-4555

CANADA, Intel Semiconductor of Canada, Ltd. 190 Attwell Drive, Suite 500 Rexdale, Ontario M9W 6H8 Tel: +416 675-2438

Printed in USA/96/POD/PMG