# 586-Drive<sup>TM</sup>

C/C++ Programmable, 133 MHz 32-bit Web Controller with Floating Point Unit, CompactFlash, 100M Ethernet, High voltage I/Os, and 24-bit ADC / DACs



# Technical Manual



1724 Picasso Avenue, Davis, CA 95618-0547, USA Tel: 530-758-0180 Fax: 530-758-0181

Email: sales@tern.com http://www.tern.com

#### **COPYRIGHT**

586-Drive, 586-Engine, 586-Engine-P, NT-Kit, and ACTF are trademarks of TERN, Inc. Am188ES and Am186ES, ElanSC520 are trademarks of Advanced Micro Devices, Inc. Paradigm C/C++ is a trademark of Paradigm Systems.

Windows95/98/2000/NT/ME/XP are trademarks of Microsoft Corporation.

#### Version 1.1

January 24, 2007

No part of this document may be copied or reproduced in any form or by any means without the prior written consent of TERN, Inc.

© 1993-2006

1724 Picasso Avenue, Davis, CA 95618, USA Tel: 530-758-0180 Fax: 530-758-0181

Email: sales@tern.com http://www.tern.com

#### **Important Notice**

TERN is developing complex, high technology integration systems. These systems are integrated with software and hardware that are not 100% defect free. TERN products are not designed, intended, authorized, or warranted to be suitable for use in life-support applications, devices, or systems, or in other critical applications. TERN and the Buyer agree that TERN will not be liable for incidental or consequential damages arising from the use of TERN products. It is the Buyer's responsibility to protect life and property against incidental failure. TERN reserves the right to make changes and improvements to its products without providing notice.

------

# **Chapter 1: Introduction**

#### INTRODUCTION

The **586-Drive**<sup>TM</sup> (**5D**) is a C/C++ programmable controller based on the 32-bit 133 MHz AMD Elan SC520. It combines this high performance 586 generation processor with an extensive set of industrial I/O features on a single-board.

The SC520 integrates an Am586 CPU and a high performance ANSI/IEEE 754 compliant hardware floating point unit (FPU). The SC520 has a total of seven timers including PIT timers and GP timers, plus a software timer. A real-time clock (RTC) provides a time-of-day calendar and 114 bytes of battery backed RAM. 13 user programmable multifunctional I/O lines are available. One synchronous serial interface (SSI) supports full-duplex, high speed bi-directional communication.

By default, 256KW low power 55 ns SRAM is installed to allow longer battery backup lifetime; this requires slower 2 wait state access to memory. Optionally, if battery backup is not required, a high speed 20 ns SRAM can be installed to allow higher performance zero wait state operation.

In additional to the on-board surface mount Flash, a 32-pin DIP IC socket allows using traditional user application plug-in ROM/Flash. A 50-pin CompactFlash interface supports low cost, removable, up to 2 GB mass storage CompactFlash cards with Windows compatible FAT file system support. The **5D** can be powered by a single unregulated DC power from 8V to 30V range with the on-board high-efficiency 5V switching regulator.

#### **NETWORKING/CONNECTIVITY**

An Fast Ethernet Module can be installed to provide 100M BaseT network connectivity. The hardware LSI TCP/IP stack implements TCP/IP, UDP, ICMP and ARP in hardware. With 16KB internal transmit and receiving buffer, the Ethernet module releases internet connectivity and protocol processing from the host processor. The system can easily handle TCP/IP traffic of up to 200 Kbyte/s. Samples for HTTP, Telnet, FTP applications are available.

Up to 4 RS232 serial ports (2 from SC520, and 2 SCC2691) are available. Two ports can be factory-configured as RS485.

#### INDUSTRIAL I/O

By default, 7 high voltage inputs(30V), and 14 high voltage sinking drivers(ULN2003A) are installed in DIP sockets. Each driver is capable of sinking 350 mA at 50V per line. They can directly drive solenoids, relays, or lights. Optionally, 12 DAC channels can be installed in their place.

Four high isolation voltage photo couplers (PS2701, NEC) can be installed to provide optically isolated inputs. Also, 16 additional input channels are provided with a 24-bit ADC(LTC2448), connected via hardware configurable buffer resistors and screw terminal blocks. Variable resistor dividers can be installed to allow variable(up to 30V) input range(as default, 0-5V). They can be processed as analog or digital signals. With a peak sample rate of 5 KHz, this ADC works well directly with analog signals from strain gages, current shunts, RTDs, resistive sensors, and also work well directly with thermocouples in the differential mode. A precision reference(LT1019/REF02) with a internal temperature sensor providing local temperature for thermocouple applications.

In addition, a 4 ch. 16-bit parallel ADC (AD7655, 0-5V) supports ultra high-speed (1 MHz conversion rate) analog signal acquisition. The AD7655 contains two low noise, high bandwidth track-and-hold amplifiers that allow *simultaneous* sampling on two channels.

Eight 16-bit digital to analog converters (TLC2600) can be installed to provide analog voltage (0-5V) outputs.

With the 388 pin BGA package of the SC520, repair support is not available. The **5D** works with TERN expansion boards including the UR8, C24, P100, and P300.

Special Note: The core of the Am520 CPU operates at +2.5V and the I/O operation at +3.3V. Also, the input for the I/O is +5V compatable. Stresses above these can cause permanent damage to the SC520 CPU. Operation above these values is not recommended, and can effect device reliability.

------

#### **Features**

#### Features:

- 151x82 mm. DIN rail mounting, 110 mA at 24V DC power
- 133 MHz, AMD SC520, program in C/C++
- High performance hardware floating point coprocessor
- Non volatile SRAM, Flash, RTC, timers, interrupts
- 4 RS232/485 Async serial ports and 1 Sync serial port
- CompactFlash and FAT16 file system support
- 4 ch 16-bit ADC (AD7655) and 8 16-bit DAC(LT2600)
- Hardware TCP/IP stack for 100M Base-T Ethernet
- 16 analog or digital inputs(30V) with 24-bit ADC(LT2448)
- 20+ Solenoid Drivers, 4 Opto-couplers, 10+ digital I/Os.

### **Physical Description**



## Expansions of the 586-Drive:

A P300 expansion board can be driven by the 586-Drive.



A low power graphic 192x128 pixel LCD can be driven by the 586-Drive.



1-3

------



Figure 1.1 Flow chart for ACTF operation

The "ACTF boot loader" resides in the 256KW on-board Flash chip (29F400). At power-on or RESET, the "ACTF" will check the STEP 2 jumper. If STEP 2 jumper is not installed, the ACTF menu will be sent out from serial port0 at 19200 baud. If STEP 2 jumper is installed, the *586Drive* will check for a valid battery back-up. If present, it will go to the jump address stored in the CPU's 114 bytes of general purpose RAM. Without a valid battery back up, it will write the address 0x80000 to the inernal RAM, and then go to that address.

There is a 32-pin ROM socket on the *5D*. The user's application program must reside in SRAM for debugging in STEP1, reside in battery-backed SRAM for the standalone field test in STEP2, and finally be programmed into Flash for a complete product. For production, the user must produce an ACTF-downloadable HEX file for the application, based on the DV-P. The "STEP2" jumper (J2 pins 38-40) must be installed for every production-version board.

#### Step 1 settings

In order to correctly download a program in STEP1 with Paradigm C++ Debugger, the 5D must meet these requirements:

- 1) 5860\_115.HEX must be pre-loaded into Flash starting address 0x80000.
- 2) The CPU's 114 bytes of RAM must have the correct jump address pointing at 5860\_115.HEX, which is the address 0x80000.
- 4) The STEP2 jumper must be installed on J2 pins 38-40.

For further information on programming the 586-Drive, refer to the Software chapter.

------

### 1.2 586-Drive Programming Overview

Steps for **5D** based product development:

Preparation for debugging (By default, this is done in the factory!):

- Connect board to PC via serial link at 19,200, N, 8, 1 with Hyper Terminal
- Power-on without STEP 2 Jumper installed (J2.38 = J2.40)
- ACTF Menu will be sent from SER0 to external HyperTerminal at 19,200 baud
- Type 'D', then <enter> to download. Send Text File \tern\586\rom\l\_29f400.hex
- Type 'G04000', then <enter> to execute HEX file just sent; will prepare flash
- Send Text File \tern\586\rom\5860 115.hex
- Type 'G80000', then <enter> to execute debug kernel
- Set STEP 2 Jumper (J2.38 = J2.40)
- Power-on/reset, ready to connect to Paradigm C/C++ for debugging



#### STEP 1: Debugging

- Launch Paradigm C/C++. Open \tern\586\samples\5d\5d.ide.
- Run sample code
- Create application using sample code.
- Use Paradigm C/C++ to edit, compile, link, locate, and remote debug



#### STEP 2: Standalone field test

- Downloaded code from STEP 1 is located by default at 0x0800:0x0000 in the battery-backed SRAM
- Set Jump Address to point to downloaded code. Remove STEP 2 Jumper and cycle power. Setup Hyper Terminal and see ACTF menu sent from 5D.
- Type 'G08000', then <enter> to jump to and execute application in SRAM
- Set STEP 2 jumper. 5D will execute application at 0x08000 at power-up.
- Test application. Return to STEP 1 as needed



#### STEP 3: Production (DV-P Kit required)

- Use Paradigm C/C++ to generate ACTF downloadable application HEX file.
- In Paradigm C/C++, open **Target Expert** for application. Set "Target Connection" to "No Target/ROM". Build target. Paradigm C/C++ will generate HEX file.
- Remove STEP 2 Jumper and see ACTF menu at hyper terminal. Download "1\_29f400.hex". Same process as above. Type 'G04000" to prepare flash
- Send you application HEX file. Type 'G80000' to execute code and set jump address to point to you application in flash.
- Set STEP2 Jumper

-----

# **Chapter 2: Installation**

#### 2.1 Software Installation

Refer to the Technical manual "Software Kit.pdf" on TERN CD under tern docs\manuals.

By manufacture default upon shipment, the 586-Drive should be ready to communicate with Paradigm C++ for debugging, with STEP2 jumper installed, CMOS RAM setup for 0x80000, and 5860\_115 debug kernel residing in Flash starting 0x80000. Power on, the on-board LED should blink twice indicating running debug kernel. You **DO NOT** have to download debug kernel into flash again. You can **SKIP** the operation discussed in 2.2 below.

## 2.2 Prepare 586-Drive for Paradigm C++ TERN Edition

This section uses RTLOAD to communicate with the ACTF utility while chapter refers to a hyper terminal. Either works fine.

1) Start Paradigm C++. Select from Top menu: Tool, RTLOAD,



A HEX file Loader window will be shown.

2-1



- 2) F5=Select Baud, to setup 19200.
- 3) F8=Select .HEX file from c:\tern\586\rom\L 29F400.HEX



3) Power on the 586-Drive with the STEP2 jumper off, the ACTF menu will show up.



4) Caps Lock on your PC keyboard, Type "D" command, then enter.



5) F6 = Upload .HEX file, from PC to 586 SRAM.



6) Type "G04000" to run the "L\_29F400" in SRAM. The first time you type "G04000" you will get an error. Doing the "G04000" again, you will see as below



The 29F400 Flash sector 0x80000 to 0xFBFFF will be erased. It will then be ready to program Flash with new DEBUG kernel file (c:\tern\586\rom\5860 115.hex), or user application .HEX.

7) F8= Select .HEX file, from c:\tern\586\rom\5860 115.hex.



8) F6=Upload .HEX file to program the 5860\_115 debug kernel into Flash starting address 0x80000.



After programming the Flash, 586-Drive will automatically reset.

"G80000" to setup the CMOS RAM Jump Address, and start the DEBUG kernel. The on-board LED should blink twice and then stay on; indicating 586-Drive is ready for remote debugging.

Power off the controller. Install the STEP2 Jumper, then power on, the LED blink twice.

Use F9 = Exit.

The 586-Drive is ready for using Paradigm C++ TERN Edition to download, debug, and run.

There are several sample projects in the c:\tern\586 directory (default working directory):

\_\_\_\_\_

tern\586\led.ide: The most basic sample project, just flashes LED

tern\586\test.ide: Gives samples for serial ports, interrupts, timers. etc

tern\586\samples\5d\5d.ide: Shows how to use hardware components on board.

To open projects, go to "File" and open the sample project file, then build and download.

There are many sample programs under c:\tern\586\samples.

After you debug your application code, you can setup the 586-Drive to run in Standalone Mode.

Standalone Mode(STEP2):

By default, the Paradigm C++ TERN Edition will download your application code starting at 0x08000 in the battery backed SRAM.

Power off 586-Drive. Remove STEP2 jumper. On PC side, click TOOL, RTLOAD.

Power on 586-Drive again without STEP2 jumper, then the ACTF menu should show up.

At the ACTF menu prompt, type "G08000" to setup the Jump Address and run your application.

Power off, install the STEP2 Jumper. Then at power on, controller will jump to 0x08000 in SRAM and run your application.

2-6

\_\_\_\_\_

### 2.3 Hardware Installation

#### Overview

- Connect debug serial cable: For debugging (STEP 1), place IDE connector on SER0 (H2) with red edge of cable at pin 1
- Connect wall transformer:

  Connect 9V wall transformer to power and plug into power jack adapter which installs in 2-pin screw terminal



2-7

# **Chapter 3: Hardware**

#### 3.1 SC520 - Introduction

The 586-Drive is based on AMD Elan SC520 Microcontroller. It includes an industry-standard Am5x86 CPU with floating point unit (FPU). It provides a General-Purpose (GP) bus with programmable timing for 8 and 16-bit devices. A ROM/Flash controller supports on-board high performance code execution. An enhanced programmable interrupt controller (PIC) prioritizes 22 interrupt levels with up to 15 external sources. Two asynchronous UARTs can operate up to 1.15 M bit/s. A Synchronous Serial Interface (SSI) offers full-duplex or half-duplex operation to support on-board ADC/DACs and user expansion. A real time clock, a software timer, 3 GP timers, and 3 programmable interval timers are all included. Thirty-two programmable I/O pins are on-board.

Please refer to the SC520 User's Manual, SC520 Data Sheet, and SC520 Register Set Manual included on TERN's CD: amd docs\sc520.

#### **3.2** *SC520* – *Features*

#### 3.2.1 Clock

One 32.768 KHz and one 33.333 MHz crystal are installed to provide all the clocks required for CPU, Real time clock, UART, timers, and clock output.

The CLKTEST (CLKT) signal is routed to J1 pin 4. Software can select to output one of 6 internal clocks, including 32.768K, 1.8443 MHz, 18.432 MHz, 1.1882 MHz, 1.47456 MHz, and 36.864 MHz.

### 3.2.2 GP Bus Chip Selects

To Program GP bus Memory or I/O Chip Selects:

- 1) Config one of the 16 PAR regs(PAR11=Flash).
- 2) Config GP bus timing
- 3) Enable Chip select function on the desired pin in the CSPFS reg.

A total of 8 GP-bus Chip selects is supported on the 586Drive:

/GPCS0=P27=J2.37(PAR13), I/O 0x1800, Free to use for expansion.

/GPCS1=/ROMCS1=U16.22, 32-pin DIP socket, memory mapping 0x40000.

/GPCS2=/ROMCS2=SRAM (PAR14), Used by 586E/P/D, 0x0000-0x7ffff, memory mapping

/GPCS3=PITG2=U2.5(PAR15), I/O=0x2000, used by 5D ADC1655, SC1, SC2, L1, L2, and CF select

/GPCS4=TMRIN1=J2.1(PAR4), I/O=0x2800, free to use for expansion

/GPCS5=TMRIN0=J1.19(PAR5), I/O=0x3000, Free to use for expansion.

/GPCS6=TMROUT1=J2.5(PAR6), I/O=0x3800, Free to use.

/GPCS7=TMROUT0=J2.8=JP1.5(PAR7), Ehernet(i2chip) select, MEM mapping 0xf0000

#### 3.2.3 Programmable interrupt controller and external Interrupts

The Programmable Interrupt Controller (PIC) prioritizes 22 interrupt levels (P1-P22) with up to 15 external sources (GPIRQ0-10 and INTA-D).

A programmable router must be programmed to map internal or external interrupt sources to the master or two of slave interrupt controllers to provide different priorities, from P1 to P22.

All 15 external interrupt requests are programmed as edge sensitive, after 586 init();

An example map for P1 to P22 is listed below and demonstrated in the sample program **586\_intx.c** and is included in the pre-built sample project \text{\text.ide}.

```
There are 22 interrupt priority levels plus NMI
     There are 15 external interrupt requests (GPIRQ0-10, /INTA-D)
//
     Example internal interrupt map by TERN:
//
     P1=Master PIC IR0, interrupt vector=0x40, PIT timer0
     P2=Master PIC IR1, interrupt vector=0x41, GPIRQ0=PIO23=J2.33
     P3=slave1 PIC IR0, interrupt vector=0x48, RTC
     P4=slave1 PIC IR1, interrupt vector=0x49, GPIRQ1=PIO22=J2.23
//
//
     P5=slave1 PIC IR2, interrupt vector=0x4a, GPIRQ2=PIO21=J2.21
//
     P6=slave1 PIC IR3, interrupt vector=0x4b, GPIRQ3=PIO20=J2.19
//
     P7=slave1 PIC IR4, interrupt vector=0x4c, GPIRQ4=PIO19=J2.20
//
     P8=slave1 PIC IR5, interrupt vector=0x4d, FPU
//
     P9=slave1 PIC IR6, interrupt vector=0x4e, /INTD=SCC=J3.14
     P10=slave1 PIC IR7, interrupt vector=0x4f, GP timer1/INTC=J3.13
//
//
     P11=Master PIC IR3, interrupt vector=0x43, SER2/0
//
     P12=Master PIC IR4, interrupt vector=0x44, SER1
//
     P13=Slave2 PIC IR0, interrupt vector=0x50, GP timer0
     P14=Slave2 PIC IR1, interrupt vector=0x51, GPIRQ5=PIO18=J2.17
//
     P15=Slave2 PIC IR2, interrupt vector=0x52, GPIRQ6=PIO17=J2.18
//
     P16=Slave2 PIC IR3, interrupt vector=0x53, GPIRQ7=PIO16=J2.15
//
     P17=Slave2 PIC IR4, interrupt vector=0x54, PIT timer1
//
     P18=Slave2 PIC IR5, interrupt vector=0x55, GPIRQ8=PIO15=J2.16
//
     P19=Slave2 PIC IR6, interrupt vector=0x56, GPIRQ9=PIO14=J2.6
//
     P20=Slave2 PIC IR7, interrupt vector=0x57, GPIRQ10=PIO13=J2.8
     P21=Master PIC IR6, interrupt vector=0x46, PIT Timer2/INTB=J3.12
//
     P22=Master PIC IR7, interrupt vector=0x47, GP timer2/INTA=J3.11
```

See the sample program in c:\tern\586\samples\5e\586 intx.c for more details.

The 586-Drive uses vector interrupt functions to respond to external interrupts. Refer to the SC520 User's manual for more information about interrupt vectors.

#### 3.2.4 Asynchronous Serial Ports

The SC520 has two 16450/16550-compatible asynchronous serial channels: SER0/2 and SER1. Both asynchronous serial ports support the following:

- Full-duplex operation,
- 5-, 6-, 7-, and 8-bit data transfers
- Odd, even, and no parity
- 1, 1.5, or 2 stop bits
- Error detection
- Hardware flow control
- Transmit and receive interrupts for each port
- Maximum baud rate, up to 1.152 MHz

The software drivers for each serial port implement a ring-buffered interrupt transmitting and receiving arrangement. See the samples files  $s1\_echo.c$  and  $s0\_echo.c$ ;  $s1\_echo.c$  is included in the pre-built sample project \tern\586\test.ide.

#### 3.2.5 GP Timers

Three 16-bit General-Purpose Timers are on-board. Two external inputs, TIN0=J1.19(Expansion) and TIN1=J2.1, can be used for the GP Timer0 and Timer1 to capture and count external pulses up to 33.333/4 MHz

Timer 0 and Timer1 can output pulses on TOUT0=J2.8(i2chip) and TOUT1=J2.5.

GP Timers support interrupt on terminal count, continuous mode, and square wave generation.

Timer2 is not connected to any external pin. It can be used as an internal timer for real-time coding or time-delay applications. It can also prescale timer0 and timer1.

The TIN0, TIN1m TOUT0, and TOUT1 pins are also shared with GP bus chip select lines.

See the sample programs *timer02.c* and *tmr\_out.c* in the tern\586\samples\5e directory. GP timer code is intergrated into several of the samples in the sample project \tern\586\test.ide.

#### 3.2.6 PIT Timers

Three 16-bit Programmable Interval Timers (PIT) are on-board. Each PIT channel has one interrupt output. Only PIT2 has an external output pin and can provide square wave output. All PITs supports interrupt on terminal count, hardware-retriggerable one-shot and timer functions. See samples at c:\tern\586\samples\5e directory for 586\_pit0.c and pit2\_out.c.

#### 3.2.7 Software timers

The "software timer" is actually a hardware timer, which is intended to provide a millisecond timebase with microsecond resolution. Ideal applications for this function include providing a system wide timebase, or a precise measurement of the time between events. The software timer has a 16-bit microsecond counter that increments with a period of one millisecond. This yields a maximum duration of 65.5 seconds. A microsecond latch register that provides the number of microseconds since the last time that the millisecond register was read.

The software timer provides a very efficient hardware timerbase for use by software. It is designed to replace the traditional method of system timebase generation.

Traditional system timebase generation is accomplished by programming a timer to generate a periodic interrupt. The interrupt service routine then increments a counter. This value is often kept in a global variable, which can then be accessed by other code that needs to track time. The problem with this traditional timebase is caused by interrupt latency and possible missed interrupt.

The software timer included can be used to resolve these problems.

See more details on AMD SC520 Users' Manual, on TERN CD\amd\_docs\sc520\sc520\_user\_manul.pdf chapter 18.

#### 3.2.8 SSI

A synchronous serial interface (SSI) provides full-duplex and half-duplex, bi-directional communication at a software programmable SSI clock speed, from 64K Hz to 8MHz.586-Drive uses the SSI to interface to the LTC2448 ADC and TLC2600 DAC. The user can use the SSI to interface many types of external serial peripheral devices.

See the sample c:\tern\586\samples\5e\ssio.c.

#### 3.2.9 RTC

A battery backed up real-time clock (RTC) is included. The RTC consists of time-of-day clock with alarm and a 100-year calendar. It has also a programmable periodic interrupt and 114 bytes of static user RAM. When the 5D is powered off, the RTC (and 114 bytes of static RAM) is backed-up by the 3V battery installed on the 586-Drive.

See samples 586\_rtc.c and rtc\_pint.c for more details.

#### 3.2.10 Watchdog timer

The Watchdog timer included in SC520 is not disabled. The 586-Drive uses a 691 supervisor chip to monitor the 5V power and provides an external watchdog. User can activate the 691 watchdog with a jumper setting on H1. The watchdog timer provided a means to monitor proper software execution. If software becomes stuck in an infinite loop, for example, the watchdog timer can reset the board to recover to proper execution. If the watchdog timer is enabled (via install a jumper at H1), it must be reset via software every 1.6 seconds, or sooner. If the watchdog timer is not reset after 1.6 seconds, the 691 supervisor will assert /RST, and reset the 586Drive. To reset the watchdog timer, user code needs to toggle the PIO pin P4. With the watchdog jumper enabled, this routine should be arragned such that it is called every 1.6 seconds are sooner.

#### 3.2.11 Analog/digital inputs based on 24-bit ADC(LTC2448) with resister dividers

A 24-bit LTC2448 sigma-delta ADC can be installed. The ADC is driven by SC520 via the high speed Sync Serial Port. The LTC2448 chip offers 8 ch. differential or 16 ch. single-ended input channels. Variable speed/resolution settings can be configured. A peak single-channel output rate of 5 KHz can be achieved.

The LTC2448 switches the analog input to a 2 pf capacitor at 1.8MHz with an equivalent input resistance of 110K ohm. The ADC works well directly with strain gages, current shunts, RTDs, resistive sensors, and 4-20mA current loop sensors. The ADC can also work well directly with thermocouples in the differential mode. Installing a precision reference with a internal temperature sensor(LT1019, 2.5V) provides local temperature measurement for thermocouple applications.

Protected resistor networkd are provided on-board to buffer the 16 channel inputs. In order to support both analog or digital inputs with variable, higher voltage level, user configurable protection resistor networks are designed into the board.



The valid input voltage at the ADC(LTC2448) is 0-1.25V with an on-board 2.5V precision reference. As default, 16 resistor dividers RP1-7 (20K+10K) are installed to provide valid analog input reading of 0-3.75V. Input voltage higher than 2.5V will be read by software as full scale.

The software source sample code on TERN CD, c:\tern\586\samples\5d\5d\_ad24.c, allows user to modify the input reading resolution. For digital inputs, only one byte reading is needed.



#### 3.2.12 SC520 PIOs

The SC520 supports 32 user-programmable I/O lines (PIO). Each of these pins can be used as a user-programmable input or output signal, if the interface function is not needed.

The 586-Drive PIO pins are 3.3V output and all inputs are 5V tolerant. Absolutely no voltage greater than 5V should be applied to any pins. Over 5V voltage input can cause permanent damage.

After power-on/reset, PIO pins default to various configurations. The initialization routine, sc\_init();, provided by TERN libraries reconfigures some of these pins as needed as:

P27=/GPCS0=J2.37 for 16-bit I/O operation.

P31=J2.38 as input for STEP2 jumper reading

P0 as output for on-board LED control

P1=/GPBHE as /BHE for 16-bit SRAM data bus high byte enable signal

P2 is used for on-board EEPROM(U0, SDA) and AD7655(U6, A0).

P3 is used by 24-bit ADC(LTC2448, Busy)

P4 is used as EEPROM SCL.

P5-P11 is for HV U19.8-2

P12=J2.12=J0.7(i2x DAT)

P13=J2.13=J0.1(i2x CLK)

P14=J2.14

Some PIO pins on the J2 header are free to use. PIO lines can be configured to operate as an output or an input with a weak internal pull-up or pull-down resistor. A PIO pin's behavior, either pull-up or pull-down, is pre-determined and shown in the table below.

These configurations, as well as the processor-internal peripheral usage configurations, are listed below in Table 3.1.

| PIO | Function  | Power-On/Reset status | 586-Drive Pin No.               | 586-Drive Initial      |
|-----|-----------|-----------------------|---------------------------------|------------------------|
| P0  | GPALE     | Input with pull-up    | LED L1 pin 2                    | Output for LED control |
| P1  | /GPBHE    | Input with pull-up    | SRAM U9.40                      | High byte enable /BHE  |
| P2  | GPRDY     | Input with pull-up    | EE(U0.5, SDA)<br>ADC7655(U5.3,  | Input with pull-up     |
| Р3  | GPAEN     | Input with pull-up    | A0)<br>LTC2448<br>(U22.2, Busy) | Input with pull-up     |
| P4  | GPTC      | Input with pull-up    | EE(U0.6, SCL)                   | Input with pull-up     |
| P5  | GPDRQ3    | Input with pull-down  | HV(U19.8)                       | Input with pull-down   |
| P6  | GPDRQ2    | Input with pull-down  | HV(U19.7)                       | Input with pull-down   |
| P7  | GPDRQ1    | Input with pull-down  | HV(U19.6)                       | Input with pull-down   |
| P8  | GPDRQ0    | Input with pull-down  | HV(U19.5)                       | Input with pull-down   |
| P9  | /GPDACK3  | Input with pull-up    | HV(U19.4)                       | Input with pull-up     |
| P10 | /GPDACK2  | Input with pull-up    | HV(U19.3)                       | Input with pull-up     |
| P11 | /GPDACK1  | Input with pull-up    | HV(U19.2)                       | Input with pull-up     |
| P12 | /GPDACK0  | Input with pull-up    | J2.12,J0.7(i2x DAT)             | Input with pull-up     |
| P13 | GPIRQ10   | Input with pull-up    | J2.13,J0.1(i2x CLK)             | Input with pull-up     |
| P14 | GPIRQ9    | Input with pull-up    | J2 pin 14                       | Input with pull-up     |
| P15 | GPIRQ8    | Input with pull-up    | J2 pin 16                       | Input with pull-up     |
| P16 | GPIRQ7    | Input with pull-up    | J2 pin 15                       | Input with pull-up     |
| P17 | GPIRQ6    | Input with pull-up    | J2 pin 18                       | Input with pull-up     |
| P18 | GPIRQ5    | Input with pull-up    | J2 pin 17                       | Input with pull-up     |
| P19 | GPIRQ4    | Input with pull-up    | OPTO 2                          | Input with pull-up     |
| P20 | GPIRQ3    | Input with pull-up    | OPTO 4                          | Input with pull-up     |
| P21 | GPIRQ2    | Input with pull-up    | OPTO 3                          | Input with pull-up     |
| P22 | GPIRQ1    | Input with pull-up    | OPTO 1                          | Input with pull-up     |
| P23 | GPIRQ0    | Input with pull-up    | J2 pin 33,                      | Input with pull-up     |
|     |           |                       | I2chip /INT=JP1.2               |                        |
| P24 | /GPBUFOE  | Input with pull-up    | J2 pin 24                       | Input with pull-up     |
| P25 | /GPIOCS16 | Input with pull-up    | J2 pin 25                       | Input with pull-up     |
| P26 | /GPMCS16  | Input with pull-up    | J2 pin 29                       | Input with pull-up*    |
| P27 | /GPCS0    | Input with pull-up    | J2 pin 37                       | 16-bit I/O operation   |
| P28 | /CTS2     | Input with pull-up    | J2 pin 36                       | Input with pull-up     |
| P29 | /DSR2     | Input with pull-up    | J2 pin 35                       | Input with pull-up     |
| P30 | /DCD2     | Input with pull-up    | J2 pin 30                       | Input with pull-up     |
| P31 | /RIN2     | Input with pull-up    | J2 pin 38                       | STEP2 Jumper           |

Table 3.1 I/O pin default configuration after power-on or reset

C function in the library **586.lib** can be used to initialize and to operate PIO pins. void *pio\_init*(char bit, char mode);

Where bit = 0-31 and mode = 0 (for interface function), 1 (for input), or 2 (for output).

Example: **pio\_init**(0, 2); will set P0 as output **pio\_init**(1, 0); will set P1 as /GPBHE

void *pio\_wr*(char bit, char dat);

```
pio\_wr(0,1); set P0 pin high and the LED is off, if P0 is in output mode pio\_wr(0,0); set P0 pin low and the LED is on, if P0 is in output mode
```

unsigned int pio\_rd(char port);

```
pio_rd (0); return 16-bit status of P0-P15, if corresponding pin is in input mode, pio_rd (1); returns 16-bit status of P16-P31, if corresponding pin is in input mode,
```

Some of the I/O lines are used or shared by the 586-Drive system for on-board components.

# Do not use these lines unless you are sure that you are not interfering with the operation of such components

#### 3.2.13 I/O Space

External I/O devices can use I/O mapping for access. You can access such I/O devices with *inportb*(port) or *outportb*(port,dat). These functions will transfer one byte or word of data to the specified I/O address. The external I/O space is 64K, ranging from 0x0000 to 0xffff.

The default GP bus timing is setup in sc\_init(); as:

```
pokeb(MMCR, _GPCSRT_, 0x01); // set the GP CS recovery time pokeb(MMCR, _GPCSPW_, 0x1f); // set the GP CS width pokeb(MMCR, _GPCSOFF_, 0x01); // set the GP CS offset pokeb(MMCR, _GPRDW_, 0x1f); // set the GP RD pulse width pokeb(MMCR, _GPRDOFF_, 0x0); // set the GP RD offset pokeb(MMCR, _GPWRW_, 0x1f); // set the GP WR pulse width pokeb(MMCR, _GPWROFF_, 0x0); // set the GP WR offset
```

User may modify the GP bus timing after sc\_init();. Details regarding this can be found in the SC520 User's Manual and SC520 Register Set Manual. Slower components, such as most LCD interfaces, might find the maximum programmable wait state of 15 cycles still insufficient.

The table below shows the I/O mapping for each peripheral.

| I/O space     | Select | Location      | Usage                    |
|---------------|--------|---------------|--------------------------|
| 0x2000-0x2004 | /AD    | U2.15 & U6.32 | AD7655 chip select       |
| 0x2020        | /CV    | U6.35         | Begin conversion on ADC  |
| 0x2040        | HIT    | H1.1          | Reset watchdog timer     |
| 0x2060-0x2074 | /SC1   | U17.14        | SCC2691 Enable           |
| 0x2080-0x2094 | /SC2   | U18.14        | SCC2691 Enable           |
| 0x20a0        | /L1    | U24.11        | 74HC273 Chip select      |
| 0x20c0        | /L2    | U25.11        | 74HC273 Chip select      |
| 0x20E0        | /CF    | U5.32         | CompactFlash chip select |

#### 3.2.14 Eight channel 16-bit DAC (LTC2600)

The LTC2600 is an eight channel 16-bit digital-to-analog converter (DAC) in an SO-8 package. It is complete with a rail-to-rail voltage output amplifier capable of driving up to 15mA. It uses a 3-wire SPI compatable serial interface and has an output range of 0-REF volts, making 1 LSB equal to REF/65535 V.

The reference voltage input is routed to J4 pin 11 and by default is shorted to VCC via jumper at J4.11=J4.9. The REF voltage must be greater than GND and less than VCC. The DAC outputs are routed to the J4 pin header, pins 13-20.

The DAC is installed on the 586-Drive at location U8 and uses /RTS0 as the chip select. The synchronous serial interface is used to send data to the device. Refer to the sample code, \ten\586\samples\5d\5d\_da.c for an example on driving the DAC.

The sample is also included in the pre-built sample project \tern\586\samples\5d\5d.ide.

Refer to the DAC data sheet for additional specifications; \tern\_docs\parts\ltc2600.pdf.

#### 3.2.15 Four channel, 16-bit ADC

The unique 16-bit parallel ADC (AD7655, 0-5V) supports ultra high-speed (1 MHz conversion rate) analog signal acquisition. The AD7655 contains two low noise, high bandwidth track-and-hold amplifiers that allow *simultaneous* sampling on two channels. Each track-and hold amplifier has a multiplexer in front to provide a total of 4 channels analog inputs. The parallel ADC achieves very high throughput by requiring only two CPU I/O operations (one start, one read) to complete a 16-bit ADC reading. With a precision external 2.5V reference, the ADC accepts 0-5V analog inputs at 16-bit resolution of 0-65,535.

See sample program \tern\586\samples\5d\5d\_ad.c for details on reading the ADC. The sample program is also included in the pre-built sample project; \tern\586\samples\5d\5d.ide.

Refer to the data sheet for additional specifications; \tern docs\parts\ad7655.pdf.

#### 3.2.16 UART SCC2691

There are two UART SCC2691 chips (U17,U18), which are mapped into the 8-bit I/O address space at 0x2060-0x2074 and 0x2080-0x2094. The SCC2691 has a full-duplex asynchronous receiver/transmitter, a quadruple buffered receiver data register, an interrupt control mechanism, programmable data format, selectable baud rate for the receiver and transmitter, a multi-functional and programmable 16-bit counter/timer, an on-chip crystal oscillator, and a multi-purpose input/output including RTS and CTS mechanism.

The U17 chip uses signals /TXDA and /RXDA (routed to H4.2,H4.3 respectively), while the U18 chip uses signals /TXDB and /RXDB (routed to H5.2,H5.3 respectively). The UART in location U17 is RS485 configurable, if the option is desired. By default, both UARTs use RS232 drivers.

See sample programs  $\sc 5d\sc 1.c$  and  $\sc 1.c$  and  $\sc 1.c$  and  $\sc 1.c$  for details on activating both UARTs. The sample programs are also included in the pre-built sample project;  $\sc 1.c$  and  $\sc 1.c$ 

#### 3.2.17 **EEPROM**

A serial EEPROM of 512 bytes (24C04), or 2K bytes (24C16) can be installed in U0. The 586-Drive uses the P4=SCL (serial clock) and P2=SDA (serial data) to interface with the EEPROM. The EEPROM can be used to store important data such as a node address, calibration coefficients, and configuration codes. It typically has 1,000,000 erase/write cycles. The data retention is more than 40 years. EEPROM can be read and written by simply calling the functions ee\_rd() and ee\_wr().

#### 3.2.18 Protective high voltage inputs

In order to support high voltage digital signal input up to 30V, a Darlington Transistor Array (ULN2003A) can be in U19. The maximum input voltage is 30V. The input pin has 12.7K resistance load to the GND. You have to provide a pulled high signal input. A valid input low voltage is less than 0.8V, while a valid input high voltage is higher than 3V and less than 30V.



Figure 3.1 Darlington Transistors used as Protective High Voltage Inputs.

U19 may be set as input or output. By factory default, U19 is input. The input and output orientation for the ULN2003A chip in the U19 socket are different.

Seven high voltage inputs cab be available on T2: O2-O8 while the ULN2003A is installed in U19 as input mode:

ULN2003A is a 16-pin DIP, U19 is a 18-pin socket.

Install ULN2003A to register ULN2003A pin 1=U19.11

Force T2.1=O1 to GND first!!

T2.2=O2 to P11

T2.3=O3 to P10

T2.4=O4 to P9

T2.5=O5 to P8

12.5-05 10 F

T2.6=O6 to P7 T2.7=O7 to P6

T2.8=O8 to P5

Input up to 24 VDC voltage on screw terminal O2-O8, and read PIO line P11-P5.

While O2=24V, P11=low; O2=0V, P11=high. In addition, the ULN2003A chip may be replaced with a resistor pack to provide digital inputs or outputs to the terminal blocks.



Figure 3.2 Locations of user configurable Darlington Transistor Arrays.

J5.3 (VS) = J5.4 (GND)

#### 3.2.19 High-Voltage, High-Current Drivers

ULN2003A has high voltage, high current Darlington transistor arrays, consisting of seven silicon NPN Darlington pairs on a common monolithic substrate. All channels feature open-collector outputs for sinking 350 mA at 50V, and integral protection diodes for driving inductive loads. Peak inrush currents of up to 600 mA sinking are allowed. By default, U21 and U20 are high-voltage driver outputs and U19 provides high-voltage inputs.



Figure 3.3 Default High Voltage Driver Configuration (U19 input; U20,U21 output).

These outputs may be paralleled to achieve high-load capability, although each driver has a maximum continuous collector current rating of 350 mA at 50V. The maximum power dissipation allowed is 2.20 W per chip at 25 degrees C (°C). The common substrate G is routed to T2 GND pins. All currents sinking in must return to the T2 GND pin. A heavy gauge (20) wire must be used to connect the T2 GND terminal to an external common ground return. K connects to the protection diodes in the ULN2003A chips and should be tied to highest voltage in the external load system. K can be connected to an unregulated on board +12V via J5 pin 1 and pin 2. ULN2003A is a *sinking* driver. An example of typical application wiring is shown below.



Figure 3.4 Drive inductive load with high voltage/current drivers.

<u>Sourcing Option</u>: U20 and U21 are also configurable as sourcing high-voltage inputs. See **Figure 3.5** for visual reference. This sourcing option MUST use UDN2982A sourcing chips, instead of the ULN2003A sinking ones. Also, note the new jumper settings on J5.



Figure 3.5 Optional High Voltage Driver Configuration (U20,U21 sourcing inputs).

#### 3.2.20 Optional 192x128 pixel LCD

The 5D has a 20 pin Flex cable connector U31 to interface a 192x128 graphics LCD.

Since this interface is located underneath solenoid driver sockets U20 and U21, these two chips <u>cannot</u> be installed with the LCD option. Also note that T2 screw terminal block <u>cannot</u> be installed as well to allow for physical ease on the cable and POT.

For contrast adjustment, the LCD uses a POT (P7). See Figure 3.6.

For more information regarding the 5D-LCD interface, see sample **5d\_lcd.c** in project **c:\tern\586\samples\5d**.



Figure 3.6 LCD interface & Contrast POT. Sockets U20, U21 cannot be installed.



Figure 3.7 586D controller with LCD cable interface.

#### 3.2.21 Opto-coupler Inputs

There are 4 opto-couplers available on the 5D.

Positive inputs OT1+ to OT4+ are routed to T1.25=OT1+, T1.24=OT2+, T1.22=OT3+, T1.21=OT4+. Negative inputs are shared in pairs, however. Input OTA- (T1.26) corresponds to the OT1+ and OT2+ opto-couplers, while input OTB- (T1.23) corresponds to the OT3+ and OT4+ optos.

Inputs are protected by current limiting resistors installed at RP9. Inputs CPU-side are read through four Port-1 PIO lines (P19-22), each corresponding to an opto-coupler.

See \tern\samples\586d\5d opto.c for more details.

#### 3.2.22 100 MHz BaseT Ethernet

An WizNet<sup>TM</sup> Fast Ethernet Module can be installed to provide 100M Base-T network connectivity. This Ethernet module has a hardware LSI TCP/IP stack. It implements TCP/IP, UDP, ICMP and ARP in hardware, supporting internet protocol DLC and MAC. It has 16KB internal transmit and receiving buffer which is mapped into host processor's direct memory. The host can access the buffer via high speed DMA transfers. The hardware Ethernet module releases internet connectivity and protocol processing from the host processor. It supports 4 independent stack connections simultaneously at a 4Mbps protocol processing speed. An RJ45 8-pin connector is on-board for connecting to 10/100 Base-T Ethernet network. A software library is available for Ethernet connectivity. Lines used by the Ethernet device include P23 and Timer 0 output TOUT0. These should not be user-activated if Ethernet option is installed.



Figure 3.8 WizNet™ Ethernet Module

#### 3.3 Power supplies and Supervisor with Watchdog Timer

Two supervisor chips monitor 5V and 3.3V and provide power failure detection, a watchdog and system reset. The 2.5V power supply is used for the SC520 core and 3.3V supports SC520 I/O operation. Signal lines on headers are 3.3V output, and 5V maximum input. Absolutely no voltage greater than 5V should be applied to any pins. The 388 pin BGA package of SC520 makes repair support not available. All components are soldered on board for highest reliability.

The **586-Drive**<sup>TM</sup> can be powered with a single regulated 5V with the on-board 3.3V and 2.5V regulators. The 586-Drive also includes an on-board switching regulator to provide 5V from an unregulated 8-30V input.

A 691 (U7) supervisor chip is used to monitor the 5V power and a MIC8114 (U4) is designed to monitor the 3.3V. The supervisor provides a watchdog timer, battery backup, power-on-reset delay, power-supply monitoring, and power-failure warning. These will significantly improve system reliability.

#### **Watchdog Timer**

Setting a jumper on H1 activates the 691 watchdog timer. The watchdog timer provides a means of verifying proper software execution. In the user's application program, calls to the function **inport(0x2040)**; that toggles the H1 pin1= HIT, should be arranged such that the HIT pin is accessed at least once every 1.6 seconds. If the H1 jumper is installed and the HIT pin is not accessed within this timeout period, the watchdog timer pulls the WDO pin low, and asserts /RST. This automatic assertion of /RST may recover the application program if something is wrong. When controllers are shipped from the factory the H1 jumper is off, which disables the watchdog timer.

The SC520's internal watchdog timer is disabled by default with sc\_init().

#### **Battery Backup Protection**

The backup battery protection protects data stored in the SRAM and RTC. The battery-switch-over circuit compares VCC to VBAT (+3 V lithium battery positive pin), and connects whichever is higher to the VRAM (power for SRAM and RTC). Thus, the SRAM and the real-time clock RTC72423 are backed up. In normal use, the lithium battery should last about 3-5 years without external power being supplied. When the external power is on, the battery-switch-over circuit will select the VCC to connect to the VRAM.

### 3.4 Headers and Connectors

#### 3.4.1 Expansion Headers J1 and J2

There are two 20x2 0.1 spacing headers for 586-Drive expansion. Most signals are directly routed to the SC520 processor. These signals are 3.3V and 5V tolerant, and any out-of-range voltages will damage the board.

| J2 Signal |    |    |        |
|-----------|----|----|--------|
| GND       | 40 | 39 | VCC    |
| P31       | 38 | 37 | P27    |
| P28       | 36 | 35 | P29    |
| TXD0      | 34 | 33 | P23    |
| RXD0      | 32 | 31 |        |
| P30       | 30 | 29 | P26    |
| TXD1      | 28 | 27 |        |
| RXD1      | 26 | 25 | P25    |
| P24       | 24 | 23 |        |
|           | 22 | 21 |        |
|           | 20 | 19 |        |
| P17       | 18 | 17 | P18    |
| P15       | 16 | 15 | P16    |
| P14       | 14 | 13 | P13    |
| P12       | 12 | 11 | SSI    |
|           | 10 | 9  | SSO    |
| TOUT0     | 8  | 7  | SSC    |
| /INTA     | 6  | 5  | TOUT1  |
|           | 4  | 3  | /PITO2 |
| GND       | 2  | 1  | TIN1   |

| J1 Signal |    |    |      |
|-----------|----|----|------|
| VCC       | 1  | 2  | GND  |
| /INTB     | 3  | 4  | CLKT |
|           | 5  | 6  | GND  |
|           | 7  | 8  | D0   |
|           | 9  | 10 | D1   |
|           | 11 | 12 | D2   |
| D15       | 13 | 14 | D3   |
| /RST      | 15 | 16 | D4   |
| RST       | 17 | 18 | D5   |
| TIN0      | 19 | 20 | D6   |
| D14       | 21 | 22 | D7   |
| D13       | 23 | 24 | GND  |
|           | 25 | 26 | A7   |
| D12       | 27 | 28 | A6   |
| /IOWR     | 29 | 30 | A5   |
| /IORD     | 31 | 32 | A4   |
| D11       | 33 | 34 | A3   |
| D10       | 35 | 36 | A2   |
| D9        | 37 | 38 | A1   |
| D8        | 39 | 40 | A0   |

#### 3.4.2 Header J4

The J4 (10x2 pins) headers on the 586-Drive provides the user with access to DAC outputs, ADC inputs. See the schematic (5d-man.sch) attached in the last page of this manual for details.

|      | J4 | Signal |     |
|------|----|--------|-----|
| AA1  | 1  | 2      | GND |
| AA2  | 3  | 4      | GND |
| AB2  | 5  | 6      | GND |
| AB1  | 7  | 8      | GND |
| VCC  | 9  | 10     | REF |
| REF1 | 11 | 12     |     |
| V2   | 13 | 14     | V3  |
| V1   | 15 | 16     | V4  |
| V7   | 17 | 18     | V6  |
| V8   | 19 | 20     | V5  |

#### 3.4.3 Headers T1 and T2

Both these headers are 26x1 screw terminals. T1 provides 16 ADC inputs and 4 opto-coupler inputs. T2 provides 7 HV inputs and 14 HV outputs. See schematic for more details. Default ULN2003A chips are installed in U20 and U21 to provide 14 high voltage sinking drivers at O10-O16 and O18-O24. The O9 and O17 lines are not used for sinking drivers.

Optionally, two UDS2892 chips can be installed in U20 and U21 to provide 16 high voltage sourcing drivers. The O9 and O17 lines will be used here for sourcing.



Figure 3.9 Default 586-Drive T1 & T2 Header Configuration.

# **Chapter 4: Software**

Please refer to the Technical Manual of the "C/C++ Development Kit for TERN 16-bit Embedded Microcontrollers" for details on debugging and programming tools.

#### Guidelines, awareness, and problems in an interrupt driven environment

Although the C/C++ Development Kit provides a simple, low cost solution to application engineers, some guidelines must be followed. If they are not followed, you may experience system crashes, PC hang-ups, and other problems.

The debugging of interrupt handlers with the Remote Debugger can be a challenge. It is possible to debug an interrupt handler, but there is a risk of experiencing problems. Most problems occur in multi-interrupt-driven situations. Because the remote kernel running on the controller is interrupt-driven, it demands interrupt services from the CPU. If an application program enables interrupt and occupies the interrupt controller for longer than the remote debugger can accept, the debugger will time-out. As a result, your PC may hang-up. In extreme cases, a power reset may be required to restart your PC.

For your reference, be aware that our system is remote kernel interrupt-driven for debugging.

The run-time environment on TERN controllers consists of an I/O address space and a memory address space. I/O address space ranges from **0x00000** to **0xffff**, or 64 KB. Memory address space ranges from **0x00000** to **0xfffff** in real-mode, or 1 MB. These are accessed differently, and not all addresses can be translated and handled correctly by hardware. I/O and memory mappings are done in software to define how translations are implemented by the hardware. Implicit accesses to I/O and memory address space occur throughout your program from TERN libraries as well as simple memory accesses to either code or global and stack data. You can, however, explicitly access any address in I/O or memory space, and you will probably need to do so in order to access processor registers and on-board peripheral components (which often reside in I/O space) or non-mapped memory.

This is done with four different sets of similar functions, described below.

#### poke/pokeb

Arguments: unsigned int segment, unsigned int offset, unsigned int/unsigned char data

Return value: none

These standard C functions are used to place specified data at any memory space location. The **segment** argument is left shifted by four and added to the **offset** argument to indicate the 20-bit address within memory space. **poke** is used for writing 16 bits at a time, and **pokeb** is used for writing 8 bits.

The process of placing data into memory space means that the appropriate address and data are placed on the address and data-bus, and any memory-space mappings in place for this particular range of memory will be used to activate appropriate chip-select lines and the corresponding hardware component responsible for handling this data.

#### peek/peekb

Arguments: unsigned int segment, unsigned int offset

Return value: unsigned int/unsigned char data

These functions retrieve the data for a specified address in memory space. Once again, the **segment** address is shifted left by four bits and added to the **offset** to find the 20-bit address. This address is then output over the address bus, and the hardware component mapped to that address should return either an

8-bit or 16-bit value over the data bus. If there is no component mapped to that address, this function will return random garbage values every time you try to peek into that address.

#### outport/outportb

Arguments: unsigned int address, unsigned int/unsigned char data

Return value: none

This function is used to place the **data** into the appropriate **address** in I/O space. It is used most often when working with processor registers that are mapped into I/O space and must be accessed using either one of these functions. This is also the function used in most cases when dealing with user-configured peripheral components.

When dealing with processor registers, be sure to use the correct function. Use **outport** if you are dealing with a 16-bit register.

#### inport/inportb

**Arguments:** unsigned int address

Return value: unsigned int/unsigned char data

This function can be used to retrieve data from components in I/O space. You will find that most hardware options added to TERN controllers are mapped into I/O space, since memory space is valuable and is reserved for uses related to the code and data. Using I/O mappings, the address is output over the address bus, and the returned 16 or 8-bit value is the return value.

For a further discussion of I/O and memory mappings, please refer to the Hardware chapter of this technical manual.

#### 4.1 586.LIB

586.LIB is a C library for basic 586Drive operations. It includes the following modules: 586.OBJ, SER0.OBJ, SER1.OBJ, SCC.OBJ. You need to link 586.LIB in your applications and include the corresponding header files. The following is a list of the header files:

| Include-file name | Description                            |
|-------------------|----------------------------------------|
| 586.H             | timer/counter, ADC, DAC, RTC, Watchdog |
| SER0.H            | Internal serial port 0/2               |
| SER1.H            | Internal serial port 1                 |
| SCC.H             | External UART SCC2691                  |

The 586.LIB was originally developed for the 586-Engine (the predecessor to the 586-Engine-D). Function prototypes for the 586-Engine's ADC and DAC were incorporated into 586.lib. The 586Drive does not uses these same routines for its analog I/O. You must refer to the sample code in the 586\samples\5d directory for examples on accessing the analog I/O on the 586Drive.

#### 4.2 Functions in 586.OBJ

#### 4.2.1 586Drive Initialization

#### sc init

586-Drive

This function should be called at the beginning of every program running on 586Drive. It provides default initialization and configuration of the various I/O pins, interrupt vectors, sets up I/O, and provides other processor-specific updates needed at the beginning of every program.

There are certain default pin modes and interrupt settings you might wish to change. With that in mind, the basic effects of **sc\_init** are described below. For details regarding register use, you will want to refer to the AMD SC520 Microcontroller User's manual.

- Initialize the programmable interrupt controller. Setup the master interrupt controller at vector 0x40. The slave1 interrupt vector at 0x48, and slave 2 interrupt vector at 0x50.
- Initialize /ROMCS1 chip select to support the 8-bit I/O starting 0x1000, and the /GPCS0 to support 16-bit I/O starting address at 0x1800.
- Disable SDRAM
- Initialize default GP-bus chip select timing as

```
pokeb (MMCR, _GPCSRT_, 0x01); // set the GP CS recovery time pokeb (MMCR, _GPCSPW_, 0x1f); // set the GP CS width pokeb (MMCR, _GPCSOFF_, 0x01); // set the GP CS offset pokeb (MMCR, _GPRDW_, 0x1f); // set the GP RD pulse width pokeb (MMCR, _GPRDOFF_, 0x0); // set the GP RD offset pokeb (MMCR, _GPWRW_, 0x1f); // set the GP WR pulse width pokeb (MMCR, _GPWROFF_, 0x0); // set the GP WR offset
```

• Initialize and configure PIO ports for default operation. All pins are set up as default input, except for P31, P27, P2, and P0.

The GP chip selects are set to 0x1f wait states, by default. This makes it possible to interface with many slower external peripheral components. If you require faster I/O access, you can modify this number down as needed.

A CLKT signal is routed to J1 pin 4 for the external user clock. The CLKT can be selected as

```
void clkt_sel
Arguments: unsigned char clk
Return value: none.

The CLKT pin is programmed as an output (CLKTEST).
When programmed as output, CLKT output one of the 6 internal clocks:

void clkt_sel(unsigned char clk);

where:
clk=000, RTC (32.768 KHz)
clk=001, UART (1.8443 MHz)
clk=010, UART (18.432 MHz)
clk=011, PIT (1.1882 MHz)
clk=100, PLL1 (1.47456 MHz)
clk=101, PLL2 (36.864 MHz)
clk=101, PLL2 (36.864 MHz)
clk=110-111, CLKT=0
```

#### 4.2.2 External Interrupt Initialization

The programmable interrupt controller consistes of a system of three individual interrupt controllers (Master, Slave1, and Slave2), each has eight interrupt channels. A total of 22 interrupt priority levels are supported. A programmable interrupt router handles routing of various external and internal interrupt sources to the 22 interrupt channels. TERN recommends an interrupt map as follows;

```
There are 22 interrupt priority levels plus NMI
There are 15 external interrupt requests (GPIRQ0-10, /INTA-D). You must
provide a low to high edge to generate an interrupt
Example internal interrupt map by TERN:
       P1=Master PIC IR0, interrupt vector=0x40, PIT timer0
        P2=Master PIC IR1, interrupt vector=0x41, GPIRQ0=PIO23=J2.33
       P3=slave1 PIC IR0, interrupt vector=0x48, RTC
        P4=slave1 PIC IR1, interrupt vector=0x49, GPIRO1=PIO22=J2.23
        P5=slave1 PIC IR2, interrupt vector=0x4a, GPIRQ2=PIO21=J2.21
        P6=slave1 PIC IR3, interrupt vector=0x4b, GPIRQ3=PIO20=J2.19
       P7=slave1 PIC IR4, interrupt vector=0x4c, GPIRO4=PIO19=J2.20
       P8=slave1 PIC IR5, interrupt vector=0x4d, FPU
        P9=slave1 PIC IR6, interrupt vector=0x4e, /INTD=SCC=J3.14
        P10=slave1 PIC IR7, interrupt vector=0x4f, GP timer1/INTC=J3.13
       P11=Master PIC IR3, interrupt vector=0x43, SER2/0
        P12=Master PIC IR4, interrupt vector=0x44, SER1
       P13=Slave2 PIC IR0, interrupt vector=0x50, GP timer0
        P14=Slave2 PIC IR1, interrupt vector=0x51, GPIRO5=PIO18=J2.17
        P15=Slave2 PIC IR2, interrupt vector=0x52, GPIRQ6=PIO17=J2.18
        P16=Slave2 PIC IR3, interrupt vector=0x53, GPIRQ7=PIO16=J2.15
        P17=Slave2 PIC IR4, interrupt vector=0x54, PIT timer1
        P18=Slave2 PIC IR5, interrupt vector=0x55, GPIRQ8=PIO15=J2.16
        P19=Slave2 PIC IR6, interrupt vector=0x56, GPIRO9=PIO14=J2.6
       P20=Slave2 PIC IR7, interrupt vector=0x57, GPIRO10=PIO13=J2.8
       P21=Master PIC IR6, interrupt vector=0x46, PIT Timer2/INTB=J3.12
       P22=Master PIC IR7, interrupt vector=0x47, /INTA=J3.11
```

A spurious interrupt is defined as a "Not Valid" interrupt.

A Spurious Interrupt on any IR line generates the same vector number as an IR7 request. The spurious interrupt, however, does not set the

in-service bit for IR7. Therefore, an IR7 isr must check the isr register to determine the interrupt source was a

valid IR7 (the in-service bit is set),

or a spurious interrupt (the in-service bit is cleared)

#### **Functions**

```
void
         nmi init(void); // nmi interrupt handler initialization
         int0 init(char i, void interrupt far (* int0 isr)())
void
         int1 init(char i, void interrupt far (* int1 isr)())
void
         int2 init(char i, void interrupt far (* int2 isr)())
void
void
         int3 init(char i, void interrupt far (* int3 isr)())
         int4 init(char i, void interrupt far (* int4 isr)())
void
         int5 init(char i, void interrupt far (* int5 isr)())
void
         int6 init(char i, void interrupt far (* int6 isr)())
void
void
         int7 init(char i, void interrupt far (* int7 isr)())
void
         int8 init(char i, void interrupt far (* int8 isr)())
         int9 init(char i, void interrupt far (* int9 isr)())
void
```

void intD\_init(char i, void interrupt far (\* intD\_isr)())

For a detailed discussion involving the interrupt, the user should refer to Chapter 15 of the AMD SC520 Microcontroller User's Manual.

TERN provides functions to enable/disable all of the external interrupts. The user can call any of the interrupt init functions listed for this purpose. The first argument indicates whether the particular interrupt should be enabled, and the second is a function pointer to an appropriate interrupt service routine that should be used to handle the interrupt. The TERN libraries will set up the interrupt vectors correctly for the specified external interrupt line.

At the end of interrupt handlers, the appropriate in-service bit for the IR signal currently being handled must be cleared. This can be done using the **Nonspecific EOI command**. At initialization time, interrupt priority was placed in **Fully Nested** mode. This means the current highest priority interrupt will be handled first, and a higher priority interrupt will interrupt any current interrupt handlers. So, if the user chooses to clear the in-service bit for the interrupt currently being handled, the interrupt service routine just needs to issue the nonspecific EOI command to clear the current highest priority IR.

void intx\_init

Arguments: unsigned char i, void interrupt far(\* intx isr) ())

Return value: none

These functions can be used to initialize any one of the external interrupt channels (for pin locations and other physical hardware details, see the Hardware chapter). The first argument i indicates whether this particular interrupt should be enabled or disabled. The second argument is a function pointer, which will act as the interrupt service routine. The overhead on the interrupt service routine, when executed, is about  $20 \, \mu s$ .

By default, the interrupts are all disabled after initialization. To disable them again, you can repeat the call but pass in 0 as the first argument.

#### I/O Initialization

Two ports of 16 I/O pins each are available on the 586Drive. Hardware details regarding these PIO lines can be found in the Hardware chapter.

Several functions are provided for access to the PIO lines. At the beginning of any application where you choose to use the PIO pins as input/output, you will probably need to initialize these pins in one of the three available modes. Before selecting pins for this purpose, make sure that the peripheral mode operation of the pin is not needed for a different use within the same application.

You should also confirm the PIO usage that is described above within **sc\_init()**. During initialization, several lines are reserved for TERN usage and you should understand that these are not available for your application. There are several PIO lines that are used for other on-board purposes. These are all described in some detail in the Hardware chapter of this technical manual. For a detailed discussion on the I/O ports, please refer to Chapter 23 of the AMD SC520 User's Manual.

Please see the sample program **586\_pio.c** in **tern\586\samples\5e**. You will also find that these functions are used throughout TERN sample files, as most applications do find it necessary to re-configure the PIO lines.

The function **pio\_wr** and **pio\_rd** can be quite slow when accessing the PIO pins. Depending on the pin being used, it might require from 5-10 us. The maximum efficiency you can get from the PIO pins occur if you instead modify the PIO registers directly with an **poke** instruction Performance in this case will be around 1-2 us to toggle any pin. For example: poke(MMCR, PIOSET15 0, m);

void pio init

**Arguments:** char bit, char mode

Return value: none

**bit** refers to any one of the 32 PIO lines, 0-31. **mode** refers to one of 3 modes of operation.

• 0, Interface operation

- 1, input with pullup/down
- 2, output

## unsigned int pio\_rd:

**Arguments:** char port

**Return value:** byte indicating PIO status

Each bit of the returned 16-bit value indicates the current I/O value for the PIO pins in the selected port.

void pio\_wr:

**Arguments:** char bit, char dat

Return value: none

Writes the passed in dat value (either 1/0) to the selected PIO.

#### 4.2.3 GPTimer Units

The three GP timers present on the 586Drive can be used for a variety of applications

These timers are controlled and configured through a mode register that is specified using the software interfaces. The mode register is described in detail in chapter 17 of the AMD SC520 User's Manual.

Two of the timers, Timer0 and Timer1 has external pulses output and counter inputs.

It is also possible to use the output of **Timer2** to pre-scale one of the other timers, since 16-bit resolution at the maximum clock rate specified gives you only 150 Hz. Only by using **Timer2** can you slow this down even further. The sample files *timer02.c* and *timer12.c*, located in *tern\586\samples\5e*, demonstrate this.

# void t0\_init void t1 init

**Arguments:** int tm, int ta, int tb, void interrupt far(\*t isr)()

Return values: none

Both of these timers have two maximum counters (MAXCOUNTA/B) available. These can all be specified using **ta** and **tb**. The argument **tm** is the value that you wish placed into the **T0CON/T1CON** mode registers for configuring the two timers.

The interrupt service routine  $\mathbf{t}$ \_isr specified here is called whenever the full count is reached, with other behavior possible depending on the value specified for the control register.

void t2 init

**Arguments:** int tm, int ta, void interrupt far(\*t\_isr)()

Return values: none.

Timer2 behaves like the other timers, except it only has one max counter available.

#### 4.2.4 Other library functions

#### On-board supervisor MAX691 or LTC691

The watchdog timer offered by the MAX691 or LTC691 offers an excellent way to monitor improper program execution. If the watchdog timer (H1) jumper is set, the function **hitwd()** must be called every 1.6 seconds of program execution. If this is not executed because of a run-time error, such as an infinite loop or stalled interrupt service routine, a hardware reset will occur.

```
void hitwd
Arguments: none
Return value: none
Resets the supervisor timer for another 1.6 seconds.

void led
Arguments: int ledd
Return value: none

Turns the on-board LED on or off according to the value of ledd.
```

#### Real-Time Clock

A real-time clock is included in the SC520, and can be used to keep track of real time. Backed up by a lithium-coin battery, the real time clock can be accessed and programmed using two interface functions.

There are two common data structure used to access and use both interfaces.

```
// SC520 RTC data structure
typedef struct{
        unsigned char sec;
        unsigned char alarm sec;
        unsigned char min;
        unsigned char alarm min;
        unsigned char hour;
        unsigned char alarm hour;
        unsigned char day week;
        unsigned char day month;
        unsigned char month:
        unsigned char year;
} RTCTIME;
// Real time data structure
typedef struct{
        unsigned char sec1; One second digit.
        unsigned char sec10; Ten second digit.
        unsigned char min1; One minute digit.
        unsigned char min10; Ten minute digit.
        unsigned char hour1; One hour digit.
        unsigned char hour 10; Ten hour digit.
        unsigned char day1; One day digit.
        unsigned char day 10; Ten day digit.
        unsigned char mon1; One month digit.
        unsigned char mon10; Ten month digit.
        unsigned char year1; One year digit.
        unsigned char year 10; Ten year digit.
        unsigned char wk; Day of the week.
} TIM;
```

int rtc\_rd

**Arguments:** TIM \*r

Return value: int error code

This function places the current value of the real time clock within the argument  $\mathbf{r}$  structure. The structure should be allocated by the user. This function returns 0 on success and returns 1 in case of error, such as the clock failing to respond.

int rtc\_rds

Arguments: char\* realTime Return value: int error code

This function places a string of the current value of the real time clock in the char\* realTime.

The text string has a format of "year1000 year100 year10 year1 month10 month1 day10 day1 hour10

hour1 min10 min1 second10 second1". For example" 19991220081020" represents year 1999, December 20<sup>th</sup>, Eight o'clock, 10 minutes, and 20 seconds.

This function returns 0 on success and returns 1 in case of error, such as the clock failing to respond.

Void rtc init

Arguments: char\* t, RTCTIME \*rtcp

Return value: none

This function is used to initialize and set a value into the real-time clock. The argument **t** should be a null-terminated byte array that contains the new time value to be used.

The RTCTIME data structure will be initialized based on the string t.

The byte array should correspond to { weekday, year10, year1, month10, month1, day10, day1, hour10, hour1, minute10, minute1, second10, second1, 0 }.

If, for example, the time to be initialized into the real time clock is June 5, 1998, Friday, 13:55:30, the byte array would be initialized to:

unsigned char  $t[14] = \{ 5, 9, 8, 0, 6, 0, 5, 1, 3, 5, 5, 3, 0 \};$ 

#### Delay

In many applications it becomes useful to pause before executing any further code. There are functions provided to make this process easy. For applications that require precision timing, you should use hardware timers, (Software Timer of SC520) provided on-board for this purpose.

### void delay0

**Arguments:** unsigned int t **Return value:** none

This function is just a simple software loop. The actual time that it waits depends on processor speed as well as interrupt latency. The code is functionally identical to:

While(t) { t--; }

Passing in a t value of 600 causes a delay of approximately 1 ms.

void delay ms

**Arguments:** unsigned int **Return value:** none

This function is similar to delay0, but the passed in argument is in units of milliseconds instead of loop iterations. Again, this function is highly dependent upon the processor speed. NOT accurate at all.

void sc\_rst

**Arguments:** none **Return value:** none

This function is similar to a hardware reset, and can be used if your program needs to re-start the board for any reason. Depending on the current hardware configuration, this might either start executing code from the DEBUG ROM or from some other address.

# 4.3 Functions in SER0.OBJ/SER1.OBJ

The functions described in this section are prototyped in the header file **ser0.h** and **ser1.h** in the directory **tern\586\include**.

The internal asynchronous serial ports are functionally identical. SER0/2 is used by the DEBUG ROM provided as part of the TERN EV/DV-P software kits for communication with the PC. As a result, you will not be able to debug code directly written for serial port 0, but you can run it in STEP2.

Two asynchronous serial ports are integrated in the SC520: SER0/2 and SER1. Both ports have baud rates based on the system clock, and can operate at a maximum of 1.152 Mbaud.

By default, SER0 is used by the DEBUG ROM for application download/debugging in STEP 1. We will use SER1 as the example in the following discussion; any of the interface functions that are specific to SER1 can be easily changed into function calls for SER0. For details, you should see both chapter 21 of the SC520 Microprocessor User's Manual and the schematic of the 586Drive provided at the end of this manual. TERN interface functions make it possible to use one of a number of predetermined baud rates. These baud rates are achieved by specifying a divisor. The following table shows the function arguments that express each baud rate, to be used in TERN functions. These are based on a 33.333 MHz external crystal. Note: Only up to 115,200 BAUD has been tested in house. (10/25/00)

| Function Argument | Baud Rate |
|-------------------|-----------|
| 1                 | 300       |
| 2                 | 600       |
| 3                 | 2400      |
| 4                 | 4800      |
| 5                 | 7200      |
| 6                 | 9600      |
| 7                 | 14,400    |
| 8                 | 19,200    |
| 9                 | 38,400    |
| 10                | 57,600    |
| 11                | 115,200   |
| 12                | 114,000   |
| 13                | 192,000   |

4-9

| Function Argument | Baud Rate |
|-------------------|-----------|
| 14                | 288,000   |
| 15                | 576,000   |
| 16                | 1,152,000 |

Table 4.1 Baud rate values

After initialization by calling **sl\_init()**, SER1 is configured as a full-duplex interrupt-driven serial port and is ready to transmit/receive serial data at one of the specified 16 baud rates.

An input buffer, **ser1\_in\_buf** (whose size is specified by the user), will automatically store the receiving serial data stream into the memory. The user only has to check the buffer status with **serhit1()** and take out the data from the buffer with **getser1()**, if any. The input buffer is used as a circular ring buffer, as shown in Figure 4.1



Figure 4.1 Circular ring input buffer

The input buffer (**ibuf**), buffer size (**isiz**), and baud rate (**baud**) are specified by the user with **sl\_init()** with a default mode of 8-bit, 1 stop bit, no parity. After **sl\_init()** you can set up a new mode with different numbers for data-bit, stop bit, or parity by directly accessing the Serial Port 0/1 Control Register if necessary, as described in chapter 21 of the SC520 manual for asynchronous serial ports.

Due to the nature of high-speed baud rates and possible effects from the external environment, serial input data will automatically fill in the buffer circularly without stopping, regardless of overwrite. If the user does not take out the data from the ring buffer with <code>getser1()</code> before the ring buffer is full, new data will overwrite the old data without warning or control. Thus it is important to provide a sufficiently large buffer if large amounts of data are transferred. For example, if you are receiving data at 9600 baud, a 4-KB buffer will be able to store data for approximately four seconds.

However, it is always important to take out data early from the input buffer, before the ring buffer rolls over. You may designate a higher baud rate for transmitting data out and a slower baud rate for receiving data. This will give you more time to do other things, without overrunning the input buffer. You can use **serhit1()** to check the status of the input buffer and return the offset of the in\_head pointer from the in tail pointer. A return value of 0 indicates no data is available in the buffer.

You can use <code>getser1()</code> to get the serial input data byte by byte using FIFO from the buffer. The in\_tail pointer will automatically increment after every <code>getser1()</code> call. It is not necessary to suspend external devices from sending in serial data with /RTS. Only a hardware reset or <code>sl\_close()</code> can stop this receiving operation.

For transmission, you can use **putser1()** to send out a byte, or use **putsers1()** to transmit a character string. You can put data into the transmit ring buffer, **s1\_out\_buf**, at any time using this method. The transmit ring buffer address (**obuf**) and buffer length (**osiz**) are also specified at the time of initialization. The transmit interrupt service will check the availability of data in the transmit buffer. If there is no more data (the head and tail pointers are equal), it will disable the transmit interrupt. Otherwise, it will continue to take out the data from the out buffer, and transmit. After you call **putser1()** and transmit functions, you are free to do other tasks with no additional software overhead on the transmitting

operation. It will automatically send out all the data you specify. After all data has been sent, it will clear the busy flag and be ready for the next transmission.

The sample program **ser1\_0.c** demonstrates how a protocol translator works. It would receive an input HEX file from SER1 and translate every ':' character to '?'. The translated HEX file is then transmitted out of SER0. This sample program can be found in **tern\586\samples\5e**.

#### **Software Interface**

Before using the serial ports, they must be initialized.

There is a data structure containing important serial port state information that is passed as argument to the TERN library interface functions. The **COM** structure should normally be manipulated only by TERN libraries. It is provided to make debugging of the serial communication ports more practical. Since it allows you to monitor the current value of the buffer and associated pointer values, you can watch the transmission process.

The two serial ports have similar software interfaces. Any interface that makes reference to either **s0** or **ser0** can be replaced with **s1** or **ser1**, for example. Each serial port should use its own **COM** structure, as defined in **586.h**.

```
typedef struct
  unsigned char ready; /* TRUE when ready */
  unsigned char baud;
  unsigned char mode;
  unsigned char iflag; /* interrupt status
  unsigned char *in buf; /* Input buffer */
  unsigned char in mt; /* Input buffer FLAG */
unsigned char in full; /* input buffer full */
  unsigned char *out_buf; /* Output buffer */
  unsigned char out_full; /* Output buffer FLAG */
unsigned char out_mt; /* Output buffer MT */
  unsigned char tmso; // transmit macro service operation
  unsigned char rts;
  unsigned char dtr;
  unsigned char en485;
  unsigned char err;
  unsigned char node;
  unsigned char cr; /* scc CR register
  unsigned char slave;
  unsigned char slave;
unsigned int in_segm; /* input buffer segment */
unsigned int in_offs; /* input buffer offset */
unsigned int out_segm; /* output buffer segment */
unsigned int out_offs; /* output buffer offset */
  unsigned char byte delay; /* V25 macro service byte delay */
} COM;
Arguments: unsigned char b, unsigned char* ibuf, int isiz, unsigned char* obuf, int osiz, COM* c
Return value: none
```

This function initializes either SER0 or SER1 with the specified parameters. **b** is the baud rate value shown in Table 4.1. Arguments **ibuf** and **isiz** specify the input-data buffer, and **obuf** and **osiz** specify the location and size of the transmit ring buffer.

The serial ports are initialized for 8-bit, 1 stop bit, no parity communication.

There are a couple different functions used for transmission of data. You can place data within the output buffer manually, incrementing the head and tail buffer pointers appropriately. If you do not call one of the following functions, however, the driver interrupt for the appropriate serial-port will be disabled, which means that no values will be transmitted. This allows you to control when you wish the transmission of data within the outbound buffer to begin. Once the interrupts are enabled, it is dangerous to manipulate the values of the outbound buffer, as well as the values of the buffer pointer.

# putser*n*

Arguments: unsigned char outch, COM \*c

Return value: int return value

This function places one byte **outch** into the transmit buffer for the appropriate serial port. The return value returns one in case of success, and zero in any other case.

#### putsers*n*

**Arguments:** char\* str, COM \*c **Return value:** int return value

This function places a null-terminated character string into the transmit buffer. The return value returns one in case of success, and zero in any other case.

serhitn() should be called before trying to retrieve data.

#### serhit*n*

**Arguments:** COM \*c **Return value:** int value

This function returns 1 as value if there is anything present in the in-bound buffer for this serial port.

#### getser*n*

Arguments: COM \*c

Return value: unsigned char value

This function returns the current byte from **sn\_in\_buf**, and increments the **in\_tail** pointer. Once again, this function assumes that **serhitn** has been called, and that there is a character present in the buffer.

## getsersn

Arguments: COM c, int len, char\* str

Return value: int value

This function fills the character buffer **str** with at most **len** bytes from the input buffer. It also stops retrieving data from the buffer if a carriage return (ASCII: **0x0d**) is retrieved.

This function makes repeated calls to **getser**, and will block until **len** bytes are retrieved. The return **value** indicates the number of bytes that were placed into the buffer.

Be careful when you are using this function. The returned character string is actually a byte array terminated by a null character. This means that there might actually be multiple null characters in the byte array, and the returned **value** is the only definite indicator of the number of bytes read. Normally, we suggest that the **getsers** and **putsers** functions only be used with ASCII character strings. If you are working with byte arrays, the single-byte versions of these functions are probably more appropriate.

#### **Miscellaneous Serial Communication Functions**

One thing to be aware of in both transmission and receiving of data through the serial port is that TERN drivers only use the basic serial-port communication lines for transmitting and receiving data. Hardware flow control in the form of **CTS** (Clear-To-Send) and **RTS** (Ready-To-Send) is not implemented. There are, however, functions available that allow you to check and set the value of these I/O pins appropriate for whatever form of flow control you wish to implement. Before using these functions, you should once again be aware that the peripheral pin function you are using might not be selected as needed. For details, please refer to the SC520 User's Manual.

#### char sn cts(void)

Retrieves value of CTS pin.

void sn\_rts(char b)

Sets the value of **RTS** to **b**.

### **Completing Serial Communications**

After completing your serial communications, you can re-initialize the serial port with s1\_init(); to reset default system resources.

## sn close

Arguments: COM \*c Return value: none

This closes down the serial port, by shutting down the hardware as well as disabling the interrupt.

The asynchronous serial I/O ports available on the SC520 have many other features that might be useful for your application. If you are truly interested in having more control, please read Chapter 21 of the manual for a detailed discussion of other features available to you.

## 4.4 Functions / Routines not in 586.lib

## 4.4.1 4 channel 16-bit ADC, AD7655

The ADC is accessed using a few different control lines. The control lines are summarized below:

| Control Line | Description / Function                                          |
|--------------|-----------------------------------------------------------------|
| /CV          | Begin conversion. I/O address 0x2020                            |
| /AD          | Read conversion. 0x2000 for AB1, AB2 and 0x2004 for AA1 and AA2 |
| P2           | MUX select. P2 high selects AA2, AB2. P2 low selects AA1, AB1   |
| A2           | A2 = /AD&0x0004.                                                |

4-13

The following process must be followed to read the ADC:

```
pio_wr (2, hi_lo);
inport (CV);
wait for 2.5 μs
inport (ADA);
inport (ADB);
```

#### Where

| hi_lo | ADA    | ADB    | Read channel |
|-------|--------|--------|--------------|
| 0     | A2 = 1 |        | AA1          |
| 0     |        | A2 = 0 | AB1          |
| 1     | A2 = 1 |        | AA2          |
| 1     |        | A2 = 0 | AB2          |

## 4.4.2 8 channel 16-bit DAC

To access the DAC use the routine called 'da\_2600' found in the \tern\586\samples\5d\5d\_da.c sample program. You must copy the routine from the source file into your application, as it is not part of a library. The function 'da\_2600' takes two arguments. The first is the channel to write to, with the valid range 0-7. The second is the output value, ranging from 0-65535. See the sample program for additional details.

## 4.4.3 16 channel 24-bit ADC

## **Delta-Sigma ADC LTC2448**

The LTC2448 ADC (U22) provides 16 channels of 0-2.5V analog single-ended (24 differential) inputs (or **0-3.75V** input with 2.5V reference chip + default input-resistor divider network). It should be noted that with a 2.5V ref chip, the input at the ADC chip itself should be 1.25V. To account for a diverse range of user-input voltages, you may reconfigure the resistor dividers to fit your input needs (as long as the input voltage ADC-side is 1.25V). For details regarding the hardware configuration, see the Hardware chapter.

The following functions will drive the 24-bit ADC. The order of functions given here should be followed in actual implementation.

```
void ad24_init(void);
void ad24_setup(unsigned char chip, unsigned int control_byte);
void ad24_ssi_rd(unsigned char* raw);
```

The control byte, *control\_byte*, drives the LTC2448 in 16 channel single-ended mode with value 0xb000. In code, the control byte is calculated this way:

```
ch_sel=0;  //select channel
control_byte=control_byte+speed[10];  //add speed desired to 0xb000
control_byte=control_byte+(ch_sel<<8);  //add channel selection w/ 8 bit left shift</pre>
```

NOTE: "ch\_sel" and the desired channel signal do not match up. Instead use this scheme to select the desired signal on the board:

| ch_sel | On U11 | On T1  |
|--------|--------|--------|
|        | chip   | header |
| 0      | B00    | 1      |
| 1      | B02    | 3      |
| 2      | B04    | 5      |
| 3      | B06    | 7      |
| 4      | B08    | 10     |
| 5      | B10    | 12     |
| 6      | B12    | 14     |
| 7      | B14    | 16     |
| 8      | B01    | 2      |
| 9      | B03    | 4      |
| 10     | B05    | 6      |
| 11     | B07    | 8      |
| 12     | B09    | 11     |
| 13     | B11    | 13     |
| 14     | B13    | 15     |
| 15     | B15    | 17     |

The LTC2448 also supports 8 channel differential mode. This can be achieved by changing the control byte passed to the 'ad24\_setup' routine to 0xa0000 (speed and channel selection is added on the same way as in single-ended mode). See the LTC2448 data sheet for details on how to define the control byte, 'LTC2448.pdf' in the tern docs\parts directory.

For a sample file demonstrating the use of the ADC, please see  $5d\_ad24.c$  in  $tern\586\samples\586d$ .

This sample is also included in the **586d.ide** test project in the **tern\586\samples\5d** directory.

## 4.4.4 Compact Flash Interface / File System Access

The 586Drive offers a 50-pin compact flash interface. TERN libraries support FAT file systems access to/from the compact flash interface. This allows uses to log data with the 586Drive and then conveniently access the file(s) with a PC.

Refer to \tern\586\include\fileio.h and \tern\586\include\filegeo.h for file system functions and descriptions. Also see \tern\586\samples\5p\586p.ide for a sample program accessing the compact flash interface. For the program in 586p.ide, you will need to install a FAT formatted CF (FAT32 is NOT allowable) card into the 586Drive and link SER1 (H3) with a PC hyper terminal at 192,000, N, 8, 1.

#### 4.4.5 UART SCC2691

Two SCC2691 UART chips may be installed on the 5D at location U17 and U18. The UART at U17 is enabled using chip select line /SC1, while U18 is enabled using /SC2.

Because this UART configuration is unique on the 586-Drive, new functions utilizing specific mapping were created to service these chips. UART U17 will use code from \ten\586\samples\586d\sc1.c, and U18 uses \ten\586\samples\586d\sc2.c. It is vital to include one or both of these files in your target, depending on which UART(s) you are using.

Also, make sure to include "sc1.h" for "sc1.c" and "sc2.h" for sc2.c in your code.

Before using a serial port, we must initialize it with the following function calls:

sc1\_init(m1, m2, baud,sc1\_in\_buf,isize,sc1\_out\_buf,osize,c1); //U17 UART initialization sc2\_init(m1, m2, baud,sc2\_in\_buf,isize,sc2\_out\_buf,osize,c2); //U18 UART initialization

## Important Notes

- These UARTs will use the same **COM** structure as the default Serial Ports 0 and 1 (located in \text{\text{tern}}\586\include\586.h).
- U17 uses /INTC as the interrupt input to the SC520. The function initializing /INTC, however, is not included in 586.lib. The user will need to paste this function and its prototype into their code. The function is found in "5d\_sc1.c", inside of \tern\586\samples\586d\ directory, and is appropriately named void intC\_init(char i,void interrupt far(\*intC\_isr)());

# **Appendix A: 586-Drive Layout**

The **586-Drive** measures 151x82 mm(5.94x3.23 inches). All dimensions are in inches. For mounting holes, arrows point to the center of the mounting hole. For pin headers, arrows point to pin 1 of the header, and to the center the pin.

The 586Drive PCB is designed to fit in a 35 mm DIN rail mount enclosure (JIT DR1570, www.jit-components.com). See datasheet of the JIT DR1570





# www.jit-components.com

# JIT DR1570 Enclosure for 35MM Din Rail

- Module enclosure for 35MM DIN Rail.
- To be used in a closed rack or installation box.
- Possibility for both horizontal and vertical PCB.
- Custom Perforations and Printing.



# **TECHNICAL DATA**

| HOUSING      |                 |
|--------------|-----------------|
| Material     | Lexan           |
| Color        | Gray (RAL 7035) |
| Max. Temp.   | 100°C           |
| Width        | 157.0 MM        |
| Flammability | UL 94 V0        |

| BASE           |                        |
|----------------|------------------------|
| Material       | Noryl                  |
| Color          | Black (RAL 7021)       |
| Max. Temp.     | 100°C                  |
| Max. Wire Dia. | 2 x 2.5mm <sup>2</sup> |
| Max. Load      | 10 A                   |
| Terminals      | 52 Fixed or Plugable   |
| Mounting       | DIN-Rail (EN50022)     |
| Flammability   | UL 94 V0               |

| 58.0 | 86.0                                    |
|------|-----------------------------------------|
|      | (00000000000000000000000000000000000000 |
|      |                                         |

| SERIES 1570                                           | ORDER CODE |
|-------------------------------------------------------|------------|
| Housing, gray, use with plugable terminals            | 4969001571 |
| Housing, gray, use with screw terminals               | 4969001572 |
| Housing, gray, closed, no terminals                   | 4969001570 |
| Base with clip and 2 screws                           | 4969021571 |
| Base only                                             | 4969021570 |
| Transparent lid                                       | 4969011570 |
| DIN rail clip                                         | 0669000604 |
| Plugable terminal block, 13 contact, PCB male portion | 1155713580 |
| Plugable terminal block, 13 contact, wire             | 1155613080 |
| female portion                                        |            |
| Screw terminal block, 13 contact                      | 3950141308 |
| Screwless type terminal block, 1 contact              | 1157801150 |
| Screw, selfcutting, 2 pieces                          | 7006102965 |







|           | TERN                         |  |
|-----------|------------------------------|--|
| Title     | e                            |  |
| 586-Drive |                              |  |
| Size      | Size Document Number         |  |
| B 5D.2    |                              |  |
| Date      | : August 31, 2006 Sheet 2 of |  |