Technical Description of the EXOS 302

High-Performance VME Bus Ethernet Adapter

(Rev. 2 PCB and later)

Don Cohrac

Excelan, Inc.

# 1. Introduction

This document describes the EXOS 302 network processor board. This is a preliminary document which describes what will be included in the initial design.

#### 2. General Description (Hardware Architecture)

0

The EXOS 302 is a microprocessor-controlled adapter which provides an Ethernet connection for computers using the VME bus. Architecturally it is compatible with the NX300 network software.

The EXOS 302 operates as a bus master, capable of reading and writing the host memory. Coordination between the host and the EXOS 302 is accomplished by the establishment of control blocks in host memory, the passing of values via special registers, and interrupts. The EXOS 302 operates as a slave to a bus master reading or writing its control registers.

The architecture is represented in Figure 1. Its major features are :

- o INTEL 80286 Microprocessor, 8MHz
- o INTEL 82586 Ethernet Controller
  - IEEE 802.3 Compliance 10 Base 5
  - ETHERNET V1 and V2 Compatibility
  - D-type 15-pin transceiver connector
- o 512KB or 1MB Local Memory
  - 32KB or 64KB Local PROM for 80286 firmware
- o INTEL 8259A Interrupt Controller
- o Dual UART for RS232 port and timer
- o Host bus interface :
  - 8- or 16-bit data
  - 24- or 32-bit address
  - Daisy-chained bus request/grant
  - Daisy-chain interrupt structure
- o EXOS 202 compatibility mode (24-bit address)



File "la: x la/jai 302pvu" dated Sep 6,1989 13:43

# Figure 1. EXOS 302 Block Diagram

#### 3. Functional Descriptiom

#### 3.1. CPU

The processing element of the EXOS 302 is an Intel 80286 microprocessor. operating at an 8 MHz clock rate. It is supported by the 82284 clock chip, 82288 bus controller, and 8259A interrupt controller.

## 3.2. Memory Address Space

In real mode the 80286 has 1Mb of memory space, which is allocated as shown in Figure 2. The local memory appears as the lowest 512KB in the memory space. The host memory is mapped into the local space, and is limited to a relocatable 128KB. All local I/O devices reside in the 80286 I/O space (see section 4.1).

The 82586 can access only the dual-ported local memory; it can access the complete 512KB. It cannot access host memory.

In protected mode the 80286 can access beyond 1MB. An additional 512KB is provided at the beginning of the second megabyte. The 82586 can also access the additional 512KB.



----- NA-- 0 1022

#### 3.3. Local Memory

Local memory is two blocks of 512KB of dynamic RAM (total 1MB). The memory is 16 bits wide, and is dual-ported to the 80286 and the 82586. It is not accessible from the host bus. The memory has a minimum 1 wait state and maximum 6 wait states for the 82586, with an average of 3. It has a minimum of zero wait states and a maximum of 7 for the 80286, with an average of 1. Wait states above the minimum are caused by memory or refresh cycles-in-progress. CAS-before-RAS refresh is used.

The upper 512KB is accessible only when the 80286 is operating in the protected mode and a control bit is set in the EXOS Status Port (see section 4.1.11).

#### 3.4. EPROM

Two sockets are provided for up to 64KB of EPROM for the 80286 firmware. EPROMs of 16K, 32K or 64K bytes can be used (see section 5.2.2).

#### 3.5. Network Controller

The network controller performs the Ethernet control functions with minimum aid from the 80286. It consists of an Intel 82586 controller and an 8023A Ethernet Serial Interface chip.

The 80286 and the 82586 communicate via the local memory and the Channel Attention and network interrupt signals. Essentially the 80286 provides buffer descriptors for receiving and stransmitting packets in the local memory, and generates the Channel Attention. Then the 82586 transmits or receives an Ethernet packet between the network and the local memory, and generates an interrupt to the 80286 when the operation is complete. Synchronization between the 80286 and the 82586 is maintained by means of the Channel Attention and interrupt. Neither the 80286 nor the 82586 can be locked out from accessing memory, so care is required in the implementation of semaphores. The 82586 documentation from INTEL describes the network interface and the 80286/82586 protocol in detail.

The 82586 reset signal is a latched bit, and is true upon reset, or by control of the 80286 via the CPU Control Port. The 82586 should be initialized before enabling the upper memory. Initialization data for the 82586 must be set up at memory location 007FFF6 (system configuration pointer) prior to the first assertion of Channel Attention. If the 80586 attempts to fetch the configuration pointer while the upper memory is enabled, it will fetch from location 017FFF6 (address bit 20 is enabled, and address bit 19 is ignored because of the discontinuity in memory between 512K and 1M).

The hardware provides two loopback features for confidence test capability. A bit in the CPU Control Port enables loopback through the transceiver interface chip. The 82586 also has facilities for internal loopback and diagnostics (see Intel documentation).

#### 3.6. DUART/Timer

An RS232C serial port for debug purposes and a periodic interrupt timer are provided by a single LSI chip, Signetics SCN2681. See section 4.1.1 for details of the implementation.

## 3.7. Host Interface

#### 3.7.1. VME Bus Conformity

EXOS 302 VME bus conformity is :

A24 MASTER, D(EO), D16; A24 SLAVE, D(O) OR A32 MASTER, D(EO), D16; A24 SLAVE, D(O) OR A32 MASTER, D(EO), D16; A32 SLAVE, D(O)

depending upon jumper configuration.

The EXOS 302 as a slave responds to two sets of address modifier codes. The slave bus size jumper defines which set of codes is valid. These codes are implemented with a PAL, so can easily be modified.

A24 SLAVE : address modifier codes (hex) 39, 3A, 3D, 3E. A32 SLAVE : address modifier codes (hex) 09, 0A, 0D, 0E.

## 3.7.2. Address Generation

The EXOS 302 acts as a bus master to transfer data between its memory and the system memory over the host bus. The EXOS 302 provides 32 address lines to the host, for a total addressable memory space of 4 Gigabytes. However, only 128KB is accessible at a time. The lower sixteen bits from the 80286 are concatenated with sixteen bits from a mapping register to form the full 32-bit address. Address line 16 from the 80286 selects which of two sixteen-bit registers is used (see Figure 3). This provides two 64KB windows, each of which can be placed on any 64KB boundary in the host's address space. The two windows appear as locations 080000-08FFFF and 090000-09FFFF to the 80286 (see Figure 2).

In the 32-bit mode the six address modifier bits, which must be driven by a bus master, are derived from an address modifier register which is written by the host (see section 4.2).

The EXOS 302 can be jumpered to operate as a 24-bit master, which makes it compatible with the EXOS 202 host driver and address modifier scheme. The difference from the 32-bit mode, besides the number of address bits, is the source of the six address modifier bits. In 202 mode the address modifiers are derived from the low six bits of the upper byte of the mapping register (see Figure 3).

Note that the EXOS 302 is HARDWARE-configured as either a 32- or 24-bit master. If configured for 32-bit, the address modifier is always driven by the addesss modifier register. 24-bit memories can be addressed by changing the address modifier register to one valid for 24-bit devices. The upper byte of the map register as the address modifier source in this case, since the map register is driving sixteen bits of address.

In 24-bit mode the address modifier from the map register is also driven onto the upper address byte (bits 31-24). According to the VME specification, if the address modifier indicates a 24-bit address, devices must not decode the upper byte.

The EXOS 302 can do eight- or sixteen-bit data operations. Bytes are automatically swapped by the hardware (high byte with low, low with high) since it is assumed that any memory device will present data to the bus in 68000 format.





#### 3.7.3. Slave Ports

Masters on the host bus can access three I/O ports on the EXOS 302. These slave ports are used for resetting the board, setting the address modifier register, setting the interrupt vector, and reading or writing program-defined status. These ports are compatible with the EXOS 202 ports, with the addition of the address modifier register.

The EXOS 302 is able to read and write its own ports, except for the interrupt vector, which is accessible only as an interrupt vector. The address modifier register can be written by the EXOS 302 under certain circumstances. Before the register is written, its contents are undefined. If the EXOS 302 does a host bus write cycle, it will likely assert a garbage address modifier. This will prevent the EXOS 302 slave machine from responding, and the bus will hang (or timeout).

The slave bus size can be configured separately from the master size. This allows both 24-bit, both 32-bit, or 24-bit slave with 32-bit master (slave 32 and master 24 is prohibited by the logic). This is accomplished by two jumpers (see section 5.2).

#### 3.7.4. Bus Errors

The bus error signal is not generated by the EXOS 302, since access is only to registers. The only error that can occur is an incorrect data alignment access by the host (double-byte or quad-byte). On such an access the EXOS 302 does not generate transfer acknowledge (DTACK), and a system bus timeout should occur. The EXOS 302 does monitor the bus error when a master, and an error causes the NMI to the 80286. The NMI is enabled via the CPU Control Port.

Signals ACFAIL and SYSFAIL are monitored, and while either is asserted the EXOS 302 bus request is inhibited. If the 80286 requests the bus during this time, it is held in wait.

#### 3.7.5. Burst Mode

۰.

The EXOS 302 has a burst mode, which is enabled by a jumper (see section 5.3). It holds the bus a maximum of 15 microseconds, unless the 80286 stops requesting the bus sooner. This allows about 20 cycles. Since the bus will be released regularly, the bus clear signal is not monitored. Note that this is not a VME BLOCK operation, wherein the slave increments the address itself after the first transfer. The EXOS 302 generates an address every cycle. The bus busy line is simply held true to avoid arbitration for each cycle.

Normally, burst mode will not be effective. However, under certain conditions burst mode may increase performance. These are, either singly or in combination : if the bus arbiter has a request-to-grant latency greater than 200 nanoseconds; if some masters in the system do not provide early release of busy, thereby preventing pipelined arbitration; if the EXOS 302 is not at the beginning of the grant daisy-chain.

#### 3.7.6. Interrupts

The EXOS 302 implements a daisy-chained interrupt structure. All bus interrupts (1-7) are available. The interrupt level is selected by a jumper. Three other jumpers are used to decode the interrupt acknowledge. See section 5.2 for jumper configurations.

The 80286 can issue a host bus interrupt by writing to an internal port (see section 4.1). The interrupt request is cleared by the hardware during interrupt acknowledge sequence (ROAK).

An interrupt to the EXOS 302 from the host is also available. This interrupt is set when one byte is data is written to the Comm register in the EXOS 302. See section 4.2.

## 3.8. Indicators (LED)

The EXOS 302 has three red LEDs to indicate operational status; one for network transmit activity, one for host bus activity, and one diagnostic LED which is connected to the CPU Control Register. Figure 4 shows the position of these LEDs on the board.



Figure 4. LED Positions

#### 4. Programming

:

4.-

This section explains the I/O ports which are used to control the operation of the EXOS 302. It contains two subsections: one for internal control, and one for host bus.

# 4.1. CPU Internal I/O

The I/O map is shown below. An I/O device uses some or all the I/O addresses in the range. When less than all the space is used, data replicates. Since incomplete decoding is used, the entire I/O map from 00-FF repeats to 7FFF.

Note that all the I/O devices are 8-bit except for the CPU Configuration/ Status Port. This means that all I/O addresses are even (except for the upper byte of the Configuration/Status Port). For the sake of simplicity, all I/O devices will be given one wait state.

A caveat using I/O: the MOS devices (PIC and DUARTS) require recovery time between I/O commands. Because of the speed of the 80286, NOP's will be required when accessing these devices. The details are described in the appropriate sections of this document.

| Base    |       | Width   |                                        |  |
|---------|-------|---------|----------------------------------------|--|
| Address | Туре  | in bits | Function                               |  |
| 00      | Rd/Wr | 8       | DUART (2681)                           |  |
| 20-     |       | •       | reserved                               |  |
| 80      |       |         |                                        |  |
| 88      | Write | -       | Channel Attention                      |  |
| 90      | Rd/Wr | 8       | CPU Control Register                   |  |
| 98      | Rd/Wr | 16      | CPU Status Register                    |  |
| AO      | Rd/Wr | 8       | Interrupt Controller (8259A)           |  |
| .B0     | Write |         | Schedule Interrupt (software)          |  |
| B8      | Write | -       | Interrupt Host                         |  |
| CO      | Read  | 8       | Ethernet Address PROM                  |  |
| E0      | Write | 16      | Address Mapping Registers              |  |
| E8      | Write |         | Local Reset of EXOS 302                |  |
| E8      | Read  | 8       | Host data register and interrupt reset |  |
| FO      | Write | 16      | EXOS 302 Status                        |  |
| 100     |       | ·       | 00-FF Repeats to FFFF                  |  |

FIGURE 3. I/O Map

.

#### EXCELAN CONFIDENTIAL

#### 4.1.1. DUART (Serial port/timer) (Read/Write at 00h)

The serial port and timer are implemented using a Signetics SCN2681 Dual Universal Asynchronous Receiver/Transmitter (!) chip. The chart below shows the connection of the chip to the RS232C port and the 8259A. Specific programming information for the chip can be found in the Signetics Microprocessor Data Manual (1986). The chip is operated from a 3.6864 MHz crystal. The interrupt request is connected to INT2 of the 8259A. Note that the chip is on the upper byte of the data bus. Therefore its ports are every other odd address. One NOP is required between successive I/O operations to the DUART to satisfy command recovery time.

| 2681<br>Pin | Function               |
|-------------|------------------------|
| RXA         | Receive Data           |
| TXA         | Transmit Data          |
| IP0         | Clear to Send          |
| IP4         | Data Terminal Ready    |
| OP0         | Request to Send        |
| OP2         | Data Set Ready         |
| OP3         | Counter output to INT2 |

#### 4.1.2. Channel Attention (Write at 88h)

A write to this port (data is irrelevant) generates a pulse to the Channel Attention input of the 82586.

#### 4.1.3. CPU Control Port (Read/Write at 90h)

The CPU Control Port is a latching register used by the 80286 for internal control of the EXOS 302. All bits are set to zero at power up or reset by the host. The port can be written and read at I/O address 90, and is defined as follows:

| Bit | Value | Function                                   |
|-----|-------|--------------------------------------------|
| 0   | 0     | Reset 82586 network controller             |
|     | 1     | Enable 82586 network controller            |
| 1   | 0     | Enable loopback mode of 8023 serial I/F    |
|     | 1     | Enable normal mode of 8023 serial I/F      |
| 2   | 0     | Disable NMI (bus parity error/timeout)     |
|     | . 1   | Enable NMI                                 |
| 3   | 0     | Light diagnostic LED                       |
|     | 1     | Extinguish diagnostic LED                  |
| 4   |       | not used                                   |
| 5   |       | not used                                   |
| 6   |       | not used                                   |
| 7   |       | Diagnostic bus control (see section 5.2.3) |

# 4.1.4. CPU Configuration/Status Port (Read at 98h)

A 16-bit I/O port at address 98 is assigned to configuration jumpers and other status bits. These bits are defined as follows:

| ·   |        | ,                                   |
|-----|--------|-------------------------------------|
| Bit | Source | Function                            |
| 0   | sense  | Xcvr +12V fuse (0=blown; 1=ok)      |
| 1   | sense  | not used                            |
| 2   | sense  | not used                            |
| 3   | sense  | Bus error (0=parity; 1=spurious)    |
| 4   | sense  | Mem size (1=512KB; 0=1MB)           |
| 5   | sense  | not used                            |
| 6   | sense  | not used                            |
| 7   | sense  | CPU Clock (1=8MHz; 0=n/a)           |
|     |        |                                     |
| 8   | J2     | Disable CRS check (=0)              |
| 9   | J3 -   | Ethernet/no SQE test (=0)           |
| 10  | J4     | Boot (0=from net; 1= wait for host) |
| 11  | J5     | reserved                            |
| 12  | J6     | reserved                            |
| 13  | J7     | reserved                            |
| 14  | J8     | NX Console (=0)                     |
| 15  | J9     | Debugger (=0)                       |

All jumper inputs are pulled up (=1). An installed jumper is read as a zero (=0).

#### 4.1.5. Interrupt Controller (PIC) (Read/Write at A0h)

The 8259A Interrupt Controller provides prioritized interrupts to the 80286. Refer to Intel literature for operation and description of its I/O registers. Note that due to timing constraints (Intel's - not ours), one no-op is required between successive identical commands, and two noops are required between successive dissimilar commands. This is required to satisfy the chip's command recovery time. Since the 8259A is a byte device, it uses even addresses A0h & A2h. Interrupt sources are listed below.

| Level | Source                       |          |
|-------|------------------------------|----------|
| INTO  | reserved                     |          |
| INT1  | Host Data Register/Interrupt | <i>.</i> |
| INT2  | 2681 Output OP3 (Timer)      |          |
| INT3  | 82586                        |          |
| INT4  | 2681 Interrupt (SIO)         |          |
| INT5  |                              |          |
| INT6  | Scheduling Int               |          |
| INT7  |                              |          |

### 4.1.6. Scheduling Interrupt (Write at B0h)

An access to this port (data is irrelevant) generates an interrupt on INT6 of the 8259A.

#### 4.1.7. EXOS 302 Interrupt (Write at B8h)

A write to this port (data is irrelevant) generates the EXOS 302 interrupt to the host. It also sets a status bit which appears in the EXOS 302 Status port for the host. The interrupt is cleared by a reset or interrupt acknowledge. The status bit is cleared by reset or by command from the host (see section 4.2).

#### 4.1.8. Ethernet Address PROM (Read At C0-DF)

The address PROM contains the Ethernet address and other configuration information. Sixteen bytes are available for use (even addresses only). A possible format is illustrated below:

| Address | Contents           |
|---------|--------------------|
| CO      | PROM rev level     |
| C2      | Byte 5 of address  |
| C4      | Byte 4 of address  |
| C6      | Byte 3 of address  |
| C8      | Byte 2 of address  |
| CA      | Byte 1 of address  |
| CC      | Byte 0 of address  |
| CE      | Product type       |
| D0      | Hardware rev level |
| D2-     |                    |
| DD      | reserved           |
| DE      | Checksum           |

The product type for the EXOS 302 is 12 (decimal). The checksum is formed by summing all the bytes modulo 256 and exclusive or ing with 55. This procedure ensures that a PROM with all zeros or all ones will not have a correct checksum.

#### 4.1.9. Extended Address Map Registers (Write at E0h)

Accesses to the host use 24 or 32 addresses, of which the lower 16 come from the 80286. The upper 16 are provided from one of two 16-bit mapping registers.

When host bus accesses are performed (addresses 8000-9FFF), address line 16 from the 80286 selects which register is concatenated with the lower 16 addresses. The mapping registers are loaded by writing data at I/O locations E0 and E2. The write data must be 16 bits wide. The lower byte furnishes host address bits 16-23. The upper byte furnishes bits 24-31 if in 32-bit-address mode. If jumpered for EXOS 202 compatibility, the upper byte furnishes address modifier bits 0-5 (register bit 8= address modifier bit 0).

| Мар      | Map Maps      |  | Master  | Map Reg | ister Bits |
|----------|---------------|--|---------|---------|------------|
| Register | CPU Addresses |  | Mode    | 15-8    | 7-0        |
| E0       | 8000-8FFF     |  | 32-bit  | A31-24  | A23-16     |
| E2       | 9000-9FFF     |  | EXOS202 | AM5-0   | A23-16     |

#### 4.1.10. Host Data Register and Interrupt Reset (Read at E8h)

A read of this port acquires the byte written into Port B when the host set the interrupt to the EXOS 302. Reading this port clears the interrupt to the 8259A, and clears status bit 3 of Port B.

#### 4.1.11. Local Reset of EXOS 302 (Write at E8h)

۰.

A write to this port (data is irrelevant) generates the board reset. This is equivalent to a host reset, and resets the entire board. The 80286 returns to real mode.

# 4.1.12. EXOS 302 Status Bytes 0 and 1 (Write at F0h, Read as Ports A and B)

A 16-bit write-only register at location F0h is available to provide status information to the host Ports A and B. All bits are software-definable. (Current definition of the status bits is explained in the Host Interface section.) However, bits 1 and 3 of Port B do not come from this register. They are derived directly from the hardware, and indicate the state of the EXOS 302 interrupt and the host interrupt, respectively. Reset clears bits 0-7; bits 8-15 are not cleared.

Bit 1 of the write-only register is used internally to enable the upper 512KB RAM when the 80286 is operated in the protected mode. This inhibits access to the upper RAM. When the bit is set, access is enabled. The bit should be set AFTER the 80286 is in protected mode, because the extended address bits from the 80286 in real mode are not in a defined state, and may cause faulty memory operation.

| Bit  | Destination    | Definition                     |
|------|----------------|--------------------------------|
| 0    | Port B         | 0=diagnostic failed; 1=passed  |
| 1    | Memory Control | 0=disable upper 512K; 1=enable |
| 2    | Port B         |                                |
| 3    | no connect     |                                |
| 4    | Port B         | none                           |
| 5    | Port B         | none                           |
| 6    | Port B         | 0=loopback passed; 1= failed   |
| 7    | Port B         | none                           |
|      |                |                                |
| 8-15 | Port A         | none                           |

Bit 3 is not connected to anything internally. Bits 8-15 are all wired directly to Port A. This register must always be written as a word register; byte operations will trash the other byte.

#### 4.2. Host Slave Ports

Three ports exist on the EXOS 302 in the host memory-mapped I/O space. Two are for status and control information to/from the EXOS 302. The third is for the host to write the 6-bit address modifier for 32-bit address mode.

There are individual compare jumpers for address bits 31-16, and two jumpers to define one of four combinations for bits 15-7. These four combinations are programmed into a PAL, and currently are compatible with the EXOS 202 rev F. The slave address configuration is shown below. (This is a program representation - bit 0 does not appear on the VMEbus.) Bits 31-24 are not decoded when in 24-bit slave mode. Bits 6-3 are not decoded, leaving a 128-byte hole above the base address. These ports must be accessed as byte values.

| 31   | 24   | 23   | 16   | 15  |       | 8 | 7 | 0    |
|------|------|------|------|-----|-------|---|---|------|
| aaaa | aaaa | aaaa | aaaa | bbb | b bbb | b | b | -ppp |

|      | Base   |       | and the second |  |
|------|--------|-------|------------------------------------------------------------------------------------------------------------------|--|
| Port | Offset | Туре  | Function                                                                                                         |  |
| A    | -1     | Read  | Read Status Byte 1                                                                                               |  |
|      |        |       | (extended status)                                                                                                |  |
| A    | ÷1     | Write | EXOS 302 Reset                                                                                                   |  |
| В    | -3     | Read  | Read Status Byte 0                                                                                               |  |
| В    | ÷3     | Write | Host Interrupt/Data Write                                                                                        |  |
| С    | -5     | Write | Write Address Modifier Register                                                                                  |  |

Write Port A Genera

Generates a hardware reset of the EXOS 302.

Read Port A Reads the EXOS 302 Status Byte 1 (extended status). All bits are register bits, and can be set by the EXOS 302 firmware. Contents are indeterminate after reset. See section 4.1.

Write Port B Host Interrupt/Data - Interrupts the 80286 via INT1 of the 8259A and writes a byte into the Host Data Register. This also sets the host interrupt status bit, which is available to the host in the EXOS 302 Status port. The bit is cleared when the 80286 reads the Host Interrupt/Data Port. This mechanism provides for handshaking and parameter passing between the EXOS 302 and the host.

- Read Port B Reads the EXOS 302 Status Byte 0. All the bits are cleared by reset, except bit 3, which is set. All bits are register bits which can be set by the EXOS 302 firmware, except bits 1 and 3, which are interrupt status bits (see section 4.1).
  - Bit 0 Set/reset by EXOS 302 firmware. This bit has previously been defined as Status of the board. When reset, indicates the on-board diagnostic has failed or is not completed yet.
  - Bit 1 Host Interrupt Status this bit is set by the hardware when the 80286 issues an interrupt to the host. It is cleared by the hardware during the interrupt acknowledge sequence.
  - Bit 3 Local Interrupt Status is set when the host writes into Port B, and is cleared when the 80286 reads the Host Data Port. On power reset the state of this bit is undefined. Therefore, the 80286 should read the Host Data Port before enabling interrupts.
  - Bit 6 Set/reset by EXOS 302 firmware. This bit has previously been defined as a qualifer of bit 0. When set, indicates a failure of net loopback.

Write Port C

A byte is written into the address modifier register in the EXOS 302. Only the low six bits are valid. This register drives the six address modifier bits during a bus cycle if the master size is jumpered for 32 bits.

13

# 5. Board Configuration

#### 5.1. Memory Option

An additional 512KB of memory may be installed in sockets U16-U19. The memory size jumper in the CPU Status Register must agree with the installed size. The chips must be 256K X 4 DRAMS, 120ns or faster, with CAS-before-RAS refresh capability (TOSHIBA 514256 or equivalent).

## 5.2. Jumpers

# 5.2.1. CPU Status Register Jumpers

All jumper inputs are pulled up (1). An installed jumper is read as zero (0). Note that these are only sense jumpers, and have no affect on the hardware, except J3 and J10.

| Bit | Jumper | Function                     |
|-----|--------|------------------------------|
| 4   | J10    | Mem size (1=512K; 0=1MB)     |
| 5   | J11    | reserved (=1)                |
| 6   | J12    | reserved (=1)                |
| 7   | J13    | 286 Clock (1=8MHz; 0=n/a)    |
| 8   | J2     | Disable CRS check (=0)       |
| 9   | J3     | V1 ethernet/no SQE test (=0) |
| 10  | J4     | Net Boot (=0)                |
| 11  | J5     | reserved (=1)                |
| 12  | J6     | reserved (=1)                |
| 13  | J7     | reserved (=1)                |
| 14  | J8     | NX Console (=0)              |
| 15  | J9     | Debugger (=0)                |

#### 5.2.2. PROM Size

ي الما من من الم

Two jumpers define the size of the PROMs used for the 80286 firmware.

| PROM  |     |     |
|-------|-----|-----|
| Туре  | J16 | J15 |
| 27128 | 1-2 | 1-2 |
| 27256 | 1-2 | 2-3 |
| 27512 | 2-3 | 2-3 |

## 5.2.3. Burst Mode Enable

This jumper enables or disables the burst mode to the host bus.

| ſ | Burst    |     |
|---|----------|-----|
| İ | Function | J25 |
| ſ | Enabled  | 1-2 |
|   | Disabled | 2-3 |

#### 5.2.4. Address Configuration Jumpers

There are two sets of address jumpers to configure : one for selecting the host bus address width (24 or 32 bits) for slave and/or master mode, and one for slave mode decode.

#### 5.2.4.1. Host Address Bus Width

The host address bus width determines how many bits are decoded on a slave cycle, and the source of the address modifier on a master cycle. Master and slave mode can use different widths. Note that the master mode cannot be smaller than the slave. However, if the address modifier register contains a 24-bit modifier, then only 24-bit slaves can respond.

| Master<br>Width | Slave<br>Width | Master<br>Jumper | Slave<br>Jumper |
|-----------------|----------------|------------------|-----------------|
| (bits)<br>24    | (bits)<br>24   | J44<br>ON        | J46<br>ON       |
| 32              | 24             | OFF              | ON              |
| 32              | 32             | ON               | OFF             |
| 32              | 32             | OFF              | OFF             |

#### 5.2.4.2. Slave Address Compare

There are individual compare jumpers for address bits 31-16, and two jumpers to define the base within 64KB (bits 15-7). The two jumpers select one of four bases which are programmed into the decode PAL.

The corresponding addresses/jumpers are shown in the two charts below. For the address compare jumpers, an installed jumper is a compare to zero. The four base addresses are compatible with the EXOS202 (rev F). Bits 6-3 are not decoded, leaving a 128-byte hole above the base address.

|   | Host Address Bits                               |        |       |      |     |
|---|-------------------------------------------------|--------|-------|------|-----|
| Α | 31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16 | 15 - 8 | 7 - 0 |      |     |
| J | 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 |        |       | 42   | 43  |
|   | Slave Address Compare Jumpers                   |        |       |      |     |
|   |                                                 | 00     | 00    | OFF  | OFF |
|   |                                                 | 7F     | 80    | OFF  | ON  |
|   |                                                 | 80     | 00    | ON · | OFF |
|   |                                                 | FF     | 80    | ON   | ON  |

## 5.2.5. Interrupt Level and Level Enable

. . . . . .

٠.

· · · · ·

٠.

The request jumper and the acknowledge jumpers MUST match for interrupt operation. An installed jumper is indicated by a 1.

| IRQ   | Request | Acknow | ledge Jur | npers |
|-------|---------|--------|-----------|-------|
| Level | Jumper  | J54    | J55       | J56   |
| IRQ1  | J47     | 0      | 0         | 1     |
| IRQ2  | J48     | 0      | 1         | 0     |
| IRQ3  | J49     | 0      | 1         | 1     |
| IRQ4  | J50     | 1      | 0         | 0     |
| IRQ5  | J51     | 1      | 0         | 1     |
| IRQ6  | J52     | 1      | 1         | 0     |
| IRQ7  | J53     | 1      | 1         | 1     |

#### 5.2.6. Bus Request/Grant In/Grant Out

Request and grant jumpers must be on the same level. One Request jumper is used. One Grant In and one Grant Out are jumpered to match the request level. The other three Grant Out lines must be jumpered to daisy-chain the unused Grant In lines back to their respective Grant TOUT Out lines.

|       | Request |       | Grant In/Out Jumpers |       |       |       |       |       |       |
|-------|---------|-------|----------------------|-------|-------|-------|-------|-------|-------|
| Level | Jumper  | J68   | J67                  | J66   | J65   | J64   | J63   | J62   | J61   |
| 0     | J60     | ON    | ON                   | J65-1 | J66-1 | J63-1 | J64-1 | J61-1 | J62-1 |
| 1     | J59     | J67-1 | J68-1                | ON    | ON    | J63-1 | J64-1 | J62-1 | J61-1 |
| 2     | J58     | J67-1 | J68-1                | J65-1 | J66-1 | ON    | ON    | J61-1 | J62-1 |
| 3     | J57     | J67-1 | J68-1                | J65-1 | J66-1 | J63-1 | J64-1 | ON    | ON    |

## 5.2.7. Host Bus Ethernet Connection

The Ethernet signals connected to the 15-pin D-type connector can be jumpered to the host bus on P2. All jumpers must be installed if used. The  $\pm$ 12V is fused (VE – 12V).

| Signal | Jumper | Host Pin |
|--------|--------|----------|
| VGND   | J24    | P2C-15   |
| VE-12V | J17    | P2C-16   |
| VTRMT- | J18    | P2A-13   |
| VTRMT- | J19    | P2A-14   |
| VRECV- | J20    | P2A-15   |
| VRECV- | J21    | P2A-16   |
| VCLSN+ | J22    | P2C-13   |
| VCLSN- | J23    | P2C-14   |

#### 5.2.8. Dynamic Address Bus Control (Diagnostic only)

The diagnostic jumper J44 is to be used only in the factory with special test equipment which can short the jumper upon command. The jumper, when shorted, allows the firmware to force the board from 24-bit mode into 32-bit mode by setting the DIAG24 bit in the CPU Control Register. This is necessary for testing the address modifier register, since testing will involve using invalid modifiers, and an invalid modifier will prevent subsequent slave cycles. Testing is accomplished by 1) forcing 24-bit mode to use address modifier from the map register, 2) writing the address modifier register with a test value, 3) switching back to 32-bit mode to use the address modifier register, 4) doing a bus cycle, which is latched in the bus tester, and 5) reading the latched address modifier value from the bus tester. The master jumper is pulled up, and, if off, indicates 32-bit mode. The master will follow the slave if the Master jumper is on. This circuit is shown below. Note that the DIAG24 and SLAVE24 signals are low true, and MASTER32 is high true. The gate is a positive AND. Note that this process cannot force the board into 32-bit mode if the slave jumper is set for 24-bit mode.

The DIAG24 bit is 0 after reset. The CPU Control Register is a write/read register. However, the DIAG24 bit (7) is not read directly. The bit which is read is an AND of the DIAG24 bit and the DIAG jumper.



# 6. Connectors

# 6.1. VME Bus Connectors

The following table shows the EXOS 302 signal assignments for the VME host bus connectors P1 and P2. Signals not used by the EXOS 302 are designated n/u. Many of the pins on P2 are not dedicated to any bus signals.

|     | VME BUS PIN ASSIGNMENTS |              |           |        |         |        |  |
|-----|-------------------------|--------------|-----------|--------|---------|--------|--|
| PIN | P1A                     | P1B          | P1C       | P2A    | P2B     | P2C    |  |
| 1   | VD00                    | VBSY-        | VD08      |        | +5V     |        |  |
| 2   | VD01                    | BCLR- n/u    | VD09      |        | GND     |        |  |
| 3   | VD02                    | ACFAIL-      | VD10      |        |         |        |  |
| 4   | VD03                    | VBG0IN-      | VD11      |        | VA24    |        |  |
| 5   | VD04                    | VBG0OUT-     | VD12      |        | VA25    |        |  |
| 6   | VD05                    | VBG1IN-      | VD13      |        | VA26    |        |  |
| 7   | VD06                    | VBG1OUT-     | VD14      |        | VA27    |        |  |
| 8   | VD07                    | VBG2IN-      | VD15      |        | VA28    |        |  |
| 9   | GND                     | VBG2OUT-     | GND       |        | VA29    |        |  |
| 10  | SYSCLK n/u              | VBG3IN-      | SYSFAIL-  |        | VA30    |        |  |
| 11  | GND                     | VBG3OUT-     | VBERR-    |        | VA31    |        |  |
| 12  | VDS1-                   | VBR0-        | SYSRESET- |        | GND     |        |  |
| 13  | VDS0-                   | VBR1-        | LWORD-    | VTRMT+ | ÷5V     | VCLSN+ |  |
| 14  | VWRITE-                 | VBR2-        | VAM5      | VTRMT- | D16 n/u | VCLSN- |  |
| 15  | GND                     | VBR3-        | VA23      | VRECV- | D17 n/u | VGND   |  |
| 16  | VDTACK-                 | VAMO         | VA22      | VRECV- | D18 n/u | VE-12V |  |
| 17  | GND                     | VAM1         | VA21      |        | D19 n/u |        |  |
| 18  | VAS                     | VAM2         | VA20      |        | D20 n/u |        |  |
| 19  | GND                     | VAM3         | VA19      |        | D21 n/u |        |  |
| 20  | IACK-                   | GND          | VA18      |        | D22 n/u |        |  |
| 21  | IACKIN-                 | SERCLK n/u   | VA17      |        | D23 n/u |        |  |
| 22  | IACKOUT-                | SERDAT n/u   | VA16      |        | GND     |        |  |
| 23  | VAM5                    | GND          | VA15      |        | D24 n/u |        |  |
| 24  | VA07                    | IRQ7-        | VA14      |        | D25 n/u |        |  |
| 25  | VA06                    | IRQ6-        | VA13      |        | D26 n/u |        |  |
| 26  | VA05                    | IRQ5-        | VA12      |        | D27 n/u |        |  |
| 27  | VA04                    | IRQ4-        | VA11      |        | D28 n/u |        |  |
| 28  | VA03                    | IRQ3-        | VA10      |        | D29 n/u |        |  |
| 29  | VA02                    | IRQ2-        | VA09      |        | D30 n/u |        |  |
| 30  | VA01                    | IRQ1-        | VA08      |        | D31 n/u |        |  |
| 31  | -12v                    | +5V STBY n/u | +12V      |        | GND     |        |  |
| 32  | - 5v                    | -∕-5V        | 5V        |        | ÷5V     |        |  |

•

# EXOS 302

# 6.1.1. ETHERNET Connector

The ETHERNET connector P3 is a 15-pin D-type. The following table gives the pin assignments.

| F   | 23     |
|-----|--------|
| Pin | Signal |
| 1   | GND    |
| 2   | CLSN+  |
| 3   | TRMT+  |
| 4   | GND    |
| 5   | RCV+   |
| 6   | GND    |
| 7   |        |
| 8   | GND    |
| 9   | CLSN-  |
| 10  | TRMT-  |
| 11  | GND    |
| 12  | RCV-   |
| 13  | E+12V  |
| 14  | GND    |
| 15  |        |



# 6.1.2. SERIAL PORT HEADER

The following diagrams show the pin orientation and signal assignments for P4, the serial port header.

|    | IC side |        |
|----|---------|--------|
| 23 | 31      | L      |
| 24 | 42      | E      |
|    |         | Ds     |
|    | 24      | 24 4.2 |

|     | P4     |  |  |  |  |  |
|-----|--------|--|--|--|--|--|
| Pin | Signal |  |  |  |  |  |
| 1   |        |  |  |  |  |  |
| 2   |        |  |  |  |  |  |
| 3   | RXD    |  |  |  |  |  |
| 4   |        |  |  |  |  |  |
| 5   | TXD    |  |  |  |  |  |
| 6   |        |  |  |  |  |  |
| 7   | CTS    |  |  |  |  |  |
| 8   |        |  |  |  |  |  |
| 9   | RTS    |  |  |  |  |  |
| 10  |        |  |  |  |  |  |
| 11  | DTR    |  |  |  |  |  |
| 12  |        |  |  |  |  |  |
| 13  | GND    |  |  |  |  |  |
| 14  | DSR    |  |  |  |  |  |
| 15  |        |  |  |  |  |  |
| 16- | not    |  |  |  |  |  |
| 26  | used   |  |  |  |  |  |

D 1 1 0 ----- 4000

PAGE 55,132

COMMENT

\$Header: c:/nx6/s\_net/net586/RCS/net586\_2.asm 1.3.1.1 91/03/07 10:21:10 karlt Exp Locker: karlt

\$Project: NX6 \$

\$Creator: Steve Grau \$

\$Locker: karlt \$

\$Source: c:/nx6/s net/net586/RCS/net586 2.asm \$

(C) Copyright 1988 by Excelan Inc. All Rights Reserved

This software is furnished under contract and may be used and copied only in accordance with the terms of such contract and with the inclusion of the above copyright notice. This software or any other copies thereof may not be provided or otherwise made available to any other person. No title to and ownership of the software is hereby transferred.

\_\_\_\_\_

\$Abstract:

NX6 Intel 82586 Ethernet Controller Driver.

This module contains the transmitter state machines.

\$

\$Implementation Notes:

The 82586 transmitter driver is implemented using four state machines to manage the transmit queues and keep the 82586 running at maximum throughput.

#### Transmitter Queues

The transmitter contains two queues. The first queue is made up of a circular list of transmit command blocks (CBLs) and a circular list of transmit buffer descriptors (TBDs). These structures (CBLs and TBDs) are described in detail in the 82586 data sheet. The two circular lists do not necessarily contain the same number of elements. Fragmentation of transmit buffers requires several TBDs per CBL. So the TBD list should be much longer than the CBL list. The actual sizes of these lists is configurable at assembly time. This queue will be referred to as the Transmit Ring.

The second queue is simply a linked list which is used to hold user requests when there is a resource limitation in the first queue (out of CBLs or out of TBDs, or both). This queue will be referred to as the Wait Queue.

Transmitter State Machines

There are four transmitter state machines. They are:

Enqueue Transmit - (entry points qtx ????)

Handles queing user transmit requests to the appropriate transmit queue. Also, starts the 82586 transmitting if it is not already doing so.

Transmit Complete - (entry points cx\_???)

Runs on the command complete interrupt, CX bit in the 82586 interrupt status. Notifies, the user when each transmit completes and frees up the CBL and TBDs used by the transmit.

Process Wait Queue - (entry points pwq ????)

Transfers transmit requests from the wait queue to the transmit ring as ring resources become available. Is run on the same interrupt as the transmit complete state machine immediately after it finishes.

Transmitter Not Active - (entry points cna ????)

Restarts the 82586 transmitter when it completes a chain of transmit requests. All requests waiting in the transmit ring are forwarded to the 82586. Gets invoked by the command not active (CNA) interrupt from the 82586.

In addition to the state machines there is also a transmit timeout interrupt that runs of a clock interrupt.

\$

\$Log: net586\_2.asm \$
Revision 1.3.1.1 91/03/07 10:21:10 karlt
temporary fix for APPLLO's transmit hang problem

Revision 1.3 89/02/22 09:04:49 steveg Added new TPS status bit.

Revision 1.2 88/12/09 15:57:31 dauber added support for concurrent link level

Revision 1.1 88/08/23 08:40:13 steveg Initial revision

\$EndLog\$

1

include nx6.inc include cfgxxx\_x.inc include cfg\_x.inc include intxxx\_x.inc include net586.inc

.186

TEXT SEG

SEGMENT

```
EXTRN
      net586 qtx return:NEAR
                                          ; qtx state machine return pt.
     net586 cna return:NEAR
                                          ; cna state machine return pt.
EXTRN
       net586 cx return:NEAR
EXTRN
                                          ; cx state machine return pt.
     net586 pwq return:NEAR
                                           ; pwq state machine return pt.
EXTRN
public cx_state
public cna state
public pwq state
public qtx state
public p_1st_586_cbl
public p_1st_rdy_cbl
public p_last_rdy_cbl
public p 1st free cbl
public p 1st tx wait
public p last tx wait
public p 1st free tbd
public p last free tbd
public last_tbs_off
public last_tbs_sel
public fragment count
; net586_tx_chain - return transmit buffer chain
;
; This function returns the TPS and TBS structures that have been queued
; to the driver. The transmit is turned off.
;
       Inputs: none
;
;
       Outputs: STACK USER ES:STACK SI points to TPS chain
;
               STACK_SI = ffff if none returned
ï
               STACK AX = 0
;
ï
       The TPS chain is linked by the QUEUE field. The end of the chain
;
       is marked by an offset of ffff.
ï
PUBLIC net586 tx chain
net586_tx_chain PROC NEAR
       pushf
                                           ; protect
       cli
; Disable transmit timeout.
       mov
              tx_timeout_enable,0
                                           ; disable the timeout
; Abort the transmitter.
       mov
              cx,Offffh
                                           ; long timeout
txc wait cmd:
              scb.SCB COMMAND,0
       cmp
                                           ; command clear?
              txc cmd clear
       je
                                           ; yes
       loop txc wait cmd
txc cmd clear:
              scb.SCB COMMAND, SCB CUC ABORT
       mov
                                           ; abort the transmitter
       CHANNEL ATTENTION
                                           ; bang on the 586
```

mov si,Offffh ; assume no buffers mov WORD PTR [bp+STACK\_SI],si

; Check to see if there are any tps's on the tps wait queue.

cmp pwq\_state,OFFSET pwq\_wait ; tps wait queue active? jne txc\_tx\_wait\_done

; Get the pointer to first waiting tps.

| mov | ax,p_1st_tx_wait      | ; get the offset             |
|-----|-----------------------|------------------------------|
| mov | [bp+STACK_SI],ax      | ÷.                           |
| mov | ax,p_1st_tx_wait+2    | ; get selector               |
| mov | [bp+STACK_USER_ES],ax | :                            |
|     |                       |                              |
| mov | si,p_last_tx_wait     | ; get pointer to last waiter |
| mov | es,p_last_tx_wait+2   |                              |

txc\_tx\_wait\_done:

.

.

; Check to see if there are any tps's queued to the 82586.

| cmp | cna_state,OFFSET cna_idle | ; 586 transmitting? |
|-----|---------------------------|---------------------|
| je  | txc_tps_done              | ; no                |

; Free up the tbds.

| mov | bx,OFFSET tbd_list | ; point anywhere in list |
|-----|--------------------|--------------------------|
| mov | p_last_free_tbd,bx | ; this one is last       |
| mov | bx,[bx].TBD_LINK   | ;next one is first       |
| mov | p_1st_free_tbd,bx  |                          |

; Shuffle through the cbls linking tps's together.

bx,p\_1st\_586\_cbl mov ; get pointer to first cbl txc cbl loop: cmp bx,p\_lst\_free\_cbl ; all cbls done? txc\_tps\_done je ; yes p\_lst\_free\_cbl,OFFSET p\_lst\_free\_cbl ; any free? cmp txc\_some\_free jne ; yes p 1st free cbl,bx ; make this the first free one mov

txc\_some\_free:

| cmp | si,Offffh    | ; | any | on | the | return | list? |
|-----|--------------|---|-----|----|-----|--------|-------|
| jne | txc_got_some | ; | yes |    |     |        |       |

; The return chain is empty. Attach the first element.

| les | si,DWORD PTR [bx].CBL_TPS_OFF | ; | get pointer        |
|-----|-------------------------------|---|--------------------|
| mov | [bp+STACK_SI],si              | ; | write return value |
| mov | [bp+STACK_USER_ES],es         |   |                    |
| mov | es:[si].TPS_QUEUE,Offffh      | ; | mark end of list   |
| mov | bx,[bx].CBL_LINK              | ; | get next CBL       |
| jmp | txc_cbl_loop                  | ; | loop through them  |

txc\_got\_some:

; Link the TPS attached to the CBL to the end of the chain. ax,[bx].CBL TPS OFF ; get offset mov ; link to previous TPS es:[si].TPS\_QUEUE,ax mov ax,[bx].CBL\_TPS\_SEL mov ; get selector es:[si].TPS QUEUE+2,ax mov ; Load pointer to this TPS. les si, DWORD PTR [bx].CBL TPS OFF ; load it es:[si].TPS QUEUE,Offffh mov ; mark end of list bx,[bx].CBL LINK ; get next CBL mov txc cbl loop ; loop through them jmp txc tps done: p\_lst\_rdy\_cbl,OFFSET p\_lst\_rdy\_cbl ; invalidate pointers mov p\_last\_rdy\_cbl,OFFSET p\_last\_rdy\_cbl mov ax,p 1st free cbl ; set 1st 586 cbl pointer mov p 1st 586 cbl,ax mov ; Set new states. mov qtx state, OFFSET qtx idle cx state, OFFSET cx idle mov mov pwq state,OFFSET pwq idle cna\_state,OFFSET cna\_idle mov popf ; restore interrupt state mov WORD PTR [bp+STACK AX],0 ; no error ret net586 tx chain ENDP ; Enqueue transmit state machine - qtx ???? ; ; This state machine is made up of 3 states.

1. qtx\_idle - transmitter is currently idle.

;

;;

;

;

;

ï

;

;;

;

ï

;

;

;;

;

;

The request is processed immediately and handed directly to the 82586 for transmission. This is the initial state of this state machine. This state is entered from qtx\_ring when all pending user transmits requests are completed which is detected in cna\_none\_queued.

2. qtx\_ring - new transmits are being queued into transmit ring.

This is the current state when the 82586 is busy transmitting a previous request and there are no requests in the wait queue. States qtx\_idle and pwq\_wait are responsible for activating this state.

3. qtx wait - new transmits are being queued into the wait queue.

This is the current state when the transmit ring has insufficient free resources to hold all transmit requests.

All requests are placed on the wait queue while in this ; state. State qtx ring is responsible for activating this ; state. ; ; qtx state machine PROC NEAR ; qtx idle - transmitter is idle PUBLIC net586 qtx idle ; symbol for publication net586 qtx idle: qtx idle: ; Get the first free CBL. The state assures that all CBLs are currently ; free. mov bx,p 1st free cbl ; get the cbl pointer ; Store pointer to user's transmit packet structure TPS in the CBL. [bx].CBL TPS OFF,si ; save pointer to TPS mov [bx].CBL TPS SEL,es mov ; Clear the TPS status field, and pick up the link fields. es:[si].TPS STATUS,0 mov ; clear the status ax,es:[si].TPS LAST TBS ; get the last TBS offset mov last tbs off,ax ; store it mov ax,es:[si].TPS LAST TBS+2 ; get the last TBS selector mov mov last tbs sel,ax si, DWORD PTR es: [si]. TPS 1ST TBS ; get pointer to first TBS les ; Get pointer to first TBD. di,p 1st free tbd ; grab the first tbd mov ; Setup the CBL. [bx].CBL STATUS,0 ; clear the status mov [bx].CBL COMMAND,CBL CMD XMIT+CBL EL BIT+CBL I BIT mov mov [bx].CBL TBD OFFSET,di ; pointer to first TBD ; Fill in the tbd. push bx ; save the cbl pointer qtxi\_build\_tbds: ; build the tbds ax,es:[si].TBS\_SIZE ; read the size from the TBS mov ; store the size in the TBD [di].TBD STATUS,ax mov ; get the buffer offset dx,es:[si].TBS BUFFER PTR mov bx,es:[si].TBS\_BUFFER\_PTR+2 mov ; get the buffer selector call map\_586\_state ; call the mapper ; store the address.. [di].TBD ADDR LO,dx mov

The reason to do this new fix was based on the observation that whenever the set dre, it dies because of an overwrite of unfinished command

And since this is the ONLY place that has the possibility to overwrite the command word of 586 (it does not check if the command has been cleared). I thought the NEW Fix' would work, but.

• • •

[di].TBD\_ADDR\_HI,bx ; .. in the tbd mov ; check for last tbs? si, last tbs off cmp qtxi next tbs ; not last, do another ine mov ax,es ; copy selector to ax cmp ax, last tbs\_sel ; check selector too qtxi\_next\_tbs ; not last, do another jne ; This is the last tbs. Set the end of frame bit in the TBD and continue on. [di].TBD STATUS,TBD EOF BIT ; set the end of frame bit or qtxi\_tbds\_done ; continue on jmp qtxi next tbs: ; advance to the next tbs si,DWORD PTR es:[si].TBS LINK les ; point es:si to new tbs di,p last free tbd cmp ; is this the last tbd? je qtxi too many frags ; yes, too many fragments di,[di].TBD\_LINK ; follow the link mov jmp qtxi build tbds ; keep looping qtxi too many frags: ; cleanup and return error ; restore cbl pointer pop bx mov WORD PTR [bp+STACK AX], NET FRAG ERROR ; set the return value net586 qtx return ; return jmp qtxi\_tbds\_done: pop bx ; restore the cbl pointer the back of previous page ; NEW FIX ï although this is perceived as fix for the real cause of the problem, ; and it did elongate the up time (without timeout), it inavitably ; introduces, or unvails, new and fetal bug(s) which will crash both ; the transmit and receive in about 20 minutes with 80% net traffic ; ; push сх ; ; public patch ;patch: cx, 0ffffh ; long timeout mov ; ;txs\_wait cmd: scb.SCB COMMAND,0 ; command clear? ; cmp je txs cmd clear ; yes ;

> mov scb.SCB\_CBL\_OFFSET,bx ; pass cbl pointer mov scb.SCB\_COMMAND,SCB\_CUC\_START ; start the command unit

txs wait cmd

; END NEW FIX -----

CX

; Setup the SCB and get the 586 started.

loop

pop

;txs cmd clear:

;;

;

;

CHANNEL ATTENTION ; alert the 586 tx time count,0 ; clear the timeout counter mov tx\_timeout\_enable,0ffffh ; enable the timeout mov ; Set new states. mov qtx state,OFFSET qtx ring ; in queueing to ring state cx\_state,OFFSET cx\_transmitting ; transmit is pending mov cna state, OFFSET cna none queued ; nothing more queued mov mov ax, [bx].CBL LINK ; advance pointer to next p 1st free cbl,ax ; currently free mov p\_last\_rdy\_cbl,OFFSET p\_last\_rdy\_cbl ; mark none ready mov ; Assuming there was more than one cbl, the last one couldn't have been used. p 1st 586 cbl,bx ; now belongs to the 586 mov ; Adjust tbd pointer. ax,[di].TBD LINK mov ; advance the pointer p 1st free tbd,ax mov ; store it ; di pointing to last used tbd di,p last free tbd ; was last tbd used? Cmp qtxi\_return jne ; no, return p\_1st\_free\_tbd,OFFSET p 1st free tbd ; mark point no good mov qtxi\_return: WORD PTR [bp+STACK AX],0 ; set return status mov jmp net586 qtx return ; qtx ring - queueing to CBL and TBD rings PUBLIC net586 qtx ring ; symbol for publication net586 qtx ring: qtx\_ring: ; Get the first free CBL if there is a free one. bx,p\_1st\_free\_cbl mov ; get the cbl pointer bx,OFFSET p 1st free cbl ; points to self if no more cmp qtxr got cbl jne ; got a cbl, continue qtxr\_wait jmp ; no cbl, put on wait queue ; Store pointer to user's transmit packet structure TPS in the CBL. qtxr got cbl: push bx ; save the cbl pointer [bx].CBL TPS OFF, si mov ; save pointer to TPS

; Clear the TPS status field, and pick up the link fields.

[bx].CBL\_TPS\_SEL,es

mov

, and prok up t

|                    | · · · ·  |                                  |    |                             |
|--------------------|----------|----------------------------------|----|-----------------------------|
|                    | mov      | es:[si].TPS STATUS,0             | ;  | clear the status            |
|                    | mov      | ax,es:[si].TPS LAST TBS          |    | get the last TBS offset     |
|                    | mov      | last tbs off,ax                  |    | store it                    |
|                    | mov      | ax,es:[si].TPS_LAST_TBS+2        | •  | get the last TBS selector   |
|                    |          | last tbs sel,ax                  | '  | get the fast iss selector   |
|                    | mov      |                                  | ~  | , and maintan to finat MDC  |
|                    | les      | si,DWORD PTR es:[si].TPS_1ST_TB  | 5  | ; get pointer to first TBS  |
| ; Get p            | ointer t | o first TBD.                     |    |                             |
|                    | mov      | di,p_1st_free_tbd                | ;  | grab the first tbd          |
|                    | cmp      | di,OFFSET p_1st_free_tbd         | ;  | is there one?               |
|                    | je       | qtxr_tbd_shortage                | ;  | nope                        |
|                    |          |                                  |    |                             |
| ; Setup            | the CBL  | •                                |    |                             |
|                    | mov      | [bx].CBL_STATUS,0                | ;  | clear the status            |
|                    | mov      | [bx].CBL COMMAND,CBL CMD XMIT+C  | BL | I BIT ; may not be last cbl |
|                    | mov      | [bx].CBL_TBD_OFFSET,di           |    | pointer to first TBD        |
|                    |          |                                  |    |                             |
| qtxr_bu            | ild_tbds | ::                               | ;  | build the tbds              |
|                    | mov      | ax,es:[si].TBS SIZE              | ;  | read the size from the TBS  |
|                    | mov      | [di].TBD STATUS,ax               | ;  | store the size in the TBD   |
|                    | mov      | dx,es:[si].TBS BUFFER PTR        | -  | get the buffer offset       |
|                    | mov      | bx,es:[si].TBS BUFFER PTR+2      |    | get the buffer selector     |
|                    |          | ~_,····()····_                   |    | -                           |
|                    | call     | map_586_state                    | ;  | call the mapper             |
|                    | mov      | [di].TBD ADDR LO,dx              | ;  | store the address           |
|                    | mov      | [di].TBD ADDR HI,bx              | •  | in the tbd                  |
|                    |          | (, <u>.</u> ,,,,,                | '  |                             |
|                    | cmp      | si,last_tbs_off                  | ;  | check for last tbs?         |
|                    | jne      | qtxr next tbs                    | ;  | not last, do another        |
|                    | mov      | ax,es                            |    | copy selector to ax         |
|                    | cmp      | ax,last tbs sel                  |    | check selector too          |
|                    | jne      | qtxr next tbs                    |    | not last, do another        |
|                    | Jile     | devi neve cos                    | '  | not fast, do another        |
| ; This             | is the l | ast tbs. Set the end of frame b  | it | in the TBD and continue on. |
|                    | or       | [di].TBD STATUS,TBD EOF BIT      | ;  | set the end of frame bit    |
|                    | jmp      | qtxr tbds done                   |    | continue on                 |
|                    | JE       | 1                                | '  |                             |
| atxr ne            | ext_tbs: |                                  | :  | advance to the next tbs     |
| 1                  | les      | si,DWORD PTR es:[si].TBS LINK    |    | point es:si to new tbs      |
|                    | TCD      | SI, DHOND FIN ES. [SI]. IDS_LINK | '  | point estar to new that     |
|                    | amp      | di,p last free tbd               |    | is this the last tbd?       |
|                    | cmp      |                                  |    |                             |
|                    | je       | qtxr_tbd_shortage                | ;  | yes, tbd shortage           |
|                    | mov      | di,[di].TBD LINK                 |    | follow the link             |
|                    | jmp      | qtxr_build_tbds                  |    | keep looping                |
|                    | Jup      | devi_patta_coas                  | '  | keep looping                |
| qtxr_tbd_shortage: |          |                                  | ;  | out of tbds, cleanup        |
|                    | non      | by                               |    | restore chi neister         |
|                    | рор      | bx                               | ì  | restore cbl pointer         |
|                    | les      | si,DWORD PTR [bx].CBL_TPS_OFF    | ;  | get pointer back to tps     |
|                    |          |                                  |    |                             |
| qtxr_wa            | it:      |                                  | ;  | put request on wait queue   |
| _                  |          |                                  |    |                             |

·

•

p\_lst\_tx\_wait,si ; put request on wait queue mov p 1st tx wait+2,es mov p\_last\_tx\_wait,si ; it is the only one mov p last tx wait+2,es mov ; Set new states. mov es:[si].TPS STATUS,0 ; clear the tps status qtx\_state,OFFSET\_qtx\_wait ; put queuer into wait mode mov pwq\_state,OFFSET pwq\_wait ; transmits in wait queue mov mov WORD PTR [bp+STACK AX],0 ; set the return value net586\_qtx\_return ; return jmp qtxr\_tbds\_done: ; restore the cbl pointer bx pop ; Set new states. mov cna state, OFFSET cna ring ; transmits waiting in ring ; Adjust pointers. ax, [bx].CBL LINK ; advance pointer to next mov p 1st free cbl,ax movp\_last\_rdy\_cbl,OFFSET p\_last\_rdy\_cbl ; any ready now? cmp jne qtxr\_cbls\_rdy ; yes, some are ready ; this is first one ready mov p\_1st\_rdy\_cbl,bx qtxr\_cbls\_rdy: ; one just done is now ready mov p\_last\_rdy\_cbl,bx ; Check to see if last cbl was used. ax,p\_1st\_586\_cbl ; last cbl used? cmp qtxr chck tbd ; no, check tbds jne mov p\_lst\_free\_cbl,OFFSET p\_lst\_free\_cbl ; all used, invalidate pointer qtxr\_chck\_tbd: ; Adjust tbd pointer. ax,[di].TBD\_LINK mov ; advance the pointer p\_lst\_free\_tbd,ax ; store it mov di,p\_last\_free\_tbd ; was last tbd used? cmp jne gtxr return ; no, return mov p\_lst\_free\_tbd,OFFSET p\_lst\_free\_tbd ; mark point no good qtxr\_return: WORD PTR [bp+STACK AX],0 mov ; set return status jmp net586 qtx return ; qtx wait - queueing to transmit resource wait queue 

PUBLIC net586\_qtx\_wait net586\_qtx\_wait: ; symbol for publication

#### qtx\_wait:

; Put the request at the end of the wait queue.

| mov | bx,p_last_tx_wait              | ; pick up last pointer |
|-----|--------------------------------|------------------------|
| mov | <pre>ax,p_last_tx_wait+2</pre> |                        |
|     |                                |                        |
| mov | p last tx wait,si              | ; update pointer       |
| mov | p last tx wait+2,es            |                        |
|     | F                              |                        |

; Make the last one on the queue point to the new one.

| mov        | cx,es                                         | ; copy new selector           |
|------------|-----------------------------------------------|-------------------------------|
| mov        | es,ax                                         | ; load selector last on queue |
| mov        | es:[bx].TPS_QUEUE,si                          | ; set offset                  |
| mov        | es:[bx].TPS_QUEUE+2,cx                        | ; set selector                |
| mov<br>jmp | WORD PTR [bp+STACK_AX],0<br>net586_qtx_return | ; set return status           |

qtx state machine ENDP

```
; Command (Transmit) Not Active state machine - cna ????
; This state machine is made up of 3 states.
;
     1. cna_idle - transmitter is currently idle.
ï
;
          Does nothing. This is the initial state. This state is
;
          activated by cna_none_queued.
;
ï
     2. cna_none_queued - transmitter is active, nothing waiting in ring
;
          Switches states to cna idle and qtx idle when the 82586
ï
          finishes the last transmit requeust. This state is activated
;
          by qtx_idle and cna_ring.
ï
;
     3. cna_ring - transmitter active, transmits waiting in ring
ï
;
          Starts 82586 transmitting next set of transmits when previous
ï
          set has finished. This state is activated by qtx ring and
;
          pwq_wait.
;
;
cna state machine
               PROC
                    NEAR
; cna_idle - transmitter is idle
cna idle
          EQU
               net586 cna return
                              ; idle is a do nothing state
; cna none queued - transmit pending but nothing new queued
```

PUBLIC net586 cna none queued ; symbol for publication net586 cna none queued: cna none queued: ; Set new states and return. mov cna state,OFFSET cna idle ; nothing pending qtx\_state,OFFSET qtx\_idle ; queuer to idle state mov net586 cna return jmp ; return ; cna ring - transmit pending, more queued to transmit ring, none waiting PUBLIC net586 cna ring ; symbol for publication net586\_cna\_ring: cna ring: bx,p\_last\_rdy\_cbl ; get pointer to last ready mov ; Set EL bit in last CBL. or [bx].CBL COMMAND,CBL EL BIT ; mark the end ; Setup the cbl link in the SCB and add start to the new SCB command. ax,p 1st rdy cbl ; get first ready cbl mov scb.SCB\_CBL\_OFFSET,ax mov ; pass cbl pointer new scb command, SCB CUC START ; start command or mov tx time count,0 ; clear timeout counter tx\_timeout\_enable,Offffh ; enable transmit timeout mov ; Set cx state to transmitting. cx state, OFFSET cx transmitting ; transmitter is running mov mov ax, [bx].CBL LINK ; next cbl ax,p\_1st\_rdy\_cbl ; full circle? cmp cnar none free ; no free buffers je p 1st free cbl,ax ; assume it is free mov ; does 586 own it? cmp ax,p\_1st\_586\_cbl cnar free set ; no, continue jne cnar none free: p\_lst\_free\_cbl,OFFSET p\_lst\_free\_cbl ; none free mov cnar free set: ; Update list pointers. p\_lst\_rdy\_cbl,OFFSET p\_lst\_rdy\_cbl ; invalidate ready pointers mov mov p last rdy cbl, OFFSET p last rdy cbl ; Check to see if state has anything is waiting. pwq state, OFFSET pwq wait ; anything waiting? cmp

cnar\_state\_set ; yes, don't change state je · cna state, OFFSET cna none queued ; no more queued mov cnar state set: jmp net586 cna return cna state machine ENDP ; Process Wait Queue state machine - pwq ???? ; This state machine is run on the transmit complete interrupt after the ; cx (transmit complete) state machine has been run. It is designed to pull ; waiting transmits in off of the wait queue and get them into the transmit ; ring as resources become available (transmits complete). ; 1. pwq idle - no transmit requests are on the wait queue. ; Nothing is done. This is the initial state. This state ; is activated by state pwq\_wait. ; 2. pwq wait - transmits are waiting in the wait queue. ; Transmit requests are transferred from the wait queue to the ; transmit ring as ring resources become available. This ; state activated by qtx ring. ; ; pwq state machine PROC NEAR ; pwq idle - nothing is in the wait queue pwq idle EQU net586 pwq return ; do nothing state ; pwq wait - transmits waiting on wait queue PUBLIC net586\_pwq\_wait ; symbol for publication net586 pwq wait: pwq wait: ; Get a CBL. mov bx,p 1st free cbl ; get first free CBL bx,OFFSET p 1st free cbl ; are any free? cmp jne pwqw\_get\_waiter ; yes, got one jmp pwqw\_wait1 ; no, still resource waiting pwqw\_get\_waiter: ; get first waiting packet les si, DWORD PTR p 1st tx wait ; load pointer ; If no CBL's are currently ready, set the pointer to the 1st ready to ; this one. Use the pointer to the last ready cbl to indicate if this

; has been done. If it doesn't change when we get through processing

; then there are no ready cbls. p 1st rdy cbl, OFFSET p 1st rdy cbl ; any ready? cmp jne pwqw queue ring ; yes, don't change pointer mov p\_1st\_rdy\_cbl,bx ; set the pointer ; Attempt to queue the packet to the transmit ring. pwqw queue ring: push bx ; save the cbl pointer ; Store pointer to user's transmit packet structure TPS in the CBL. mov [bx].CBL TPS OFF,si ; save pointer to TPS mov [bx].CBL TPS SEL,es ; Pick up the TPS link fields. ax,es:[si].TPS LAST TBS mov ; get the last TBS offset last tbs off,ax mov ; store it ax,es:[si].TPS LAST TBS+2 ; get the last TBS selector mov mov last tbs sel,ax si, DWORD PTR es:[si].TPS\_1ST\_TBS ; get pointer to first TBS les fragment count,1 mov ; keep count of fragments ; Get pointer to first TBD. mov di,p 1st free tbd ; grab the first tbd di,OFFSET p\_1st\_free\_tbd ; is there one? cmp je pwqw tbd shortage ; nope ; Setup the CBL. mov [bx].CBL STATUS,0 ; clear the status [bx].CBL COMMAND,CBL CMD XMIT+CBL I BIT ; may not be last cbl mov [bx].CBL TBD OFFSET,di ; pointer to first TBD mov pwqw\_build\_tbds: ; build the tbds ax,es:[si].TBS\_SIZE ; read the size from the TBS mov [di].TBD STATUS,ax ; store the size in the TBD mov dx,es:[si].TBS BUFFER PTR ; get the buffer offset mov mov bx,es:[si].TBS BUFFER PTR+2 ; get the buffer selector call map\_586\_state ; call the mapper [di].TBD ADDR LO,dx mov ; store the address.. mov [di].TBD ADDR HI,bx ; .. in the tbd si, last tbs off ; check for last tbs? CMD pwqw\_next\_tbs ; not last, do another jne mov ax,es ; copy selector to ax ax, last tbs sel cmp ; check selector too jne pwqw next tbs ; not last, do another ; This is the last tbs. Set the end of frame bit in the TBD and continue on.

> ; set the end of frame bit [di].TBD STATUS, TBD EOF BIT or

jmp , pwqw\_tbds\_done ; continue on pwqw\_next tbs: ; advance to the next tbs si,DWORD PTR es:[si].TBS LINK les ; es:si now points to new tbs inc fragment count ; got another fragment ; is this the last tbd? cmp di,p last free tbd pwqw tbd shortage ; yes, tbd shortage je mov di,[di].TBD LINK ; follow the link jmp pwqw\_build\_tbds ; keep looping pwqw\_tbd\_shortage: ; out of tbds, cleanup ; restore cbl pointer pop bx si, DWORD PTR [bx].CBL TPS OFF les ; get pointer back to tps fragment\_count,NET586\_N\_TBD ; more fragments than TBDs? cmp ; no jbe pwgw wait jmp pwqw fragment error ; some still waiting pwqw tbds done: ; restore the cbl pointer pop bx p\_last\_rdy\_cbl,bx ; CBL just done is now ready mov ; Adjust 1st tbd pointer. ax,[di].TBD LINK mov ; advance the pointer p\_1st\_free\_tbd,ax ; store it mov cmp di,p last free tbd ; was last tbd used? ; no pwqw chck next jne p\_1st\_free\_tbd,OFFSET p\_1st\_free\_tbd ; mark point no good mov pwqw\_chck\_next: ; Pickup the pointer to the TPS out of the CBL. ax, [bx].CBL TPS SEL mov ; selector si,[bx].CBL TPS OFF ; offset mov ; See if another packet is on the wait queue. Must compare both selector and ; offset with end of queue marker. cmp si,p\_last\_tx\_wait ; was this last? je pwqw chk selector ; maybe, keep checking pwqw\_do\_another ; no, do another jmp pwqw\_chk\_selector: ax,p last tx wait+2 cmp ; was this last? je pwqw\_queue\_empty ; yes pwqw\_do\_another: ; Pick up the next queued item. ; load the selector mov es,ax

. si, DWORD PTR es: [si]. TPS QUEUE ; get new pointer les ; Get another CBL. bx, [bx].CBL LINK ; advance pointer mov p 1st free cbl,bx ; assume it is free mov ; 586 own this one? bx,p 1st 586 cbl cmp ; yes, out of cbls pwqw cbls out je pwqw\_queue\_ring ; no, queue to ring jmp pwqw cbls out: ; ran out of cbls p\_lst\_free\_cbl,OFFSET p\_lst\_free\_cbl ; invalidate pointer mov pwqw wait: ; leave request on wait queue mov p 1st tx wait, si ; leave request on wait queue p 1st tx wait+2,es mov pwqw wait1: ; States don't change. Check to see if any CBLs are ready. No CBLs are ; ready if the last ready pointer points to itself. p\_last\_rdy\_cbl,OFFSET p\_last\_rdy\_cbl ; any ready? cmp jne pwqw some rdy ; yes p\_lst\_rdy\_cbl,OFFSET p\_lst\_rdy\_cbl ; no, invalidate pointer mov jmp net586 pwg return pwqw\_some\_rdy: cna state, OFFSET cna ring ; transmits waiting in ring movnet586 pwq return ; all done, return jmp ; wait queue is now empty pwqw\_queue\_empty: ; No more are waiting. Set new states. pwq state,OFFSET pwq idle ; no more queued mov mov qtx state,OFFSET qtx ring ; queue new stuff to the ring cna\_state,OFFSET cna\_ring ; request waiting in ring mov ax, [bx].CBL LINK ; next cbl mov p 1st free cbl,ax ; assume it is free mov ; does 586 own it? ax,p 1st 586 cbl cmp pwqw\_ptrs\_set ; no, continue jne p\_lst\_free\_cbl,OFFSET p\_lst\_free\_cbl ; none free mov pwqw\_ptrs\_set: jmp net586 pwq return ; all done, return pwqw fragment error:

; A transmit request has been found that has more fragments than there are ; tbds. This should never happend since the driver should be configured ; with lots of tbds. But, if it did happen it would lock up the transmitter. ; Any packets that come through here are pushed back up to the user with an ; error so that the transmitter can continue on. This code can't be entered ; before all preceeding transmits have completed since the fragment counter ; can never exceed the count of tbd's unless all tbds are free. This will ; insure that the user gets the packet back in sequence with other transmit ; requests.

es:[si].TPS\_STATUS,TPS\_COMP\_BIT+TPS\_UNSPEC\_ERR ; complete, error mov push si push es EXECUTE 1 APNDG 11 appendage ; notify link level es pop pop si ; was packet link levels? ax,0 cmp pwqw\_skip\_app ; skip user appendage if so je push si push es EXECUTE 1 APNDG tx\_appendage ; notify user pop es si pop pwqw\_skip\_app: ax,es:[si].TPS\_QUEUE ; update pointers... mov p\_1st\_tx\_wait,ax ; ... incase we start over mov ax,es:[si].TPS QUEUE+2 mov p 1st tx wait+2,ax mov p last rdy\_cbl,OFFSET p last\_rdy\_cbl ; any ready? cmp pwqw frag rdy jne ; yes p 1st\_rdy cbl,OFFSET p 1st\_rdy\_cbl ; no, invalidate pointer mov pwqw frag rdy: ; copy selector to ax ax,es mov ; See if another packet is on the wait queue. Must compare both selector and ; offset with end of queue marker. If nothing is waiting, must set new states ; and return. ; was this last? cmp si,p last tx wait je pwqw\_frag\_selector ; maybe, keep checking jmp pwq\_wait ; no, start over pwqw frag selector: ax,p\_last tx\_wait+2 ; was this last? cmp je pwqw\_frag\_exit ; yes jmp pwq wait ; no, start over pwqw frag exit: mov pwq\_state,OFFSET pwq\_idle ; nothing in queue jmp net586 pwq return pwq state machine ENDP ; Command (Transmit) Complete state machine - cx ???? ; This state machine is made up of 2 states. 1. cx idle - transmitter is currently idle.

; ;

;

;

٠

```
. Nothing is done in this state. This is the initial state.
;
            States cx transmitting and the transmit timeout routine
ï
            activate this state.
;
;
      2. cx_transmitting - transmitter is currently transmitting.
;
ï
            Transmits are passed back to the user as they complete.
;
            The transmit status is set appropriately. States qtx_idle
;
            and cna ring are responsible for activating this state.
;
;
cx_state_machine
                   PROC
                         NEAR
; cx idle - transmitter currently idle
cx idle
            EQU
                   net586 cx return
                                      ; do nothing state
; cx transmitting - transmitter currently active state
PUBLIC net586_cx_transmitting
                                     ; symbol for publication
net586 cx transmitting:
                                      ; transmitter is transmitting
cx transmitting:
            bx,p 1st 586 cbl
      mov
                                      ; get pointer to cbl
cxt check:
                                      ; check the cbl status
      mov
            cx,[bx].CBL STATUS
                                      ; get the status
      test
            cx,CBL C BIT
                                     ; check completion bit
            cxt_return_tbds
                                      ; complete, handle it
      jnz
                                      ; not complete, exit
      jmp
            cxt return
; Free up the tbds.
cxt_return_tbds:
            di, [bx].CBL TBD OFFSET
                                      ; get pointer to tbd
      mov
; If there were no free tbds then make this into a free tbd.
            p 1st free tbd, OFFSET p 1st free tbd ; any free?
      cmp
      jne
            cxt tbd loop
                                      ; yes
            p 1st free tbd,di
      mov
                                      ; reset the pointer
cxt tbd loop:
            p last free tbd,di
                                      ; free this one up
      mov
            [di].TBD_STATUS,TBD_EOF_BIT ; last one in the chain?
      test
            cxt_tbds_returned
                                     ; all tbds have been returned
      jnz
      mov
            di,[di].TBD_LINK
                                     ; link to next one
            cxt_tbd_loop
                                     ; loop until got 'em all
      jmp
```

```
; Determine completion status.
```

۲

|                                                                                                                                                                                                                                                  | mov<br>mov  | ax,TPS_COMP_BIT<br>dx,TPS_OK_BIT                               | ; build status in ax<br>; dx cleared if error found  |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------|----------------------------------------------------------------|------------------------------------------------------|
|                                                                                                                                                                                                                                                  | cmp<br>je   | crs_check_flag,0<br>cxt c2                                     | ; CRS checking?<br>; nope                            |
|                                                                                                                                                                                                                                                  | test<br>jz  | -<br>cx,CBL_S10_BIT<br>cxt c2                                  | ; check carrier sense                                |
|                                                                                                                                                                                                                                                  | or<br>xor   | ax,TPS_NO_CRS_BIT<br>dx,dx                                     | ; set no carrier bit<br>; found an error             |
| ant a).                                                                                                                                                                                                                                          |             |                                                                |                                                      |
| cxt_c2:                                                                                                                                                                                                                                          | cmp<br>je   | sqe_check_flag,0<br>cxt_c3                                     | ; SQE checking?<br>; no SQE checking                 |
|                                                                                                                                                                                                                                                  | test<br>jnz | cx,CBL_S6_BIT<br>cxt_c3                                        | ; check SQE bit, should be set<br>; is set, no error |
|                                                                                                                                                                                                                                                  | test<br>jnz | cx,Ofh<br>cxt_c3                                               | ; any collisions?<br>; yes, S6 bit meaningless       |
|                                                                                                                                                                                                                                                  | or<br>xor   | ax,TPS_SQE_BIT<br>dx,dx                                        | ; signal quality error<br>; found an error           |
| cxt c3:                                                                                                                                                                                                                                          |             |                                                                |                                                      |
| CXC_C3.                                                                                                                                                                                                                                          | test<br>jnz | cx,CBL_OK_BIT<br>cxt_c4                                        | ; error free completion?<br>; yes                    |
|                                                                                                                                                                                                                                                  | mov         | dx,TPS_UNSPEC_ERR                                              | ; assume unspecified error                           |
|                                                                                                                                                                                                                                                  | test<br>jz  | cx,CBL_S5_BIT<br>cxt c4                                        | ; excessive collisions bit                           |
|                                                                                                                                                                                                                                                  | or          | ax, TPS_CLSN_BIT                                               | ; set the bit                                        |
|                                                                                                                                                                                                                                                  | xor         | dx, dx                                                         | ; found an error                                     |
| cxt_c4:                                                                                                                                                                                                                                          |             |                                                                |                                                      |
|                                                                                                                                                                                                                                                  | test<br>jz  | cx,CBL_S8_BIT<br>cxt c5                                        | ; DMA underrun?                                      |
|                                                                                                                                                                                                                                                  | or          | ax, TPS_UNDER_BIT                                              | ; underrun bit                                       |
|                                                                                                                                                                                                                                                  | xor         | dx,dx                                                          | ; found an error                                     |
|                                                                                                                                                                                                                                                  |             |                                                                |                                                      |
| cxt_c5:                                                                                                                                                                                                                                          | add         | conoria stata STS TY COUNT 1                                   | ; increment transmit count                           |
|                                                                                                                                                                                                                                                  | adc         | generic_stats.STS_TX_COUNT,1<br>generic_stats.STS_TX_COUNT+2,0 | , increment transmit count                           |
|                                                                                                                                                                                                                                                  | or          | ax,dx                                                          | ; or in other bits                                   |
|                                                                                                                                                                                                                                                  | les         | si,DWORD PTR [bx].CBL_TPS_OFF                                  | ; get new pointer                                    |
|                                                                                                                                                                                                                                                  | mov         | es:[si].TPS_STATUS,ax                                          | ; write the status                                   |
| ; Execute the transmit complete appendage, if one is setup. First, give<br>; link level a chance at the packet. If the packet was sent by link level,<br>; it will return 0 in AX. If this is so, the user appendage should not be<br>: executed |             |                                                                |                                                      |

; executed.

| push    | bx                    | ; | save bx               |
|---------|-----------------------|---|-----------------------|
| EXECUTE | _1_APNDG ll_appendage | ; | execute the appendage |

The for here is in effect of NRG. 1x. It is virtually a kick on SE (an interrupt) Since we (80286) issued Command(s) but no completion interropt coming back from 186,

The assumption was sold didn't see the Command (s) after the (st one. and we simply tell 586 again.

The clean up logic is ignored completely. Since it really does work.

; restore bx pop bx J ax,0 ; was this link level's packet cmp ; skip user appendage if so je cxt skip app ; save bx push bx EXECUTE 1 APNDG tx appendage ; execute the appendage  $\mathbf{b}\mathbf{x}$ ; restore bx pop cxt skip app: ; If no cbl's are free, this one becomes the 1st free one. p 1st free cbl,OFFSET p 1st free cbl ; any free? CMD jne cxt advance ptr ; yes p 1st free cbl,bx ; set new free pointer mov cxt\_advance\_ptr: bx,[bx].CBL LINK ; advance to next cbl mov cmp bx,p 1st free cbl ; was this the last transmit? cxt enter idle ; yes, no more are queued je bx,p 1st rdy cbl ; hit the ready list cmp je cxt enter idle ; gone full circle? cmp bx,p\_1st\_586\_cbl cxt enter idle jе cxt check ; check if new one is complete jmp ; Transmitter empty, update state machines. cxt enter idle: cx state, OFFSET cx idle ; enter idle state mov tx timeout enable,0 ; disable the timeout mov cxt return: tx time count,0 ; clear the timeout counter mov p\_1st\_586\_cbl,bx ; will be first one next time mov net586 cx return ; jump to return location jmp cx state machine ENDP ; net586\_xmit\_timeout - transmitter timeout handler ; ; This module just removes a timed-out transmit from the 82586 transmit ring ; and returns it to the user. No attempt is made to revive the 586 if it ; has died. see the back of previous page \*\*\*\*\* PUBLIC net586\_xmit\_timeout net586 xmit timeout PROC NEAR cx,Offffh mov tmout wait cmd: scb.SCB COMMAND,0 ; command clear? cmp

\_ tmout\_cmd\_clear je ; yes loop f tmout\_wait\_cmd tmout cmd clear: mov scb.SCB\_COMMAND,SCB\_CUC\_START ; restart the transmitter CHANNEL ATTENTION ; bang on the 586 tx time count, 0 mov tx timeout enable, Offffh mov ret mov bx,p\_1st\_586\_cbl ; get pointer to cbl ; Free up the tbds. xt\_return\_tbds: di,[bx].CBL\_TBD\_OFFSET mov ; get pointer to tbd ; If there were no free tbds then make this first free tbd. p 1st free tbd, OFFSET p 1st free tbd ; any free? cmp jne xt tbd loop ; yes mov p 1st free tbd,di ; reset the pointer xt\_tbd\_loop: p last free tbd,di ; free this one up mov [di].TBD\_STATUS,TBD\_EOF\_BIT ; last one in the chain? test xt\_tbds\_returned ; all tbds have been returned jnz di,[di].TBD\_LINK mov ; link to next one jmp xt tbd loop ; loop until got 'em all xt tbds returned: si,DWORD PTR [bx].CBL TPS OFF ; get tps pointer les es:[si].TPS\_STATUS,TPS\_COMP\_BIT+TPS\_TIMEOUT\_BIT ; write status mov ; Execute the transmit complete appendage, if one is setup. First, give ; link level a chance at the packet. If the packet was sent by link level, ; it will return 0 in AX. If this is so, the user appendage should not be ; executed. push bx ; save bx EXECUTE 1 APNDG 11 appendage ; execute the appendage ; restore bx pop bx ax,0 cmp ; was this link level's packet xt skip app je ; skip user appendage if so push bx ; save bx EXECUTE 1 APNDG tx\_appendage ; execute the appendage pop bx ; restore bx xt\_skip\_app:

; If no cbl's are free, this one becomes the 1st free one.

cmp p\_lst\_free\_cbl,OFFSET p\_lst\_free\_cbl ; any free?
jne xt\_advance\_ptr ; yes

|         | mov             | p_1st_free_cbl,bx                  | ; | set new free pointer                                |
|---------|-----------------|------------------------------------|---|-----------------------------------------------------|
| xt_adva | nce_ptr:<br>mov | bx,[bx].CBL_LINK                   | ; | advance to next cbl                                 |
|         | cmp<br>je       | bx,p_1st_free_cbl<br>xt_enter_idle | • | was this the last transmit? yes, no more are queued |
|         | cmp<br>je       | bx,p_lst_rdy_cbl<br>xt_enter_idle  | ; | hit the ready list                                  |
|         | cmp<br>je       | bx,p_1st_586_cbl<br>xt_enter_idle  | ; | gone full circle?                                   |
|         | jmp             | xt_return                          | ; | return                                              |
| xt_ente | r_idle:         |                                    |   |                                                     |
|         | mitter e        | mpty, update state machines.       |   |                                                     |

| public xt_en | ter_idle                       |                               |
|--------------|--------------------------------|-------------------------------|
| mov          | cx_state,OFFSET cx_idle        | ; enter idle state            |
| mov          | <pre>tx_timeout_enable,0</pre> | ; disable the timeout         |
| xt return:   |                                |                               |
| mov          | <pre>tx_time_count,0</pre>     | ; clear the timeout counter   |
| mov          | p_1st_586_cbl,bx               | ; will be first one next time |
| ret          |                                |                               |
|              |                                |                               |

net586\_xmit\_timeout ENDP

\_TEXT

٠

\*

ENDS END

## UPDATE NOTE NX300 Network Executive Release 6.2

This update note for NX300 Network Executive, Release 6.2 describes the features of Release 6.2 and lists the fixes that have been made since previous releases.

If you should have any questions, comments, or suggestions, please contact:

Customer Service Center Federal Technology Corporation 207 South Peyton Street Alexandria, VA 22314 1-800-FON4LAN

## FEEDBACK

Your feedback on this product is appreciated. An overall evaluation form, as well as a problem report form, are included at the end of this update note. Please complete the evaluation form and return it to Federal Technology Corporation. Please use the problem report as necessary.

## DIFFERENCES BETWEEN NX VERSION 5 AND VERSION 6 IN LINK-LEVEL MODE

The transmit buffering area is now 32 Kbytes. All remaining memory is now used for receive buffering.

NX Version 6 uses approximately 16 Kbytes for internal data area and 32 Kbytes for transmit buffering. All remaining memory space is available for receive buffering. The maximum memory used in any case is limited to 512 Kbytes. Buffers are allocated to hold a maximum size packet. For 1518-byte Ethernet packets the approximate board buffering capability is as follows:

|             | Transmit | Receive |
|-------------|----------|---------|
| Memory Size | Buffers  | Buffers |
| 128 Kbytes  | 20       | 50      |
| 256 Kbytes  | 20       | 140     |
| 512 Kbytes  | 20       | 310     |

Part No. 4230202-01, Rev. B

- \* The number of multicast addresses is no longer configurable. Sixty-four multicast slots are always configured.
- \* I/O-mapped and memory-mapped host interrupts are no longer supported.
- \* Configuration error codes, fatal error codes, and warning codes have been changed. The error codes for the two versions of the NX firmware are mutually exclusive. Status codes in link-level reply messages remain unchanged.
- The configuration message has been changed. Although the NX Version 6 configuration message is compatible with that of Version 5, several changes have been made. The following fields have become reserved fields in Version 6:
  - Memory Map Size
  - Memory Map
  - NX Movable Block Address
  - Number of Processes
  - Number of Mailboxes
  - Number of Multicast Slots
  - Host-to-EXOS Message Queue Interrupt Value
  - EXOS-to-Host Message Queue Interrupt Value

Version 6 of the NX firmware ignores these fields. The request and reply values of these fields are undefined.

The following fields have recommended specified values in the NX Version 6 configuration. Other nonspecified values are supported for compatibility with Version 5 but the use of these values is discouraged. Version 6 provides the best performance if the specified values are used.

- EXOS Mode = 0
- Host Data Format Option = 0101H
- Host Address Mode = 3
- Number of Hosts = 1

For the fields Host-to-EXOS Message Queue Interrupt Type and EXOS-to-Host Message Interrupt Type, option 1 (I/O mapped) and option 2 (memory mapped) are no longer valid. Option 3 (level interrupt) and option 4 (bus-vectored interrupt) have been combined into a single option, option 4 (bus hardware

interrupt). Option 3 is still supported, but its use is not recommended.

- \* The NET TRANSMIT reply no longer returns the slot number the transmit request occurred on. This field in the request/reply message is now reserved and always returns a value of zero.
- \* The NET TRANSMIT return code 10H is now a general transmit error that indicates the transmit was unsuccessful.
- \* A new NET TRANSMIT request (request code OFH) has been added. This request returns a reply message to the host as soon as the data in the host buffer have been copied to the EXOS board. With this form of message, the transmit status is not returned to the host. Therefore, protocol on the host must detect transmit failures.
- \* The network statistics counters no longer stick at their maximum value. Instead, they roll over and continue counting.
- \* The loopback error bit available to the host in the status port is now a generic warning bit. Any warning, including loopback test failure, causes this bit to be set.
- \* The debug jumper is now the NX mode jumper. When NX PROMs that support both Versions 5 and 6 are used, this jumper selects the mode to run in. Installing the jumper causes Version 6 to run. Removing the jumper causes Version 5 to run.

## UPGRADING TO NX 6.2

s 1 - •

If you have an EXOS 301, EXOS 302, or EXOS 304 board installed in your system, you must replace both EPROMs on the board before you can use Release 4.0 of the LAN Service network software. The EPROMs you are removing contain Version 5.x of the Network Executive (NX) firmware. The EPROMs you are replacing them with contain Version 6.2 of NX (NX 6.2) as well as the latest NX 5.x firmware for your board (the latter is provided for downward compatibility).

Follow these steps to upgrade to the NX 6.2 EPROM:

- 1. Power down your system.
- 2. Remove the EXOS board from your system.
- 3. Locate the EPROM marked NX300L and make a note of its orientation.
- 4. Remove the EPROM from the board.
- 5. Insert the new EPROM (NX300L 6.2) in the board. Make sure it is oriented the same way as the old EPROM.

- 6. Locate the EPROM marked NX300H and make a note of its orientation.
- 7. Remove the EPROM from the board.
- 8. Insert the new EPROM (NX300H 6.2) in the board. Make sure it is oriented the same way as the old EPROM.
- 9. Jumper the EXOS board for 27256 EPROMs, as follows:

| Board    | Jumpers                          |
|----------|----------------------------------|
| EXOS 301 | J15 1-2 present                  |
| EXOS 302 | J15 2-3 present; J16 1-2 present |
| EXOS 304 | J18 1-2 present; J19 1-2 present |

10. Insert the NX 6.x select jumper. This jumper selects the operation mode of NX. The EXOS board operates in NX 6.x mode when the jumper is present and in NX 5.x mode when the jumper is absent.

| Board    | Jumpers                    |
|----------|----------------------------|
| EXOS 301 | J9 present for NX 6.x mode |
| EXOS 302 | J9 present for NX 6.x mode |
| EXOS 304 | J9 present for NX 6.x mode |

- 11. Re-insert the EXOS board in the system and reconnect the Ethernet cables.
- 12. Power on the system. Ensure that the status LED on the EXOS board starts flashing at a steady rate approximately 3 seconds after you power on the system. If the status LED remains lit constantly, check that you have replaced the NX EPROMs properly.