

# CHAPTER

## Computer Number Systems, Codes, and Digital Devices

Before starting our discussion of microprocessors and microcomputers, we need to make sure that some key concepts of the number systems, codes, and digital devices used in microcomputers are fresh in your mind. If the short summaries of these concepts in this chapter are not enough to refresh your memory, then you may want to consult some of the chapters in *Digital Circuits and Systems*, McGraw-Hill, 1989, before going on in this book.

### OBJECTIVES

At the conclusion of this chapter you should be able to:

1. Convert numbers between the following codes: binary, hexadecimal, and BCD.
2. Define the terms *bit*, *nibble*, *byte*, *word*, *most significant bit*, and *least significant bit*.
3. Use a table to find the ASCII or EBCDIC code for a given alphanumeric character.
4. Perform addition and subtraction of binary, hexadecimal, and BCD numbers.
5. Describe the operation of gates, flip-flops, latches, registers, ROMs, PALs, dynamic RAMs, static RAMs, and buses.
6. Describe how an arithmetic logic unit can be instructed to perform arithmetic or logical operations on binary words.

### ✓ COMPUTER NUMBER SYSTEMS AND CODES

#### Review of Decimal System

To understand the structure of the binary number system, the first step is to review the familiar decimal or base-10 number system. Here is a decimal number with the value of each place holder or digit expressed as a power of 10.

|        |        |        |        |   |           |           |
|--------|--------|--------|--------|---|-----------|-----------|
| 5      | 3      | 4      | 6      | . | 7         | 2         |
| $10^3$ | $10^2$ | $10^1$ | $10^0$ |   | $10^{-1}$ | $10^{-2}$ |

The digits in the decimal number 5346.72 thus tell you that you have 5 thousands, 3 hundreds, 4 tens, 6 ones, 7 tenths, and 2 hundredths. The number of symbols needed in any number system is equal to the base number. In the decimal number system, then, there are 10 symbols, 0 through 9. When the count in any digit position passes that of the highest-value symbol, the digit rolls back to 0 and the next higher digit is incremented by 1. A car odometer is a good example of this.

A number system can be built using powers of any number as place holders or digits, but some bases are more useful than others. It is difficult to build electronic circuits which can store and manipulate 10 different voltage levels but relatively easy to build circuits which can handle two levels. Therefore, a *binary*, or *base-2*, number system is used to represent numbers in digital systems.)

#### ✓ The Binary Number System

Figure 1-1a, p. 2, shows the value of each digit in a binary number. Each binary digit represents a power of 2. A *binary digit* is often called a *bit*. Note that digits to the right of the *binary point* represent fractions used for numbers less than 1. The binary system uses only two symbols, zero (0) and one (1), so in binary you count as follows: 0, 1, 10, 11, 100, 101, 110, 111, 1000, etc. For reference, Figure 1-1b shows the powers of 2 from  $2^1$  to  $2^{32}$ .

Binary numbers are often called *binary words* or just *words*. Binary words with certain numbers of bits have also acquired special names. A 4-bit binary word is called a *nibble*, and an 8-bit binary word is called a *byte*. A 16-bit binary word is often referred to just as a *word*, and a 32-bit binary word is referred to as a *doubleword*. The rightmost or *least significant bit* of a binary word is usually referred to as the *LSB*. The leftmost or *most significant bit* of a binary word is usually called the *MSB*.

To convert a binary number to its equivalent decimal number, multiply each digit times the decimal value of the digit and just add these up. The binary number 101, for example, represents:  $(1 \times 2^2) + (0 \times 2^1) + (1 \times 2^0)$ .

|                                                                   |             |                   |                       |                          |
|-------------------------------------------------------------------|-------------|-------------------|-----------------------|--------------------------|
|                                                                   | $2^1 = 2$   | $2^8 = 512$       | $2^{17} = 131,072$    | $2^{25} = 33,554,432$    |
|                                                                   | $2^2 = 4$   | $2^{10} = 1,024$  | $2^{18} = 262,144$    | $2^{26} = 67,108,864$    |
|                                                                   | $2^3 = 8$   | $2^{11} = 2,048$  | $2^{19} = 524,288$    | $2^{27} = 134,217,728$   |
|                                                                   | $2^4 = 16$  | $2^{12} = 4,096$  | $2^{20} = 1,048,576$  | $2^{28} = 268,435,456$   |
| 1 0 1 1 0 . 1 1                                                   | $2^5 = 32$  | $2^{13} = 8,192$  | $2^{21} = 2,097,152$  | $2^{29} = 536,870,912$   |
| $2^7 \ 2^8 \ 2^5 \ 2^4 \ 2^3 \ 2^2 \ 2^1 \ 2^0 \ 2^{-1} \ 2^{-2}$ | $2^6 = 64$  | $2^{14} = 16,384$ | $2^{22} = 4,194,304$  | $2^{30} = 1,073,741,824$ |
| 128 64 32 16 8 4 2 1 $\frac{1}{2} \frac{1}{4}$                    | $2^7 = 128$ | $2^{15} = 32,768$ | $2^{23} = 8,388,608$  | $2^{31} = 2,147,483,648$ |
|                                                                   | $2^8 = 256$ | $2^{16} = 65,536$ | $2^{24} = 16,777,216$ | $2^{32} = 4,294,967,296$ |

(a)

(b)

FIGURE 1-1 (a) Digit values in binary. (b) Powers of 2.

or  $4 + 0 + 1 =$  decimal 5. For the binary number 10110.11, you have:

$$(1 \times 2^4) + (0 \times 2^3) + (1 \times 2^2) + (1 \times 2^1) + (0 \times 2^0) + (1 \times 2^{-1}) + (1 \times 2^{-2}) = \\ 16 + 0 + 4 + 2 + 0 + 0.5 + 0.25 = \text{decimal } 22.75$$

To convert a decimal number to binary, there are two common methods. The first (Figure 1-2a) is simply a reverse of the binary-to-decimal method. For example, to convert the decimal number 21 (sometimes written as  $21_{10}$ ) to binary, first subtract the largest power of 2 that will fit in the number. For  $21_{10}$  the largest power of 2 that will fit is 16 or  $2^4$ . Subtracting 16 from 21 gives a remainder of 5. Put a 1 in the  $2^4$  digit position and see if the next lower power of 2 will fit in the remainder. Since  $2^3$  is 8 and 8 will not fit in the remainder of 5, put a 0 in the  $2^3$  digit position. Then try the next lower power of 2. In this case the next is  $2^2$  or 4, which will fit in the remainder of 5. A 1 is therefore put in the  $2^2$  digit position. When  $2^2$  or 4 is subtracted from the old remainder of 5, a new remainder of 1 is left. Since  $2^1$  or 2 will not fit into this remainder, a 0 is put in that position. A 1 is put in the  $2^0$  position because  $2^0$  is equal to 1 and this fits exactly into the remainder of 1. The result shows that  $21_{10}$  is equal to 10101 in binary. This conversion process is somewhat messy to describe but easy to do. Try converting  $46_{10}$  to binary. You should get 10110.

Another method of converting a decimal number to binary is shown in Figure 1-2b. Divide the decimal number by 2 and write the quotient and remainder as shown. Divide this quotient and following quotients by 2 until the quotient reaches 0. The column of remainders will be the binary equivalent of the given decimal number. Note that the MSD is on the bottom of the column and the LSD is on the top of the column if you perform the divisions in order from the top to the bottom of the page. You can demonstrate that the binary number is correct by reconverting from binary to decimal, as shown in the right-hand side of Figure 1-2b.

You can convert decimal numbers less than 1 to binary by successive multiplication by 2, recording carries until the quantity to the right of the decimal point becomes zero, as shown in Figure 1-2c. The carries represent the binary equivalent of the decimal number, with the most significant bit at the top of the column. Decimal 0.625 equals 0.101 in binary. For decimal values that do not convert exactly the way this one did (the quantity to the

right of the decimal never becomes zero), you can continue the conversion process until you get the number of binary digits desired.

At this point it is interesting to compare the number of digits required to express numbers in decimal with the number required to express them in binary. In

|                                     |
|-------------------------------------|
| $2^5 \ 2^4 \ 2^3 \ 2^2 \ 2^1 \ 2^0$ |
| 32 16 8 4 2 1                       |
| $21_{10} = 0 \ 1 \ 0 \ 1 \ 0 \ 1_2$ |

(a)

$$227_{10} = ? \text{ Binary}$$

|                                |
|--------------------------------|
| Least Significant Binary Digit |
| ↓                              |
| $2\overline{)227} = 113$       |
| $2\overline{)113} = 56$        |
| $2\overline{)56} = 28$         |
| $2\overline{)28} = 14$         |
| $2\overline{)14} = 7$          |
| $2\overline{)7} = 3$           |
| $2\overline{)3} = 1$           |
| $2\overline{)1} = 0$           |
| R1 × 1 = 1                     |
| R1 × 2 = 2                     |
| R0 × 4 = 0                     |
| R0 × 8 = 0                     |
| R0 × 16 = 0                    |
| R1 × 32 = 32                   |
| R1 × 64 = 64                   |
| R1 × 128 = 128                 |
| ↑                              |
| Most Significant Binary Digit  |
| 227 Check                      |

$$\therefore 227_{10} = 11100011_2$$

(b)

| MSB                    | Check           |
|------------------------|-----------------|
| $2 \times .625 = 1.25$ | $1 \times .5$   |
| $2 \times .25 = 0.50$  | $0 \times .25$  |
| $2 \times .50 = 1.00$  | $1 \times .125$ |
| LSB                    | .625            |

(c)

FIGURE 1-2 Converting decimal to binary. (a) Digit value method. (b) Divide by 2 method. (c) Decimal fraction conversion.

decimal, one digit can represent  $10^1$  numbers, 0 through 9; two digits can represent  $10^2$  or 100 numbers, 0 through 99; and three digits can represent  $10^3$  or 1000 numbers, 0 through 999. In binary, a similar pattern exists. One binary digit can represent 2 numbers, 0 and 1; two binary digits can represent  $2^2$  or 4 numbers, 0 through 11; and three binary digits can represent  $2^3$  or 8 numbers, 0 through 111. The pattern, then, is that N decimal digits can represent  $10^N$  numbers and N binary digits can represent  $2^N$  numbers. Eight binary digits can represent  $2^8$  or 256 numbers, 0 through 255 in decimal.

## Hexadecimal

Binary is not a very compact code. This means that it requires many more digits to express a number than does, for example, decimal. Twelve binary digits can only describe a number up to  $4095_{10}$ . Computers require binary data, but people working with computers have trouble remembering long binary words. One solution to the problem is to use the *hexadecimal* or base-16 number system.

Figure 1-3a shows the digit values for hexadecimal, which is often just called hex. Since hex is base 16, you have to have 16 possible symbols, one for each digit. The table of Figure 1-3b shows the symbols for hex code.

After the decimal symbols 0 through 9 are used up, <sup>you</sup> use the letters A through F for values 10 through 15.

As mentioned above, each hex digit is equal to four binary digits. To convert the binary number 11010110 to hex, mark off the binary bits in groups of 4, moving to the left from the binary point. Then write the hex symbol for the value of each group of 4.

|        |      |      |
|--------|------|------|
| Binary | 1101 | 0110 |
| Hex    | D    | 6    |

The 0110 group is equal to 6 and the 1101 group is equal to 13. Since 13 is D in hex, 11010110 binary is equal to D6 in hex. "H" is usually used after a number to indicate that it is a hexadecimal number. For example, D6 hex is usually written D6H. As you can see, 8 bits can be represented with only 2 hex digits.

If you want to convert a number from decimal to hexadecimal, Figure 1-3c shows a familiar trick for doing this. The result shows that  $227_{10}$  is equal to E3H. As you can see, hex is an even more compact code than decimal. Two hexadecimal digits can represent a decimal number up to 255. Four hex digits can represent a decimal number up to 65,535.

To illustrate how hexadecimal numbers are used in digital logic, a service manual tells you that the 8-bit-wide data bus of an 8088A microprocessor should contain 3FH during a certain operation. Converting 3FH to binary gives the pattern of 1's and 0's (0011 1111) you would expect to find with your oscilloscope or logic analyzer on the parallel lines. The 3FH is simply a shorthand which is easier to remember and less prone to errors than the binary equivalent.

$$16^3 \quad 16^2 \quad 16^1 \quad 16^0 \cdot \frac{1}{16} \quad \frac{1}{16^2} \quad \frac{1}{16^3}$$

(a)

| <b>Dec</b> | <b>Hex</b> | <b>Dec</b> | <b>Hex</b> |
|------------|------------|------------|------------|
| 0 = 0      |            | 8 = 8      |            |
| 1 = 1      |            | 9 = 9      |            |
| 2 = 2      |            | 10 = A     |            |
| 3 = 3      |            | 11 = B     |            |
| 4 = 4      |            | 12 = C     |            |
| 5 = 5      |            | 13 = D     |            |
| 6 = 6      |            | 14 = E     |            |
| 7 = 7      |            | 15 = F     |            |

16

$$\begin{array}{r}
 227_D = \underline{\quad ? \quad}_{\text{Hex}} \quad \text{LSD} \\
 16 \overline{) 227} \quad = \quad 14 \quad R3 \times 1 = 3 \\
 16 \overline{) 14} \quad = \quad 0 \quad RE \times 16 = \underline{224} \\
 \end{array}$$

$$-227_{10} = E3_{16}$$

(c)

## **BCD Codes**

## STANDARD BCD

In applications such as frequency counters, digital voltmeters, or calculators, where the output is a decimal display, a binary-coded decimal or BCD code is often used. BCD uses a 4-bit binary code to individually represent each decimal digit in a number. As you can see in Table 1-1, p. 4, the simplest BCD code uses the first 10 numbers of standard binary code for the BCD numbers 0 through 9. The hex codes A through F are invalid BCD codes. To convert a decimal number to its BCD equivalent, just represent each decimal digit by its 4-bit binary equivalent, as shown here.

Decimal      5      2      9  
               BCD      0101      0010      1001

To convert a BCD number to its decimal equivalent, reverse the process.

## GRAY CODE

Gray code is another important binary code; it is often used for encoding shaft position data from machines such as computer-controlled lathes. This code has the same possible combinations as standard binary, but as you can see in the 4-bit example in Table 1-1, they are

TABLE 1-1  
COMMON NUMBER CODES

| Decimal | Binary | Octal | Hex | Binary-Coded Decimal |      |           | Reflected Gray Code | 7-Segment Display (1 = on) |         |
|---------|--------|-------|-----|----------------------|------|-----------|---------------------|----------------------------|---------|
|         |        |       |     | 8421                 | BCD  | EXCESS-3  |                     | a b c d e f g              | Display |
| 0       | 0000   | 0     | 0   |                      | 0000 | 0011 0011 | 0000                | 1 1 1 1 1 1 0              | 0       |
| 1       | 0001   | 1     | 1   |                      | 0001 | 0011 0100 | 0001                | 0 1 1 0 0 0 0              | 1       |
| 2       | 0010   | 2     | 2   |                      | 0010 | 0011 0101 | 0011                | 1 1 0 1 1 0 1              | 2       |
| 3       | 0011   | 3     | 3   |                      | 0011 | 0011 0110 | 0010                | 1 1 1 1 0 0 1              | 3       |
| 4       | 0100   | 4     | 4   |                      | 0100 | 0011 0111 | 0110                | 0 1 1 0 0 1 1              | 4       |
| 5       | 0101   | 5     | 5   |                      | 0101 | 0011 1000 | 0111                | 1 0 1 1 0 1 1              | 5       |
| 6       | 0110   | 6     | 6   |                      | 0110 | 0011 1001 | 0101                | 1 0 1 1 1 1 1              | 6       |
| 7       | 0111   | 7     | 7   |                      | 0111 | 0011 1010 | 0100                | 1 1 1 0 0 0 0              | 7       |
| 8       | 1000   | 10    | 8   |                      | 1000 | 0011 1011 | 1100                | 1 1 1 1 1 1 1              | 8       |
| 9       | 1001   | 11    | 9   |                      | 1001 | 0011 1100 | 1101                | 1 1 1 0 0 1 1              | 9       |
| 10      | 1010   | 12    | A   | 0001                 | 0000 | 0100 0011 | 1111                | 1 1 1 1 1 0 1              | A       |
| 11      | 1011   | 13    | B   | 0001                 | 0001 | 0100 0100 | 1110                | 0 0 1 1 1 1 1              | B       |
| 12      | 1100   | 14    | C   | 0001                 | 0010 | 0100 0101 | 1010                | 0 0 0 1 1 0 1              | C       |
| 13      | 1101   | 15    | D   | 0001                 | 0011 | 0100 0110 | 1011                | 0 1 1 1 1 0 1              | D       |
| 14      | 1110   | 16    | E   | 0001                 | 0100 | 0100 0111 | 1001                | 1 1 0 1 1 1 1              | E       |
| 15      | 1111   | 17    | F   | 0001                 | 0101 | 0100 1000 | 1000                | 1 0 0 0 1 1 1              | F       |

arranged in a different order. Notice that only one binary digit changes at a time as you count up in this code.

If you need to construct a Gray-code table larger than that in Table 1-1, a handy way to do so is to observe the pattern of 1's and 0's and just extend it. The least significant digit column starts with one 0 and then has alternating groups of two 1's and two 0's as you go down the column. The second most significant digit column starts with two 0's and then has alternating groups of four 1's and four 0's. The third column starts with four 0's, then has alternating groups of eight 1's and eight 0's. By now you should see the pattern. Try to figure out the Gray code for the decimal number 16. You should get 11000.

an odd number of 1's, the word is said to have odd parity. The binary word 0110111 with five 1's has odd parity. The binary word 0110000 has an even number of 1's (two), so it has even parity.

In practice the parity bit is used as follows. The system that is sending a data word checks the parity of the word. If the parity of the data word is odd, the system will set the parity bit to a 1. This makes the parity of the data word plus parity bit even. If the parity of the data word is even, the sending system will reset the parity bit to a 0. This again makes the parity of the data word plus parity even. The receiving system checks the

## 7-Segment Display Code

Figure 1-4a shows the segment identifiers for a 7-segment display such as those commonly used in digital instruments. Table 1-1 shows the logic levels required to display 0 to 9 and A to F on a common-cathode LED display such as that shown in Figure 1-4b. For a common-anode LED display such as that in Figure 1-4c, simply invert the segment codes shown in Table 1-1.

## Alphanumeric Codes

When communicating with or between computers, you need a binary-based code which can represent letters of the alphabet as well as numbers. Common codes used for this have 7 or 8 bits per word and are referred to as alphanumeric codes. To detect possible errors in these codes, an additional bit, called a parity bit, is often added as the most significant bit.

Parity is a term used to identify whether a data word has an odd or even number of 1's. If a data word contains



FIGURE 1-4 7-segment LED display. (a) Segment labels. (b) Schematic of common-cathode type. (c) Schematic of common-anode type.

parity of the data word plus parity bit that it receives. If the receiving system detects odd parity in the received data word plus parity, it assumes an error has occurred and tells the sending system to send the data again. The system is then said to be using even parity. The system could have been set up to use (maintain) odd parity in a similar manner.

## ASCII

Table 1-2 shows several alphanumeric codes. The first of these is ASCII, or American Standard Code for Information Interchange. This is shown in the table as a 7-bit code. With 7 bits you can code up to 128 characters, which is enough for the full uppercase and lowercase

**TABLE 1-2  
COMMON ALPHANUMERIC CODES**

| ASCII<br>Symbol | HEX Code<br>for 7-Bit<br>ASCII | EBCDIC<br>Symbol | HEX Code<br>for<br>EBCDIC | ASCII<br>Symbol | HEX Code<br>for 7-Bit<br>ASCII | EBCDIC<br>Symbol | HEX Code<br>for<br>EBCDIC | ASCII<br>Symbol | HEX Code<br>for 7-Bit<br>ASCII | EBCDIC<br>Symbol | HEX Code<br>for<br>EBCDIC |
|-----------------|--------------------------------|------------------|---------------------------|-----------------|--------------------------------|------------------|---------------------------|-----------------|--------------------------------|------------------|---------------------------|
| NUL             | 00                             | NUL              | 00                        | *               | 2A                             | *                | 5C                        | T               | 54                             | T                | E3                        |
| SOH             | 01                             | SOH              | 01                        | +               | 2B                             | +                | 4E                        | U               | 55                             | U                | E4                        |
| STX             | 02                             | STX              | 02                        | ,               | 2C                             | ,                | 6B                        | V               | 56                             | V                | E5                        |
| ETX             | 03                             | ETX              | 03                        | .               | 2D                             | .                | 60                        | W               | 57                             | W                | E6                        |
| EOT             | 04                             | EOT              | 37                        | -               | 2E                             | -                | 4B                        | X               | 58                             | X                | E7                        |
| ENQ             | 05                             | ENQ              | 2D                        | /               | 2F                             | /                | 61                        | Y               | 59                             | Y                | E8                        |
| ACK             | 06                             | ACK              | 2E                        | 0               | 30                             | 0                | F0                        | Z               | 5A                             | Z                | E9                        |
| BEL             | 07                             | BEL              | 2F                        | 1               | 31                             | 1                | F1                        | !               | 5B                             | [                | AD                        |
| BS              | 08                             | BS               | 16                        | 2               | 32                             | 2                | F2                        | x               | 5C                             | NL               | 15                        |
| HT              | 09                             | HT               | 05                        | 3               | 33                             | 3                | F3                        | !               | 5D                             | ]                | DD                        |
| LF              | 0A                             | LF               | 25                        | 4               | 34                             | 4                | F4                        | ,               | 5E                             | ^                | 5F                        |
| VT              | 0B                             | VT               | 0B                        | 5               | 35                             | 5                | F5                        | -               | 5F                             | -                | 6D                        |
| FF              | 0C                             | FF               | 0C                        | 6               | 36                             | 6                | F6                        | -               | 60                             | RES              | 14                        |
| CR              | 0D                             | CR               | 0D                        | 7               | 37                             | 7                | F7                        | a               | 61                             | a                | 81                        |
| S0              | 0E                             | S0               | 0E                        | 8               | 38                             | 8                | F8                        | b               | 62                             | b                | 82                        |
| S1              | 0F                             | S1               | 0F                        | 9               | 39                             | 9                | F9                        | c               | 63                             | c                | 83                        |
| DLE             | 10                             | DLE              | 10                        | -               | 3A                             | -                | 7A                        | d               | 64                             | d                | 84                        |
| DC1             | 11                             | DC1              | 11                        | -               | 3B                             | -                | 5E                        | e               | 65                             | e                | 85                        |
| DC2             | 12                             | DC2              | 12                        | -               | 3C                             | -                | 4C                        | f               | 66                             | f                | 86                        |
| DC3             | 13                             | DC3              | 13                        | -               | 3D                             | -                | 7E                        | g               | 67                             | g                | 87                        |
| DC4             | 14                             | DC4              | 35                        | -               | 3E                             | -                | 6E                        | h               | 68                             | h                | 88                        |
| NAK             | 15                             | NAK              | 3D                        | ?               | 3F                             | ?                | 6F                        | i               | 69                             | i                | 89                        |
| SYN             | 16                             | SYN              | 32                        | @               | 40                             | @                | 7C                        | j               | 6A                             | j                | 91                        |
| ETB             | 17                             | EOB              | 26                        | A               | 41                             | A                | C1                        | k               | 6B                             | k                | 92                        |
| CAN             | 18                             | CAN              | 18                        | B               | 42                             | B                | C2                        | l               | 6C                             | l                | 93                        |
| EM              | 19                             | EM               | 19                        | C               | 43                             | C                | C3                        | m               | 6D                             | m                | 94                        |
| SUB             | 1A                             | SUB              | 3F                        | D               | 44                             | D                | C4                        | n               | 6E                             | n                | 95                        |
| ESC             | 1B                             | BYP              | 24                        | E               | 45                             | E                | C5                        | o               | 6F                             | o                | 96                        |
| FS              | 1C                             | FLS              | 1C                        | F               | 46                             | F                | C6                        | p               | 70                             | p                | 97                        |
| GS              | 1D                             | GS               | 1D                        | G               | 47                             | G                | C7                        | q               | 71                             | q                | 98                        |
| RS              | 1E                             | RDS              | 1E                        | H               | 48                             | H                | C8                        | r               | 72                             | r                | 99                        |
| US              | 1F                             | US               | 1F                        | I               | 49                             | I                | C9                        | s               | 73                             | s                | A2                        |
| SP              | 20                             | SP               | 40                        | J               | 4A                             | J                | D1                        | t               | 74                             | t                | A3                        |
| !               | 21                             | !                | 5A                        | K               | 4B                             | K                | D2                        | u               | 75                             | u                | A4                        |
| "               | 22                             | "                | 7F                        | L               | 4C                             | L                | D3                        | v               | 76                             | v                | A5                        |
| #               | 23                             | #                | 7B                        | M               | 4D                             | M                | D4                        | w               | 77                             | w                | A6                        |
| \$              | 24                             | \$               | 5B                        | N               | 4E                             | N                | D5                        | x               | 78                             | x                | A7                        |
| %               | 25                             | %                | 6C                        | O               | 4F                             | O                | D6                        | y               | 79                             | y                | A8                        |
| &               | 26                             | &                | 50                        | P               | 50                             | P                | D7                        | z               | 7A                             | z                | A9                        |
| '               | 27                             | '                | 7D                        | Q               | 51                             | Q                | D8                        | {               | 7B                             | {                | 8B                        |
| (               | 28                             | (                | 4D                        | R               | 52                             | R                | D9                        |                 | 7C                             |                  | 9B                        |
| )               | 29                             | )                | 5D                        | S               | 53                             | S                | E2                        | }               | 7D                             | }                | 4A                        |
|                 |                                |                  |                           |                 |                                |                  | DEL                       | DEL             | 7E                             | DEL              | 07                        |

**TABLE 1-3**  
**DEFINITIONS OF CONTROL CHARACTERS**

|      |                     |     |                        |
|------|---------------------|-----|------------------------|
| NULL | Null                | DC1 | Direct control 1       |
| SOH  | Start of heading    | DC2 | Direct control 2       |
| STX  | Start text          | DC3 | Direct control 3       |
| ETX  | End text            | DC4 | Direct control 4       |
| EOT  | End of transmission | NAK | Negative acknowledge   |
| ENQ  | Enquiry             | SYN | Synchronous idle       |
| ACK  | Acknowledge         | ETB | End transmission block |
| BEL  | BS                  | CAN | Cancel                 |
| BS   | Backspace           | EM  | End of medium          |
| HT   | Horizontal tab      | SUB | Substitute             |
| LF   | Line feed           | ESC | Escape                 |
| VT   | Vertical tab        | FS  | Form separator         |
| FF   | Form feed           | GS  | Group separator        |
| CR   | Carriage return     | RS  | Record separator       |
| SO   | Shift out           | US  | Unit separator         |
| SI   | Shift in            |     |                        |
| DLE  | Data link escape    |     |                        |

| INPUTS |   |                 | OUTPUTS |                  |
|--------|---|-----------------|---------|------------------|
| A      | B | C <sub>IN</sub> | S       | C <sub>OUT</sub> |
| 0      | 0 | 0               | 0       | 0                |
| 0      | 0 | 1               | 1       | 0                |
| 0      | 1 | 0               | 1       | 0                |
| 0      | 1 | 1               | 0       | 1                |
| 1      | 0 | 0               | 1       | 0                |
| 1      | 0 | 1               | 0       | 1                |
| 1      | 1 | 0               | 0       | 1                |
| 1      | 1 | 1               | 1       | 1                |

$$S = A \oplus B \oplus C_{IN}$$

$$C_{OUT} = A \cdot B + C_{IN} (A \oplus B)$$

(a)

$$\begin{array}{r}
 10011010 \\
 + 11011100 \\
 \hline
 10111010
 \end{array}$$

↑ Carry ✓

(b)

FIGURE 1-5 Binary addition. (a) Truth table for 2 bits plus carry. (b) Addition of two 8-bit words.

✓ alphabet, numbers, punctuation marks, and control characters. The code is arranged so that if only uppercase letters, numbers, and a few control characters are needed, the lower 6 bits are all that are required. If a parity check is wanted, a parity bit is added to the basic 7-bit code in the MSB position. The binary word 1100 0100, for example, is the ASCII code for uppercase D with odd parity. Table 1-3 gives the meanings of the control character symbols used in the ASCII code table.

### EBCDIC

Another alphanumeric code commonly encountered in IBM equipment is the Extended Binary-Coded Decimal Interchange Code or EBCDIC. This is an 8-bit code without parity. A ninth bit can be added for parity. To save space in Table 1-2, the eight binary digits of EBCDIC are represented by their 2-digit hex equivalent.

## ARITHMETIC OPERATIONS ON BINARY, HEX, AND BCD NUMBERS

### Binary Arithmetic

#### ADDITION

Figure 1-5a shows the truth table for addition of two binary digits and a carry in ( $C_{IN}$ ) from addition of previous digits. Figure 1-5b shows the result of adding two 8-bit binary numbers together using these rules. Assuming that  $C_{IN} = 1$ ,  $1 + 0 + C_{IN} =$  a sum of 0 and a carry into the next digit, and  $1 + 1 + C_{IN} =$  a sum of 1 and a carry into the next digit because the result in any digit position can only be a 1 or a 0.

### 2'S-COMPLEMENT SIGNS-AND-MAGNITUDE BINARY

When you handwrite a number that represents some physical quantity such as temperature, you can simply put a + sign in front of the number to indicate that the

number is positive, or you can write a - sign to indicate that the number is negative. However, if you want to store values such as temperatures, which can be positive or negative, in a computer memory, there is a problem. Since the computer memory can store only 1's and 0's, some way must be established to represent the sign of the number with a 1 or a 0.

✓ A common way to represent signed numbers is to reserve the most significant bit of the data word as a sign bit and to use the rest of the bits of the data word to represent the size (magnitude) of the quantity. A computer that works with 8-bit words will use the MSB (bit 7) as the sign bit and the lower 7 bits to represent the magnitude of the numbers. The usual convention is to represent a positive number with a 0 sign bit and a negative number with a 1 sign bit.

To make computations with signed numbers easier, the magnitude of negative numbers is represented in a special form called 2's complement. The 2's complement of a binary number is formed by inverting each bit of the data word and adding 1 to the result. Some examples should help clarify all of this.

The number  $+7_{10}$  is represented in 8-bit sign-and-magnitude form as 00000111. The sign bit is 0, which indicates a positive number. The magnitude of positive numbers is represented in straight binary, so 00000111 in the least significant bits represents  $7_{10}$ .

✓ To represent  $-7_{10}$  in 8-bit 2's-complement sign-and-magnitude form, start with the 8-bit code for  $+7$ , 0000 0111. Invert each bit, including the MSB, to get 1111 1000. Then add 1 to get 11111001. This result is the correct representation of  $-7_{10}$ . Figure 1-6 shows some more examples of positive and negative numbers expressed in 8-bit sign-and-magnitude form. For practice, try generating each of these yourself to see if you get the same result.

To reverse this procedure and find the magnitude of a number expressed in sign-and-magnitude form, proceed as follows. If the number is positive, as indicated

| Sign bit |           |
|----------|-----------|
| + 7      | 0 0000111 |
| + 46     | 0 0101110 |
| + 105    | 0 1101001 |
| - 12     | 1 1110100 |
| - 54     | 1 1001010 |
| - 117    | 1 0001011 |
| - 46     | 1 1010010 |

} Sign and  
two's complement  
of magnitude

FIGURE 1-6 Positive and negative numbers represented with a sign bit and 2's complement.

✓ the sign bit being a 0, then the least significant 7 bits represent the magnitude directly in binary. If the number is negative, as indicated by the sign bit being a 1, then the magnitude is expressed in 2's complement. To get the magnitude of this negative number expressed in standard binary, invert each bit of the data word, including the sign bit, and add 1 to the result. For example, given the word 11101011, invert each bit to get 00010100. Then add 1 to get 00010101. This equals  $21_{10}$ , so you know that the original numbers represent  $-21_{10}$ . Again, try reconverting a few of the numbers in Figure 1-6 for practice.

Figure 1-7 shows some examples of addition of signed binary numbers of this type. Sign bits are added together just as the other bits are. Figure 1-7a shows the results of adding two positive numbers. The sign bit of the result is zero, so the result is positive. The second example, in Figure 1-7b, adds a -9 to a +13 or, in effect, subtracts 9 from 13. As indicated by the zero sign bit, the result of 4 is positive and in true binary form.

Figure 1-7c shows the result of adding a -13 to a smaller positive number, +9. The sign bit of the result is a 1. This indicates that the result is negative and the magnitude is in 2's-complement form. To reconvert a 2's complement result to a signed number in true binary form:

- ✓ 1. Invert each bit to produce the 1's complement.
- ✓ 2. Add 1.
- ✓ 3. Put a minus sign in front to indicate that the result is negative.

The final example, in Figure 1-7d, shows the result of adding two negative numbers. The sign bit of the result is a 1, so the result is negative and in 2's-complement form. Again, inverting each bit, adding 1, and prefixing a minus sign will put the result in a more recognizable form.

Now let's consider the range of numbers that can be represented with 8 bits in sign-and-magnitude form. Eight bits can represent a maximum of  $2^8$  or 256 numbers. Since we are representing both positive and negative numbers, half of this range will be positive and

half negative. Therefore, the range is -128 to +127. Here are the sign-and-magnitude binary representations for these values:

|                     |                   |
|---------------------|-------------------|
| <del>128</del>      | +128 Out of Range |
| 0 1111111           | +127              |
| 0 0000001           | +1                |
| 0 0000000           | zero              |
| <del>11111111</del> | -1                |
| <del>F F</del>      |                   |
| 1 0000001           | -127              |
| 1 0000000           | -128              |
| <del>8 0 h</del>    |                   |

If you like number patterns, you might notice that this scheme shifts the normal codes for 128 to 255 downward to represent -128 to -1.

If a computer is storing signed numbers as 16-bit words, then a much larger range of numbers can be represented. Since 16 bits gives  $2^{16}$  or 65,536 possible values, the range for 16-bit sign-and-magnitude numbers is -32,768 to +32,767. Operations with 16-bit sign-and-magnitude numbers are done the same way as operations with 8-bit sign-and-magnitude numbers.

|                         |                                                                                                                                                                                    |
|-------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| $+13$<br>$+ 9$<br>$+22$ | $00001101$<br>$00001001$<br>$00010110$                                                                                                                                             |
|                         | ↑ Sign bit is 0<br>so result is positive                                                                                                                                           |
|                         | (a)                                                                                                                                                                                |
| $+13$<br>$- 9$<br>$+ 4$ | $00001101$<br>$11110111$ 2's complement for -9 with sign bit<br>$1 00000100$                                                                                                       |
|                         | ↑ Sign bit is 0<br>so result is positive<br>Ignore carry                                                                                                                           |
|                         | (b)                                                                                                                                                                                |
| $+ 9$<br>$-13$<br>$- 4$ | $00001001$<br>$11110011$ 2's complement for -13 with sign bit<br>$1 11111100$ Sign bit is 1<br>$00000011$ So invert each bit<br>$+ 1$ Add 1<br>$-00000100$ Prefix with minus sign  |
|                         | (c)                                                                                                                                                                                |
| $- 9$<br>$-13$<br>$-22$ | $11110111$ 2's complement,<br>$11110011$ sign-and-magnitude form<br>$11101010$ Sign bit is 1<br>$00010101$ So invert each bit<br>$+ 1$ Add 1<br>$-00010110$ Prefix with minus sign |
|                         | (d)                                                                                                                                                                                |

FIGURE 1-7 Addition of signed binary numbers. (a) +9 and +13. (b) -9 and +13. (c) +9 and -13. (d) -9 and -13.

| INPUTS |   |          | OUTPUTS |           |
|--------|---|----------|---------|-----------|
| A      | B | $B_{IN}$ | D       | $B_{OUT}$ |
| 0      | 0 | 0        | 0       | 0         |
| 0      | 0 | 1        | 1       | 1         |
| 0      | 1 | 0        | 1       | 1         |
| 0      | 1 | 1        | 0       | 1         |
| 1      | 0 | 0        | 1       | 0         |
| 1      | 0 | 1        | 0       | 0         |
| 1      | 1 | 0        | 0       | 0         |
| 1      | 1 | 1        | 1       | 1         |

$$\text{DIFFERENCE} = A + B + B_{IN}$$

$$\text{BORROW} = A \cdot B + (A \oplus B) \cdot B_{IN}$$

(a)

$$\begin{array}{r} 91_{10} \\ - 46_{10} \\ \hline 45_{10} \end{array}$$

$$\begin{array}{r} 10101010 \\ - 01100100 \\ \hline 01000110 \end{array}$$

(b)

$$\begin{array}{r} 01011011 \\ - 00101110 \\ \hline \text{One's comp} & 11010001 \\ \text{Add 1} & + 1 \\ \hline \text{Two's comp} & 11010010 \end{array}$$

Invert each bit

$+11010010$

$\boxed{1} 00101101 = 45_{10}$

Indicates result positive and in true binary form

Carry

(c)

$$\begin{array}{r} 77_{10} \quad 01001101 \\ - 88_{10} \quad 01011000 \\ \hline -11_{10} \end{array}$$

Complement

$10100111$

$+ 1$

$10101000$

Two's comp

Carry

$$\begin{array}{r} 01001101 \\ + 10101000 \\ \hline \boxed{0} 11110101 \end{array}$$

Complement

$00001010$

Add one

$+ 1$

$- 1011 = -11_{10}$

Indicates result negative and in two's complement form

(d)

FIGURE 1-8 Binary subtraction. (a) Truth table for 2 bits and borrow. (b) Pencil method. (c) 2's-complement positive result. (d) 2's-complement negative result.

## SUBTRACTION

There are two common methods for doing binary subtraction. These are the pencil method and the 2's-complement add method. Figure 1-8a shows the truth table for binary subtraction of two binary digits A and B. Also included in the truth table is the effect of a borrow-in,  $B_{IN}$ , from subtracting previous digits. Figure 1-8b shows an example of the "pencil" method of subtracting two 8-bit numbers. Using the truth table, this method is done the same way that you do decimal subtraction.

A second method of performing binary subtraction is by adding the 2's-complement representation of the bottom number (subtrahend) to the top number (minuend). Figure 1-8c shows how this is done. First represent the top number in sign-and-magnitude form. Then form the 2's-complement sign-and-magnitude representation for the negative of the bottom number. Finally, add the two parts formed. For the example in Figure 1-8c, the sign of the result is a 0, which indicates that the result is positive and in true form. The final carry produced by the addition can be ignored. Figure 1-8d shows another example of this method of subtraction. In this case the bottom number is larger than the top number. Again, represent the top number in sign-and-magnitude form, produce the 2's-complement sign-and-magnitude form for the negative of the bottom number, and add the two together. The sign bit of the result is a 1 for this example. This indicates that the result is negative and its magnitude is represented in 2's-complement form. To

get the result into a form that is more recognizable to you, invert each bit of the result, add 1 to it, and put a minus sign in front of it as shown in Figure 1-8d.

Problems that may occur when doing signed addition or subtraction are overflow and underflow. If the magnitude of the number produced by adding two signed numbers is larger than the number of bits available to represent the magnitude, the result will "overflow" into the sign bit position and give an incorrect result. For example, if the signed positive number 01001001 is added to the signed positive number 01101101, the result is 10110110. The 1 in the MSB of this result indicates that it is negative, which is obviously incorrect for the sum of two positive numbers. In a similar manner, doing an 8-bit signed subtraction that produces a magnitude greater than  $-128$  will cause an "underflow" into the sign bit and produce an incorrect result.

For simplicity the examples shown use 8 bits, but the method works for any number of bits. This method may seem awkward, but it is easy to do in a computer or microprocessor because it requires only the simple operations of inverting and adding.

## MULTIPLICATION

There are several methods of doing binary multiplication. Figure 1-9 shows what is called the pencil method because it is the same way you learned to multiply decimal numbers. The top number, or multiplicand, is multiplied by the least significant digit of the bottom number, or multiplier. The partial product is written

$$\begin{array}{r}
 & 1011 \\
 & \times 1001 \\
 \hline
 & 1011 \\
 & 0000 \\
 & 0000 \\
 & 1011 \\
 \hline
 & 1100011
 \end{array}
 \begin{array}{l}
 \text{MULTIPLICAND} \\
 \text{MULTIPLIER} \\
 \text{PARTIAL PRODUCTS} \\
 \text{PRODUCT}
 \end{array}$$

FIGURE 1-9 Binary multiplication.

down. The top number is then multiplied by the next digit of the multiplier. The resultant partial product is written down under the last, but shifted one place to the left. Adding all the partial products gives the total product. This method works well when doing multiplication by hand, but it is not practical for a computer because the type of shifts required makes it awkward to implement.

\* One of the multiplication methods used by computers is repeated addition. To multiply  $7 \times 55$ , for example, the computer can just add up seven 55's. For large numbers, however, this method is slow. To multiply  $786 \times 253$ , for example, requires 252 add operations.

Most computers use an add-and-shift-right method. This method takes advantage of the fact that for binary multiplication, the partial product can only be either the top number exactly if the multiplier digit is a 1 or a 0 if the multiplier digit is a 0. The method does the same thing as the pencil method, except that the partial products are added as they are produced and the sum of the partial products is shifted right rather than each partial product being shifted left.

A point to note about multiplying numbers is the number of bits the product requires. For example, multiplying two 4-bit numbers can give a product with as many as 8 bits, and two 8-bit numbers can give a 16-bit product.

## DIVISION

Binary division can also be performed in several ways. Figure 1-10 shows two examples of the pencil method. This is the same process as decimal long division. However, it is much simpler than decimal long division

$$\begin{array}{r}
 & 01100 \quad \text{QUOTIENT} \\
 \text{DIVISOR} \quad 110 \overline{)1001000} \quad \text{DIVIDEND} & 12 \\
 -110 \\
 \hline
 & 110 \\
 -110 \\
 \hline
 & 0
 \end{array}$$

(a)

$$\begin{array}{r}
 & 6.25 \\
 100 \overline{)11001.00} & 4 \overline{)26 \\
 -100 \\
 \hline
 & 100 \\
 -100 \\
 \hline
 & 0100
 \end{array}$$

(b)

FIGURE 1-10 Binary division.

because the digits of the result (quotient) can only be 0 or 1. A division is attempted on part of the dividend. If this is not possible because the divisor is larger than that part of the dividend, a 0 is entered in the quotient. Another attempt is then made to divide using one more digit of the dividend. When a division is possible, a 1 is entered in the quotient. The divisor is then subtracted from the portion of the dividend used. As with standard long division, the process is continued until all the dividend is used. As shown in Figure 1-10b, 0's can be added to the right of the binary point and division continued to convert a remainder to a binary equivalent.

Another method of division that is easier for computers and microprocessors to perform uses successive subtractions. The divisor is subtracted from the dividend and from each successive remainder until a borrow is produced. The desired quotient is 1 less than the number of subtractions needed to produce a borrow. This method is simple, but for large numbers it is slow.

For faster division of large numbers, computers use a subtract-and-shift-left method that is essentially the same process you go through with a pencil long division.

## Hexadecimal Addition and Subtraction

People working with computers or microprocessors often use hexadecimal as a shorthand way of representing long binary numbers such as memory addresses. It is therefore useful to be able to add and subtract hexadecimal numbers.

### ADDITION

As shown in Figure 1-11a, one way to add two hexadecimal numbers is to convert each hexadecimal number to its binary equivalent, add the two binary numbers, and convert the binary result back to its hex equivalent. For converting to binary, remember that each hex digit represents 4 binary digits.

A second method, shown in Figure 1-11b, works directly with the hex numbers. When adding hex digits, a carry is produced whenever the sum is 16 decimal or greater. Another way of saying this is that the value of a carry in hex is 16 decimal. For the least significant digits in Figure 1-11b, an A in hex is 10 in decimal and an F is 15 in decimal. These add to give 25 decimal. This is greater than 16, so mentally subtract 16 from the 25 to give a carry and a remainder of 9. The 9 is written down and the carry is added to the next digit column. In this column 7 plus 3 plus a carry gives a decimal 11, or B in hex.

$$\begin{array}{r}
 & \text{Carry} \\
 & \downarrow \\
 & 7A & 0111 & 1010 & 7^1 & A_{16} \\
 +3F & +0011 & 1111 & & +3 & F_{16} \\
 \hline
 & B9 & 1011 & 1001 & 11_{10} & 25_{10} \\
 & & & & \checkmark & B_{16} \\
 & \checkmark & (a) & & & 9_{16} \\
 & & & & & \checkmark (b)
 \end{array}$$

FIGURE 1-11 Hexadecimal addition.

$$\begin{array}{r}
 77_{16} = 119_{10} \\
 - 3B_{16} = - 59_{10} \\
 \hline
 3C_{16} = 80_{10}
 \end{array}$$

FIGURE 1-12 Hexadecimal subtraction.

You may use whichever method seems easier to you and gives you consistently right answers. If you are doing a great deal of hexadecimal arithmetic, you might buy an electronic calculator specifically designed to do decimal, binary, and hexadecimal arithmetic.

## SUBTRACTION

Hexadecimal subtraction is similar to decimal subtraction except that when a borrow is needed, 16 is borrowed from the next most significant digit. Figure 1-12 shows an example of this. It may help you to follow the example if you do partial conversions to decimal in your head. For example, 7 plus a borrowed 16 is 23. Subtracting B or 11 leaves 12 or C in hexadecimal. Then 3 from the 6 left after a borrow leaves 3, so the result is 3CH.

## BCD Addition and Subtraction

In systems where the final result of a calculation is to be displayed, such as a calculator, it may be easier to work with numbers in a BCD format. These codes, as shown in Table 1-1, represent each decimal digit, 0 through 9, by its 4-bit binary equivalent.

## ADDITION

BCD can have no digit-word with a value greater than 9. Therefore, a carry must be generated if the result of a BCD addition is greater than 1001 or 9. Figure 1-13

$$\begin{array}{r}
 \text{BCD} \\
 \begin{array}{r}
 35 \\
 + 23 \\
 \hline
 58
 \end{array}
 \end{array}
 \quad
 \begin{array}{r}
 \text{BCD} \\
 \begin{array}{r}
 0011\ 0101 \\
 + 0010\ 0011 \\
 \hline
 0101\ 1000
 \end{array}
 \end{array}
 \quad (a) \\[10mm]
 \begin{array}{r}
 \text{BCD} \\
 \begin{array}{r}
 7 \\
 + 5 \\
 \hline
 12
 \end{array}
 \end{array}
 \quad
 \begin{array}{r}
 \text{BCD} \\
 \begin{array}{r}
 0111 \\
 + 0101 \\
 \hline
 1100 \quad \text{INCORRECT BCD} \\
 + 0110 \quad \text{ADD 6}
 \end{array}
 \end{array}
 \quad (b) \\[10mm]
 \begin{array}{r}
 \text{BCD} \\
 \begin{array}{r}
 9 \\
 + 8 \\
 \hline
 17
 \end{array}
 \end{array}
 \quad
 \begin{array}{r}
 \text{BCD} \\
 \begin{array}{r}
 1001 \\
 + 1000 \\
 \hline
 0001\ 0001 \quad \text{INCORRECT BCD} \\
 0000\ 0110 \quad \text{ADD 6} \\
 \hline
 0001\ 0111 \quad \text{CORRECT BCD 17}
 \end{array}
 \end{array}
 \quad (c)$$

FIGURE 1-13 BCD addition. (a) No correction needed. (b) Correction needed because of illegal BCD result. (c) Correction needed because of carry-out of BCD digit.

$$\begin{array}{r}
 17 = 0001\ 0111 \\
 - 9 = 0000\ 1001 \\
 \hline
 8 = 0000\ 1110 \quad \text{ILLEGAL BCD} \\
 - 0110 \\
 \hline
 0000\ 1000 \quad \text{SUBTRACT 6} \\
 \hline
 0000\ 1000 \quad \text{CORRECT BCD}
 \end{array}$$

FIGURE 1-14 BCD subtraction.

shows three examples of BCD addition. The first, in Figure 1-13a, is very straightforward because the sum for each BCD digit is less than 9. The result is the same as it would be for adding standard binary.

For the second example, in Figure 1-13b, adding BCD 7 to BCD 5 produces 1100. This is a correct binary result of 12, but it is an illegal BCD code. To convert the result to BCD format, a correction factor of 6 is added. The result of adding 6 is 0001 0010, which is the legal BCD code for 12.

Figure 1-13c shows another case where a correction factor must be added. The initial addition of 9 and 8 produces 0001 0001. Even though the lower four digits are less than 9, this is an incorrect BCD result because a carry out of bit 3 of the BCD digit-word was produced. This carry out of bit 3 is often called an auxiliary carry. Adding the correction factor of 6 gives the correct BCD result of 0001 0111 or 17.

To summarize, a correction factor of 6 must be added if the result in the lower 4 bits is greater than 9 or if the initial addition produces a carry out of bit 3 of any BCD digit-word. This correction is sometimes called a decimal adjust operation.

The reason for the correction factor of 6 is that in BCD we want a carry into the next digit after 1001 or 9, but in binary a carry out of the lower 4 bits does not occur until after 1111 or 15. The difference between the two carry points is 6, so you have to add 6 to produce the desired carry if the result of an addition in any BCD digit is more than 1001.

## SUBTRACTION

Figure 1-14 shows a subtraction, BCD 17 (0001 0111) minus BCD 9 (0000 1001). The initial result, 0000 1110, is not a legal BCD number. Whenever this occurs in BCD subtraction, 6 must be subtracted from the initial result to produce the correct BCD result. For the example shown in Figure 1-14, subtracting 6 gives a correct BCD result of 0000 1000 or 8.

The correction factor of 6 must be subtracted from any BCD digit-word if that digit-word is greater than 1001, or if a borrow from the next higher digit was required to do the subtraction.

## BASIC DIGITAL DEVICES

Microcomputers such as those we discuss throughout this book often contain basic logic gates as "glue" between LSI (large-scale integration) devices. For troubleshooting these systems, it is important to be able to predict logic levels at any point directly from the schematic rather than having to work your way through a

truth table for each gate. This section should help refresh your memory of basic logic functions and help you remember how to quickly analyze logic gate circuits.

### Inverting and Noninverting Buffers

Figure 1-15 shows the schematic symbols and truth tables for simple buffers and logic gates. The first thing to remember about these symbols is that the shape of the symbol indicates the logic function performed by the device. The second thing to remember about these symbols is that a bubble or no bubble indicates the assertion level for an input or output signal. Let's review how modern logic designers use these symbols.

The first symbol for a buffer in Figure 1-15a has no bubbles on the input or output. Therefore, the input is active high and the output is active high. We read this symbol as follows: If the input A is asserted high, then the output Y will be asserted high. The rest of the truth table is covered by the assumption that if the A input is not asserted high, then the Y output will not be asserted high.

The next two symbols for a buffer each contain a bubble. The bubble on the output of the first of these

indicates that the output is active low. The input has no bubble, so it is active high. You can read the function of the device directly from the schematic symbol as follows. If the A Input is asserted high, then the Y output will be asserted low. This device simply changes the assertion level of a signal. The output Y will always have a logic state which is the complement or inverse of that on the input, so the device is usually referred to as an inverter.

The second schematic symbol for an inverter in Figure 1-15a has the bubble on the input. We draw the symbol this way when we want to indicate that we are using the device to change an asserted-low signal to an asserted-high signal. For example, if we pass the signal CS through this device, it becomes CS. The symbol tells you directly that if the input is asserted low, then the output will be asserted high. Now let's review how you express the functions of logic gates using this approach.

### Logic Gates

Figure 1-15b shows the symbols and truth tables for simple logic gates. A symbol with a flat back and a round front indicates that the device performs the logical AND function. This means that the output will be asserted if the A input is asserted and the B input is asserted. Again, bubbles or no bubbles are used to indicate the assertion level of each input and output. The first AND symbol in Figure 1-15b has no bubbles, so the inputs and the output are active high. The output then will be asserted high if the A input is asserted high and the B input is asserted high. The bubble on the output of the second AND symbol in Figure 1-15b indicates that this device, commonly called a NAND gate, has an active low output. If the A input is asserted high and the B input is asserted high, then the Y output will be asserted low. Look at the truth table in Figure 1-15b to see if you agree with this.

Figure 1-15c shows the other two possible cases for the AND symbol. The first of these has bubbles on the inputs and on the output. If you see this symbol in a schematic, you should immediately see that the output will be asserted low if the A input is asserted low and the B input is asserted low. The second AND symbol in Figure 1-15c has no bubble on the output, so the output will be asserted high if the A and B inputs are both asserted low.

A logic symbol with a curved back indicates that the output of the device will be asserted if the A input is asserted or the B input of the device is asserted. Again, bubbles or no bubbles are used to indicate the assertion level for inputs and outputs. Note in Figure 1-15b and c that each of the AND symbol forms has an equivalent OR symbol form. An AND symbol with active high inputs and an active high output, for example, represents the same device (a 74LS08 perhaps) as an OR symbol with active low inputs and an active low output. Use the truth table in Figure 1-15b to convince yourself of this. The bubbled-OR representation tells you that if one input is asserted low, the output will be low, regardless of the state of the other input. As we will show later in this chapter, this is often a useful way to think of the operation of an AND gate.



FIGURE 1-15 Buffers and logic gates. (a) Buffers. (b) AND-NAND. (c) OR-NOR. (d) Exclusive OR.



(a)



(b)



(c)

FIGURE 1-16 FPLA, PROM, and PAL programmed to implement some simple logic functions. (a) FPLA. (b) PROM. (c) PAL.

Figure 1-15d shows the symbol and truth table for an exclusive OR gate and for an exclusive NOR gate. The output of an exclusive OR gate will be high if the logic levels on the two inputs are different. The output of an exclusive NOR gate will be high if the logic levels on the two inputs are the same.

You need to be familiar with all these symbols, because most logic designers will use the symbol that best describes the function they want a device to perform in a particular circuit.

## Programmable Logic Devices

Instead of using discrete gates, modern microcomputer systems usually use *programmable logic devices* such as PLAs, PROMs, or PALs to implement the "glue" logic between LSI devices. To refresh your memory, Figure 1-16 shows the internal structure of each of these devices. As you can see, they all consist of a programmable AND-OR matrix, so they can easily implement any sum-of-products logic expression. Each AND gate in these figures has up to four inputs, but to simplify the drawing only a single input line is shown. Likewise, the OR gates have several inputs, but are shown with a single input line to simplify the drawing. These devices are programmed by blowing out fuses, which are represented in the figure by Xs. An X in the figure indicates that the fuse is intact and makes a connection between, for example, the output of an AND gate and one of the inputs of an OR gate. A dot at the intersection of two wires indicates a hard-wired connection implemented during manufacture.

In a *programmable logic array* (PLA) or *field programmable logic array* (FPLA), both the AND matrix and the OR matrix are programmable by leaving in fuses or blowing them out. The two programmable matrixes make FPLAs very flexible, but difficult to program.

In a *programmable read-only memory* or PROM, the AND matrix is fixed and just the OR matrix is programmable by leaving in fuses or blowing them out. PROMs implement all the possible product terms for the input variables, so they are useful as code converters.

In a *programmable array logic* device or PAL, the connections in the OR matrix are fixed and the AND matrix connections are programmable. PALs are often used to implement combinational logic and address decoders in microcomputer systems.

A computer program is usually used to develop the fuse map for an FPLA, PROM, or PAL. Once developed, the fuse-map file is downloaded to a programmer which blows fuses or stores charges to actually program the device.

## Latches, Flip-Flops, Registers, and Counters

### THE D LATCH

A *latch* is a digital device that stores a 1 or a 0 on its output. Figure 1-17a shows the schematic symbol and truth table for a D latch. The device functions as follows. If the enable input CK is low, the logic level present on the D input will have no effect on the Q and Q̄ outputs.



FIGURE 1-17 Latches and flip-flops. (a) D latch. (b) D flip-flop.

This is indicated in the truth table by an X in the D column. If the enable input is high, a high or a low on the D input will be passed to the Q output. In other words, the Q output will follow the D input as long as the enable input is high. The  $\bar{Q}$  output will contain the complement of the logic state on Q. When the enable input is made low again, the state on Q at that time will be latched there. Any changes on D will have no effect on Q until the enable input is made high again. When the enable input goes low, then, the state present on D just before the enable goes low will be stored on the Q output. Keep this operation in mind as you read about the D flip-flop in the next section.

### THE D FLIP-FLOP

Figure 1-17b shows the schematic symbol and the truth table for a typical D flip-flop. The small triangle next to the CK input of this device tells you that the Q and  $\bar{Q}$  outputs are updated when a rising signal edge is applied to the CK input. The up arrows in the clock column of the truth table also indicate that a 1 or 0 on the D input will be copied to the Q output when the clock input goes from low to high. In other words, the D flip-flop takes a snapshot of whatever state is on the D input when the clock goes high, and displays the "photo" on the Q output. If the clock input is low, a change on D will have no effect on the output. Likewise, if the clock input is high, a change on D will have no effect on the Q output. Contrast this operation with that of the D latch to make sure you understand the difference between the two devices.

The D flip-flop in Figure 1-17b also has direct set (S) and reset (R) inputs. A flip-flop is considered set if its Q output is a 1. It is reset if its Q output is a 0. The bubbles on the set and reset inputs tell you that these inputs are active low. The truth table for the D flip-flop in Figure 1-17b indicates that the set and reset inputs are asynchronous. This means that if the set input is asserted low, the output will be set, regardless of the

states on the D and the clock inputs. Likewise, if the reset input is asserted low, the Q output will be reset, regardless of the state of the D and clock inputs. The Xs in the D and CK columns of the truth table remind you that these inputs are "don't cares" if set or reset is asserted. The condition indicated by the asterisks (\*) is a nonstable condition; that is, it will not persist when reset or clear inputs return to their inactive (high) level.

### REGISTERS

Flip-flops can be used individually or in groups to store binary data. A register is a group of D flip-flops connected in parallel, as shown in Figure 1-18a. A binary word applied to the data inputs of this register will be transferred to the Q outputs when the clock input is made high. The binary word will remain stored on the Q outputs until a new binary word is applied to the D inputs and a low-to-high signal is applied to the clock input. Other circuitry can read the stored binary word from the Q outputs at any time without changing its value.

If the Q output of each flip-flop in the register is connected to the D input of the next as shown in Figure 1-18b, then the register will function as a *shift register*. A 1 applied to the first D input will be shifted to the first Q output by a clock pulse. The next clock pulse will shift this 1 to the output of the second flip-flop. Each additional clock pulse will shift the 1 to the next flip-flop in the register. Some shift registers allow you to load a binary word into the register and shift the loaded word left or right when the register is clocked. As we will show later, the ability to shift binary numbers is very useful.

### COUNTERS

Flip-flops can also be connected to make devices whose outputs step through a binary or other count sequence



FIGURE 1-18 Registers. (a) Simple data storage. (b) Shift register.



(a)

| Q3 | Q2 | Q1 | Q0 |
|----|----|----|----|
| 0  | 0  | 0  | 0  |
| 0  | 0  | 0  | 1  |
| 0  | 0  | 1  | 0  |
| 0  | 0  | 1  | 1  |
| 0  | 1  | 0  | 0  |
| 0  | 1  | 0  | 1  |
| 0  | 1  | 1  | 0  |
| 0  | 1  | 1  | 1  |
| 1  | 0  | 0  | 0  |
| 1  | 0  | 0  | 1  |
| 1  | 0  | 1  | 0  |
| 1  | 0  | 1  | 1  |
| 1  | 1  | 0  | 0  |
| 1  | 1  | 0  | 1  |
| 1  | 1  | 1  | 0  |
| 1  | 1  | 1  | 1  |

(b)

FIGURE 1-19 Four-bit, presettable binary counter. (a) Schematic symbol. (b) Count sequence.

when they are clocked. Figure 1-19a shows a schematic symbol and count sequence for a presettable 4-bit binary counter. The main point we want to review here is how a presettable counter functions, so there is no need to go into the internal circuitry of the device. If the reset input is asserted, the Q outputs will all be made 0's. After the reset signal is unasserted, each clock pulse will cause the binary count on the outputs to be incremented by 1. As shown in Figure 1-19b, the count sequence will go from 0000 to 1111. If the outputs are at 1111, then the next clock pulse will cause the outputs to "roll over" to 0000 and a carry pulse to be sent out the carry output.

This carry pulse can be used as the clock input for another counter. Counters can be cascaded to produce as large a count sequence as is needed for a particular application. The maximum count for a binary counter is  $2^N - 1$ , where N is the number of flip-flops.

Now, suppose that we want the counter to start counting from some number other than 0000. We can do this by applying the desired number to the four data inputs and asserting the load input. For example, if we apply a binary 6, 0110, to the data inputs and assert the load input, this value will be transferred to the Q outputs. After the load signal is unasserted, the next clock signal will increment the Q outputs to 0111 or 7.

### ROMs, RAMs, and Buses

The next topics we need to review are the devices that store large numbers of binary words and how several of these devices can be connected on common data lines.

#### ROMS

The term ROM stands for *read-only memory*. There are several types of ROM that can be written to, read, erased, and written to with new data, but the main feature of ROMs is that they are *nonvolatile*. This means that the information stored in them is not lost when the power is removed from them.

Figure 1-20a shows the schematic symbol of a common ROM. As indicated by the eight data outputs, D0 to D7, this ROM stores 8-bit data words. The data outputs are three-state outputs. This means that each output can be at a logic low state, a logic high state, or a high-impedance floating state. In the high-impedance state an output is essentially disconnected from anything connected to it. If the CE input of the ROM is not asserted, then all the outputs will be in the high-



(a)



(b)

FIGURE 1-20 ROMs. (a) Schematic symbol. (b) Connection in parallel.

impedance state. Most ROMs also switch to a lower-power-consumption standby mode if  $\overline{CE}$  is not asserted. If the  $\overline{CE}$  input is asserted, the device will be powered up, and the output buffers will be enabled. Therefore, the outputs will be at a normal logic low or logic high state. If you don't happen to remember, you will soon see why this is important.

You can think of the binary words stored in the ROM as being in a long, numbered list. The number that identifies the location of each stored word in the list is called its **address**. You can tell the number of binary words stored in the ROM by the number of address inputs. The number of words is equal to  $2^N$ , where N is the number of address lines. The device in Figure 1-20a has 15 address lines, A0 to A14, so the number of words is  $2^{15}$  or 32,768. In a data sheet this device would be referred to as a 32K  $\times$  8 ROM. This means it has 32K addresses with 8 bits per address.

In order to get a particular word onto the outputs of the ROM, you have to do two things. You have to apply the address of that word to the address inputs, A0 to A14, and you have to assert the  $\overline{CE}$  input to power up the device and to enable the three-state outputs.

Now, let's see why we want three-state outputs on this ROM. Suppose that we want to store more than 32K data words. We can do this by connecting two or more ROMs in parallel, as shown in Figure 1-20b. The address lines connect to each device in parallel, so we can address one of the 32,768 words in each. A set of parallel lines used to send addresses or data to several devices in this way is called a **bus**. The data outputs of the ROMs are likewise connected in parallel so that any one of the ROMs can output data on the common data bus. If these ROMs had standard two-state outputs, a serious problem would occur when both ROMs tried to output data words on the bus. The resulting argument between data outputs would probably destroy some of the outputs and give meaningless information on the data bus. Since the ROMs have three-state outputs, however, we can use external circuitry to make sure that only one ROM at a time has its outputs enabled. The very important principle here is that whenever several outputs are interconnected on a bus, the outputs should all be three-state, and only one set of outputs should be enabled at a time.

At the beginning of this section we mentioned that some ROMs can be erased and rewritten or reprogrammed with new data. Here's a summary of the different types of ROMs.

**Mask-programmed ROM**—Programmed during manufacture; cannot be altered.

**PROM**—User programs by blowing fuses; cannot be altered except to blow additional fuses.

**EPROM**—Electrically programmable by user; erased by shining ultraviolet light on quartz window in package.

**EEPROM**—Electrically programmable by user; erased with electrical signals, so it can be reprogrammed in circuit.

**Flash EEPROM**—Electrically programmable by user; erased electrically, so it can be reprogrammed in circuit.

## STATIC AND DYNAMIC RAMS

The name RAM stands for *random-access memory*, but since ROMs are also random access, the name probably should be *read-write memory*. RAMs are also used to store binary words. A static RAM is essentially a matrix of flip-flops. Therefore, we can write a new data word in a RAM location at any time by applying the word to the flip-flop data inputs and clocking the flip-flops. The stored data word will remain on the flip-flop outputs as long as the power is left on. This type of memory is volatile because data is lost when the power is turned off.

Figure 1-21 shows the schematic symbol for a common RAM. This RAM has 12 address lines, A0 to A11, so it stores  $2^{12}$  (4096) binary words. The eight data lines tell you that the RAM stores 8-bit words. When we are reading a word from the RAM, these lines function as outputs. When we are writing a word to the RAM, these lines function as inputs. The chip enable input,  $\overline{CE}$ , is used to enable the device for a read or for a write. The  $\overline{R/W}$  input will be asserted high if we want to read from the RAM or asserted low if we want to write a word to the RAM. Here's how all these lines work for reading from and writing to the device.

To write to the RAM, we apply the desired address to the address inputs, assert the  $\overline{CE}$  input low to turn on the device, and assert the  $\overline{R/W}$  input low to tell the RAM we want to write to it. We then apply the data word we want to store to the data lines of the RAM for a specified time. To read a word from the RAM, we address the desired word, assert  $\overline{CE}$  low to turn on the device, and assert  $\overline{R/W}$  high to tell the RAM we want to read from it. For a read operation the output buffers on the data lines will be enabled and the addressed data word will be present on the outputs.

The static RAMs we have just reviewed store binary words in a matrix of flip-flops. In dynamic RAMs (DRAMs), binary 1's and 0's are stored as an electric charge or no charge on a tiny capacitor. Since these tiny capacitors take up less space on a chip than a flip-flop



FIGURE 1-21 RAM schematic symbol.

would, a dynamic RAM chip can store many more bits than the same size static RAM chip. The disadvantage of dynamic RAMs is that the charge leaks off the tiny capacitors. The logic state stored in each capacitor must be refreshed every 2 milliseconds (ms) or so. A device called a *dynamic RAM refresh controller* can be used to refresh a large number of dynamic RAMs in a system. Some newer dynamic RAM devices contain built-in refresh circuitry, so they appear static to external circuitry.

## Arithmetic Logic Units

An *arithmetic logic unit*, or *ALU*, is a device that can AND, OR, add, subtract, and perform a variety of other operations on binary words. Figure 1-22a shows a block diagram for the 74LS181, which is a 4-bit ALU. This device can perform any one of 16 logic functions or any one of 16 arithmetic functions on two 4-bit binary words. The function performed on the two words is determined by the logic level applied to the mode input M and by the 4-bit binary code applied to the select inputs S0 to S3.

Figure 1-22b shows the truth table for the 74LS181. In this truth table, A represents the 4-bit binary word applied to the A0 to A3 inputs, and B represents the 4-bit binary word applied to the B0 to B3 inputs. F represents the 4-bit binary word that will be produced on the F0 to F3 outputs. If the mode input M is high,

the device will perform one of 16 logic functions on the two words applied to the A and B inputs. For example, if M is high and we make S3 high, S2 low, S1 high, and S0 high, the 4-bit word on the A inputs will be ANDed with the 4-bit word on the B inputs. The result of this ANDing will appear on the F outputs. Each bit of the A word is ANDed with the corresponding bit of the B word to produce the result on F. Figure 1-22c shows an example of ANDing two words with this device. As you can see in this example, an output bit is high only if the corresponding bit is high in both the A word and the B word.

For another example of the operation of the 74LS181, suppose that the M input is high, S3 is high, S2 is high, S1 is high, and S0 is low. According to the truth table, the device will now OR each bit in the A word with the corresponding bit in the B word and give the result on the corresponding F output. Figure 1-22c shows the result that will be produced by ORing two 4-bit words. Figure 1-22c also shows for your reference the result that would be produced by exclusive ORing these two 4-bit words together.

If the M input of the 74LS181 is low, then the device will perform one of 16 arithmetic functions on the A and B words. Again, the result of the operation will be put on the F outputs. Several 74LS181s can be cascaded to operate on words longer than 4 bits. The ripple-carry input,  $\bar{C}_N$ , allows a carry from an operation on previous words to be included in the current operation. If the  $\bar{C}_N$

**(a)**

| SELECTION |    |    |    | M = H<br>LOGIC<br>FUNCTIONS | ACTIVE-HIGH DATA             |                                     |
|-----------|----|----|----|-----------------------------|------------------------------|-------------------------------------|
| S3        | S2 | S1 | S0 |                             | M = L; ARITHMETIC OPERATIONS | $\bar{C}_N = H$ (NO CARRY)          |
| L         | L  | L  | L  | F = $\bar{A}$               | F = A                        | F = A PLUS 1                        |
| L         | L  | L  | H  | F = $\bar{A} + B$           | F = A + B                    | F = (A + B) PLUS 1                  |
| L         | L  | H  | L  | F = $\bar{A}B$              | F = A + $\bar{B}$            | F = (A + $\bar{B}$ ) PLUS 1         |
| L         | L  | H  | H  | F = 0                       | F = MINUS 1 (2's COMPL)      | F = 0                               |
| L         | H  | L  | L  | F = $\bar{A}\bar{B}$        | F = A PLUS $\bar{A}\bar{B}$  | F = A PLUS $\bar{A}\bar{B}$ PLUS 1  |
| L         | H  | L  | H  | F = B                       | F = (A + B) PLUS AB          | F = (A + B) PLUS AB PLUS 1          |
| L         | H  | H  | L  | F = A $\oplus$ B            | F = A MINUS B MINUS 1        | F = A MINUS B                       |
| L         | H  | H  | H  | F = $\bar{A}\bar{B}$        | F = $\bar{A}\bar{B}$ MINUS 1 | F = $\bar{A}\bar{B}$                |
| H         | L  | L  | L  | F = $\bar{A} + B$           | F = A PLUS AB                | F = A PLUS AB PLUS 1                |
| H         | L  | L  | H  | F = A $\oplus$ B            | F = A PLUS B                 | F = A PLUS B PLUS 1                 |
| H         | L  | H  | L  | F = B                       | F = (A + $\bar{B}$ ) PLUS AB | F = (A + $\bar{B}$ ) PLUS AB PLUS 1 |
| H         | L  | H  | H  | F = AB                      | F = AB MINUS 1               | F = AB                              |
| H         | H  | L  | L  | F = 1                       | F = A PLUS A*                | F = A PLUS A PLUS 1                 |
| H         | H  | L  | H  | F = A + $\bar{B}$           | F = (A + B) PLUS A           | F = (A + B) PLUS A PLUS 1           |
| H         | H  | H  | L  | F = A + B                   | F = (A + $\bar{B}$ ) PLUS A  | F = (A + $\bar{B}$ ) PLUS A PLUS 1  |
| H         | H  | H  | H  | F = A                       | F = A MINUS 1                | F = A                               |

\*EACH BIT IS SHIFTED TO THE NEXT MORE SIGNIFICANT BIT POSITION

**(b)**

|                          |                          |                          |                  |                              |                  |                               |
|--------------------------|--------------------------|--------------------------|------------------|------------------------------|------------------|-------------------------------|
| $A = A_3\ A_2\ A_1\ A_0$ | $B = B_3\ B_2\ B_1\ B_0$ | $A = 1\ 0\ 1\ 0$         | $B = 0\ 1\ 1\ 0$ | $A = 1\ 0\ 1\ 0$             | $B = 0\ 1\ 1\ 0$ | $A = 1\ 0\ 1\ 0$              |
| $F = F_3\ F_2\ F_1\ F_0$ |                          | $B = 0\ 1\ 1\ 0$         |                  | $F = A \cdot B = 0\ 0\ 1\ 0$ |                  | $B = 0\ 1\ 1\ 0$              |
|                          |                          | $F = A + B = 1\ 1\ 1\ 0$ |                  |                              |                  | $F = A \oplus B = 1\ 1\ 0\ 0$ |

**(c)**

FIGURE 1-22 Arithmetic logic unit (ALU). (a) Schematic symbol. (b) Truth table. (c) Sample AND, OR, and XOR operations.

# CHAPTER

## Computers, Microcomputers, and Microprocessors—An Introduction

We live in a computer-oriented society, and we are constantly bombarded with a multitude of terms relating to computers. Before getting started with the main flow of the book, we will try to clarify some of these terms and to give an overview of computers and computer systems.

### OBJECTIVES

At the conclusion of this chapter, you should be able to:

1. Define the terms *microcomputer*, *microprocessor*, *hardware*, *software*, *firmware*, *timesharing*, *multitasking*, *distributed processing*, and *multiprocessing*.
2. Describe how a microcomputer fetches and executes an instruction.
3. List the registers and other parts in the 8086/8088 execution unit and bus interface unit.
4. Describe the function of the 8086/8088 queue.
5. Demonstrate how the 8086/8088 calculates memory addresses.

### TYPES OF COMPUTERS

#### Mainframes

Computers come in a wide variety of sizes and capabilities. The largest and most powerful are often called **mainframes**. Mainframe computers may fill an entire room. They are designed to work at very high speeds with large data words, typically 64 bits or greater, and they have massive amounts of memory. Computers of this type are used for military defense control, for business data processing (in an insurance company, for example), and for creating computer graphics displays for science fiction movies. Examples of this type of computer are the IBM 4381, the Honeywell DPS8, and the Cray Y-MP/832. The fastest and most powerful mainframes are called **supercomputers**. Figure 2-1a, p. 20, shows a photograph of a Cray Y-MP/832 supercom-

puter, which contains eight central processors and 32 million 64-bit words of memory.

#### Minicomputers

Scaled-down versions of mainframe computers are often called **minicomputers**. The main unit of a minicomputer usually fits in a single rack or box. A minicomputer runs more slowly, works directly with smaller data words (often 32-bit words), and does not have as much memory as a mainframe. Computers of this type are used for business data processing, industrial control (for an oil refinery, for example), and scientific research. Examples of this type of computer are the Digital Equipment Corporation VAX 6360 and the Data General MV/8000II. Figure 2-1b shows a photograph of a Digital Equipment Corporation's VAX 6360 minicomputer.

#### Microcomputers

As the name implies, **microcomputers** are small computers. They range from small controllers that work directly with 4-bit words and can address a few thousand bytes of memory to larger units that work directly with 32-bit words and can address billions of bytes of memory. Some of the more powerful microcomputers have all or most of the features of earlier minicomputers. Therefore, it has become very hard to draw a sharp line between these two types. One distinguishing feature of a microcomputer is that the CPU is usually a single integrated circuit called a **microprocessor**. Older books often used the terms **microprocessor** and **microcomputer** interchangeably, but actually the microprocessor is the CPU to which you add ROM, RAM, and ports to make a **microcomputer**. A later section in this chapter discusses the evolution of different types of microprocessors. Microcomputers are used in everything from smart sewing machines to computer-aided design systems. Examples of microcomputers are the Intel 8051 single-chip controller; the SDK-86, a single-board computer design kit; the IBM Personal Computer (PC); and the Apple Macintosh computer. The Intel 8051 microcontroller is contained in a single 40-pin chip. Figure 2-2a, p. 21, shows the SDK-86 board, and Figure 2-2b shows the Compaq 386/25 system.



(a)



(b)

FIGURE 2-1 (a) Photograph of Cray Y-MP/832 computer. (Courtesy Cray Research, Inc., and photographer, Paul Shambroom.) (b) Photograph of VAX 6360 minicomputer. (Courtesy Digital Equipment Corp.)

## Computerizing an Electronics Factory—Problem

Now, suppose that we want to "computerize" an electronics company. By this we mean that we want to make computer use available to as many people in the company as possible as cheaply as possible. We want the engineers to have access to a computer which can help them design circuits. People in the drafting department should have access to a computer which can be used for computer-aided drafting. The accounting department should have access to a computer for doing all the financial bookkeeping. The warehouse should have access to a computer to help with inventory control. The manufacturing department should have access to a computer for controlling machines and testing finished products. The president, vice presidents, and supervisors should have access to a computer to help them with long-range planning. Secretaries should have access to a computer for word processing. Salespeople should have access to a computer to help them keep track of current pricing, product availability, and commissions. There are several ways to provide all the needed computer power. One solution is to simply give everyone an individual personal computer. The problem with this approach is that it makes it difficult for different people to access commonly needed data. In the next sections we show you two ways to provide computer power and common data to many users.

### TIMESHARING AND MULTITASKING SYSTEMS

One common method of providing computer access is a *timesharing* system such as shown in Figure 2-3, p. 22. Several video terminals are connected to the computer through direct wires or through telephone lines. The terminal can be on the user's desk or even in the user's home. The rate at which a user usually enters data is very slow compared with the rate at which a computer can process the data. Therefore, the computer can serve many users by dividing its time among them in small increments. In other words, the computer works on user

1's program for perhaps 20 milliseconds (ms), then works on user 2's program for 20 ms, then works on user 3's program for 20 ms, and so on, until all the users have had a turn. In a few milliseconds the computer will get back to user 1 again and repeat the cycle. To each user it will appear as if he or she has exclusive use of the computer because the computer processes data as fast as the user enters it. A timesharing system such as this allows several users to interact with the computer at the same time. Each user can get information from or store information in the large memory attached to the computer. Each user can have an inexpensive printer attached to the terminal or can direct program or data output to a high-speed printer attached directly to the computer.

An airline ticket reservation computer might use a timesharing system such as this to allow users from all over the country to access flight information and make reservations. A time-multiplexed or time-sliced system such as this can also allow a computer to control many machines or processes in a factory. A computer is much faster than the machines or processes. Therefore, it can

## HOW COMPUTERS AND MICROCOMPUTERS ARE USED—AN EXAMPLE

The following sections are intended to give you an overview of how computers are interfaced with users to do useful work. These sections should help you understand many of the features designed into current microprocessors and where this book is heading.



(a)



(b)

FIGURE 2-2 (a) Photograph of Intel SDK-86 board. (Intel Corp.) (b) Photograph of Compaq 386/25. (Compaq Corp.)

check and adjust many pressures, temperatures, motor speeds, etc., before it needs to get back and recheck the first one. A system such as this is often called a **multitasking system** because it appears to be doing many tasks at the same time.

Now let's take another look at our problem of computerizing the electronics company. We could put a powerful computer in some central location and run wires from it to video display terminals on users' desks. Each user could then run the program needed to do a particular task. The accountant could run a ledger program, the secretary could run a word processing program, etc. Each user could access the computer's large data memory. Incidentally, a large collection of data stored in a computer's memory is often referred to as a *data base*. For a small company a system such as this might be adequate. However, there are at least two potential problems.

✓ The first potential problem is, "What happens if the computer is not working?" The answer to this question is that everything grinds to a halt. In a situation where people have become dependent on the computer, not much gets done until the computer is up and running again. The old saying about putting all your eggs in one basket comes to mind here.

✓ The second potential problem of the simple timesharing system is saturation. As the number of users increases, the time it takes the computer to do each user's task increases also. Eventually the computer's response time to each user becomes unreasonably long. People get very upset about the time they have to wait.

### DISTRIBUTED PROCESSING OR MULTIPROCESSING

A partial solution for the two potential problems of a simple timesharing system is to use a distributed



✓ FIGURE 2-3 Block diagram of a computer timesharing system.

**processing system.** Figure 2-4 shows a block diagram for such a system. The system has a powerful central computer with a large memory and a high-speed printer, as does the simple timesharing system described previously. However, in this system each user has a microcomputer instead of simply a video display terminal. In other words, each user station is an independently functioning microcomputer with a CPU, ROM, RAM, and probably magnetic or optical disk memory. This means that a person can do many tasks locally on the microcomputer

without having to use the large computer at all. Since the microcomputers are connected to the large computer through a network, however, a user can access the computing power, memory, or other resources of the large computer when needed.

✓ Distributing the processing to multiple computers or processors in a system has several advantages. First, if the large computer goes down, the local microcomputers can continue working until they need to access the large computer for something. Second, the burden on the



✓ FIGURE 2-4 Block diagram of a distributed processing computer system.

large computer is reduced greatly, because much of the computing is done by the local microcomputers. Finally, the distributed processing approach allows the system designer to use a local microcomputer that is best suited to the task it has to do.

## COMPUTERIZED ELECTRONICS COMPANY OVERVIEW

Distributed processing seems to be the best way to go about computerizing our electronics factory. Engineers can have personal computers or engineering workstations on their desks. With these they can use available programs to design and test circuits. They can access the large computer if they need data from its memory. Through the telephone lines, the engineer with a personal computer can access data in the memory of other computers all over the world. The drafting people can have personal computers for simple work, or large computer-aided design systems for more complex work. Completed work can be stored in the memory of the large computer. The production department can have networked computers to keep track of product flow and to control the machines which actually mount components on circuit boards, etc. The accounting department can use personal computers with spreadsheet programs to work with financial data kept in the memory of the large computer. The warehouse supervisor can likewise use a personal computer with an inventory program to keep personal records and those in the large computer's memory updated. Corporate officers can have personal computers tied into the network. They then can interact with any of the other systems on the network. Salespeople can have portable personal computers that they can carry with them in the field. They can communicate with the main computer over the telephone lines using a modem. Secretaries doing word processing can use individual word processing units or personal computers. Users can also send messages to one another over the network. The specifics of a computer system such as this will obviously depend on the needs of the individual company for which the system is designed.

## SUMMARY AND DIRECTION FROM HERE

The main concepts that you should take with you from this section are timesharing or multitasking and distributed processing or multiprocessing. As you work your way through the rest of this book, keep an overview

of the computerized electronics company in the back of your mind. The goal of this book is to teach you how the microcomputers and other parts of a system such as this work, how the parts are connected together, and how the system is programmed at different levels.

## OVERVIEW OF MICROCOMPUTER STRUCTURE AND OPERATION

Figure 2-5 shows a block diagram for a simple microcomputer. The major parts are the central processing unit or CPU, memory, and the input and output circuitry or I/O. Connecting these parts are three sets of parallel lines called buses. The three buses are the address bus, the data bus, and the control bus. Let's take a brief look at each of these parts.

### Memory

The memory section usually consists of a mixture of RAM and ROM. It may also have magnetic floppy disks, magnetic hard disks, or optical disks. Memory has two purposes. The first purpose is to store the binary codes for the sequences of instructions you want the computer to carry out. When you write a computer program, what you are really doing is writing a sequential list of instructions for the computer. The second purpose of the memory is to store the binary-coded data with which the computer is going to be working. This data might be the inventory records of a supermarket, for example.

### Input/Output

The input/output or I/O section allows the computer to take in data from the outside world or send data to the outside world. Peripherals such as keyboards, video display terminals, printers, and modems are connected to the I/O section. These allow the user and the computer to communicate with each other. The actual physical devices used to interface the computer buses to external systems are often called ports. Ports in a computer function just as shipping ports do for a country. An input port allows data from a keyboard, an A/D converter, or some other source to be read into the computer under control of the CPU. An output port is used to send data from the computer to some peripheral, such as a video display terminal, a printer, or a D/A converter. Physically,



FIGURE 2-5 Block diagram of a simple microcomputer.

the simplest type of input or output port is just a set of parallel D flip-flops. If they are being used as an input port, the D inputs are connected to the external device, and the Q outputs are connected to the data bus which runs to the CPU. Data will then be transferred through the latches when they are enabled by a control signal from the CPU. In a system where they are being used as an output port, the D inputs of the latches are connected to the data bus, and the Q outputs are connected to some external device. Data sent out on the data bus by the CPU will be transferred to the external device when the latches are enabled by a control signal from the CPU.

## Central Processing Unit

The central processing unit or CPU controls the operation of the computer. In a microcomputer the CPU is a microprocessor, as we discussed in an earlier section of the chapter. The CPU fetches binary-coded instructions from memory, decodes the instructions into a series of simple actions, and carries out these actions in a sequence of steps.

The CPU also contains an address counter or instruction pointer register, which holds the address of the next instruction or data item to be fetched from memory; general-purpose registers, which are used for temporary storage of binary data; and circuitry, which generates the control bus signals.

12

## Address Bus

The address bus consists of 16, 20, 24, or 32 parallel signal lines. On these lines the CPU sends out the address of the memory location that is to be written to or read from. The number of memory locations that the CPU can address is determined by the number of address lines. If the CPU has N address lines, then it can directly address  $2^N$  memory locations. For example, a CPU with 16 address lines can address  $2^{16}$  or 65,536 memory locations, a CPU with 20 address lines can address  $2^{20}$  or 1,048,576 locations, and a CPU with 24 address lines can address  $2^{24}$  or 16,777,216 locations. When the CPU reads data from or writes data to a port, it sends the port address out on the address bus.

## Data Bus

The data bus consists of 8, 16, or 32 parallel signal lines. As indicated by the double-ended arrows on the data bus line in Figure 2-5, the data bus lines are bidirectional. This means that the CPU can read data in from memory or from a port on these lines, or it can send data out to memory or to a port on these lines. Many devices in a system will have their outputs connected to the data bus, but only one device at a time will have its outputs enabled. Any device connected on the data bus must have three-state outputs so that its outputs can be disabled when it is not being used to put data on the bus.

## Control Bus

The control bus consists of 4 to 10 parallel signal lines. The CPU sends out signals on the control bus to enable the outputs of addressed memory devices or port devices. Typical control bus signals are Memory Read, Memory Write, I/O Read, and I/O Write. To read a byte of data from a memory location, for example, the CPU sends out the memory address of the desired byte on the address bus and then sends out a Memory Read signal on the control bus. The Memory Read signal enables the addressed memory device to output a data word onto the data bus. The data word from memory travels along the data bus to the CPU.

## Hardware, Software, and Firmware

When working around computers, you hear the terms hardware, software, and firmware almost constantly. *Hardware* is the name given to the physical devices and circuitry of the computer. *Software* refers to the programs written for the computer. *Firmware* is the term given to programs stored in ROMs or in other devices which permanently keep their stored information.

## Summary of Important Points So Far

- A computer or microcomputer consists of memory, a CPU, and some input/output circuitry.
- These three parts are connected by the address bus, the data bus, and the control bus.
- The sequence of instructions or program for a computer is stored as binary numbers in successive memory locations.
- The CPU fetches an instruction from memory, decodes the instruction to determine what actions must be done for the instruction, and carries out these actions.

## EXECUTION OF A THREE-INSTRUCTION PROGRAM

To give you a better idea of how the parts of a microcomputer function together, we will now describe the actions a simple microcomputer might go through to carry out (execute) a simple program. The three instructions of the program are

1. Input a value from a keyboard connected to the port at address 05H.
2. Add 7 to the value read in.
3. Output the result to a display connected to the port at address 02H.

Figure 2-6 shows in diagram form and sequential list form the actions that the computer will perform to execute these three instructions.



#### PROGRAM

1. INPUT A VALUE FROM PORT 05.
2. ADD 7 TO THIS VALUE.
3. OUTPUT THE RESULT TO PORT 02.

#### SEQUENCE

- 1A CPU SENDS OUT ADDRESS OF FIRST INSTRUCTION TO MEMORY.
- 1B CPU SENDS OUT MEMORY READ CONTROL SIGNAL TO ENABLE MEMORY.
- 1C INSTRUCTION BYTE SENT FROM MEMORY TO CPU ON DATA BUS.
- 2A ADDRESS NEXT MEMORY LOCATION TO GET REST OF INSTRUCTION.
- 2B SEND MEMORY READ CONTROL SIGNAL TO ENABLE MEMORY.
- 2C PORT ADDRESS BYTE SENT FROM MEMORY TO CPU ON DATA BUS.
- 2D CPU SENDS OUT PORT ADDRESS ON ADDRESS BUS.
- 2E CPU SENDS OUT INPUT READ CONTROL SIGNAL TO ENABLE PORT.
- 2F DATA FROM PORT SENT TO CPU ON DATA BUS.
- 3A CPU SENDS ADDRESS OF NEXT INSTRUCTION TO MEMORY.
- 3B CPU SENDS MEMORY READ CONTROL SIGNAL TO ENABLE MEMORY.
- 3C INSTRUCTION BYTE FROM MEMORY SENT TO CPU ON DATA BUS.
- 4A CPU SENDS NEXT ADDRESS TO MEMORY TO GET REST OF INSTRUCTION.
- 4B CPU SENDS MEMORY READ CONTROL SIGNAL TO ENABLE MEMORY.
- 4C NUMBER 07H SENT FROM MEMORY TO CPU ON DATA BUS.
- 5A CPU SENDS ADDRESS OF NEXT INSTRUCTION TO MEMORY.
- 5B CPU SENDS MEMORY READ CONTROL SIGNAL TO ENABLE MEMORY.
- 5C INSTRUCTION BYTE FROM MEMORY SENT TO CPU ON DATA BUS.
- 6A CPU SENDS OUT NEXT ADDRESS TO GET REST OF INSTRUCTION.
- 6B CPU SENDS OUT MEMORY READ CONTROL SIGNAL TO ENABLE MEMORY.
- 6C PORT ADDRESS BYTE SENT FROM MEMORY TO CPU ON DATA BUS.
- 6D CPU SENDS OUT PORT ADDRESS ON ADDRESS BUS.
- 6E CPU SENDS OUT DATA TO PORT ON DATA BUS.
- 6F CPU SENDS OUT OUTPUT WRITE SIGNAL TO ENABLE PORT.

| (a)            |                   |                |
|----------------|-------------------|----------------|
| MEMORY ADDRESS | CONTENTS (BINARY) | CONTENTS (HEX) |
| 00100H         | 11100100          | E4             |
| 00101H         | 00000101          | 05             |
| 00102H         | 00000100          | 04             |
| 00103H         | 00000111          | 07             |
| 00104H         | 11100110          | E6             |
| 00105H         | 00000010          | 02             |

(b)

FIGURE 2-6 (a) Execution of a three-step computer program. (b) Memory addresses and memory contents for a three-step program.

For this example, assume that the CPU fetches instructions and data from memory 1 byte at a time, as is done in the original IBM PC and its clones. Also assume that the binary codes for the instructions are in sequential memory locations starting at address 00100H. Figure 2-6b shows the actual binary codes that would be required in successive memory locations to execute this program on an IBM PC-type microcomputer.

The CPU needs an instruction before it can do anything, so its first action is to fetch an instruction byte from memory. To do this, the CPU sends out the address of the first instruction byte, in this case 00100H, to memory on the address bus. This action is represented by line 1A in Figure 2-6a. The CPU then sends out a Memory Read signal on the control bus (line 1B in the figure). The Memory Read signal enables the memory to output the addressed byte on the data bus. This action is represented by line 1C in the figure. The CPU reads in this first instruction byte (E4H) from the data bus and decodes it. By decode we mean that the CPU determines from the binary code read in what actions it is supposed to take. If the CPU is a microprocessor, it selects the sequence of microinstructions needed to

carry out the instruction read from memory. For the example instruction here, the CPU determines that the code read in represents an Input instruction. From decoding this instruction byte, the CPU also determines that it needs more information before it can carry out the instruction. The additional information the CPU needs is the address of the port that the data is to be input from. This port address part of the instruction is stored in the next memory location after the code for the input instruction.

To fetch this second byte of the instruction, the CPU sends out the next sequential address (00101H) to memory, as shown by line 2A in the figure. To enable the addressed memory device, the CPU also sends out another Memory Read signal on the control bus (line 2B). The memory then outputs the addressed byte on the data bus (line 2C). When the CPU has read in this second byte, 05H in this case, it has all the information it needs to execute the instruction.

To execute the input instruction, the CPU sends out the port address (05H) on the address bus (line 2D) and sends out an I/O Read signal on the control bus (line 2E). The I/O Read signal enables the addressed port

device to put a byte of data on the data bus (line 2F). The CPU reads in the byte of data and stores it in an internal register. This completes the fetching and execution of the first instruction.

Having completed the first instruction, the CPU must now fetch its next instruction from memory. To do this, it sends out the next sequential address (00102H) on the address bus (line 3A) and sends out a Memory Read signal on the control bus (line 3B). The Memory Read signal enables the memory device to put the addressed byte (04H) on the data bus (line 3C). The CPU reads in this instruction byte from the data bus and decodes it. From this instruction byte the CPU determines that it is supposed to add some number to the number stored in the internal register. The CPU also determines from decoding this instruction byte that it must go to memory again to get the next byte of the instruction, which contains the number that it is supposed to add. To get the required byte, the CPU will send out the next sequential address (00103H) on the address bus (line 4A) and another Memory Read signal on the control bus (line 4B). The memory will then output the contents of the addressed byte (the number 07H) on the data bus (line 4C). When the CPU receives this number, it will add it to the contents of the internal register. The result of the addition will be left in the internal register. This completes the fetching and executing of the second instruction.

The CPU must now fetch the third instruction. To do this, it sends out the next sequential address (00104H) on the address bus (line 5A) and sends out a Memory Read signal on the control bus (line 5B). The memory then outputs the addressed byte (E6H) on the data bus (line 5C). From decoding this byte, the CPU determines that it is now supposed to do an Output operation to a port. The CPU also determines from decoding this byte that it must go to memory again to get the address of the output port. To do this, it sends out the next sequential address (00105H) on the address bus (line 6A), sends out a Memory Read signal on the control bus (line 6B), and reads in the byte (02H) put on the data bus by the memory (line 6C). The CPU now has all the information that it needs to execute the Output instruction.

To output a data byte to a port, the CPU first sends out the address of the desired port on the address bus (line 6D). Next it outputs the data byte from the internal register on the data bus (line 6E). The CPU then sends out an I/O Write signal on the control bus (line 6F). This signal enables the addressed output port device so that the data from the data bus lines can pass through it to the LED displays. When the CPU removes the I/O Write signal to proceed with the next instruction, the data will remain latched on the output pins of the port device. The data will remain latched on the port until the power is turned off or until a new data word is output to the port. This is important because it means that the computer does not have to keep outputting a value over and over in order for it to remain on the output.

All the steps described above may seem like a great deal of work just to input a value from a keyboard, add 7 to it, and output the result to a display. Even a simple

microcomputer, however, can run through all these steps in a few microseconds.

## Summary of Simple Microcomputer Bus Operation

1. A microcomputer fetches each program instruction in sequence, decodes the instruction, and executes it. (Memory read)
2. The CPU in a microcomputer fetches instructions or reads data from memory by sending out an address on the address bus and a Memory Read signal on the control bus. The memory outputs the addressed instruction or data word to the CPU on the data bus. (Memory write)
3. The CPU writes a data word to memory by sending out an address on the address bus, sending out the data word on the data bus, and sending a Memory Write signal to memory on the control bus.
4. To read data from a port, the CPU sends out the port address on the address bus and sends an I/O Read signal to the port device on the control bus. Data from the port comes into the CPU on the data bus.
5. To write data to a port, the CPU sends out the port address on the address bus, sends out the data to be written to the port on the data bus, and sends an I/O Write signal to the port device on the control bus.

## MICROPROCESSOR EVOLUTION AND TYPES

As we told you in the preceding section, a microprocessor is used as the CPU in a microcomputer. There are now many different microprocessors available, so before we dig into the details of a specific device, we will give you a short microprocessor history lesson and an overview of the different types.

### Microprocessor Evolution

A common way of categorizing microprocessors is by the number of bits that their ALU can work with at a time. In other words, a microprocessor with a 4-bit ALU will be referred to as a 4-bit microprocessor, regardless of the number of address lines or the number of data bus lines that it has. The first commercially available microprocessor was the Intel 4004, produced in 1971. It contained 2300 PMOS transistors. The 4004 was a 4-bit device intended to be used with some other devices in making a calculator. Some logic designers, however, saw that this device could be used to replace PC boards full of combinational and sequential logic devices. Also, the ability to change the function of a system by just changing the programming, rather than redesigning the hardware, is very appealing. It was these factors that pushed the evolution of microprocessors.

In 1972 Intel came out with the 8008, which was capable of working with 8-bit words. The 8008, however,

required 20 or more additional devices to form a functional CPU. In 1971 Intel announced the 8080, which had a much larger instruction set than the 8008 and required only two additional devices to form a functional CPU. Also, the 8080 used NMOS transistors, so it operated much faster than the 8008. The 8080 is referred to as a second-generation microprocessor.

Soon after Intel produced the 8080, Motorola came out with the MC6800, another 8-bit general-purpose CPU. The 6800 had the advantage that it required only a +5-V supply rather than the -5-V, +5-V, and +12-V supplies required by the 8080. For several years the 8080 and the 6800 were the top-selling 8-bit microprocessors. Some of their competitors were the MOS Technology 6502, used as the CPU in the Apple II microcomputer, and the Zilog Z80, used as the CPU in the Radio Shack TRS-80 microcomputer.

As designers found more and more applications for microprocessors, they pressured microprocessor manufacturers to develop devices with architectures and features optimized for doing certain types of tasks. In response to the expressed needs, microprocessors have evolved in three major directions during the last 15 years.

## Dedicated or Embedded Controllers

One direction has been dedicated or embedded controllers. These devices are used to control "smart" machines, such as microwave ovens, clothes washers, sewing machines, auto ignition systems, and metal lathes. Texas Instruments has produced millions of their TMS-1000 family of 4-bit microprocessors for this type of application. In 1976 Intel introduced the 8048, which contains an 8-bit CPU, RAM, ROM, and some I/O ports all in one 40-pin package. Other manufacturers have followed with similar products. These devices are often referred to as microcontrollers. Some currently available devices in this category—the Intel 8051 and the Motorola MC6801, for example—contain programmable counters and a serial port (UART) as well as a CPU, ROM, RAM, and parallel I/O ports. A more recently introduced single-chip microcontroller, the Intel 8096, contains a 16-bit CPU, ROM, RAM, a UART, ports, timers, and a 10-bit analog-to-digital converter.

## Bit-Slice Processors

A second direction of microprocessor evolution has been bit-slice processors. For some applications, general-purpose CPUs such as the 8080 and 6800 are not fast enough or do not have suitable instruction sets. For these applications, several manufacturers produce devices which can be used to build a custom CPU. An example is the Advanced Micro Devices 2900 family of devices. This family includes 4-bit ALUs, multiplexers, sequencers, and other parts needed for custom-building a CPU. The term slice comes from the fact that these parts can be connected in parallel to work with 8-bit words, 16-bit words, or 32-bit words. In other words, a designer can add as many slices as needed for a particu-

lar application. The designer not only custom-designs the hardware of the CPU, but also custom-makes the instruction set for it using microcode.

## General-Purpose CPUs

The third major direction of microprocessor evolution has been toward general-purpose CPUs which give a microcomputer most or all of the computing power of earlier minicomputers. After Motorola came out with the MC6800, Intel produced the 8085, an upgrade of the 8080 that required only a +5-V supply. Motorola then produced the MC6809, which has a few 16-bit instructions, but is still basically an 8-bit processor. In 1978 Intel came out with the 8086, which is a full 16-bit processor. Some 16-bit microprocessors, such as the National PACE and the Texas Instruments 9900 family of devices, had been available previously, but the market apparently wasn't ready. Soon after Intel came out with the 8086, Motorola came out with the 16-bit MC68000, and the 16-bit race was off and running. The 8086 and the 68000 work directly with 16-bit words instead of with 8-bit words; they can address a million or more bytes of memory instead of the 64 Kbytes addressable by the 8-bit processors, and they execute instructions much faster than the 8-bit processors. Also, these 16-bit processors have single instructions for functions such as multiply and divide, which required a lengthy sequence of instructions on the 8-bit processors.

The evolution along this last path has continued on to 32-bit processors that work with gigabytes ( $10^9$  bytes) or terabytes ( $10^{12}$  bytes) of memory. Examples of these devices are the Intel 80386, the Motorola MC68020, and the National 32032.

Since we could not possibly describe in this book the operation and programming of even a few of the available processors, we confine our discussions primarily to one group of related microprocessors. The family we have chosen is the Intel 8086, 8088, 80186, 80188, 80286, 80386, 80486 family. Members of this family are very widely used in personal computers, business computer systems, and industrial control systems. Our experience has shown that learning the programming and operation of one family of microcomputers very thoroughly is much more useful than looking at many processors superficially. If you learn one processor family well, you will most likely find it quite easy to learn another when you have to.

## THE 8086 MICROPROCESSOR FAMILY—OVERVIEW

The Intel 8086 is a 16-bit microprocessor that is intended to be used as the CPU in a microcomputer. The term 16-bit means that its arithmetic logic unit, its internal registers, and most of its instructions are designed to work with 16-bit binary words. The 8086 has a 16-bit data bus, so it can read data from or write data to memory and ports either 16 bits or 8 bits at a time. The 8086 has a 20-bit address bus, so it can address any one of  $2^{20}$ , or 1,048,576, memory locations.

Each of the 1,048,576 memory addresses of the 8086 represents a byte-wide location. Sixteen-bit words will be stored in two consecutive memory locations. If the first byte of a word is at an even address, the 8086 can read the entire word in one operation. If the first byte of the word is at an odd address, the 8086 will read the first byte with one bus operation and the second byte with another bus operation. Later we will discuss this in detail. The main point here is that if the first byte of a 16-bit word is at an even address, the 8086 can read the entire word in one operation.

The Intel 8088 has the same arithmetic logic unit, the same registers, and the same instruction set as the 8086. The 8088 also has a 20-bit address bus, so it can address any one of 1,048,576 bytes in memory. The 8088, however, has an 8-bit data bus, so it can only read data from or write data to memory and ports 8 bits at a time. The 8086, remember, can read or write either 8 or 16 bits at a time. To read a 16-bit word from two successive memory locations, the 8088 will always have to do two read operations. Since the 8086 and the 8088 are almost identical, any reference we make to the 8086 in the rest of the book will also pertain to the 8088 unless we specifically indicate otherwise. This is done to make reading easier. The Intel 8088, incidentally, is used as the CPU in the original IBM Personal Computer, the IBM PC/XT, and several compatible personal computers.

The Intel 80186 is an improved version of the 8086, and the 80188 is an improved version of the 8088. In addition to a 16-bit CPU, the 80186 and 80188 each have programmable peripheral devices integrated in the same package. In a later chapter we will discuss these integrated peripherals. The instruction set of the 80186 and 80188 is a superset of the instruction set of the 8086. The term *superset* means that all the 8086 and 8088 instructions will execute properly on an 80186 or an 80188, but the 80186 and the 80188 have a few additional instructions. In other words, a program written for an 8086 or an 8088 is upward-compatible to an 80186 or an 80188, but a program written for an 80186 or an 80188 may not execute correctly on an 8086 or an 8088. In the instruction set descriptions in Chapter 6, we specifically indicate which instructions work only with the 80186 or 80188.

The Intel 80286 is a 16-bit, advanced version of the 8086 which was specifically designed for use as the CPU in a multilayer or multitasking microcomputer. When operating in its *real address mode*, the 80286 functions mostly as a fast 8086. Most programs written for an 8086 can be run on an 80286 operating in its real address mode. When operating in its *virtual address mode*, an 80286 has features which make it easy to keep users' programs separate from one another and to protect the system program from destruction by users' programs. In Chapter 15 we discuss the operation and use of the 80286. The 80286 is the CPU used in the IBM PC/AT personal computer.

The Intel 80386 is a 32-bit microprocessor which can directly address up to 4 gigabytes of memory. The 80386 contains more sophisticated features than the 80286 for use in multilayer and multitasking microcomputer

systems. In Chapter 15 we discuss the features of the 80386 and the 80486, which is an evolutionary step up from the 80386.

## 8086 INTERNAL ARCHITECTURE

Before we can talk about how to write programs for the 8086, we need to discuss its specific internal features, such as its ALU, flags, registers, instruction byte queue, and segment registers.

As shown by the block diagram in Figure 2-7, the 8086 CPU is divided into two independent functional parts, the *bus interface unit* or BIU, and the *execution unit* or EU. Dividing the work between these two units speeds up processing.

The BIU sends out addresses, fetches instructions from memory, reads data from ports and memory, and writes data to ports and memory. In other words, the BIU handles all transfers of data and addresses on the buses for the execution unit.

The execution unit of the 8086 tells the BIU where to fetch instructions or data from, decodes instructions, and executes instructions. Let's take a look at some of the parts of the execution unit.

### The Execution Unit

#### CONTROL CIRCUITRY, INSTRUCTION DECODER, AND ALU

As shown in Figure 2-7, the EU contains control circuitry which directs internal operations. A decoder in the EU translates instructions fetched from memory into a series of actions which the EU carries out. The EU has a 16-bit *arithmetic logic unit* which can add, subtract, AND, OR, XOR, increment, decrement, complement, or shift binary numbers.

### FLAG REGISTER

A *flag* is a flip-flop which indicates some condition produced by the execution of an instruction or controls certain operations of the EU. A 16-bit *flag register* in the EU contains nine active flags. Figure 2-8 shows the location of the nine flags in the flag register. Six of the nine flags are used to indicate some condition produced by an instruction. For example, a flip-flop called the *carry flag* will be set to a 1 if the addition of two 16-bit binary numbers produces a carry out of the most significant bit position. If no carry out of the MSB is produced by the addition, then the carry flag will be a 0. The EU thus effectively runs up a "flag" to tell you that a carry was produced.

The six conditional flags in this group are the *carry flag* (CF), the *parity flag* (PF), the *auxiliary carry flag* (AF), the *zero flag* (ZF), the *sign flag* (SF), and the *overflow flag* (OF). The names of these flags should give you hints as to what conditions affect them. Certain 8086 instructions check these flags to determine which of two alternative actions should be done in executing the instruction.



FIGURE 2-7 8086 internal block diagram. (Intel Corp.)

The three remaining flags in the flag register are used to control certain operations of the processor. These flags are different from the six conditional flags described above in the way they are set or reset. The six conditional flags are set or reset by the EU on the basis of the results of some arithmetic or logic operation. The control flags, are deliberately set or reset with specific instructions you put in your program. The three control flags are the trap flag (TF), which is used for single stepping through a program; the interrupt flag (IF), which is used to allow or prohibit the interruption of a program; and the direction flag (DF), which is used with string instructions.

Later we will discuss in detail the operation and use of the nine flags.

#### GENERAL-PURPOSE REGISTERS

Observe in Figure 2-7 that the EU has eight general-purpose registers, labeled AH, AL, BH, BL, CH, CL, DH, and DL. These registers can be used individually for temporary storage of 8-bit data. The AL register is also called the accumulator. It has some features that the other general-purpose registers do not have.

Certain pairs of these general-purpose registers can be used together to store 16-bit data words. The acceptable



FIGURE 2-8 8086 flag register format. (Intel Corp.)

register pairs are AH and AL, BH and BL, CH and CL, and DH and DL. The AH–AL pair is referred to as the AX register, the BH–BL pair is referred to as the BX register, the CH–CL pair is referred to as the CX register, and the DH–DL pair is referred to as the DX register.

The 8086 general-purpose register set is very similar to those of the earlier-generation 8080 and 8085 microprocessors. It was designed this way so that the many programs written for the 8080 and 8085 could easily be translated to run on the 8086 or the 8088. The advantage of using internal registers for the temporary storage of data is that, since the data is already in the EU, it can be accessed much more quickly than it could be accessed in external memory. Now let's look at the features of the BIU.

## The BIU

### THE QUEUE

While the EU is decoding an instruction or executing an instruction which does not require use of the buses,

the BIU fetches up to six instruction bytes for the following instructions. The BIU stores these prefetched bytes in a first-in-first-out register set called a queue.

When the EU is ready for its next instruction, it simply reads the instruction byte(s) for the instruction from the queue in the BIU. This is much faster than sending out an address to the system memory and waiting for memory to send back the next instruction byte or bytes. The process is analogous to the way a bricklayer's assistant fetches bricks ahead of time and keeps a queue of bricks lined up so that the bricklayer can just reach out and grab a brick when necessary. Except in the cases of JMP and CALL instructions, where the queue must be dumped and then reloaded starting from a new address, this prefetch-and-queue scheme greatly speeds up processing. Fetching the next instruction while the current instruction executes is called *pipelining*.

### SEGMENT REGISTERS

The 8086 BIU sends out 20-bit addresses, so it can address any of  $2^{20}$  or 1,048,576 bytes in memory. However, at any given time the 8086 works with only four 65,536-byte (64-Kbyte) segments within this 1,048,576-byte (1-Mbyte) range. Four segment registers in the BIU are used to hold the upper 16 bits of the starting addresses of four memory segments that the 8086 is working with at a particular time. The four segment registers are the code segment (CS) register, the stack segment (SS) register, the extra segment (ES) register, and the data segment (DS) register.

Figure 2-9 shows how these four segments might be positioned in memory at a given time. The four segments can be separated as shown, or, for small programs which do not need all 64 Kbytes in each segment, they can overlap.

To repeat, then, a segment register is used to hold the upper 16 bits of the starting address for each of the segments. The code segment register, for example, holds the upper 16 bits of the starting address for the segment from which the BIU is currently fetching instruction code bytes. The BIU always inserts zeros for the lowest



FIGURE 2-9 One way four 64-Kbyte segments might be positioned within the 1-Mbyte address space of an 8086.

4 bits (nibble) of the 20-bit starting address for a segment. If the code segment register contains 348AH, for example, then the code segment will start at address 348AOH. In other words, a 64-Kbyte segment can be located anywhere within the 1-Mbyte address space, but the segment will always start at an address with zeros in the lowest 4 bits. This constraint was put on the location of segments so that it is only necessary to store and manipulate 16-bit numbers when working with the starting address of a segment. The part of a segment starting address stored in a segment register is often called the *segment base*.

A *stack* is a section of memory set aside to store addresses and data while a *subprogram* executes. The stack segment register is used to hold the upper 16 bits of the starting address for the program stack. We will discuss the use and operation of a stack in detail later.

The extra segment register and the data segment register are used to hold the upper 16 bits of the starting addresses of two memory segments that are used for data.

### INSTRUCTION POINTER

The next feature to look at in the BIU is the instruction pointer (IP) register. As discussed previously, the code



FIGURE 2-10 Addition of IP to CS to produce the physical address of the code byte. (a) Diagram. (b) Computation.

segment register holds the upper 16 bits of the starting address of the segment from which the BIU is currently fetching instruction code bytes. The instruction pointer register holds the 16-bit address, or offset, of the next code byte within this code segment. The value contained in the IP is referred to as an offset because this value must be offset from (added to) the segment base address in CS to produce the required 20-bit physical address sent out by the BIU. Figure 2-10a shows in diagram form how this works. The CS register points to the base, or start of the current code segment. The IP contains the distance or offset from this base address to the next instruction byte to be fetched. Figure 2-10b shows how the 16-bit offset in IP is added to the 16-bit segment base address in CS to produce the 20-bit physical address. Notice that the two 16-bit numbers are not added directly in line, because the CS register contains only the upper 16 bits of the base address for the code segment. As we said before, the BIU automatically inserts zeros for the lowest 4 bits of the segment base address.

If the CS register, for example, contains 348AH, you know that the starting address for the code segment is 348A0H. When the BIU adds the offset of 4214H in the IP to this segment base address, the result is a 20-bit physical address of 38AB4H.

An alternative way of representing a 20-bit physical address is the segment base offset form. For the address of a code byte, the format for this alternative form will be CS:IP. As an example of this, the address constructed in the preceding paragraph, 38AB4H, can also be represented as 348A:4214.

To summarize, then, the CS register contains the upper 16 bits of the starting address of the code segment in the 1-Mbyte address range of the 8086. The instruction pointer register contains a 16-bit offset which

tells where in that 64-Kbyte code segment the next instruction byte is to be fetched from. The actual physical address sent to memory is produced by adding the offset contained in the IP register to the segment base represented by the upper 16 bits in the CS register.

Any time the 8086 accesses memory, the BIU produces the required 20-bit physical address by adding an offset to a segment base value represented by the contents of one of the segment registers. As another example of this, let's look at how the 8086 uses the contents of the stack segment register and the contents of the stack pointer register to produce a physical address.

### STACK SEGMENT REGISTER AND STACK POINTER REGISTER

A stack, remember, is a section of memory set aside to store addresses and data while a subprogram is executing. The 8086 allows you to set aside an entire 64-Kbyte segment as a stack. The upper 16 bits of the starting address for this segment are kept in the stack segment register. The stack pointer (SP) register in the execution unit holds the 16-bit offset from the start of the segment to the memory location where a word was most recently stored on the stack. The memory location where a word was most recently stored is called the top of stack. Figure 2-11a shows this in diagram form.

The physical address for a stack read or a stack write is produced by adding the contents of the stack pointer register to the segment base address represented by the upper 16 bits of the base address in SS. Figure 2-11b shows an example. The 5000H in SS represents a segment base address of 50000H. When the FFE0H in the SP is added to this, the resultant physical address for the top of the stack will be 5FFEOH. The physical address can be represented either as a single number, 5FFEOH, or in SS:SP form as 5000:FFEOH.



FIGURE 2-11 Addition of SS and SP to produce the physical address of the top of the stack. (a) Diagram. (b) Computation.

The operation and use of the stack will be discussed in detail later as need arises.

## POINTER AND INDEX REGISTERS IN THE EXECUTION UNIT

In addition to the stack pointer register (SP), the EU contains a 16-bit base pointer (BP) register. It also contains a 16-bit source index (SI) register and a 16-bit destination index (DI) register. These three registers can be used for temporary storage of data just as the general-purpose registers described above. However, their main use is to hold the 16-bit offset of a data word in one of the segments. SI, for example, can be used to hold the offset of a data word in the data segment. The physical address of the data in memory will be generated in this case by adding the contents of SI to the segment base address represented by the 16-bit number in the DS register. After we give you an overview of the different levels of languages used to program a microcomputer, we will show you some examples of how we tell the 8086 to read data from or write data to a desired memory location.

## INTRODUCTION TO PROGRAMMING THE 8086

### Programming Languages

Now that you have an overview of the 8086 CPU, it is time to start thinking about how it is programmed. To run a program, a microcomputer must have the program stored in binary form in successive memory locations, as shown in Figure 2-12. There are three language levels that can be used to write a program for a microcomputer.

### MACHINE LANGUAGE

You can write programs as simply a sequence of the binary codes for the instructions you want the microcomputer to execute. The three-instruction program in Figure 2-6b is an example. This binary form of the program is referred to as machine language because it is the form required by the machine. However, it is difficult, if not impossible, for a programmer to memorize the thousands of binary instruction codes for a CPU such as the 8086. Also, it is very easy for an error to occur when working with long series of 1's and 0's. Using hexadecimal representation for the binary codes might help some, but there are still thousands of instruction codes to cope with.

### ASSEMBLY LANGUAGE

To make programming easier, many programmers write programs in assembly language. They then translate

the assembly language program to machine language so that it can be loaded into memory and run. Assembly language uses two-, three-, or four-letter mnemonics to represent each instruction type. A mnemonic is just a device to help you remember something. The letters in an assembly language mnemonic are usually initials or a shortened form of the English word(s) for the operation performed by the instruction. For example, the mnemonic for subtract is SUB, the mnemonic for Exclusive OR is XOR, and the mnemonic for the instruction to copy data from one location to another is MOV.

Assembly language statements are usually written in a standard form that has four fields, as shown in Figure 2-12. The first field in an assembly language statement is the label field. A label is a symbol or group of symbols used to represent an address which is not specifically known at the time the statement is written. Labels are usually followed by a colon. Labels are not required in a statement, they are just inserted where they are needed. We will show later many uses of labels.

The opcode field of the instruction contains the mnemonic for the instruction to be performed. Instruction mnemonics are sometimes called operation codes, or opcodes. The ADD mnemonic in the example statement in Figure 2-12 indicates that we want the instruction to do an addition.

The operand field of the statement contains the data, the memory address, the port address, or the name of the register on which the instruction is to be performed. Operand is just another name for the data item(s) acted on by an instruction. In the example instruction in Figure 2-12, there are two operands. AL and 07H, specified in the operand field. AL represents the AL register, and 07H represents the number 07H. This assembly language statement thus says, "Add the number 07H to the contents of the AL register." By Intel convention, the result of the addition will be put in the register or the memory location specified before the comma in the operand field. For the example statement in Figure 2-12, then, the result will be left in the AL register. As another example, the assembly language statement ADD BH, AL, when converted to machine language and run, will add the contents of the AL register to the contents of the BH register. The results will be left in the BH register.

The final field in an assembly language statement such as that in Figure 2-12 is the comment field, which starts with a semicolon. Comments do not become part of the machine language program, but they are very important. You write comments in a program to remind you of the function that an instruction or group of instructions performs in the program.

To summarize why assembly language is easier to use than machine language, let's look a little more closely at the assembly language ADD statement. The general format of the 8086 ADD instruction is

ADD destination, source

The source can be a number written in the instruction, the contents of a specified register, or the contents of a memory location. The destination can be a specified register or a specified memory location. However, the

| LABEL FIELD | OP CODE FIELD | OPERAND FIELD | COMMENT FIELD           |
|-------------|---------------|---------------|-------------------------|
| NEXT:       | ADD           | AL, 07H       | ; ADD CORRECTION FACTOR |

FIGURE 2-12 Assembly language program statement format.

source and the destination in an instruction cannot both be memory locations.

A later section on 8086 addressing modes will show all the ways in which the source of an operand and the destination of the result can be specified. The point here is that the single mnemonic ADD, together with a specified source and a specified destination, can represent a great many 8086 instructions in an easily understandable form.

The question that may occur to you at this point is, "If I write a program in assembly language, how do I get it translated into machine language which can be loaded into the microcomputer and executed?" There are two answers to this question. The first method of doing the translation is to work out the binary code for each instruction a bit at a time using the templates given in the manufacturer's data books. We will show you how to do this in the next chapter, but it is a tedious and error-prone task. The second method of doing the translation is with an assembler. An assembler is a program which can be run on a personal computer or microcomputer development system. It reads the file of assembly language instructions you write and generates the correct binary code for each. For developing all but the simplest assembly language programs, an assembler and other program development tools are essential. We will introduce you to these program development tools in the next chapter and describe their use throughout the rest of this book.

## HIGH-LEVEL LANGUAGES

Another way of writing a program for a microcomputer is with a *high-level language*, such as BASIC, Pascal, or C. These languages use program statements which are even more English-like than those of assembly language. Each high-level statement may represent many machine code instructions. An *interpreter program* or a *compiler program* is used to translate higher-level language statements to machine codes which can be loaded into memory and executed. Programs can usually be written faster in high-level languages than in assembly language because the high-level language works with bigger building blocks. However, programs written in a high-level language and interpreted or compiled almost always execute more slowly and require more memory than the same programs written in assembly language. Programs that involve a lot of hardware control, such as robots and factory control systems, or programs that must run as quickly as possible are usually best written in assembly language. Complex data processing programs that manipulate massive amounts of data, such as insurance company records, are usually best written in a high-level language. The decision concerning which language to use has recently been made more difficult by the fact that current assemblers allow the use of many high-level language features, and the fact that some current high-level languages provide assembly language features.

## OUR CHOICE

For most of this book we work very closely with hardware, so assembly language is the best choice. In later chap-

ters, however, we do show you how to write programs which contain modules written in assembly language and modules written in the high-level language C. In the next chapter we introduce you to assembly language programming techniques. Before we go on to that, however, we will use a few simple 8086 instructions to show you more about accessing data in registers and memory locations.

## How the 8086 Accesses Immediate and Register Data

In a previous discussion of the 8086 BIU, we described how the 8086 accesses code bytes using the contents of the CS and IP registers. We also described how the 8086 accesses the stack using the contents of the SS and SP registers. Before we can teach you assembly language programming techniques, we need to discuss some of the different ways in which an 8086 can access the data that it operates on. The different ways in which a processor can access data are referred to as its *addressing modes*. In assembly language statements, the addressing mode is indicated in the instruction. We will use the 8086 MOV instruction to illustrate some of the 8086 addressing modes.

The MOV instruction has the format

MOV destination, source

When executed, this instruction copies a word or a byte from the specified source location to the specified destination location. The source can be a number written directly in the instruction, a specified register, or a memory location specified in 1 of 24 different ways. The destination can be a specified register or a memory location specified in any 1 of 24 different ways. The source and the destination cannot both be memory locations in an instruction.

### IMMEDIATE ADDRESSING MODE

Suppose that in a program you need to put the number 437BH in the CX register. The MOV CX, 437BH instruction can be used to do this. When it executes, this instruction will put the *immediate* hexadecimal number 437BH in the 16-bit CX register. This is referred to as *immediate addressing mode* because the number to be loaded into the CX register will be put in the two memory locations immediately following the code for the MOV instruction. This is similar to the way the port address was put in memory immediately after the code for the input instruction in the three-instruction program in Figure 2-6b.

A similar instruction, MOV CL, 48H, could be used to load the 8-bit immediate number 48H into the 8-bit CL register. You can also write instructions to load an 8-bit immediate number into an 8-bit memory location or to load a 16-bit number into two consecutive memory locations, but we are not yet ready to show you how to specify these.

### REGISTER ADDRESSING MODE

*Register addressing mode* means that a register is the source of an operand for an instruction. The instruction

`MOV CX, AX`, for example, copies the contents of the 16-bit AX register into the 16-bit CX register. Remember that the destination location is specified in the instruction before the comma, and the source is specified after the comma. Also note that the contents of AX are just copied to CX, not actually moved. In other words, the previous contents of CX are written over, but the contents of AX are not changed. For example, if CX contains 2A84H and AX contains 4971H before the `MOV CX, AX` instruction executes, then after the instruction executes, CX will contain 4971H and AX will still contain 4971H. You can `MOV` any 16-bit register to any 16-bit register, or you can `MOV` any 8-bit register to any 8-bit register. However, you cannot use an instruction such as `MOV CX, AL` because this is an attempt to copy a byte-type operand (AL) into a word-type destination (CX). The byte in AL would fit in CX, but the 8086 would not know which half of CX to put it in. If you try to write an instruction like this and you are using a good assembler, the assembler will tell you that the instruction contains a type error. To copy the byte from AL to the high byte of CX, you can use the instruction `MOV CH, AL`. To copy the byte from AL to the low byte of CX, you can use the instruction `MOV CL, AL`.

## Accessing Data in Memory

### OVERVIEW OF MEMORY ADDRESSING MODES

The addressing modes described in the following sections are used to specify the location of an operand in memory. To access data in memory, the 8086 must also produce a 20-bit physical address. It does this by adding a 16-bit value called the effective address to a segment base address represented by the 16-bit number in one of the four segment registers. The effective address (EA) represents the displacement or offset of the desired operand from the segment base. In most cases, any of the segment bases can be specified, but the data segment is the one most often used. Figure 2-13a shows in graphic form how the EA is added to the data segment base to point to an operand in memory. Figure 2-13b shows how the 20-bit physical address is generated by the BIU. The starting address for the data segment in Figure 2-13b is 20000H, so the data segment register will contain 2000H. The BIU adds the effective address, 437AH, to the data segment base address of 20000H to produce the physical address sent out to memory. The 20-bit physical address sent out to memory by the BIU will then be 2437AH. The physical address can be represented either as a single number 2437AH or in the segment base:offset form as 2000:437AH.

The execution unit calculates the effective address for an operand using information you specify in the instruction. You can tell the EU to use a number in the instruction as the effective address, to use the contents of a specified register as the effective address, or to compute the effective address by adding a number in the instruction to the contents of one or two specified registers. The following section describes one way you can tell the execution unit to calculate an effective address. In later chapters we show other ways of specifying the effective address. Later we also show how the



FIGURE 2-13 Addition of data segment register and effective address to produce the physical address of the data byte. (a) Diagram. (b) Computation.

addressing modes this provides are used to solve some common programming problems.

### DIRECT ADDRESSING MODE

For the simplest memory addressing mode, the effective address is just a 16-bit number written directly in the instruction. The instruction `MOV BL, [437AH]` is an example. The square brackets around the 437AH are shorthand for "the contents of the memory location(s) at a displacement from the segment base of." When executed, this instruction will copy "the contents of the memory location at a displacement from the data segment base of" 437AH into the BL register, as shown by the rightmost arrow in Figure 2-13a. The BIU calculates the 20-bit physical memory address by adding the effective address 437AH to the data segment base, as shown in Figure 2-13b. This addressing mode is called direct because the displacement of the operand from the segment base is specified directly in the instruction. The displacement in the instruction will be added to the data segment base in DS unless you tell the BIU to add it to some other segment base. Later we will show you how to do this.

Another example of the direct addressing mode is the instruction `MOV BX, [437AH]`. When executed, this instruction copies a 16-bit word from memory into the BX register. Since each memory address of the 8086 represents a byte of storage, the word must come from two memory locations. The byte at a displacement of 437AH from the data segment base will be copied into BL, as shown by the right arrow in Figure 2-13a. The contents of the next higher address, displacement 437BH, will be copied into the BH register, as shown by the left arrow in Figure 2-13a. From the instruction

coding, the 8086 will automatically determine the number of bytes that it must access in memory.

An important point here is that an 8086 always stores the low byte of a word in the lower of the two addresses and stores the high byte of a word in the higher address. To stick this in your mind, remember:

### Low byte-low address, high byte-high address

The previous two examples showed how the direct addressing mode can be used to specify the source of an operand. Direct addressing can also be used to specify the destination of an operand in memory. The instruction `MOV [437AH], BX`, for example, will copy the contents of the `BX` register to two memory locations in the data segment. The contents of `BL` will be copied to the memory location at a displacement of `437AH`. The contents of `BH` will be copied to the memory location at a displacement of `437BH`. This operation is represented by simply reversing the direction of the arrows in Figure 2-13a.

**NOTE:** When you are hand-coding programs using direct addressing of the form shown above, make sure to put in the square brackets to remind you how to code the instruction. If you leave the brackets out of an instruction such as `MOV BX, [437AH]`, you will code it as if it were the instruction `MOV BX, 437AH`. This second instruction will load the immediate number `437AH` into `BX`, rather than loading a word from memory at a displacement of `437AH` into `BX`. Also note that if you are writing an instruction using direct addressing such as this for an assembler, you must write the instruction in the form `MOV BL, DS:BYTE PTR [437AH]` to give the assembler all the information it needs. As we will show you in the next chapter, when you are using an assembler, you usually use a name to represent the direct address rather than the actual numerical value.

## A FEW WORDS ABOUT SEGMENTATION

At this point you may be wondering why Intel designed the 8086 family devices to access memory using the segment:offset approach rather than accessing memory directly with 20-bit addresses. The segment:offset scheme requires only a 16-bit number to represent the base address for a segment, and only a 16-bit offset to access any location in a segment. This means that the 8086 has to manipulate and store only 16-bit quantities instead of 20-bit quantities. This makes for an easier interface with 8- and 16-bit-wide memory boards and with the 16-bit registers in the 8086.

The second reason for segmentation has to do with the type of microcomputer in which an 8086-family CPU is likely to be used. A previous section of this chapter described briefly the operation of a timesharing microcomputer system. In a timesharing system, several users share a CPU. The CPU works on one user's program for perhaps 20 ms, then works on the next user's program for 20 ms. After working 20 ms for each of the other users, the CPU comes back to the first user's program

again. Each time the CPU switches from one user's program to the next, it must access a new section of code and new sections of data. Segmentation makes this switching quite easy. Each user's program can be assigned a separate set of logical segments for its code and data. The user's program will contain offsets or displacements from these segment bases. To change from one user's program to a second user's program, all that the CPU has to do is to reload the four segment registers with the segment base addresses assigned to the second user's program. In other words, segmentation makes it easy to keep users' programs and data separate from one another, and segmentation makes it easy to switch from one user's program to another user's program. In Chapter 15 we tell you much more about the use of segmentation in multiuser systems.

## CHECKLIST OF IMPORTANT TERMS AND CONCEPTS IN THIS CHAPTER

If you do not remember any of the terms or concepts in the following list, use the index to find them in the chapter.

Microcomputer, microprocessor

Hardware, software, firmware

Timesharing computer system

Multitasking computer system

Distributed processing system

Multiprocessing

CPU

Memory, RAM, ROM

I/O ports

Address, data, and control buses

Control bus signals

ALU

Segmentation

Bus interface unit (BIU)

Instruction byte queue, pipelining,  
ES, CS, SS, DS registers, IP register

Execution unit (EU)

AX, BX, CX, DX registers, flag register,  
ALU, SP, BP, SI, DI registers

Machine language, assembly language, high-level language

Mnemonic, opcode, operand, label, comment

Assembler, compiler

Immediate address mode, register address mode, direct address mode

Effective address

```

2 ; 8086 PROGRAM F4-05.ASM
3 ; ABSTRACT : Program produces a packed BCD byte from 2 ASCII-encoded digits
4 ; The first ASCII digit (5) is loaded in BL.
5 ; The second ASCII digit (9) is loaded in AL.
6 ; The result (packed BCD) is left in AL.
7 ; REGISTERS : Uses CS, AL, BL, CL
8 ; PORTS : None used
9 0000
10
11 0000 B3 35
12 0002 B0 39
13 0004 80 E3 0F
14 0007 24 0F
15 0009 B1 04
16 000B D2 C3
17 000D 0A C3
18 000F
19
CODE SEGMENT
ASSUME CS:CODE
START: MOV BL, '5' ; Load first ASCII digit into BL
       MOV AL, '9' ; Load second ASCII digit into AL
       AND BL, 0FH ; Mask upper 4 bits of first digit
       AND AL, 0FH ; Mask upper 4 bits of second digit
       MOV CL, 04H ; Load CL for 4 rotates required
       ROL BL, CL ; Rotate BL 4 bit positions
       OR AL, BL ; Combine nibbles, result in AL
CODE ENDS
END START

```

FIGURE 4-5 List file of 8086 assembly language program to produce packed BCD from two ASCII characters.

For the example program here, we use the instruction OR AL, BL to pack the two BCD nibbles. Bits ORed with 0's will not be changed. Bits ORed with 1's will become or stay 1's. Again look at Figure 4-2 to help you visualize this operation.

### SUMMARY OF BCD PACKING PROGRAM

If you compare the algorithm for this program with the finished program in Figure 4-5, you should see that each step in the algorithm translates to one or two assembly language instructions. As we told you before, developing the assembly language program from a good algorithm is really quite easy because you are simply translating one step at a time to its equivalent assembly language instructions. Also, debugging a program developed in this way is quite easy because you simply single-step or breakpoint your way through it and check the results after each step. In the next section we discuss the 8086 JMP instructions and flags so we can show you how you implement some of the other programming structures in assembly language.

## JUMPS, FLAGS, AND CONDITIONAL JUMPS

### Introduction

The real power of a computer comes from its ability to choose between two or more sequences of actions based on some condition, repeat a sequence of instructions as long as some condition exists, or repeat a sequence of instructions until some condition exists. Flags indicate whether some condition is present or not. Jump instructions are used to tell the computer the address to fetch its next instruction from. Figure 4-6 shows in diagram form the different ways a Jump instruction can direct

the 8086 to fetch its next instruction from some place in memory other than the next sequential location.

The 8086 has two types of Jump instructions, conditional and unconditional. When the 8086 fetches and decodes an Unconditional Jump instruction, it always goes to the specified jump destination. You might use this type of Jump instruction at the end of a program so that the entire program runs over and over, as shown in Figure 4-6.

When the 8086 fetches and decodes a Conditional Jump instruction, it evaluates the state of a specified flag to determine whether to fetch its next instruction from the jump destination location or to fetch its next instruction from the next sequential memory location.



FIGURE 4-6 Change in program flow that can be caused by jump instructions.

```

; 8086 PROGRAM
; This program illustrates a forward jump
; ABSTRACT
; REGISTERS : CS, AX
; PORTS : None used
;
; CODE SEGMENT
ASSUME CS:CODE
JMP THERE ; skip over a series of instructions
NOP ; dummy instructions to represent those
NOP ; instructions skipped over
THERE: MOV AX, 0000H ; zero accumulator before addition instructions
NOP ; dummy instruction to represent continuation of execution
CODE ENDS
END

```

(A) Just past the  
marked line

FIGURE 4-9 List file of program demonstrating "forward" JMP.

JMP. The displacement is calculated by counting the number of bytes from the next address after the JMP instruction to the destination. If the displacement is negative (backward in the program), then it must be expressed in 2's complement form before it can be written in the instruction code template.

Now let's look at another simple example program, in Figure 4-9, to see how you can jump ahead over a group of instructions in a program. Here again we use a label to give a name to the address that we want to JMP to. We also use NOP instructions to represent the instructions that we want to skip over and the instructions that continue after the JMP. Let's see how this JMP instruction is coded.

When the assembler reads through the source file for this program, it will find the label "THERE" after the JMP mnemonic. At this point the assembler has no way of knowing whether it will need 1 or 2 bytes to represent the displacement to the destination address. The assembler plays it safe by reserving 2 bytes for the displacement. Then the assembler reads on through the rest of the program. When the assembler finds the specified label, it calculates the displacement from the instruction after the JMP instruction to the label. If the assembler finds the displacement to be outside the range of -128 bytes to +127 bytes, then it will code the instruction as a direct within-segment near JMP with 2 bytes of displacement. If the assembler finds the displacement to be within the -128 to +127 byte range, then it will code the instruction as a direct within-segment short-type JMP with a 1-byte displacement. In the latter case, the assembler will put the code for a NOP instruction, 90H, in the third byte it had reserved for the JMP instruction. The instruction codes for the JMP THERE instruction on line 8 of Figure 4-9 demonstrate this. As is the basic opcode for the direct within-segment short JMP. The 03H represents the displacement to the JMP destination. Since we are jumping forward in this case, the displacement is a positive number. The 90H in the next memory byte is the code for a NOP instruction. The displacement is calculated from the offset of this NOP instruction, 0002H, to the offset of the destination label, 0005H. The difference of 03H between these two is the displacement you see coded in the instruction.

If you are hand coding a program such as this, you

will probably know how far it is to the label, and you can leave just 1 byte for the displacement if that is enough. If you are using an assembler and you don't want to waste the byte of memory or the time it takes to fetch the extra NOP instruction, you can write the instruction as JMP SHORT label. The SHORT operator is a promise to the assembler that the destination will not be outside the range of -128 to +127 bytes. Trusting your promise, the assembler then reserves only 1 byte for the displacement.

Note that if you are making a JMP from an address near the start of a 64-Kbyte segment to an address near the end of the segment, you may not be able to get there with a jump of +32,767. The way you get there is to JMP backward around to the desired destination address. An assembler will automatically do this for you.

One advantage of the direct near- and short-type JMPs is that the destination address is specified relative to the address of the instruction after the JMP instruction. Since the JMP instruction in this case does not contain an absolute address or offset, the program can be loaded anywhere in memory and still run correctly. A program which can be loaded anywhere in memory to be run is said to be relocatable. You should try to write your programs so that they are relocatable.

Now that you know about unconditional JMP instructions, we will discuss the 8086 flags, so that we can show how the 8086 Conditional Jump instructions are used to implement the rest of the standard programming structures.

## The 8086 Conditional Flags

The 8086 has six conditional flags. They are the carry flag (CF), the parity flag (PF), the auxiliary carry flag (AF), the zero flag (ZF), the sign flag (SF), and the overflow flag (OF). Chapter 1 shows numerical examples of some of the conditions indicated by these flags. Here we review these conditions and show how some of the important 8086 instructions affect these flags.

### THE CARRY FLAG WITH ADD, SUBTRACT, AND COMPARE INSTRUCTIONS

If the addition of two 8-bit numbers produces a sum greater than 8 bits, the carry flag will be set to a 1 to indicate a carry into the next bit position. Likewise, if

the addition of two 16-bit numbers produces a sum greater than 16 bits, then the carry flag will be set to a 1 to indicate that a final carry was produced by the addition.

During subtraction, the carry flag functions as a borrow flag. If the bottom number in a subtraction is larger than the top number, then the carry/borrow flag will be set to indicate that a borrow was needed to perform the subtraction.

The 8086 compare instruction has the format CMP destination, source. The source can be an immediate number, a register, or a memory location. The destination can be a register or a memory location. The comparison is done by subtracting the contents of the specified source from the contents of the specified destination. Flags are updated to reflect the result of the comparison, but neither the source nor the destination is changed. If the source operand is greater than the specified destination operand, then the carry/borrow flag will be set to indicate that a borrow was needed to do the comparison (subtraction). If the source operand is the same size as or smaller than the specified destination operand, then the carry/borrow flag will not be set after the compare. If the two operands are equal, the zero flag will be set to a 1 to indicate that the result of the compare (subtraction) was all 0's. Here's an example and summary of this for your reference.

| CMP BX, CX |    |    | BX - CX |     |
|------------|----|----|---------|-----|
| condition  | CF | ZF |         |     |
| CX > BX    | 1  | 0  | 6-2     |     |
| CX < BX    | 0  | 0  |         | 463 |
| CX = BX    | 0  | 1  |         |     |

The compare instruction is very important because it allows you to easily determine whether one operand is greater than, less than, or the same size as another operand.

### THE PARITY FLAG

Parity is a term used to indicate whether a binary word has an even number of 1's or an odd number of 1's. A binary number with an even number of 1's is said to have *even parity*. The 8086 parity flag will be set to a 1 after an instruction if the lower 8 bits of the destination operand has an even number of 1's. Probably the most common use of the parity flag is to determine whether ASCII data sent to a computer over phone lines or some other communications link contains any errors. In Chapter 14 we describe this use of parity.

### THE AUXILIARY CARRY FLAG

This flag has significance in BCD addition or BCD subtraction. If a carry is produced when the least significant nibbles of 2 bytes are added, the auxiliary carry flag will be set. In other words, a carry out of bit 3 sets the auxiliary carry flag. Likewise, if the subtraction of the least significant nibbles requires a borrow, the auxiliary carry/borrow flag will be set. The auxiliary carry/borrow flag is used only by the DAA and DAS instructions. Consult the DAA and DAS instruction descriptions in Chapter 6 and the BCD operation exam-

ples section of Chapter 1 for further discussion of addition and subtraction of BCD numbers.

### THE ZERO FLAG WITH INCREMENT, DECREMENT, AND COMPARE INSTRUCTIONS

As the name implies, this flag will be set to a 1 if the result of an arithmetic or logic operation is zero. For example, if you subtract two numbers which are equal, the zero flag will be set to indicate that the result of the subtraction is zero. If you AND two words together and the result contains no 1's, the zero flag will be set to indicate that the result is all 0's.

Besides the more obvious arithmetic and logic instructions, there are a few other very useful instructions which also affect the zero flag. One of these is the compare instruction CMP, which we discussed previously with the carry flag. As shown there, the zero flag will be set to a 1 if the two operands compared are equal.

Another important instruction which affects the zero flag is the decrement instruction, DEC. This instruction will decrement (or, in other words, subtract 1 from) a number in a specified register or memory location. If, after decrementing, the contents of the register or memory location are zero, the zero flag will be set. Here's a preview of how this is used. Suppose that we want to repeat a sequence of actions nine times. To do this, we first load a register with the number 09H and execute the sequence of actions. We then decrement the register and look at the zero flag to see if the register is down to zero yet. If the zero flag is not set, then we know that the register is not yet down to zero, so we tell the 8086, with a Jump instruction, to go back and execute the sequence of instructions again. The following sections will show many specific examples of how this is done.

The increment instruction, INC destination, also affects the zero flag. If an 8-bit destination containing FFH or a 16-bit destination containing FFFFH is incremented, the result in the destination will be all 0's. The zero flag will be set to indicate this.

### THE SIGN FLAG—POSITIVE AND NEGATIVE NUMBERS

When you need to represent both positive and negative numbers for an 8086, you use 2's complement sign-and-magnitude form as described in Chapter 1. In this form, the most significant bit of the byte or word is used as a sign bit. A 0 in this bit indicates that the number is positive. A 1 in this bit indicates that the number is negative. The remaining 7 bits of a byte or the remaining 15 bits of a word are used to represent the magnitude of the number. For a positive number, the magnitude will be in standard binary form. For a negative number, the magnitude will be in 2's complement form. After an arithmetic or logic instruction executes, the sign flag will be a copy of the most significant bit of the destination byte or the destination word. In addition to its use with signed arithmetic operations, the sign flag can be used to determine whether an operand has been decremented beyond zero. Decrementing 00H, for example, will give FFH. Since the MSB of FFH is a 1, the sign flag will be set.

## THE OVERFLOW FLAG

This flag will be set if the result of a signed operation is too large to fit in the number of bits available to represent it. To remind you of what overflow means, here is an example. Suppose you add the 8-bit signed number 01110101 (+117 decimal) and the 8-bit signed number 00110111 (+55 decimal). The result will be 10101100 (+172 decimal), which is the correct binary result in this case, but is too large to fit in the 7 bits allowed for the magnitude in an 8-bit signed number. For an 8-bit signed number, a 1 in the most significant bit indicates a negative number. The overflow flag will be set after this operation to indicate that the result of the addition has overflowed into the sign bit.

## The 8086 Conditional Jump Instructions

As we stated previously, much of the real power of a computer comes from its ability to choose between two courses of action depending on whether some condition is present or not. In the 8086 the six conditional flags indicate the conditions that are present after an instruction. The 8086 Conditional Jump instructions look at the state of a specified flag(s) to determine whether the jump should be made or not.

Figure 4-10 shows the mnemonics for the 8086 Conditional Jump instructions. Next to each mnemonic is a brief explanation of the mnemonic. Note that the terms *above* and *below* are used when you are working with unsigned binary numbers. The 8-bit unsigned number 11000110 is above the 8-bit unsigned number 00111001, for example. The terms *greater* and *less* are used when you are working with signed binary numbers. The 8-bit signed number 00111001 is greater (more

positive) than the 8-bit signed number 11000110, which represents a negative number. Also shown in Figure 4-10 is an indication of the flag conditions that will cause the 8086 to do the jump. If the specified flag conditions are not present, the 8086 will just continue on to the next instruction in sequence. In other words, if the jump condition is not met, the Conditional Jump instruction will effectively function as a NOP. Suppose, for example, we have the instruction JC SAVE, where SAVE is the label at the destination address. If the carry flag is set, this instruction will cause the 8086 to jump to the instruction at the SAVE label. If the carry flag is not set, the instruction will have no effect other than taking up a little processor time.

All conditional jumps are *short-type* jumps. This means that the destination label must be in the same code segment as the jump instruction. Also, the destination address must be in the range of -128 bytes to +127 bytes from the address of the instruction after the Jump instruction. As we show in later examples, it is important to be aware of this limit on the range of conditional jumps as you write your programs.

The Conditional Jump instructions are usually used after arithmetic or logic instructions. They are very commonly used after Compare instructions. For this case, the Compare instruction syntax and the Conditional Jump instruction syntax are such that a little trick makes it very easy to see what will cause a jump to occur. Here's the trick. Suppose that you see the instruction sequence:

CMP BL, DH  
JAE HEATER\_OFF

In a program, and you want to determine what these instructions do. The CMP instruction compares the byte

| MNEMONIC | CONDITION TESTED      | "JUMP IF . . ."            |
|----------|-----------------------|----------------------------|
| JA/JNBE  | (CF or ZF)=0          | above/not below nor equal  |
| JAE/JNB  | CF=0                  | above or equal/not below   |
| JB/JNAE  | CF=1                  | below/not above nor equal  |
| JBE/JNA  | (CF or ZF)=1          | below or equal/not above   |
| JC       | CF=1                  | carry                      |
| JE/JZ    | ZF=1                  | equal/zero                 |
| JG/JNLE  | ((SF xor OF) or ZF)=0 | greater/not less nor equal |
| JGE/JNL  | (SF xor OF)=0         | greater or equal/not less  |
| JL/JNGE  | (SF xor OF)=1         | less/not greater nor equal |
| JLE/JNG  | ((SF xor OF) or ZF)=1 | less or equal/not greater  |
| JNC      | CF=0                  | not carry                  |
| JNE/JNZ  | ZF=0                  | not equal/not zero         |
| JNO      | OF=0                  | not overflow               |
| JNP/JPO  | PF=0                  | not parity/parity odd      |
| JNS      | SF=0                  | not sign                   |
| JO       | OF=1                  | overflow                   |
| JP/JPE   | PF=1                  | parity/parity equal        |
| JS       | SF=1                  | sign                       |

Note: "above" and "below" refer to the relationship of two unsigned values;  
 "greater" and "less" refer to the relationship of two signed values.

FIGURE 4-10 8086 Conditional Jump instructions.

# INTEL 8085

This chapter describes hardware, software, and interfacing aspects of the Intel 8085. Topics include 8085 register architecture, addressing modes, instruction set, input/output, and system design.

## 2.1 Introduction

The Intel 8085 is an 8-bit microprocessor. The 8085 is designed using NMOS in 40-in DIP (Dual In-line Package). The 8085 can be operated from either 3.03 MHz maximum (8085A) or 5 MHz maximum (8085A-2) internal clock frequency.

The 8085 has three enhanced versions, namely, the 8085AH, 8085AH-2, and 8085AH-1. These enhanced processors are designed using the HMOS (High-density MOS) technology. Each is packaged in a 40-pin DIP like the 8085. These enhanced microprocessors consume 20% lower power than the 8085A. The internal clock frequencies of the 8085AH, 8085AH-2, and 8085AH-1 are 3, 5, 6 MHz, respectively. These HMOS 8-bit microprocessors are expensive compared to the NMOS 8-bit 8085A.

Figure 2.1 shows a simplified block diagram of the 8085 microprocessor. The accumulator connects to the data bus and the Arithmetic and Logic Unit (ALU). The ALU performs all data manipulation, such as incrementing a number or adding two numbers.

The temporary register feeds the ALU's other input. This register is invisible to the programmer and is controlled automatically by the microprocessor's control circuitry.

The flags are a collection of flip-flops that indicate certain characteristics of the result of the most recent operation performed by the ALU. For example, the zero flag is set if the result of an operation is zero. The zero flag is tested by the JZ instruction.

The instruction register, instruction decoder, program counter, and control and timing logic are used for fetching instructions from memory and directing their execution.

## 2.2 Register Architecture

The 8085 registers and status flags are shown in Figure 2.2.

The accumulator (A) is an 8-bit register. Most arithmetic and logic operations are performed using the accumulator. All I/O data transfers between the 8085 and the I/O devices are performed via the accumulator. Also, there are a number of instructions that move data between the accumulator and memory.

The B, C, D, E, H, and L are each 8 bits long. Registers H and L are the memory address register or data counter. This means that these two registers are used to store the 16-bit address



**FIGURE 2.1** Simplified 8085 block diagram.

of 8-bit data being accessed from memory. This is the implied or register indirect addressing mode. There are a number of instructions, such as `MOV reg, M`, and `MOV M, reg`, which move data between any register and memory location addressed by H and L. However, using any other memory reference instruction, data transfer takes place between a memory location and the only 8085 register, the accumulator. The instruction `LDAX B` is a typical example.



**FIGURE 2.2** 8085 microprocessor registers and status flags.

A, B, C, D, and E are secondary accumulators or data counters. There are a number of instructions to move data between any two registers. There are also a few instructions that combine registers B and C or D and E, as a 16-bit data counter with high byte of a pair contained in the first register and low byte in the second. These instructions typically include LDAX B, LDAX D, STAX B, and STAX D, which transfer data between memory and the accumulator.

Each of these 8-bit registers can be incremented and decremented by a single byte instruction. There are a number of instructions which combine two of these 8-bit registers to form 16-bit register pairs as follows:

|   |     |     |
|---|-----|-----|
| A | and | PSW |
| B | and | C   |
| D | and | E   |
| H | and | L   |

high-order byte      low-order byte

The 16-bit register pair obtained by combining the accumulator and the program status word (PSW) is used only for stack operations. Sixteen bit arithmetic operations use B and C, D and E, or H and L as 16-bit data registers.

The program status word consists of five status flags. These are described below.

The carry flag (Cy) reflects the final carry out of the most significant bit of any arithmetic operation. Any logic instruction resets or clears the carry flag. This flag is also used by the shift and rotate instructions. The 8085 does not have any CLEAR CARRY instruction. One way of clearing the carry will be by ORing or ANDing the accumulator with itself.

The parity status flag (P) is set to 1 if an arithmetic or logic instruction generates an answer with even parity, that is, containing an even number of 1 bits. This flag is 0 if the arithmetic or logic instruction generates an answer with odd parity, that is, containing an odd number of 1s.

The auxiliary carry flag (Ac) reflects any carry from bit 3 to bit 4 (assuming 8-bit data with bit 0 as the least significant bit and bit 7 as the most significant bit) due to an arithmetic operation. This flag is useful for BCD operations.

The zero flag (Z) is set to 1 whenever an arithmetic or logic operation produces a result of 0. The zero flag is cleared to zero for a nonzero result due to arithmetic or logic operation.

The sign status flag (S) is set to the value of the most significant bit of the result in the accumulator after an arithmetic or logic operation. This provides a range of  $-128_{10}$  to  $+127_{10}$  (with 0 being considered positive) as the 8085's data-handling capacity.

The 8085 does not have an overflow flag. Note that execution of arithmetic or logic instructions in the 8085 affects the flags. All conditional instructions in the 8085 instruction set use one of the status flags as the required condition.

The stack pointer (SP) is 16 bits long. All stack operations with the 8085 use 16-bit register pairs. The stack pointer contains the address of the last data byte written into the stack. It is decremented by 2 each time 2 bytes of data are written or pushed onto the stack and is incremented by 2 each time 2 bytes of data are read from or pulled (popped) off the stack, that is, the top of the stack has the lowest address in the stack that grows downward.

The program counter (PC) is 16 bits long to address up to 64K of memory. It usually addresses the next instruction to be executed.

## 2.3 Memory Addressing

When addressing a memory location, the 8085 uses either register indirect or direct memory addressing. With register indirect addressing, the H and L registers perform the function of the memory address register or data counter; that is, the H, L pair holds the address of the data.

With this mode, data transfer may occur between the addressed memory location and any one of the registers A, B, C, D, E, H, or L.

Also, some instructions, such as LDAX B, LDAX D, STAX B, and STAX D, use registers B and C or D and E to hold the address of data. These instructions transfer data between the accumulator and the memory location addressed by registers B and C or D and E using the register indirect mode.

There are also a few instructions, such as the STA ppqq, which use the direct-memory addressing mode to move data between the accumulator and the memory. These instructions use 3 bytes, with the first byte as the OP code followed by 2 bytes of address.

The stack is basically a part of the RAM. Therefore, PUSH and POP instructions are memory reference instructions.

All 8085 JUMP instructions use direct or absolute addressing and are 3 bytes long. The first byte of this instruction is the OP code followed by a 2-byte address. This address specifies the memory location to which the program would branch.

## 2.4 8085 Addressing Modes

---

The 8085 has five addressing modes:

1. **Direct** — Instructions using this mode specify the effective address as a part of the instruction. These instructions contain 3 bytes, with the first byte as the OP code followed by 2 bytes of address of data (the low-order byte of the address in byte 2, the high-order byte of the address in byte 3). Consider LDA 2035H. This instruction loads accumulator with the contents of memory location  $2035_{16}$ . This mode is also called the absolute mode.
2. **Register** — This mode specifies the register or register pair that contains data. For example, MOV B, C moves the contents of register C to register B.
3. **Register Indirect** — This mode contains a register pair which stores the address of data (the high-order byte of the address in the first register of the pair, and the low-order byte in the second). As an example, LDAX B loads the accumulator with the contents of a memory location addressed by B, C register pair.
4. **Implied or Inherent** — The instructions using this mode have no operands. Examples include STC (Set the Carry Flag).
5. **Immediate** — For an 8-bit datum, this mode uses 2 bytes, with the first byte as the OP code, followed by 1 byte of data. On the other hand, for 16-bit data, this instruction contains 3 bytes, with the first byte as the OP code followed by 2 bytes of data. For example, MVI B, 05 loads register B with the value 5, and LXI H, 2050H loads H with 20H and L with 50H.

A JUMP instruction interprets the address that it would branch to in the following ways:

1. **Direct** — The JUMP instructions, such as JZ ppqq, use direct addressing and contain 3 bytes. The first byte is the OP code, followed by 2 bytes of the 16-bit address where it would branch to unconditionally or based on a condition if satisfied. For example, JMP 2020 unconditionally branches to location 2020H.
2. **Implied or Inherent Addressing** — This JUMP instruction using this mode is 1 byte long. A 16-bit register pair contains the address of the next instruction to be executed. The instruction PCHL unconditionally branches to a location addressed by the H, L pair.



FIGURE 2.3 8085 addressing structures.

## 2.5 8085 Instruction Set

As mentioned before, the 8085 uses a 16-bit address. Since the 8085 is a byte-addressable machine, it follows that it can directly address 65,536 ( $2^{16}$ ) distinct memory locations. The addressing structure of the 8085 processor is shown in Figure 2.3.

From this figure, we notice that two consecutive memory locations may be used to represent a 16-bit data item. However, according to the Intel convention, the high-order byte of a 16-bit quantity is always assigned to the high memory address.

The 8085 instructions are 1 to 3 bytes long and these formats are shown in Figure 2.4. The 8085 instruction set contains 74 basic instructions and supports conventional addressing modes such as immediate, register, absolute, and register indirect addressing modes.

Table 2.1 lists the 8085 instructions in alphabetical order; the object codes and instruction cycles are also included. When two instruction cycles are shown, the first is for "condition not met", while the second is for "condition met". Table 2.2 provides the 8085 instructions affecting the status flags. Note that not all 8085 instructions affect the status flags. The 8085 arithmetic and logic instructions normally affect the status flags.

In describing the 8085 instruction set, we will use the symbols in Table 2.3.

The 8085 move instruction transfers 8-bit data from one register to another, register to memory, and vice versa. A complete summary of these instructions is presented in Table 2.4.



FIGURE 2.4 8085 instructions format.

Table 2.1 Summary of 8085 Instruction Set

| Instruction | OP Code | Bytes | Cycles | Operations performed                                         |
|-------------|---------|-------|--------|--------------------------------------------------------------|
| ACI DATA    | CE      | 2     | 7      | $[A] \leftarrow [A] + \text{second instruction byte} + [Cy]$ |
| ADC A       | 8F      | 1     | 4      | $[A] \leftarrow [A] + [A] + [Cy]$                            |
| ADC B       | 88      | 1     | 4      | $[A] \leftarrow [A] + [B] + [Cy]$                            |
| ADC C       | 89      | 1     | 4      | $[A] \leftarrow [A] + [C] + [Cy]$                            |
| ADC D       | 8A      | 1     | 4      | $[A] \leftarrow [A] + [D] + [Cy]$                            |
| ADC E       | 8B      | 1     | 4      | $[A] \leftarrow [A] + [E] + [Cy]$                            |
| ADC H       | 8C      | 1     | 4      | $[A] \leftarrow [A] + [H] + [Cy]$                            |
| ADC L       | 8D      | 1     | 4      | $[A] \leftarrow [A] + [L] + [Cy]$                            |
| ADC M       | 8E      | 1     | 7      | $[A] \leftarrow [A] + [[H\ L]] + [Cy]$                       |
| ADD A       | 87      | 1     | 4      | $[A] \leftarrow [A] + [A]$                                   |
| ADD B       | 80      | 1     | 4      | $[A] \leftarrow [A] + [B]$                                   |
| ADD C       | 81      | 1     | 4      | $[A] \leftarrow [A] + [C]$                                   |
| ADD D       | 82      | 1     | 4      | $[A] \leftarrow [A] + [D]$                                   |
| ADD E       | 83      | 1     | 4      | $[A] \leftarrow [A] + [E]$                                   |
| ADD H       | 84      | 1     | 4      | $[A] \leftarrow [A] + [H]$                                   |
| ADD L       | 85      | 1     | 4      | $[A] \leftarrow [A] + [L]$                                   |
| ADD M       | 86      | 1     | 7      | $[A] \leftarrow [A] + [[H\ L]]$                              |
| ADI DATA    | C6      | 2     | 7      | $[A] \leftarrow [A] + \text{second instruction byte}$        |
| ANA A       | A7      | 1     | 4      | $[A] \leftarrow [A] \wedge [A]$                              |
| ANA B       | A0      | 1     | 4      | $[A] \leftarrow [A] \wedge [B]$                              |
| ANA C       | A1      | 1     | 4      | $[A] \leftarrow [A] \wedge [C]$                              |
| ANA D       | A2      | 1     | 4      | $[A] \leftarrow [A] \wedge [D]$                              |
| ANA E       | A3      | 1     | 4      | $[A] \leftarrow [A] \wedge [E]$                              |
| ANA H       | A4      | 1     | 4      | $[A] \leftarrow [A] \wedge [H]$                              |
| ANA L       | A5      | 1     | 4      | $[A] \leftarrow [A] \wedge [L]$                              |
| ANA M       | A6      | 1     | 4      | $[A] \leftarrow [A] \wedge [[H\ L]]$                         |
| ANI DATA    | E6      | 2     | 7      | $[A] \leftarrow [A] \wedge \text{second instruction byte}$   |
| CALL ppqq   | CD      | 3     | 18     | Call A subroutine addressed by ppqq                          |
| CC ppqq     | DC      | 3     | 9/18   | Call a subroutine addressed by ppqq if Cy = 1                |
| CM ppqq     | FC      | 3     | 9/18   | Call a subroutine addressed by ppqq if S = 1                 |
| CMA         | 2F      | 1     | 4      | $[A] \leftarrow 1's \text{ complement of } [A]$              |
| CMC         | 3F      | 1     | 4      | $[Cy] \leftarrow 1's \text{ complement of } [Cy]$            |
| CMP A       | B7      | 1     | 4      | $[A] - [A] \text{ and affects flags}$                        |
| CMP B       | B8      | 1     | 4      | $[A] - [B] \text{ and affects flags}$                        |

Table 2.1 Summary of 8085 Instruction Set (continued)

| Instruction | OP Code | Bytes | Cycles | Operations performed                            |
|-------------|---------|-------|--------|-------------------------------------------------|
| CMP C       | B9      | 1     | 4      | [A] - [C] and affects flags                     |
| CMP D       | BA      | 1     | 4      | [A] - [D] and affects flags                     |
| CMP E       | BB      | 1     | 4      | [A] - [E] and affects flags                     |
| CMP H       | BC      | 1     | 4      | [A] - [H] and affects flags                     |
| CMP L       | BD      | 1     | 4      | [A] - [L] and affects flags                     |
| CMP M       | BE      | 1     | 7      | [A] - [(HL)] and affects flags                  |
| CNC ppqq    | D4      | 3     | 9/18   | Call a subroutine addressed by ppqq if Cy = 0   |
| CNZ ppqq    | C4      | 3     | 9/18   | Call a subroutine addressed by ppqq if Z = 0    |
| CP ppqq     | F4      | 3     | 9/18   | Call a subroutine addressed by ppqq if S = 0    |
| CPE ppqq    | EC      | 3     | 9/18   | Call a subroutine addressed by ppqq if P = 1    |
| CPI DATA    | FE      | 2     | 7      | [A] - second instruction byte and affects flags |
| CPO ppqq    | E4      | 3     | 9/18   | Call a subroutine addressed by ppqq if P = 0    |
| CZ ppqq     | CC      | 3     | 9/18   | Call a subroutine addressed by ppqq if Z = 1    |
| DAA         | 27      | 1     | 4      | Decimal adjust accumulator                      |
| DAD B       | 09      | 1     | 10     | [HL] ← [HL] + [BC]                              |
| DAD D       | 19      | 1     | 10     | [HL] ← [HL] + [DE]                              |
| DAD H       | 29      | 1     | 10     | [HL] ← [HL] + [HL]                              |
| DAD SP      | 39      | 1     | 10     | [HL] ← [HL] + [SP]                              |
| DCR A       | 3D      | 1     | 4      | [A] ← [A] - 1                                   |
| DCR B       | 05      | 1     | 4      | [B] ← [B] - 1                                   |
| DCR C       | 0D      | 1     | 4      | [C] ← [C] - 1                                   |
| DCR D       | 15      | 1     | 4      | [D] ← [D] - 1                                   |
| DCR E       | 1D      | 1     | 4      | [E] ← [E] - 1                                   |
| DCR H       | 25      | 1     | 4      | [H] ← [H] - 1                                   |
| DCR L       | 2D      | 1     | 4      | [L] ← [L] - 1                                   |
| DCR M       | 35      | 1     | 4      | [(HL)] ← [(HL)] - 1                             |
| DCX B       | 0B      | 1     | 6      | [BC] ← [BC] - 1                                 |
| DCX D       | 1B      | 1     | 6      | [DE] ← [DE] - 1                                 |
| DCX H       | 2B      | 1     | 6      | [HL] ← [HL] - 1                                 |
| DCX SP      | 3B      | 1     | 6      | [SP] ← [SP] - 1                                 |
| DI          | F3      | 1     | 4      | Disable interrupts                              |
| EI          | FB      | 1     | 4      | Enable interrupts                               |
| HLT         | 76      | 1     | 5      | Halt                                            |
| IN PORT     | DB      | 2     | 10     | [A] ← [specified port]                          |
| INR A       | 3C      | 1     | 4      | [A] ← [A] + 1                                   |
| INR B       | 04      | 1     | 4      | [B] ← [B] + 1                                   |
| INR C       | 0C      | 1     | 4      | [C] ← [C] + 1                                   |
| INR D       | 14      | 1     | 4      | [D] ← [D] + 1                                   |
| INR E       | 1C      | 1     | 4      | [E] ← [E] + 1                                   |
| INR H       | 24      | 1     | 4      | [H] ← [H] + 1                                   |
| INR L       | 2C      | 1     | 4      | [L] ← [L] + 1                                   |
| INR M       | 34      | 1     | 4      | [(HL)] ← [(HL)] + 1                             |
| INX B       | 03      | 1     | 6      | [BC] ← [BC] + 1                                 |
| INX D       | 13      | 1     | 6      | [DE] ← [DE] + 1                                 |
| INX H       | 23      | 1     | 6      | [HL] ← [HL] + 1                                 |
| INX SP      | 33      | 1     | 6      | [SP] ← [SP] + 1                                 |
| JC ppqq     | DA      | 3     | 7/10   | Jump to ppqq if Cy = 1                          |
| JM ppqq     | FA      | 3     | 7/10   | Jump to ppqq if S = 1                           |
| JMP ppqq    | C3      | 3     | 10     | Jump to ppqq                                    |
| JNC ppqq    | D2      | 3     | 7/10   | Jump to ppqq if Cy = 0                          |
| JNZ ppqq    | C2      | 3     | 7/10   | Jump to ppqq if Z = 0                           |
| JP ppqq     | F2      | 3     | 7/10   | Jump to ppqq if S = 0                           |
| JPE ppqq    | EA      | 3     | 7/10   | Jump to ppqq if P = 1                           |
| JPO ppqq    | E2      | 3     | 7/10   | Jump to ppqq if P = 0                           |
| JZ ppqq     | CA      | 3     | 13     | Jump to ppqq if Z = 1                           |
| LDA ppqq    | 3A      | 1     | 7      | [A] ← [ppqq]                                    |
| LDAX B      | 0A      | 1     | 7      | [A] ← [(BC)]                                    |
| LDAX D      | 1A      | 1     | 7      | [A] ← [(DE)]                                    |
|             | 2A      | 3     | 16     | [L] ← [ppqq], [H] ← [ppqq + 1]                  |

| Instruction | OP Code | Bytes | Cycles | Operations performed                      |
|-------------|---------|-------|--------|-------------------------------------------|
| LXI B       | 01      | 3     | 10     | [BC] ← second and third instruction bytes |
| LXI D       | 11      | 3     | 10     | [DE] ← second and third instruction bytes |
| LXI H       | 21      | 3     | 10     | [HL] ← second and third instruction bytes |
| LXI SP      | 31      | 3     | 10     | [SP] ← second and third instruction bytes |
| MOV A,A     | 7F      | 1     | 4      | [A] ← [A]                                 |
| MOV A,B     | 78      | 1     | 4      | [A] ← [B]                                 |
| MOV A,C     | 79      | 1     | 4      | [A] ← [C]                                 |
| MOV A,D     | 7A      | 1     | 4      | [A] ← [D]                                 |
| MOV A,E     | 7B      | 1     | 4      | [A] ← [E]                                 |
| MOV A,H     | 7C      | 1     | 4      | [A] ← [H]                                 |
| MOV A,L     | 7D      | 1     | 4      | [A] ← [L]                                 |
| MOV A,M     | 7E      | 1     | 7      | [A] ← [(HL)]                              |
| MOV B,A     | 47      | 1     | 4      | [B] ← [A]                                 |
| MOV B,B     | 40      | 1     | 4      | [B] ← [B]                                 |
| MOV B,C     | 41      | 1     | 4      | [B] ← [C]                                 |
| MOV B,D     | 42      | 1     | 4      | [B] ← [D]                                 |
| MOV B,E     | 43      | 1     | 4      | [B] ← [E]                                 |
| MOV B,H     | 44      | 1     | 4      | [B] ← [H]                                 |
| MOV B,L     | 45      | 1     | 4      | [B] ← [L]                                 |
| MOV B,M     | 46      | 1     | 7      | [B] ← [(HL)]                              |
| MOV C,A     | 4F      | 1     | 4      | [C] ← [A]                                 |
| MOV C,B     | 48      | 1     | 4      | [C] ← [B]                                 |
| MOV C,C     | 49      | 1     | 4      | [C] ← [C]                                 |
| MOV C,D     | 4A      | 1     | 4      | [C] ← [D]                                 |
| MOV C,E     | 4B      | 1     | 4      | [C] ← [E]                                 |
| MOV C,H     | 4C      | 1     | 4      | [C] ← [H]                                 |
| MOV C,L     | 4D      | 1     | 4      | [C] ← [L]                                 |
| MOV C,M     | 4E      | 1     | 7      | [C] ← [(HL)]                              |
| MOV D,A     | 57      | 1     | 4      | [D] ← [A]                                 |
| MOV D,B     | 50      | 1     | 4      | [D] ← [B]                                 |
| MOV D,C     | 51      | 1     | 4      | [D] ← [C]                                 |
| MOV D,D     | 52      | 1     | 4      | [D] ← [D]                                 |
| MOV D,E     | 53      | 1     | 4      | [D] ← [E]                                 |
| MOV D,H     | 54      | 1     | 4      | [D] ← [H]                                 |
| MOV D,L     | 55      | 1     | 4      | [D] ← [L]                                 |
| MOV D,M     | 56      | 1     | 7      | [D] ← [(HL)]                              |
| MOV E,A     | 5F      | 1     | 4      | [E] ← [A]                                 |
| MOV E,B     | 58      | 1     | 5      | [E] ← [B]                                 |
| MOV E,C     | 59      | 1     | 4      | [E] ← [C]                                 |
| MOV E,D     | 5A      | 1     | 4      | [E] ← [D]                                 |
| MOV E,E     | 5B      | 1     | 4      | [E] ← [E]                                 |
| MOV E,H     | 5C      | 1     | 4      | [E] ← [H]                                 |
| MOV E,L     | 5D      | 1     | 4      | [E] ← [L]                                 |
| MOV E,M     | 5E      | 1     | 7      | [E] ← [(HL)]                              |
| MOV H,A     | 67      | 1     | 4      | [H] ← [B]                                 |
| MOV H,B     | 60      | 1     | 4      | [H] ← [A]                                 |
| MOV H,C     | 61      | 1     | 4      | [H] ← [C]                                 |
| MOV H,D     | 62      | 1     | 4      | [H] ← [D]                                 |
| MOV H,E     | 63      | 1     | 4      | [H] ← [E]                                 |
| MOV H,H     | 64      | 1     | 4      | [H] ← [H]                                 |
| MOV H,L     | 65      | 1     | 4      | [H] ← [L]                                 |
| MOV H,M     | 66      | 1     | 4      | [H] ← [L]                                 |
| MOV L,A     | 6F      | 1     | 7      | [H] ← [(HL)]                              |
| MOV L,B     | 68      | 1     | 4      | [L] ← [A]                                 |
| MOV L,C     | 69      | 1     | 4      | [L] ← [B]                                 |
| MOV L,D     | 6A      | 1     | 4      | [L] ← [C]                                 |
| MOV L,E     | 6B      | 1     | 4      | [L] ← [D]                                 |
| MOV L,H     | 6C      | 1     | 4      | [L] ← [E]                                 |
| MOV L,L     | 6D      | 1     | 4      | [L] ← [H]                                 |

Table 2.1 Summary of 8085 Instruction Set (continued)

| Instruction | OP Code | Bytes | Cycles | Operations performed                                                                                                                           |
|-------------|---------|-------|--------|------------------------------------------------------------------------------------------------------------------------------------------------|
| MOV L,M     | 6B      | 1     | 7      | $[L] \leftarrow [[HL]]$                                                                                                                        |
| MOV M,A     | 77      | 1     | 7      | $[[HL]] \leftarrow [A]$                                                                                                                        |
| MOV M,B     | 70      | 1     | 7      | $[[HL]] \leftarrow [B]$                                                                                                                        |
| MOV M,C     | 71      | 1     | 7      | $[[HL]] \leftarrow [C]$                                                                                                                        |
| MOV M,D     | 72      | 1     | 7      | $[[HL]] \leftarrow [D]$                                                                                                                        |
| MOV M,E     | 73      | 1     | 7      | $[[HL]] \leftarrow [E]$                                                                                                                        |
| MOV M,H     | 74      | 1     | 7      | $[[HL]] \leftarrow [H]$                                                                                                                        |
| MOV M,L     | 75      | 1     | 7      | $[[HL]] \leftarrow [L]$                                                                                                                        |
| MVI A, DATA | 3E      | 2     | 7      | $[A] \leftarrow \text{second instruction byte}$                                                                                                |
| MVI B, DATA | 06      | 2     | 7      | $[B] \leftarrow \text{second instruction byte}$                                                                                                |
| MVI C, DATA | 0E      | 2     | 7      | $[C] \leftarrow \text{second instruction byte}$                                                                                                |
| MVI D, DATA | 16      | 2     | 7      | $[D] \leftarrow \text{second instruction byte}$                                                                                                |
| MVI E, DATA | 1E      | 2     | 7      | $[E] \leftarrow \text{second instruction byte}$                                                                                                |
| MVI H, DATA | 26      | 2     | 7      | $[H] \leftarrow \text{second instruction byte}$                                                                                                |
| MVI L, DATA | 2E      | 2     | 7      | $[L] \leftarrow \text{second instruction byte}$                                                                                                |
| MVI M, DATA | 36      | 2     | 10     | $[[HL]] \leftarrow \text{second instruction byte}$                                                                                             |
| NOP         | 00      | 1     | 4      | No operation                                                                                                                                   |
| ORA A       | B7      | 1     | 4      | $[A] \leftarrow [A] \vee [A]$                                                                                                                  |
| ORA B       | B0      | 1     | 4      | $[A] \leftarrow [A] \vee [B]$                                                                                                                  |
| ORA C       | B1      | 1     | 4      | $[A] \leftarrow [A] \vee [C]$                                                                                                                  |
| ORA D       | B2      | 1     | 4      | $[A] \leftarrow [A] \vee [D]$                                                                                                                  |
| ORA E       | B3      | 1     | 4      | $[A] \leftarrow [A] \vee [E]$                                                                                                                  |
| ORA H       | B4      | 1     | 4      | $[A] \leftarrow [A] \vee [H]$                                                                                                                  |
| ORA L       | B5      | 1     | 4      | $[A] \leftarrow [A] \vee [L]$                                                                                                                  |
| ORA M       | B6      | 1     | 7      | $[A] \leftarrow [A] \vee [[HL]]$                                                                                                               |
| ORI DATA    | F6      | 2     | 7      | $[A] \leftarrow [A] \vee \text{second instruction byte}$                                                                                       |
| OUT PORT    | D3      | 2     | 10     | $[\text{specified port}] \leftarrow [A]$                                                                                                       |
| PCHL        | E9      | 1     | 6      | $[\text{PCH}]^* \leftarrow [H], [\text{PCL}]^* \leftarrow [L]$                                                                                 |
| POP B       | C1      | 1     | 10     | $[C] \leftarrow [[SP]], [SP] \leftarrow [SP] + 2$<br>$[B] \leftarrow [[SP] + 1]$                                                               |
| POP D       | D1      | 1     | 10     | $[E] \leftarrow [[SP]], [SP] \leftarrow [SP] + 2$<br>$[D] \leftarrow [[SP] + 1]$                                                               |
| POP H       | E1      | 1     | 10     | $[L] \leftarrow [[SP]], [SP] \leftarrow [SP] + 2$<br>$[H] \leftarrow [[SP] + 1]$                                                               |
| POP PSW     | F1      | 1     | 10     | $[A] \leftarrow [[SP] + 1], [\text{PSW}] \leftarrow [[SP]], [SP] \leftarrow [SP] + 2$<br>$[[SP] - 1] \leftarrow [B], [SP] \leftarrow [SP] - 2$ |
| PUSH B      | C5      | 1     | 12     | $[[SP] - 2] \leftarrow [C]$                                                                                                                    |
| PUSH D      | D5      | 1     | 12     | $[[SP] - 1] \leftarrow [D], [[SP] - 2] \leftarrow [E]$<br>$[SP] \leftarrow [SP] - 2$                                                           |
| PUSH H      | E5      | 1     | 12     | $[[SP] - 1] \leftarrow [H], [SP] \leftarrow [SP] - 2$<br>$[[SP] - 2] \leftarrow [L]$                                                           |
| PUSH PSW    | F5      | 1     | 12     | $[[SP] - 1] \leftarrow [A], [SP] \leftarrow [SP] - 2$<br>$[[SP] - 2] \leftarrow [\text{PSW}]$                                                  |
| RAL         | 17      | 1     | 4      |                                                                                                                                                |
| RAR         | 1F      | 1     | 4      |                                                                                                                                                |
| RC          | D8      | 1     | 6/12   | Return if carry; $[\text{PC}] \leftarrow [SP]$                                                                                                 |
| RET         | C9      | 1     | 10     | $[\text{PCL}]^* \leftarrow [[SP]], [SP] \leftarrow [SP] + 2$<br>$[\text{PCH}]^* \leftarrow [[SP] + 1]$                                         |
| RIM         | 20      | 1     | 4      | Read interrupt mask                                                                                                                            |
| RLC         | 07      | 1     | 4      |                                                                                                                                                |
| RM          | F8      | 1     | 6/12   | Return if minus; $[\text{PC}] \leftarrow [[SP]]$                                                                                               |

Table 2.1 Summary of 8085 Instruction Set (continued)

| Instruction | OP Code | Bytes | Cycles | Operations performed                                                               |
|-------------|---------|-------|--------|------------------------------------------------------------------------------------|
| RNC         | D0      | 1     | 6/12   | Return if no carry; [PC] $\leftarrow$ [[SP]]                                       |
| RNZ         | C0      | 1     | 6/12   | Return if result not zero; [PC] $\leftarrow$ [[SP]]                                |
| RP          | F0      | 1     | 6/12   | Return if result positive; [PC] $\leftarrow$ [[SP]], [SP] $\leftarrow$ [SP] + 2    |
| RPS         | E8      | 1     | 6/12   | Return if parity even; [PC] $\leftarrow$ [[SP]], [SP] $\leftarrow$ [SP] + 2        |
| RPO         | ED      | 1     | 6/12   | Return if parity odd; [PC] $\leftarrow$ [[SP]], [SP] $\leftarrow$ [SP] + 2         |
| RRC         | 0F      | 1     | 4      |                                                                                    |
|             |         |       |        |  |
| RST0        | C7      | 1     | 12     | Restart                                                                            |
| RST1        | CF      | 1     | 12     | Restart                                                                            |
| RST2        | D7      | 1     | 12     | Restart                                                                            |
| RST3        | DF      | 1     | 12     | Restart                                                                            |
| RST4        | E7      | 1     | 12     | Restart                                                                            |
| RST5        | EF      | 1     | 12     | Restart                                                                            |
| RST6        | F7      | 1     | 12     | Restart                                                                            |
| RST7        | FF      | 1     | 12     | Restart                                                                            |
| RZ          | C8      | 1     | 6/12   | Return if zero; [PC] $\leftarrow$ [[SP]]                                           |
| SBB A       | 9F      | 1     | 4      | [A] $\leftarrow$ [A] - [A] - [Cy]                                                  |
| SBB B       | 98      | 1     | 4      | [A] $\leftarrow$ [A] - [B] - [Cy]                                                  |
| SBB C       | 99      | 1     | 4      | [A] $\leftarrow$ [A] - [C] - [Cy]                                                  |
| SBB D       | 9A      | 1     | 4      | [A] $\leftarrow$ [A] - [D] - [Cy]                                                  |
| SBB E       | 9B      | 1     | 4      | [A] $\leftarrow$ [A] - [E] - [Cy]                                                  |
| SBB H       | 9C      | 1     | 4      | [A] $\leftarrow$ [A] - [H] - [Cy]                                                  |
| SBB L       | 9D      | 1     | 4      | [A] $\leftarrow$ [A] - [L] - [Cy]                                                  |
| SBB M       | 9E      | 1     | 7      | [A] $\leftarrow$ [A] - [[HL]] - [Cy]                                               |
| SBI DATA    | DE      | 2     | 7      | [A] $\leftarrow$ [A] - second instruction byte - [Cy]                              |
| SHLD ppqq   | 22      | 3     | 16     | [ppqq] $\leftarrow$ [L], [ppqq + 1] $\leftarrow$ [H]                               |
| SIM         | 30      | 1     | 4      | Set interrupt mask                                                                 |
| SPHL        | F9      | 1     | 6      | [SP] $\leftarrow$ [HL]                                                             |
| STA ppqq    | 32      | 3     | 13     | [ppqq] $\leftarrow$ [A]                                                            |
| STAX B      | 02      | 1     | 7      | [[BC]] $\leftarrow$ [A]                                                            |
| STAX D      | 12      | 1     | 7      | [[DE]] $\leftarrow$ [A]                                                            |
| STC         | 37      | 1     | 4      | [Cy] $\leftarrow$ 1                                                                |
| SUB A       | 97      | 1     | 4      | [A] $\leftarrow$ [A] - [A]                                                         |
| SUB B       | 90      | 1     | 4      | [A] $\leftarrow$ [A] - [B]                                                         |
| SUB C       | 91      | 1     | 4      | [A] $\leftarrow$ [A] - [C]                                                         |
| SUB D       | 92      | 1     | 4      | [A] $\leftarrow$ [A] - [D]                                                         |
| SUB E       | 93      | 1     | 4      | [A] $\leftarrow$ [A] - [E]                                                         |
| SUB H       | 94      | 1     | 4      | [A] $\leftarrow$ [A] - [H]                                                         |
| SUB L       | 95      | 1     | 4      | [A] $\leftarrow$ [A] - [L]                                                         |
| SUB M       | 96      | 1     | 7      | [A] $\leftarrow$ [A] - [[HL]]                                                      |
| SUI DATA    | D6      | 2     | 7      | [A] $\leftarrow$ [A] - second instruction byte                                     |
| XCHG        | EB      | 1     | 4      | [D] $\leftrightarrow$ [H], [E] $\leftrightarrow$ [L]                               |
| XRA A       | AF      | 1     | 4      | [A] $\leftarrow$ [A] $\oplus$ [A]                                                  |
| XRA B       | A8      | 1     | 4      | [A] $\leftarrow$ [A] $\oplus$ [B]                                                  |
| XRA C       | A9      | 1     | 4      | [A] $\leftarrow$ [A] $\oplus$ [C]                                                  |
| XRA D       | AA      | 1     | 4      | [A] $\leftarrow$ [A] $\oplus$ [D]                                                  |
| XRA E       | AB      | 1     | 4      | [A] $\leftarrow$ [A] $\oplus$ [E]                                                  |
| XRA H       | AC      | 1     | 4      | [A] $\leftarrow$ [A] $\oplus$ [H]                                                  |
| XRA L       | AD      | 1     | 4      | [A] $\leftarrow$ [A] $\oplus$ [L]                                                  |
| XRA M       | AE      | 1     | 7      | [A] $\leftarrow$ [A] $\oplus$ [[HL]]                                               |
| XRI DATA    | EE      | 2     | 7      | [A] $\leftarrow$ [A] $\oplus$ second instruction byte                              |
| XTHL        | E3      | 1     | 16     | [[SP]] $\leftrightarrow$ [L], [[SP] + 1] $\leftrightarrow$ [H]                     |

\*PCl — program counter low byte; PCH — program counter high byte.

Table 2.2 8085 Instructions Affecting the Status Flags

| Instruction <sup>a</sup> | Status flags <sup>b</sup> |                |   |   |   |
|--------------------------|---------------------------|----------------|---|---|---|
|                          | C <sub>f</sub>            | N <sub>c</sub> | Z | S | P |
| ADD DATA                 | +                         | +              | + | + | + |
| ADC reg                  | +                         | +              | + | + | + |
| ADC M                    | +                         | +              | + | + | + |
| ADD reg                  | +                         | +              | + | + | + |
| ADD M                    | +                         | +              | + | + | + |
| ADN DATA                 | +                         | 0              | + | + | + |
| ANA reg                  | 0                         | 0              | + | + | + |
| ANA M                    | 0                         | 0              | + | + | + |
| ANI DATA                 | 0                         | 0              | + | + | + |
| CMC                      | +                         |                |   |   |   |
| CMP reg                  | +                         | +              | + | + | + |
| CMP M                    | +                         | +              | + | + | + |
| CPI DATA                 | +                         | +              | + | + | + |
| DAA                      | +                         | +              | + | + | + |
| DAD rp                   | +                         |                |   |   |   |
| DCR reg                  |                           | +              | + | + | + |
| DCR M                    |                           | +              | + | + | + |
| INR reg                  |                           | +              | + | + | + |
| INR M                    |                           | +              | + | + | + |
| ORA reg                  | 0                         | 0              | + | + | + |
| ORA M                    | 0                         | 0              | + | + | + |
| ORI DATA                 | 0                         | 0              | + | + | + |
| RAL                      | +                         |                |   |   |   |
| RAR                      | +                         |                |   |   |   |
| RLC                      | +                         |                |   |   |   |
| RRC                      | +                         |                |   |   |   |
| SBB reg                  | +                         | +              | + | + | + |
| SBB M                    | +                         | +              | + | + | + |
| SBI DATA                 | +                         | +              | + | + | + |
| STC                      | +                         |                |   |   |   |
| SUB reg                  | +                         | +              | + | + | + |
| SUB M                    | +                         | +              | + | + | + |
| SUI DATA                 | +                         | +              | + | + | + |
| XRA reg                  | 0                         | 0              | + | + | + |
| XRA M                    | 0                         | 0              | + | + | + |
| XRI DATA                 | 0                         | 0              | + | + | + |

<sup>a</sup>reg — 8-bit register; M — memory; rp — 16-bit register pair.<sup>b</sup>Note that instructions which are not shown in the table do not affect the flags; + indicates that the particular flag is affected; 0 or 1 indicates that these flags are always 0 or 1 after the corresponding instructions are executed.

All mnemonics copyright Intel Corporation 1976.

Table 2.3 Symbols to be Used in 8085 Instruction Set

| Symbol                          | Interpretation                                                     |
|---------------------------------|--------------------------------------------------------------------|
| r <sub>1</sub> , r <sub>2</sub> | 8-bit register                                                     |
| rp                              | Register pair                                                      |
| data8                           | 8-bit data                                                         |
| data16                          | 16-bit data                                                        |
| M                               | Memory location indirectly addressed through the register pair H,L |
| addr16                          | 16-bit memory address                                              |

**Table 2.4 8085 MOVE Instructions**

| Instruction  | Symbolic<br>description           | Addressing mode   |                   | Illustration |                                                                                                                  |
|--------------|-----------------------------------|-------------------|-------------------|--------------|------------------------------------------------------------------------------------------------------------------|
|              |                                   | Source            | Destination       | Example      | Comments                                                                                                         |
| MOV r1, r2   | $(r1) \leftarrow (r2)$            | Register          | Register          | MOV A, B     | Copy the contents of the register B into the register A                                                          |
| MOV r, M     | $(r) \leftarrow M((HL))$          | Register indirect | Register          | MOV A, M     | Copy the contents of the memory location whose address is specified in the register pair H,L into the A register |
| MVI r, data8 | $(r) \leftarrow \text{data8}$     | Immediate         | Register          | MVI A, 08    | Initialize the A register with the value 08                                                                      |
| MOV M, r     | $M((HL)) \leftarrow (r)$          | Register          | Register indirect | MOV M, B     | Copy the contents of the B register into the memory location addressed by H,L pair                               |
| MVI M, data8 | $M((HL)) \leftarrow \text{data8}$ | Immediate         | Register indirect | MVI M, 07    | Initialize the memory location whose address is specified in the register pair H,L with the value 07             |

Now the program for the 5-s delay can be

```

LXI SP, 5000H ; Set stack pointer
MVI C, 32H ; Do DELAY loop 5016 times by loading C
START LXI D, 30D3H ; with count 3216
CALL DELAY ; Load initial count
DCR C ; Call DELAY loop
; Decrement C and check if zero: if
JNZ START ; not, do another delay
HLT ; Loop back
; STOP

```

In the above, since execution times of DCR C and JNZ START are very small compared to 5 s, they are not considered in computing the delay.

## 2.7 8085 Pins and Signals

The 8085 is housed in a 40-pin dual in-line package (DIP). Figure 2.8 shows the 8085 pins and signals.

The low-order address byte and data lines AD0 to AD7 are multiplexed. These lines are bidirectional. The beginning of an instruction is indicated by the rising edge of the ALE signal. At the falling edge of ALE, the low byte of the address is automatically latched by some of the 8085 support chips such as 8155 and 8355: AD0 to AD7 lines can then be used as data lines. Note that ALE is an input to these support chips. However, if the support chips do not latch AD0 to AD7, then external latches are required to generate eight separate address lines A7 to A0 at the falling edge of ALE.

Pins A8 to A15 are unidirectional and contain the high byte of the address.

Table 2.11 lists the 8085 pins along with a brief description of each.

The RD pin signal is output LOW by the 8085 during a memory or I/O READ operation. Similarly, the WR pin signal is output LOW during a memory or I/O WRITE.

Next, we explain the purpose of IO/M, S0, and S1 signals. The IO/M signal is output HIGH by the 8085 to indicate execution of an I/O instruction such as IN or OUT. This pin is output LOW during execution of a memory instruction such as LDA 2050H.

The IO/M, S0, and S1 are output by the 8085 during its internal operations, which can be interpreted as follows:

| IO/M | S1 | S0 | Operation performed by the 8085 |
|------|----|----|---------------------------------|
| 0    | 0  | 1  | Memory WRITE                    |
| 0    | 1  | 0  | Memory READ                     |
| 1    | 0  | 1  | I/O WRITE                       |
| 1    | 1  | 0  | I/O READ                        |
| 0    | 1  | 1  | OP code fetch                   |
| 1    | 1  | 1  | Interrupt acknowledge           |



FIGURE 2.8 8085 microprocessor signals and pin assignments.

Table 2.11 8085 Signal Description Summary

| Pin name      | Description                         | Type                     |
|---------------|-------------------------------------|--------------------------|
| AD0-AD7       | Address/data bus                    | Bi-directional, tristate |
| A8-A15        | Address bus                         | Output, tristate         |
| <u>ALE</u>    | Address latch enable                | Output, tristate         |
| <u>RD</u>     | Read control                        | Output, tristate         |
| <u>WR</u>     | Write control                       | Output, tristate         |
| <u>IO / M</u> | I/O or memory indicator             | Output, tristate         |
| S0, S1        | Bus state indicators                | Output                   |
| READY         | Wait state request                  | Input                    |
| SID           | Serial data input                   | Input                    |
| SOD           | Serial data output                  | Output                   |
| HOLD          | Hold request                        | Input                    |
| HLDA          | Hold acknowledge                    | Output                   |
| INTR          | Interrupt request                   | Input                    |
| TRAP          | Nonmaskable interrupt request       | Input                    |
| RST5.5        | Hardware vectored                   | Input                    |
| RST6.5        | Hardware vectored interrupt request | Input                    |
| <u>RST7.5</u> | Hardware vectored                   | Input                    |
| <u>INTA</u>   | Interrupt acknowledge               | Output                   |
| RESET IN      | System reset                        | Input                    |
| RESET OUT     | Peripherals reset                   | Output                   |
| X1, X2        | Crystal or RC connection            | Input                    |
| CLK (OUT)     | Clock signal                        | Output                   |
| Vcc, Vss      | Power, ground                       |                          |



(a) 8085 - 2716 interface using internal latches.



(b) 8085 - 8155 interface using ALE and AD0 to AD7

FIGURE 2.9 8085's interface to external device using ALE and the multiplexed AD0 to AD7 pins.

Figure 2.9 illustrates the utilization of ALE and AD0 to AD7 signals for interfacing an EPROM and a RAM.

The 2716 is a  $2K \times 8$  EPROM with separate address and data lines without any built-in latches. This means that a separate latch such as the 74LS373 must be used to isolate the 8085 low byte address and D0-D7 data lines at the falling edge of ALE (Figure 2.9a).

The 8155 contains 256-byte static RAM, three user ports, and a 14-bit timer. The 8155 is designed for 8085 in the sense that it has built-in latches with ALE as input along with multiplexed address (low byte) and data lines, AD0 to AD7. Therefore, as shown in Figure 2.9b, external latches are not required.

The READY input can be used by the slower external devices for obtaining extra time in order to communicate with the 8085. The READY signal (when LOW) can be utilized to provide wait-state clock periods in the 8085 machine. If READY is HIGH during a read or write cycle, it indicates that the memory or peripheral is ready to send or receive data. If not used, it must be tied high.

The Serial Input Data (SID) and Serial Output Data (SOD) lines are associated with the 8085 serial I/O transfer. The SOD line can be used to output the most significant bit of the accumulator. The SID signal can be input into the most significant bit of the accumulator.

The HOLD and HLDA signals are used for the Direct Memory Access (DMA) type of data transfer. The external devices place a HIGH on HOLD line in order to take control of the system bus. The HOLD function is acknowledged by the 8085 by placing a HIGH output on the HLDA pin and driving the tristate outputs high impedance.

The signals on the TRAP, RST7.5, RST6.5, RST5.5, INTR, and INTA are related to the 8085 interrupt signals. TRAP is a nonmaskable interrupt; that is, it cannot be enabled or disabled.



FIGURE 2.10 Crystal connection to  $X_1$  and  $X_2$  pins.

by an instruction. The TRAP has the highest priority. RST7.5, RST6.5, and RST5.5 are maskable interrupts used by the devices whose vector addresses are generated automatically. INTA is an interrupt acknowledge signal which is pulsed LOW by the 8085 in response to the interrupt INTR request. In order to service INTR, one of the eight OP codes (RST0 to RST7) has to be provided on the 8085 AD0-AD7 bus by external logic. The 8085 then executes this instruction and vectors to the appropriate address to service the interrupt.

All unused control pins such as interrupts and HOLD must be disabled by grounding them. (READY must be tied high).

The 8085 has the clock generation circuit on the chip and, therefore, no external oscillators need to be designed. The 8085A can operate with a maximum clock frequency of 3.03 MHz and the 8085A-2 can be driven with a maximum of 5 MHz clock. The 8085 clock frequency can be generated by a crystal, an LC tuned circuit, or an external clock circuit. The frequency at  $X_1X_2$  is divided by 2 internally. This means that in order to obtain 3.03 MHz, a clock source of 6.06 MHz must be connected to  $X_1X_2$ . For crystals of less than 4 MHz, a capacitor of 20 pF should be connected between  $X_2$  and a ground to ensure the starting up of the crystal at the right frequency (Figure 2.10).

There is a TTL signal which is output on pin 37, called the CLK (OUT) signal. This signal can be used by other external microprocessors or support chips.

The RESET IN signal, when pulsed LOW then high, causes the 8085 to execute the first instruction at the  $0000_{16}$  location. In addition, the 8085 resets instruction register, interrupt mask (RST5.5, RST6.5, and RST7.5) bits, and other registers. The RESET IN must be held LOW for at least three clock periods. A typical 8085 reset circuit is shown in Figure 2.11. In this circuit, when the switch is activated, RESET IN is driven to LOW with a large time constant providing adequate time to reset the system.

The 8085 requires a minimum operating voltage of 4.75 V. Upon applying power, the 8085A attains this voltage after 500  $\mu\text{s}$ . The reset circuit of Figure 2.11 resets the 8085 upon activation



FIGURE 2.11 8085 reset circuit.

| M1 |    |    |    | M2 |    |    | M3 |    |    | M4 |    |    | M5 |    |  |
|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|--|
| T1 | T2 | T3 | T4 | T5 | T6 | T1 | T2 | T3 | T1 | T2 | T3 | T1 | T2 | T3 |  |

FIGURE 2.12 8085 machine cycles.

of the switch. The voltage across the  $0.1\text{-}\mu\text{F}$  capacitor is zero on power-up. The capacitor then charges  $V_{cc}$  after a definite time determined by the time constant  $RC$ . The chosen values of  $RC$  in the figure will drive the RESET IN pin to low for at least three clock periods. In this case, after activating the switch, RESET IN will be low (assuming capacitance charge time is equal to the discharge time) for  $10K \cdot 0.1\text{ }\mu\text{F} = 1\text{ ms}$ , which is greater than three clock periods ( $3 \cdot 1/3\text{ }\mu\text{s} = 1\text{ }\mu\text{s}$ ) of the 3-MHz 8085A. During normal operation of the 8085, activation of the switch will short the capacitor to ground and will discharge it. When the switch is opened, the capacitor charges and the RESET IN pin becomes HIGH. Upon hardware reset, the 8085 clears PC, IR, HALT flip-flop, and some other registers; the 8085 registers PSW, A, B, C, D, E, H, and L are unaffected. Upon activation of the RESET IN to low, the 8085 outputs HIGH at the RESET OUT pin which can be used to reset the memory and I/O chips connected to the 8085. Note that since hardware reset initializes PC to 0, the 8085 fetches the first instruction for address  $0000_{16}$  after reset.

## 2.8 8085 Instruction Timing and Execution

An 8085's instruction execution consists of a number of machine cycles (MCs). These cycles vary from one to five (M1 to M5) depending on the instruction. Each machine cycle contains a number of 320-ns clock periods. The first machine cycle will be executed in either four or six clock periods, and the machine cycles that follow will have three clock periods. This is shown in Figure 2.12.

The shaded MCs indicate that these machine cycles are required by certain instructions. Similarly, the shaded clock periods (T5 and T6) mean that they are needed in M1 by some instructions.

The clock periods within a machine cycle can be illustrated as shown in Figure 2.13. Note that the beginning of a new machine cycle is indicated on the 8085 by outputting the Address Latch Enable (ALE) signal HIGH. During this time, lines AD0 to AD7 are used for placing the low byte of the address.

When the ALE signal goes LOW, the low byte of the address is latched so that the AD0 to AD7 lines can be used for transferring data.

We now discuss the timing diagrams for instruction fetch, READ, and WRITE.



FIGURE 2.13 Clock period within a machine cycle.

Figure 2.14 shows the 8085 basic system timing. An instruction execution includes two operations: OP code fetch and execution.

The OP code fetch cycle requires either four (for one-byte instructions such as `MOV A,B`) or six cycles (for 3 byte instructions such as `LDA 2030H`). The machine cycles that follow will need three clock periods.

The purpose of an instruction fetch is to read the contents of a memory location containing an instruction addressed by the program counter and to place it in the instruction register. The 8085 instruction fetch timing diagram shown in Figure 2.14 can be explained in the following way:

1. The 8085 puts a LOW on the  $\overline{IO/M}$  line of the system bus, indicating a memory operation.
2. The 8085 sets  $S0 = 1$  and  $S1 = 1$  on the system bus, indicating the memory fetch operation.
3. The 8085 places the program counter high byte on the A8 to A15 lines and the program counter low byte on the AD0 to AD7 lines of the system bus. The 8085 also sets the ALE signal to HIGH. As soon as the ALE signal goes to LOW, the program counter low byte on the AD0 to AD7 is latched automatically by some 8085 support chips such as 8155 (if 8085 support chips are not used, these lines must be latched using external latches), since these lines will be used as data lines for reading the OP code.
4. At the beginning of T2 in M1, the 8085 puts the RD line to LOW indicating a READ operation. After some time, the 8085 loads the OP code (the contents of the memory location addressed by the program counter) into the instruction register.
5. During the T4 clock period in M1, the 8085 decodes the instruction.

The Machine Cycle M2 of Figure 2.14 shows a memory (or I/O) READ operation as seen by the external logic, and the status of the S0 and S1 signals indicates whether the operation is instruction fetch or memory READ; for example,  $S1 = 1, S0 = 1$  during instruction fetch and  $S1 = 1, S0 = 0$  during memory READ provided  $\overline{IO/M} = 0$ .

The purpose of the memory READ is to read the contents of a memory location addressed by a register pair, such as the H,L pair, or a memory location specified with the instruction and the data placed in a microprocessor register such as the accumulator. In contrast, the purpose of the memory fetch is to read the contents of a memory location addressed by PC into IR. The machine cycle M3 of Figure 2.14 indicates a memory (or I/O) write operation. In this case,  $S1 = 0$  and  $S0 = 1$  indicate a memory write operation when  $\overline{IO/M} = 0$  and an I/O write operation when  $\overline{IO/M} = 1$ .

## 2.8.2 8085 Memory READ ( $\overline{IO/M} = 0, \overline{RD} = 0$ ) and I/O READ ( $\overline{IO/M} = 1, \overline{RD} = 0$ )

Figure 2.15a shows an 8085A clock timing diagram. The machine cycle of M2 of Figure 2.14 shows a memory READ timing diagram.

The purpose of the memory READ is to read the contents of a memory location addressed by a register pair, such as HL. Let us explain the 8085 memory READ timing diagram of Figure 2.15b along with the READ timing signals of Figure 2.14:

1. The 8085 uses machine cycle M1 to fetch and decode the instruction. It then performs the memory READ operation in M2.
2. The 8085 continues to maintain  $\overline{IO/M}$  at LOW in M2 indicating a memory READ operation (or  $\overline{IO/M} = 1$  for I/O READ).
3. The 8085 puts  $S1 = 1, S0 = 0$ , indicating a READ operation.
4. The 8085 places the contents of the high byte of the memory address register, such as the contents of the H register, on lines A8 to A15.



FIGURE 2.14 8085 Basic system timing.

5. The 8085 places the contents of the low byte of the memory address register, such as the contents of the L register, in lines AD0 to AD7.
6. The 8085 sets ALE to high, indicating the beginning of M2. As soon as ALE goes to low, the memory or support chip must latch the low byte of the address lines, since the same lines are going to be used as data lines.
7. The 8085 puts the RD signal to LOW, indicating a READ operation.
8. The 8085 gets the data from the memory location addressed by the memory address register, such as the H,L pair, and places the data into a register such as the accumulator. In case of I/O, the 8085 inputs data from the I/O port into the accumulator.

### 2.8.3 8085 Memory WRITE ( $\overline{IO/M} = 0$ , $\overline{WR} = 0$ ) and I/O WRITE ( $\overline{IO/M} = 1$ , $\overline{WR} = 0$ )

The machine cycle M3 of Figure 2.14 shows a memory WRITE timing diagram. As seen by the external logic, the signals  $S1 = 0$ ,  $S0 = 1$ , and  $\overline{WR} = 0$  indicate a memory WRITE operation.

The purpose of a memory WRITE is to store the contents of the 8085 register, such as the accumulator, into a memory location addressed by a pair, such as H,L.

The WRITE timing diagram of Figure 2.14 can be explained as follows:

1. The 8085 uses machine cycle M1 to fetch and decode the instruction. It then executes the memory WRITE instruction in M3.
2. The 8085 continues to maintain  $\overline{IO/M}$  at LOW, indicating a memory operation (or  $\overline{IO/M} = 1$  for I/O WRITE).
3. The 8085 puts  $S1 = 0$ ,  $S0 = 1$ , indicating a WRITE operation.
4. The 8085 places the Memory Address Register high byte, such as the contents of the H register, on lines A8 to A15.
5. The 8085 places the Memory Address Register low byte, such as the contents of L register, on lines AD0 to AD7.
6. The 8085 sets ALE to HIGH, indicating the beginning of M3. As soon as ALE goes to LOW, the memory or support chip must latch the low byte of the address lines, since the same lines are going to be used as data lines.
7. The 8085 puts the WR signal to LOW, indicating a WRITE operation.
8. It also places the contents of the register, say, accumulator, on data lines AD0 to AD7.
9. The external logic gets data from the lines AD0 to AD7 and stores the data in the memory location addressed by the Memory Address Register, such as the H,L pair. In case of I/O, the 8085 outputs [A] to an I/O port.

Figures 2.15a through c show the 8085A clock and read and write timing diagrams.

## 2.9 8085 Input/Output (I/O)

The 8085 I/O transfer techniques are discussed. The 8355/8755 and 8155/8156 I/O ports and 8085 SID and SOD lines are also included.

### 2.9.1 8085 Programmed I/O

There are two I/O instructions in the 8085, namely, IN and OUT. These instructions are 2 bytes long. The first byte defines the OP code of the instruction and the second byte specifies the I/O port number. Execution of the IN PORT instruction causes the 8085 to receive one byte of data into the accumulator from a specified I/O port. On the other hand, the OUT PORT instruction, when executed, causes the 8085 to send one byte of data from the accumulator into a specified I/O port.



FIGURE 2.15a 8085A clock.



FIGURE 2.15b 8085 Read Timing diagram.



FIGURE 2.15c 8085 Write Timing diagram.

The 8085 can access I/O ports using either standard I/O or memory-mapped I/O. In standard I/O, the 8085 inputs or outputs data using IN or OUT instructions.

In memory-mapped I/O, the 8085 maps I/O ports as memory addresses. Hence, LDA addr or STA addr instructions are used to input or output data to or from the 8085. The 8085's programmed I/O capabilities are obtained via the support chips, namely, 8355/8755 and 8155/8156. The 8355/8755 contains a 2K-byte ROM/EPROM and two 8-bit I/O ports (ports A and B).

The 8155/8156 contains 256-byte RAM, two 8-bit and one 6-bit I/O ports, and a 14-bit programmable timer. The only difference between the 8155 and 8156 is that chip enable is LOW on the 8155 and HIGH on the 8156.