

**USB3FPGA** 

User Manual C 1030-2805

## SPARTAN-3E FPGA board with USB 2.0 interface

Order number: C 1030-2805



# Copyright information

Copyright © 2007 CESYS GmbH. All Rights Reserved. The information in this document is proprietary to CESYS GmbH. No part of this document may be reproduced in any form or by any means or used to make derivative work (such as translation, transformation or adaptation) without written permission from CESYS GmbH.

CESYS GmbH provides this documentation without warranty, term or condition of any kind, either express or implied, including, but not limited to, express and implied warranties of merchantability, fitness for a particular purpose, and non-infringement. While the information contained herein is believed to be accurate, such information is preliminary, and no representations or warranties of accuracy or completeness are made. In no event will CESYS GmbH be liable for damages arising directly or indirectly from any use of or reliance upon the information contained in this document. CESYS GmbH will make improvements or changes in the product(s) and/or program(s) described in this documentation at any time.

CESYS GmbH retains the right to make changes to this product at any time, without notice. Products may have minor variations to this publication, known as errata. CESYS GmbH assumes no liability whatsoever, including infringement of any patent or copyright, for sale and use of CESYS GmbH products.

CESYS GmbH and the CESYS logo are registered trademarks.

All product names are trademarks, registered trademarks, or service marks of their respective owner.

 $\Rightarrow$  Please check <u>www.cesys.com</u> to get the latest version of this document.

CESYS Gesellschaft für angewandte Mikroelektronik mbH Zeppelinstrasse 6a D – 91074 Herzogenaurach Germany

# A Overview

# Summary of USB3FPGA

USB3FPGA can be used as a development platform for designs with Xilinx SPARTAN 3E FPGAs as well as a OEM-component for job lot production. A 96-pin VG connector allows the attachment of external hardware to the FPGA. The board is equipped with a XC3S500E-4PQ208C XILINX FPGA, a member of the Spartan 3E family. This programmable logic device receives its internal functions after it has been configured by downloading a bitstream that represents the design. The change of logic functions (reconfiguration) is possible at any time.

The bistream is loaded from the PC via USB to the FPGA. The software that comes with the board permits to load new configurations anytime.

Two clock oscillators supply basic clocks that can be used by the FPGA. A third clock oszillator can be populated on request.

The 96-pin VG expansion connector of the USB3FPGA allows connections to I/O pins of the FPGA as well as to 3.3 V and GND. Many extensions can be attached directly without the need of an additional external power supply

# Feature list

- XILINX XC3S500E-4PQ208C FPGA
- CYPRESS FX2LP USB controller
- 2MByte FAST SRAM (1M x 16, 10ns) connected to FPGA
- 64 kByte SRAM connected to USB 2.0 controller (optional)
- USB 2.0 compliant device
- Selectable self-powered or bus-powered
- Up to three individual onboard clock oscillators
- Expansion port (44 IO pins + 15 INPUT pins + 22 auxiliary IO pins)
- 3 LEDS
- All FPGA Pins routed to test connectors
- Driver for Windows XP<sup>™</sup>, Firmware and Benchmark-program included
- Sample code (C++ Source) of test-program included

### Minimum requirements

- ☑ PC with USB 2.0 interface running Windows XP<sup>™</sup>
- ☑ 10 MByte free harddisk space
- ☑ USB3FPGA board with USB2.0 compliant cable
- ☑ CESYS USB 2.0 drivers

# FPGA Design Tools

To simulate and synthesize FPGA designs appropriate tools are needed. Xilinx offers a toolset called "ISE WebPack" free of charge on their website: <u>http://www.xilinx.com</u>. The ISE WebPack fully supports the XC3S500E Spartan3E-FPGA. There are also other commercial tools available from Xilinx and various other vendors.

# Windows XP Quick-start installation guide

- 1. Connect *USB3FPGA* to one free USB2.0 port with shipped USB2.0 compliant cable.
- 2. Install USB3FPGA driver via Windows XP<sup>™</sup> installation guide from included driver and utilities CD.
- 3. With Windows XP<sup>™</sup> USB3FPGA device driver has to be installed twice.
- 4. Locate folder "bin" on shipped CD and double-click "diag.exe". By pressing the 'Memory Test' button on the left a first device test can be started.



Figure 1: USB3FPGA block diagram

# SPARTAN-3E FPGA

| Device                | XC3S500E-4PQ208C |
|-----------------------|------------------|
| System Gates          | 500k             |
| CLB Rows              | 46               |
| CLB Columns           | 34               |
| Total CLBs            | 1164             |
| Total Slices          | 4.656            |
| Distributed RAM bits  | 73 k             |
| Block RAM bits        | 360 k            |
| Dedicated Multipliers | 20               |
| DCMs                  | 4                |

For details on SPARTAN-3E<sup>™</sup> FPGA, please refer to data sheet at: <u>http://www.xilinx.com/support/documentation/data\_sheets/ds312.pdf</u>

Connector diagram



Figure 2: USB3FPGA connector diagram

**<u>Attention</u>**: The Testpin numbering printed on the PCB v1.0 is not correct in the area marked above. This error has been fixed in PCB v1.1. This document is correct.

# Board dimensions



Figure 3: USB3FPGA board dimensions

### Power supply

The USB3FPGA can be configured to be self powered or bus powered. The default setting is "bus powered". This means power is provided by the USB bus. If the USB3FPGA is the only device on the USB bus, most computers should allow a maximum current of about 500 mA. This may not be true for notebooks.

| J1 Power source select |                                                                                 |
|------------------------|---------------------------------------------------------------------------------|
| Pin 1 – 2              | Self powered (External +5 Volt power supply must be attached to connector CON1) |
| Pin 2 – 3              | Bus powered (USB power supply)                                                  |

The option "self powered" requires an external power supply connected to CON1. Use this method if your design draws more current than your USB bus can deliver.

Attention: Be careful when using the external power connector. If you apply more than 5,5 Volts or if you reverse the polarity, the board will permanently fail and may not be reparable.

Directly after the USB3FPGA board is connected to the USB bus, it must not consume more then 100mA. This is the current limit (defined by the USB standard) until the software has called "SetConfiguration". Most PC's ignore this restriction but some USB-hubs measure the current-flow and will report a shortcut condition when a USB device draws more than 100 mA immediately after it has been plugged.

The option "software controlled" keeps the FPGA, the SRAM and the Pins on CON1 (A3, B3, C3) powered off until "SetConfiguration" was called. As long as the FPGA is not powered, its I/O pins must be held low (+/- 0.3 Volt).

Attention: When you use "software controlled" power-on behaviour, make sure no FPGA I/O pin is driven as long as the FPGA is not powered.

| J4 Optional power sequencing |                                                                                                                                                                                                    |
|------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Pin 1 – 2                    | All power supplies will ramp up as soon as +5 Volt are attached.                                                                                                                                   |
| Pin 2 – 3                    | Software controlled power on. At startup only FX2LP power supply will ramp up. Only after the attached USB host grants more than 100mA FX2LP will enable power up of other onboard power supplies. |

# USB 2.0 interface

The USB 2.0 interface of the board is implemented using an USB I/O controller outside the FPGA. Therefore FPGA designs do not need to include USB specific code. Developers do not need to know details about the USB bus. To enable communication between the FPGA and a program running on the PC, a easy-to-use API and VHDL sample code come with the board.

If your design works "stand-alone" and does not require any communication with the PC, you may ignore the USB interface details and use it only for downloading your design.

# FPGA pin connections

All FPGA VCCO-Pins on the USB3FPGA board are connected to 3,3 Volt. The I/O pins of the SPARTAN-3E FPGA do **NOT** accept 5 Volt Input signals.

When 5 Volt signals are connected without proper level-shifters or series resistors, the FPGA will immediately become damaged. When 3,3 Volt signals are used with long traces or cables in conjunction with improper termination, the resulting overshoot and undershoot can damage the FPGA as well. Please read Xilinx application note http://direct.xilinx.com/bvdocs/appnotes/xapp659.pdf for details.

Don't apply any voltage outside the interval [-0.5V....+3.8V], not even for a few Nanoseconds. Take care of overshoot / undershoot conditions.

# LEDs

| LEDs        |                              |
|-------------|------------------------------|
| USER- LED   | FPGA I/O pin 140             |
| Power- LED  | FPGA is powered ON           |
| Config- LED | FPGA is configured           |
| PWR2- LED   | USB controller is powered ON |

Power-LED lights up when the FPGA gets power.

PWR2-LED lights up when the board gets power from the USB-bus

Config-LED lights up when the FPGA is configured successfully.

User-LED light up when there is a low level at the corresponding FPGA Pin. The meaning of this LEDs is defined by the user's FPGA design.

# FPGA Testpins

All FPGA pins are routed to testpoints to ease the connection of measurement equipment like Logic Analyzers. The relationship between FPGA pins and Testpoints is printed on the USB3FPGA board and shown in the Connector-Diagram above.

# Expansion port CON1

The 96-pin "VG96 abc reverse" external expansion connector (DIN 41612) is of type "female". Please use the connector diagram to indicate pin 1. On some connectors, the numbers are printed upside down.

Mating connectors among others are: RS Components 476-025 or Farnell 104-986 or HARTING order number 0903 196 7921.

Most pins of the FPGA can be configured as input (IN), output (OUT), or bidirectional (I/O). Make sure your FPGA design does not drive pins that are already driven by external connected logic. This is also important for bi-directional signals.

| CON1 | 96-pin VG Expansion | connector          |                    |
|------|---------------------|--------------------|--------------------|
| Pin  | Α                   | В                  | С                  |
| 1    | V5EXT               | V5EXT              | V5EXT              |
| 2    | GND                 | GND                | GND                |
| 3    | V3                  | V3                 | V3                 |
| 4    | * FPGA I/O Pin 199  | * FPGA I/O Pin 197 | * FPGA I/O Pin 196 |
| 5    | ▼ FPGA IN Pin 194   | * FPGA I/O Pin 193 | * FPGA I/O Pin 192 |
| 6    | * FPGA I/O Pin 190  | * FPGA I/O Pin 189 | * FPGA I/O Pin 187 |
| 7    | ▼ FPGA IN Pin 175   | ▼ FPGA IN Pin 174  | * FPGA I/O Pin 172 |
| 8    | ▼ FPGA IN Pin 169   | * FPGA I/O Pin 168 | * FPGA I/O Pin 167 |
| 9    | * FPGA I/O Pin 161  | * FPGA I/O Pin 160 | ▼ FPGA IN Pin 159  |
| 10   | ▼ FPGA IN Pin 6     | ▼ FPGA IN Pin 204  | ▼ FPGA IN Pin 154  |
| 11   | * FPGA I/O Pin 153  | * FPGA I/O Pin 152 | * FPGA I/O Pin 151 |
| 12   | * FPGA I/O Pin 150  | ▼ FPGA IN Pin 148  | * FPGA I/O Pin 147 |
| 13   | * FPGA I/O Pin 146  | * FPGA I/O Pin 145 | * FPGA I/O Pin 144 |
| 14   | ▼ FPGA IN Pin 142   | * FPGA I/O Pin 9   | FPGA I/O Pin 139   |
| 15   | FPGA I/O Pin 138    | FPGA I/O Pin 137   | ▼ FPGA IN Pin 136  |
| 16   | FPGA I/O Pin 135    | FPGA I/O Pin 134   | FPGA I/O Pin 133   |
| 17   | FPGA I/O Pin 132    | ▼ FPGA IN Pin 130  | FPGA I/O Pin 129   |
| 18   | FPGA I/O Pin 128    | FPGA I/O Pin 127   | FPGA I/O Pin 126   |
| 19   | ▼ FPGA IN Pin 124   | FPGA I/O Pin 123   | FPGA I/O Pin 122   |
| 20   | FPGA I/O Pin 120    | FPGA I/O Pin 119   | ▼ FPGA IN Pin 118  |
| 21   | FPGA I/O Pin 116    | FPGA I/O Pin 115   | FPGA I/O Pin 113   |
| 22   | GND                 | GND                | GND                |
| 23   | FPGA I/O Pin 112    | ▼ FPGA IN Pin 110  | FPGA I/O Pin 109   |
| 24   | FPGA I/O Pin 108    | FPGA I/O Pin 107   | FPGA I/O Pin 106   |
| 25   | FPGA I/O Pin 55     | FPGA I/O Pin 61    | FPGA I/O Pin 62    |
| 26   | FPGA I/O Pin 63     | FPGA I/O Pin 64    | FPGA I/O Pin 65    |
| 27   | FPGA I/O Pin 68     | FPGA I/O Pin 69    | ▼ FPGA I/O Pin 71  |
| 28   | ▼ FPGA I/O Pin 72   | FPGA I/O Pin 90    | ▼ FPGA I/O Pin 91  |
| 29   | FPGA I/O Pin 93     | FPGA I/O Pin 94    | FPGA I/O Pin 96    |
| 30   | FPGA I/O Pin 97     | FPGA I/O Pin 98    | ▼ FPGA I/O Pin 101 |
| 31   | FPGA I/O Pin 99     | FPGA I/O Pin 100   | FPGA I/O Pin 102   |
| 32   | GND                 | GND                | GND                |

| V5EXT: | If Jumper J1 is set to Position 1-2 (self powered mode), a 5 VDC    |
|--------|---------------------------------------------------------------------|
|        | power-supply must be connected here. For bus-powered applications   |
|        | this pins can be left unconnected.                                  |
| V3:    | This pin is connected to the 3,3 Volt power supply of the board.    |
|        | Depending on the loaded FPGA design it can source up to 200 mA.     |
| *      | Attention:                                                          |
|        | This pin is also connected to the on-board RAM. It can only be used |
|        | when the RAM is not needed and disabled by pulling RAM_CE high.     |
| ▼      | Attention:                                                          |

This pin can only be used as an Input to the FPGA.

| Clock signals |                         |
|---------------|-------------------------|
| FX2CLK        | FPGA IN GCLK9 Pin 184   |
|               | FPGA I/O Pin 103        |
| IFCLK         | FPGA IN GCLK8 Pin 183   |
| CLK50         | FPGA I/O GCLK10 Pin 185 |
| CLK_optional  | FPGA I/O GCLK11 Pin 186 |
| RESET         | FPGA I/O Pin 28         |

### Clock signals and RESET

There are 4 clock sources on the USB3FPGA evaluation board. No matter which of them you use as the main clock for your design, you should synchronize all incoming asynchronous signals to it with at least one FlipFlop before using them internally. If you fail to do so, your design may work sometimes but not every time. One-hot state machines might lose their "hot"-state and become inoperable. Encoded state machines might enter wrong or illegal states.

#### FX2CLK

This is the clock, the USB controller FX2 uses internally and for its Program/Data memory interface.

#### IFCLK

This is the interface clock of the USB controller FX2 GPIF. If you want to transmit or receive data using the USB interface, it is the easiest way to choose this clock as the main clock source for your design. It defaults to 48 MHz. Its frequency can be switched to 30 MHz by the software (See API documentation).

#### CLK50

The CLK50 clock signal is connected to a on-board oscillator running at 50 MHz.

#### CLK\_optional

The CLK\_optional clock signal is connected to an empty oscillator position. Please contact CESYS if you need this clock.

#### RESET

The RESET-signal can be used as an active high input to reset the whole design. It is active during FPGA configuration and for a few milliseconds after configuration has finished. It also can get activated by the Host software.

# FPGA JTAG port

The JTAG port of the SPARTAN-3E FPGA is accessible through connector CON2. This 14-pin connector can be used with XILINX download cables. Although configuration of the FPGA can be made via USB, many other tools (i.e. ChipScope) require JTAG.

| CON2 JTAG connector |               |
|---------------------|---------------|
| Pin 1,3,5,7,9,11,13 | GND           |
| Pin 2               | +2,5 Volt     |
| Pin 4               | TMS           |
| Pin 6               | тск           |
| Pin 8               | TDO           |
| Pin 10              | TDI           |
| Pin 12,14           | Not connected |

Attention: Don't connect JTAG adapters that use 3,3 Volt signaling. The FPGA only accepts 2,5 Volt signal levels.

# Memory interface

CESYS USB3FPGA is equipped with 2MByte of FAST SRAM (1M x 16, 10ns, CYPRESS CY7C1061AV33-10ZC). This type of memory is static. This means it can be used very easily because it does not need refresh, bank management, address multiplexing or other techniques known from dynamic memories. The access time is 10ns. The maximum data rate is 200 MByte/s.

| FPGA <-> Memory interface |                  |
|---------------------------|------------------|
| SRAM signal name          | FPGA pin number  |
| RAM_A0                    | FPGA I/O Pin 189 |
| RAM_A1                    | FPGA I/O Pin 192 |
| RAM_A2                    | FPGA I/O Pin 196 |
| RAM_A3                    | FPGA I/O Pin 199 |

| FPGA <-> Memory interface |                  |  |
|---------------------------|------------------|--|
| SRAM signal name          | FPGA pin number  |  |
| RAM_A4                    | FPGA I/O Pin 202 |  |
| RAM_A5                    | FPGA I/O Pin 200 |  |
| RAM_A6                    | FPGA I/O Pin 197 |  |
| RAM_A7                    | FPGA I/O Pin 193 |  |
| RAM_A8                    | FPGA I/O Pin 190 |  |
| RAM_A9                    | FPGA I/O Pin 187 |  |
| RAM_A10                   | FPGA I/O Pin 171 |  |
| RAM_A11                   | FPGA I/O Pin 167 |  |
| RAM_A12                   | FPGA I/O Pin 164 |  |
| RAM_A13                   | FPGA I/O Pin 162 |  |
| RAM_A14                   | FPGA I/O Pin 160 |  |
| RAM_A15                   | FPGA I/O Pin 161 |  |
| RAM_A16                   | FPGA I/O Pin 163 |  |
| RAM_A17                   | FPGA I/O Pin 165 |  |
| RAM_A18                   | FPGA I/O Pin 168 |  |
| RAM_A19                   | FPGA I/O Pin 172 |  |
| RAM_D0                    | FPGA I/O Pin 150 |  |
| RAM_D1                    | FPGA I/O Pin 151 |  |
| RAM_D2                    | FPGA I/O Pin 152 |  |
| RAM_D3                    | FPGA I/O Pin 153 |  |
| RAM_D4                    | FPGA I/O Pin 144 |  |
| RAM_D5                    | FPGA I/O Pin 145 |  |
| RAM_D6                    | FPGA I/O Pin 146 |  |
| RAM_D7                    | FPGA I/O Pin 147 |  |
| RAM_D8                    | FPGA I/O Pin 203 |  |
| RAM_D9                    | FPGA I/O Pin 5   |  |
| RAM_D10                   | FPGA I/O Pin 8   |  |
| RAM_D11                   | FPGA I/O Pin 9   |  |
| RAM_D12                   | FPGA I/O Pin 2   |  |
| RAM_D13                   | FPGA I/O Pin 3   |  |
| RAM_D14                   | FPGA I/O Pin 4   |  |
| RAM_D15                   | FPGA I/O Pin 205 |  |
| RAM_WE#                   | FPGA I/O Pin 178 |  |

| FPGA <-> Memory interface |                  |
|---------------------------|------------------|
| SRAM signal name          | FPGA pin number  |
| RAM_OE#                   | FPGA I/O Pin 179 |
| RAM_CE#                   | FPGA I/O Pin 180 |
| RAM_BLE                   | FPGA I/O Pin 177 |
| RAM_BHE                   | FPGA I/O Pin 181 |

# **Undocumented IO**

FPGA IO pins that are not documented are reserved for USB communication and should not be connected in user designs.

# C Software

# Files

The files which ships with our source package are sorted in subdirectories, the structure is described below.

| Folder  | Contains                                                                                                                                                                                                             |
|---------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| bin     | Compiled sample applications, including the diagnostic tool<br>'diag.exe' which is can be used for simple tasks like device testing<br>and FPGA configuration. How to use it can be found later in this<br>document. |
| designs | Holds the generated design which is used for all of our sample applications, including the diagnostic tool. Source code can be found in the source folder.                                                           |
| doc     | All documentation, including this document can be found here.                                                                                                                                                        |
| drivers | System drivers for our device is located in this folder, if the operating system asks for a driver, point it to this folder.                                                                                         |
| source  | Source files, for the FPGA and the host system, mainly sample applications can be found in this folder, in addition, the API is located in subfolders 'lib' and 'include'.                                           |

### Driver installation

After plugging the device to a PC the first time, the operating system, in this case Windows XP will pop up an information about the detection of a new hardware. As Windows doesn't know the USB3FPGA board, it asks the user several questions to install the correct drivers for the device. The steps are shortly described below:

In the first Dialog, select the second option, 'Install from a list or specific location (Advanced)', press 'Next'. In the second dialog, select the first option 'Search for the best driver in these location.' and include only the second option there, 'Include this location in the search:'. Then press 'Browse' and select the 'drivers' folder which is included in the source package that ships with our board, press 'Next'. Windows will try to install the driver now, for security reasons it will ask the user to allow this via another dialog box, you have to select 'Continue Anyway' here. Finish the installation by pressing 'Finish' in the next dialog box, the loader driver is installed now. Because the used USB chip needs a two step driver loading mechanism, Windows will now pop up another dialog which informs the user about a new hardware detection. To install this second driver, follow the installation instructions for first driver step by step, this will be exactly the same. After finishing the installation for the second driver, the device should work correctly. To verify the installation, have a look in the device manager, expand the 'Universal Serial Bus controllers' tree and look for a device named 'Cesys USB3FPGA compatible device'. In addition, start the diagnostic tool which is located in the 'bin' folder in our source

package and try the memory test and benchmark options there. How this works is described in the following chapter.

# Diagnostics

| Ceusb3 Diagnostics |                                                                                | × |
|--------------------|--------------------------------------------------------------------------------|---|
| <b>CES\S</b>       | \\?\usb#vid_10f8&pid_c381#5&f635d31&0&5#{53b029df-bbc6-4db8-841e-69ce4f163588} |   |
| ReEnum Device List |                                                                                |   |
| Device Info        |                                                                                |   |
| Download Design    |                                                                                |   |
| Dump EEPROM        |                                                                                |   |
| RBT to BIN         |                                                                                |   |
| Register I/O       |                                                                                |   |
| Memory Test        |                                                                                |   |
|                    |                                                                                |   |
| Benchmark          |                                                                                |   |
| Close              |                                                                                |   |

Figure 4: Ceusb3 Diagnostics

The list box on top of the window lists all available devices. All device specific operations are bound to the device selected in this box. After a change in the hardware configuration (device plug, unplug, replug), this list has to be re-enumerated to ensure stability and functionality. This can be done by simply pressing the '**ReEnum Device List**' button.

To get detailed information about a selected device, a click on '**Device Info**' will print all known information to the log window below the device list box. This includes information about the driver, firmware, hardware composition plus some software details.

The devices main functionality is given by the on board FPGA, which must be configured to work (which is not case after power on). The configuration process is quite simply, after clicking the 'Download Design' button, a file selection dialog will pop up, the chosen file will be used to configure the FPGA after the selection. Two file types are supported at the moment, .RBT (raw bit streams) and .FPGA (binary equivalents for raw bit streams, creation described below).

The 'Dump EEPROM' button will pop up a file selection dialog, where users can choose a file where all EEPROM contents will be stored in a hexadecimal like text format.

As described above the design importer is able to use .RBT and .FPGA files. The .FPGA format is simply a binary representation for .RBT files, which are smaller than there .RBT equivalents and they will be parsed much faster. To create a .FPGA out of a .RBT, a click on '**RBT to BIN**' will pop up a source selection dialog. Afterwards a

destination file must be chosen in a second file selection dialog, thats all, the conversion should be done.

The 'Register I/O' button hides the log window and pops up additional controls to test the register read and write functionality based on the underlying FPGA design, a compatible design must be downloaded. The 'Write' button writes the given value to the selected register, just as the 'Read' button reads the value from the selected register and displays the value in the 'Value' field. The 'Return' button on the left return to normal functionality.

| Ceusb3 Diagnostics |                                                                                | × |
|--------------------|--------------------------------------------------------------------------------|---|
| <b>CESYS</b>       | \\?\usb#vid_10f8&pid_c381#5&f635d31&0&5#{53b029df-bbc6-4db8-841e-69ce4f163588} |   |
| ReEnum Device List | Register: 0x0000 Value: 0x0000                                                 |   |
| Device Info        | Write Read                                                                     |   |
| Download Design    |                                                                                |   |
| Dump EEPROM        |                                                                                |   |
| RBT to BIN         |                                                                                |   |
| Return             |                                                                                |   |
| Memory Test        |                                                                                |   |
|                    |                                                                                |   |
| Benchmark          |                                                                                |   |
| Close              |                                                                                |   |

Figure 5: Ceusb3 Diagnostics: Register I/O

The 2 MB on board memory can be tested by pressing the 'Memory Test' button. A message box will pop up afterwards, showing the results of the test, either success or failure.

The 'Benchmark' button shows a list of 3 different benchmark options, Read, Write, Read and Write. After choosing one of these options, the log window disappears and the benchmark graph pops up. The blue graph line shows the unchanged value of bytes per second transferred between PC and device, while the purple line shows an averaged value. Benchmarking can be stopped by pressing the 'Stop' button on the left.



Figure 6: Ceusb3 Diagnostics: Benchmark

# API

### General

#### Language compatibility

The CEUSB3 API is designed to work with C++ native code to ensure best performance. Due to the wide popularity the library is compatible with Visual C++ 7.1 and higher versions. In addition to this, a wrapper for the .NET Framework 1.1 is also available, so applications written in C++ NET, C# and Visual Basic have access to the API too. The NET wrapper consists of the same classes and methods as the C++ API, but global functions, macros and constants are encapsulated in an additional class (ceUSB3API), based on the fact that NET doesn't support this. Furthermore some data types are not available in all languages, so a few of the parameters have a different value type in the C++ and .NET API. The best place to analyze the differences are the sample sources that ship with the API. The test application shows many parts from the API and is available in C++ native (cntest), C++ NET (cnettest), C# (cstest) and VB (vbtest).

### Backward compatibility

The CEUSB3 API is newly designed so there's no compatibility with API's from USB2FPGA or other devices.

#### ceUSB3 C++ API specs

#### Basics

The API contains the necessary library and include files. To use the API you have to follow the steps below:

- Include the main header file (ceusb3api.h).
- Link the executable with the main library (ceusb3api.lib).

The whole API is located in namespace ceUSB3, so either tell the compiler to use this namespace (using namespace ceUSB3;) or scope all elements with this namespace separately (e.g. ceUSB3::ceDevice \*pDev = ceUSB3::ceDevice::GetDevice(0);). Pointers retrieved by the API must NOT be deleted, this is done by the API internally. Affected classes are ceDevice and ceInfo.

#### Error handling

Most functions return a HRESULT code, so you can use the SUCCEEDED() and FAILED() macros defined in the windows API. To retrieve a printable error string from a failed call use GetHRESULTMessage(), which returns the description string of a given error code (Only error codes used by the API).

#### ceUSB3 NET API specs

#### Basics

The API can be used by adding a new reference to the project, choose the file browser there and select ceusb3apinet.dll. To be able to use the classes, namespace cesys.ceUSB3NET must be used, the syntax is based on the NET language that is used.

#### Error handling

Most functions return a System::Int32 code, the C++ macros for error checking are encapsulated in two static methods, ceUSB3API.ceSUCCEEDED() and ceUSB3API.ceFAILED(). In addition, all possible error codes used by the API are defined as constants in that class (ceUSBAPI.ceS\_\* / ceUSBAPI.ceE\_\*). To retrieve a printable error string from a failed call use ceUSB3API.GetHRESULTMessage(), which returns the description string of a given error code (Only error codes used by the API).

#### Additional differences to the C++ API

Because NET doesn't support global functions, Init() and Delnit() are encapsulated in class ceUSB3API too. Furthermore Init() doens't expect a GUID, but a value from the ceUSB3API.ceDeviceType enumeration.

#### How to use the API

#### Initialization / Deinitialization

To use the API it must be initialized, this is done by a call to Init() (NET: ceUSB3API.Init()). This function searches for all devices plugged to the computer which matches the given GUID (NET: ceUSB3API.ceDeviceType). You can call this function with different GUID's which builds an internal list of all of them.

After using the API it must be freed, this is done by calling **Delnit()** (NET:

ceUSB3API::Delnit()). To detect any changes in the list of connected devices, you have to call Delnit() and start again with one or more lnit() - calls. This forces a reenumeration of all devices. Attention! This invalidates all pointers you get from the API!

#### How to communicate with devices

If the API is initialized correctly, you can retrieve the count of available devices by calling the static member function GetDeviceCount() from class ceDevice. To access one of the devices, call function GetDevice() from the same class and use an index in range of 0 < index < GetDeviceCount() to specify one of the devices. The pointer returned by this function is constant and valid until you call Delnit() (the same call will return the same pointer, so you do not have to store this pointer anywhere).

All communication with the device is done using this class pointer. Before any data can be sent or received, the device must be opened. To do this call method Open() which internally opens the device, set default parameters and retrieves some information about the device. After a successful call to this function you can do those things (descriptions below):

- Configure device (SetGPIFSpeed()).
- Retrieve information (GetInfo()).
- Download FPGA designs (ProgramFPGA()).
- Read and write FPGA registers (ReadRegister()/WriteRegister()).
- Read and write huge blocks of data (ReadBulk()/WriteBulk()).
- Read and write parts of the EEPROM (ReadEeprom()/WriteEeprom()).

To properly finish the use of the device, call Close().

#### Function description (methods in alphabetic order)

All methods are listed twice, the first one is the C++ native notation, the other one is the NET counterpart in C# notation.

# Global functions (NET: class ceUSB3API)

| C++:    | <pre>void DeInit()</pre>                                                                      |
|---------|-----------------------------------------------------------------------------------------------|
| NET:    | <pre>void ceUSB3API.DeInit()</pre>                                                            |
| Info    | Frees all resources allocated by $\texttt{Init}()$ , this must be called after using the API. |
| Returns | -                                                                                             |
| Errors  | -                                                                                             |

| C++:    | const char *GetHRESULTMessage(HRESULT hr)                  |
|---------|------------------------------------------------------------|
| NET:    | String ceUSB3API.GetHRESULTMessage(System.Int32 hr)        |
| Info    | Returns the error string bound to the given error code hr. |
| Returns | Error string                                               |
| Errors  | -                                                          |

| C++:    | HRESULT Init(const GUID &Guid)                                                                                                                                                                                                                                                                                                                                                                                    |
|---------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| NET:    | <pre>Int32 ceUSBAPI.Init(ceUSB3API.DeviceType T)</pre>                                                                                                                                                                                                                                                                                                                                                            |
| Info    | Initializes the API and searches for devices with the given type (via<br>GUID or device type). The function can be called multiple times with<br>different types to enumerate and use different devices. Resources<br>allocated by that call must be freed by calling DeInit() after use.<br>Possible GUID's (C++):<br>GUID_INTERFACE_CEUSB3<br>GUID_INTERFACE_PSAA4096V2<br>Possible Types (NET):<br>ceDT_CEUSB3 |
|         | ceDT_PSAA4096V2                                                                                                                                                                                                                                                                                                                                                                                                   |
| Returns | Error code                                                                                                                                                                                                                                                                                                                                                                                                        |
| Errors  | S_OK: no error                                                                                                                                                                                                                                                                                                                                                                                                    |
|         | E_FAIL: error searching devices                                                                                                                                                                                                                                                                                                                                                                                   |

### class ceDevice

| C++:    | HRESULT ceDevice::Abo                         | rtPipe(uint uiPipeNumber)                                                       |
|---------|-----------------------------------------------|---------------------------------------------------------------------------------|
| NET:    | Int32 ceDevice.AbortPipe(UInt32 uiPipeNumber) |                                                                                 |
| Info    | Forces the USB bus driver to (uiPipeNumber).  | abort the transfer on a given pipe                                              |
| Returns | Error code                                    |                                                                                 |
| Errors  | E_OPEN: d<br>E_FAIL: c                        | o error<br>evice not open<br>all to driver fails<br>iPipeNumber is out of range |

| C++:    | <pre>void ceDevice::Close()</pre> |
|---------|-----------------------------------|
| NET:    | <pre>void ceDevice.Close()</pre>  |
| Info    | Closes the device.                |
| Returns | -                                 |
| Errors  | -                                 |

| C++:    | <pre>HRESULT ceDevice::GetAsyncResult(ceAsyncHandle *pH, uint *uiTransfered)</pre>                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |
|---------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| NET:    | <pre>Int32 ceDevice::GetAsyncResult(ref ceAsyncHandle pH, ref UInt32 uiTransfered)</pre>                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |
|         | This method is needed when using any of the following functions<br>using the async call convention: ReadBulk(), WriteBulk().<br>Usage: After starting an async operation, use the async handle (pH)<br>to check if the transfer is complete. Afterwards you have to call<br>GetAsyncResult() to cleanup the call and retrieve the count of<br>bytes transferred via this operation (uiTransfered).<br>A good example on how to use this can be found in the test<br>application that ships with the API, which is available in all supported<br>languages. |
| Returns | Error code                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                 |
| Errors  | S_OK:no errorE_FAIL:the function failsE_INVALIDARG:pH is NULL                                                                                                                                                                                                                                                                                                                                                                                                                                                                                              |

| C++:    | <pre>ceDevice *ceDevice::GetDevice(uint uiIdx)</pre>                                                                                                                                        |
|---------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| NET:    | ceDevice ceDevice.GetDevice(UInt32 uiIdx)                                                                                                                                                   |
| Info    | Returns a pointer to a device which is selected by a zero based index (uildx). This pointer is valid until DeInit() is called. Never try to delete this object, this is done automatically. |
| Returns | Pointer to device with the given index, NULL otherwise.                                                                                                                                     |
| Errors  | -                                                                                                                                                                                           |

| C++:    | <pre>uint ceDevice::GetDeviceCount()</pre>                                                                                     |
|---------|--------------------------------------------------------------------------------------------------------------------------------|
| NET:    | UInt32 ceDevice.GetDeviceCount()                                                                                               |
| Info    | Returns the count of devices find during the call of Init(), if Init() is called multiple times, the total number is returned. |
| Returns | Count of devices found in the system.                                                                                          |
| Errors  | -                                                                                                                              |

| C++:    | <pre>ceInfo *ceDevice::GetInfo()</pre>                                                                                                                                                                        |
|---------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| NET:    | ceInfo ceDevice.GetInfo()                                                                                                                                                                                     |
| Info    | Returns a static pointer to a ceInfo class instance bound to the device. This holds additional information about the device. Never try to delete the returned object, this is done by DeInit() automatically. |
| Returns | Pointer to info class.                                                                                                                                                                                        |
| Errors  | _                                                                                                                                                                                                             |

| C++:    | <pre>uint ceDevice::GetLastError()</pre>                                                    |
|---------|---------------------------------------------------------------------------------------------|
| NET:    | UInt32 ceDevice.GetLastError()                                                              |
| Info    | Returns the last error occurred in the driver. This may help to find out unexpected errors. |
| Returns | Driver error code.                                                                          |
| Errors  | -                                                                                           |

| C++:    | <pre>uint ceDevice::GetLastFirmwareError()</pre>                                              |
|---------|-----------------------------------------------------------------------------------------------|
| NET:    | <pre>UInt32 ceDevice.GetLastFirmwareError()</pre>                                             |
| Info    | Returns the last error occurred in the firmware. This may help to find out unexpected errors. |
| Returns | Firmware error code.                                                                          |
| Errors  | -                                                                                             |

| C++:    | HRESULT ceDevice::Open()                                                                                    |  |
|---------|-------------------------------------------------------------------------------------------------------------|--|
| NET:    | Int32 ceDevice.Open()                                                                                       |  |
| Info    | Opens the device.                                                                                           |  |
| Returns | Error code.                                                                                                 |  |
| Errors  | S_OK:no errorS_FALSE:device already openE_FAIL:error retrieving informationfrom driverfailed to open device |  |

| C++:    | HRESULT ceDevice::ProgramFPGA(ceFPGA *pFPGA)<br>Int32 ceDevice.ProgramFPGA(ceFPGA pFPGA)                                                                |                                                                                                                                                        |
|---------|---------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------|
| Info    | Downloads a FPGA design to the device. This should be the first step after opening the device. Without a running design the hardware won't do anything. |                                                                                                                                                        |
| Returns | Error code.                                                                                                                                             |                                                                                                                                                        |
| Errors  | S_OK:<br>E_OPEN:<br>E_FAIL:<br>E_INVALIDARG:<br>E_NOPIPE:<br>E_FPGA_INIT:<br>E_FPGA_NC:                                                                 | no error<br>device not open<br>call to driver fails<br>invalid design<br>no matching pipe found<br>fpga init pin doesn't switch<br>fpga not configured |

| C++:<br>NET: | <pre>HRESULT ceDevice::ReadBulk(uchar *pucData, uint<br/>uiSize, uint &amp;uiTransfered, ceAsyncHandle *pH,<br/>uint uiPipe, uint uiTimeOut)<br/>Int32 ceDevice.ReadBulk(Byte[] pucData, UInt32<br/>uiSize, ref UInt32 uiTransfered, ref ceAsyncHandle<br/>pH, UInt32 uiPipe, UInt32 uiTimeOut)</pre>                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      |  |
|--------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
| Info         | pH, UInt32 uiPipe, UInt32 uiTimeOut) This function should be used to transfer huge blocks of data from device to host. It is able to work in sync or async mode, depending on the given parameters. Parameter pucData should point to a buffer that is able to hold the requested data, while uiSize must be data count of bytes that should be received. The maximum allowed count of bytes in one call can be retrieved by method GetPipeBufferSize() from attached class celnfo (use GetInfo() to get it). Furthermore this count must be dividable by 512. If pH is NULL, than synced I/O is active, if pH is a valid async handle, async I/O will be used. Using synced I/O, uiTransfered will return the count of bytes transferred, which can be unequal to the requested transfer count, otherwise this return value is undefined. To specify a special pipe for the transfer, uiPipe can be used, but in most cases a value of 0xfffffff let the API decide the best pipe. The last parameter, uiTimeOut is only valid using synced I/O, a timeout for transfer completion in milliseconds can be specified here. |  |
| Returns      | Error code.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |  |
| Errors       | S_OK: no error<br>E_OPEN: device not open<br>E_FAIL: call to driver fails<br>E_INVALIDARG: invalid data ptr, uiSize = 0<br>or uiSize not dividable by 512<br>E_NOPIPE: no matching pipe found/uiPipe<br>invalid<br>E_TIMEOUT: call is timed out (sync)<br>E_PENDING: device is in pending mode<br>(async)                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |  |

| C++:    | <pre>HRESULT ceDevice::ReadEeprom(uint uiAddress, uchar *pucData, uint uiSize)</pre>                                                                                                                                                                                              |
|---------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| NET:    | <pre>Int32 ceDevice.ReadEeprom(uint uiAddress, Byte[] pucData, UInt32 uiSize)</pre>                                                                                                                                                                                               |
| Info    | Reads data from on board EEPROM. 7 KB are free for use, starting<br>at address 0. Maximum transfer size is 4 KB. uiAddress sets the<br>base offset, pucData should be huge enough to hold the requested<br>data, while uiSize sets the count of bytes that should be transferred. |
| Returns | Error code.                                                                                                                                                                                                                                                                       |

| Errors | S_OK:no errorE_OPEN:device not openE_FAIL:call to driver fails    |
|--------|-------------------------------------------------------------------|
|        | E_INVALIDARG: uiAddress+uiSize > 7k, uiSize>4096 or<br>0==pucData |

| C++:    | <pre>HRESULT ceDevice::ReadRegister(ushort usAddress,<br/>ushort &amp;usValue)</pre> |  |
|---------|--------------------------------------------------------------------------------------|--|
| NET:    | <pre>Int32 ceDevice.ReadRegister(UInt16 usAddress, ref UInt16 usValue)</pre>         |  |
| Info    | Read the value of FPGA register usAddress, the result will be stored in usValue      |  |
| Returns | Error code.                                                                          |  |
| Errors  | S_OK:no errorE_OPEN:device not openE_FAIL:call to driver fails                       |  |

| C++:    | HRESULT ceDevice::ResetFPGA()                                  |  |
|---------|----------------------------------------------------------------|--|
| NET:    | Int32 ceDevice.ResetFPGA()                                     |  |
| Info    | Pulses the FPGA reset pin.                                     |  |
| Returns | Error code.                                                    |  |
| Errors  | S_OK:no errorE_OPEN:device not openE_FAIL:call to driver fails |  |

| C++:    | HRESULT ceDevice::ResetPipe(uint uiPipeNumber)               |                                                                            |
|---------|--------------------------------------------------------------|----------------------------------------------------------------------------|
| NET:    | Int32 ceDevice.ResetPipe(UInt32 uiPipeNumber)                |                                                                            |
| Info    | Forces the USB bus driver to reset pipe number uiPipeNumber. |                                                                            |
| Returns | Error code.                                                  |                                                                            |
| Errors  | E_OPEN: de<br>E_FAIL: ca                                     | error<br>evice not open<br>all to driver fails<br>.PipeNumber out of range |

| C++: | HRESULT ceDevice:SetGPIFSpeed(ceGPIFSpeed Speed) |
|------|--------------------------------------------------|
| NET: | Int32 ceDevice.SetGPIFSpeed(ceDevice.ceGPIFSpeed |

|         | Speed)                                                                                                                                                                                                                               |                                                                                  |
|---------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------|
| Info    | Allows the adjustment of the GPIF speed between 30 and 48 MHz.<br>Default value is 48 MHz. It is not necessary to change this value<br>except for some special cases.<br>Possible enumerators are:<br>ceGPIFS_30MHz<br>ceGPIFS_48MHz |                                                                                  |
| Returns | Error code.                                                                                                                                                                                                                          |                                                                                  |
| Errors  | S_OK:<br>E_OPEN:<br>E_FAIL:<br>E_INVALIDARG:                                                                                                                                                                                         | no error<br>device not open<br>call to driver fails<br>uiPipeNumber out of range |

| C++:<br>NET: | <pre>HRESULT ceDevice::WriteBulk(uchar *pucData, uint<br/>uiSize, uint &amp;uiTransfered, ceAsyncHandle *pH,<br/>uint uiPipe, uint uiTimeOut)<br/>Int32 ceDevice.WriteBulk(Byte[] pucData, UInt32<br/>uiSize, ref UInt32 uiTransfered, ref ceAsyncHandle<br/>pH, UInt32 uiPipe, UInt32 uiTimeOut)</pre>                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    |  |
|--------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|
| Info         | pH, UInt32 uiPipe, UInt32 uiTimeOut) This function should be used to transfer huge blocks of data from host to device. It is able to work in sync or async mode, depending on the given parameters. Parameter pucData should point to the buffer which contains the data to send, while uiSize must be data count of bytes that should be transfered. The maximum allowed count of bytes in one call can be retrieved by method GetPipeBufferSize() from attached class celnfo (use GetInfo() to get it). Furthermore this count must be even. If pH is NULL, than synced I/O is active, if pH is a valid async handle, async I/O will be used. Using synced I/O, uiTransfered will return the count of bytes transferred, which can be unequal to the requested transfer count, otherwise this return value is undefined. To specify a special pipe for the transfer, uiPipe can be used, but in most cases a value of 0xfffffff let the API decide the best pipe. The last parameter, uiTimeOut is only valid using synced I/O, a timeout for transfer completion in milliseconds can be specified here. |  |
| Returns      | Error code.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |  |
| Errors       | S_OK: no error<br>E_OPEN: device not open<br>E_FAIL: call to driver fails<br>E_INVALIDARG: invalid data ptr, uiSize = 0<br>or uiSize not dividable by 512<br>E_NOPIPE: no matching pipe found/uiPipe                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |  |

| invalid    |                           |
|------------|---------------------------|
| E_TIMEOUT: | call is timed out (sync)  |
| E_PENDING: | device is in pending mode |
| (async)    |                           |

| C++:    | <pre>HRESULT ceDevice::WriteEeprom(uint uiAddress,<br/>uchar *pucData, uint uiSize)<br/>Int32 ceDevice.WriteEeprom(uint uiAddress, Byte[]</pre>                                                                                           |
|---------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| INE I.  | pucData, UInt32 uiSize)                                                                                                                                                                                                                   |
| Info    | Writes data to on board EEPROM. 7 KB are free for use, starting at address 0. Maximum transfer size is 4 KB. uiAddress sets the base offset, pucData must hold the data, while uiSize sets the count of bytes that should be transferred. |
| Returns | Error code.                                                                                                                                                                                                                               |
| Errors  | S_OK: no error<br>E_OPEN: device not open<br>E_FAIL: call to driver fails<br>E_INVALIDARG: uiAddress+uiSize > 7k,<br>uiSize>4096 or 0==pucData                                                                                            |

| C++:    | HRESULT ceDevice::WriteRegister(ushort usAddress,<br>ushort &usValue) |
|---------|-----------------------------------------------------------------------|
| NET:    | Int32 ceDevice.WriteRegister(UInt16 usAddress, ref<br>UInt16 usValue) |
| Info    | Write value usValue to FPGA register usAddress                        |
| Returns | Error code.                                                           |
| Errors  | S_OK:no errorE_OPEN:device not openE_FAIL:call to driver fails        |

### class celnfo

| C++:    | <pre>const char *ceInfo::GetDeviceName()</pre>                              |
|---------|-----------------------------------------------------------------------------|
| NET:    | String ceInfo.GetDeviceName()                                               |
| Info    | Returns the name of the device (Same name as listed in the device manager). |
| Returns | Requested information.                                                      |
| Errors  | -                                                                           |

| C++:    | <pre>const char *ceInfo::GetDevicePath()</pre>                                            |
|---------|-------------------------------------------------------------------------------------------|
| NET:    | <pre>String ceInfo.GetDevicePath()</pre>                                                  |
| Info    | Returns the internal name of windows path to the device. For informational purposes only. |
| Returns | Requested information.                                                                    |
| Errors  | -                                                                                         |

| C++:    | <pre>const char *ceInfo::GetDriverInfo()</pre>                                           |
|---------|------------------------------------------------------------------------------------------|
| NET:    | String ceInfo.GetDriverInfo()                                                            |
| Info    | Returns the description and version of the used driver. For informational purposes only. |
| Returns | Requested information.                                                                   |
| Errors  | -                                                                                        |

| C++:    | <pre>const char *ceInfo::GetFirmwareInfo()</pre>                                           |
|---------|--------------------------------------------------------------------------------------------|
| NET:    | <pre>String ceInfo.GetFirmwareInfo()</pre>                                                 |
| Info    | Returns the description and version of the used firmware. For informational purposes only. |
| Returns | Requested information.                                                                     |
| Errors  | -                                                                                          |

| C++:    | <pre>const char *ceInfo::GetHostController()</pre>                                                           |
|---------|--------------------------------------------------------------------------------------------------------------|
| NET:    | String ceInfo.GetHostController()                                                                            |
| Info    | Returns the description of the host controller this device is connected to. For informational purposes only. |
| Returns | Requested information.                                                                                       |
| Errors  | -                                                                                                            |

| C++:    | <pre>uint ceInfo::GetPipeBufferSize()</pre>                                                                                                    |
|---------|------------------------------------------------------------------------------------------------------------------------------------------------|
| NET:    | <pre>UInt32 ceInfo.GetPipeBufferSize()</pre>                                                                                                   |
| Info    | Returns the buffer size of each pipe inside the driver. This is the maximum count of bytes usable by block transfers via <pre>ReadBulk()</pre> |
| Returns | Requested information.                                                                                                                         |
| Errors  | -                                                                                                                                              |

| C++:    | <pre>uint ceInfo::GetPipeCount()</pre>                                                          |
|---------|-------------------------------------------------------------------------------------------------|
| NET:    | <pre>UInt32 ceInfo.GetPipeCount()</pre>                                                         |
| Info    | Count of pipes supported by the current host-device interface. For informational purposes only. |
| Returns | Requested information.                                                                          |
| Errors  | -                                                                                               |

| C++:    | <pre>const char *ceInfo::GetUSBPath()</pre>                                                                                                                     |
|---------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------|
| NET:    | <pre>String ceInfo.GetUSBPath()</pre>                                                                                                                           |
| Info    | Returns the connection path from device to host controller, including<br>any hub in between. Used ports are enclosed in squared brackets in<br>back of any hub. |
| Returns | Requested information.                                                                                                                                          |
| Errors  | _                                                                                                                                                               |

| C++:    | <pre>bool ceInfo::GetUSBPath()</pre>                                                                               |
|---------|--------------------------------------------------------------------------------------------------------------------|
| NET:    | Boolean ceInfo.GetUSBPath()                                                                                        |
| Info    | Returns true if the transfer between host and device is in highspeed mode (480MBit/s), false otherwise (15MBit/s). |
| Returns | Requested information.                                                                                             |
| Errors  | -                                                                                                                  |

#### class ceFPGA

This class is able to import and export different formats of FPGA designs. This time, rawbit (.RBT) and binary streams (.FPGA, cesys internally used format) are supported. Except ceDevice and ceInfo this class has an public constructor and destructor, so you have to take care about the lifetime of this object.

| C++:    | ceFPGA::ceFPGA()   |
|---------|--------------------|
| NET:    | ceFPGA.ceFPGA()    |
| Info    | Class constructor. |
| Returns | -                  |
| Errors  | -                  |

| C++:    | ceFPGA::~ceFPGA() |
|---------|-------------------|
| NET:    | _                 |
| Info    | Class destructor. |
| Returns | -                 |
| Errors  | -                 |

| C++:    | HRESULT ceFPGA::LoadBin(const char *pszFileName)                                            |  |  |
|---------|---------------------------------------------------------------------------------------------|--|--|
| NET:    | <pre>Int32 ceFPGA.LoadBin(String sFileName)</pre>                                           |  |  |
| Info    | Load design from pszFileName / sFileName using bin format importer (created via SaveBin()). |  |  |
| Returns | Error Code.                                                                                 |  |  |
| Errors  | S_OK: no error<br>E_OPEN: can't open file<br>E_OUTOFMEMORY: not enough memory available     |  |  |

| C++:    | HRESULT ceFPGA::LoadRBT(const char *pszFileName)                    |                                                                              |
|---------|---------------------------------------------------------------------|------------------------------------------------------------------------------|
| NET:    | Int32 ceFPGA.LoadRBT(String sFileName)                              |                                                                              |
| Info    | Load design from pszFileName / sFileName using RBT format importer. |                                                                              |
| Returns | Error Code.                                                         |                                                                              |
| Errors  | S_OK:<br>E_OPEN:<br>E_FAIL:<br>E_OUTOFMEMORY:                       | no error<br>can't open file<br>unknown format<br>not enough memory available |

| C++:    | <pre>HRESULT ceFPGA::SaveBin(const char *pszFileName)</pre>           |                             |
|---------|-----------------------------------------------------------------------|-----------------------------|
| NET:    | Int32 ceFPGA.SaveBin(String sFileName)                                |                             |
| Info    | Save design in bin format (smaller and faster loading via LoadBin()). |                             |
| Returns | Error Code.                                                           |                             |
| Errors  | S_OK:<br>E OPEN:                                                      | no error<br>can't open file |

| C++:    | HRESULT ceFPGA::SetBin(uchar *pucData, uint<br>uiSize)                       |  |  |  |
|---------|------------------------------------------------------------------------------|--|--|--|
| NET:    | Int32 ceFPGA.SetBin(Byte[] pucData, uint uiSize)                             |  |  |  |
| Info    | Set design based on the binary equivalent given by pucData with size uiSize. |  |  |  |
| Returns | Error Code.                                                                  |  |  |  |
| Errors  | S_OK: no error<br>E_OUTOFMEMORY: not enough memory available                 |  |  |  |

#### class ceAsyncHandle

This class is a helper class for async operations. It holds all necessary informations about an active transfer in background and is needed for completion. The methods of this class are designed to help to detect transfer finishing.

| C++:    | ceAsyncHandle::ceAsyncHandle() |
|---------|--------------------------------|
| NET:    | ceAsyncHandle.ceAsyncHandle()  |
| Info    | Class constructor.             |
| Returns | -                              |
| Errors  | -                              |

| C++:    | ceAsyncHandle::~ceAsyncHandle() |  |
|---------|---------------------------------|--|
| NET:    | _                               |  |
| Info    | Class destructor.               |  |
| Returns | -                               |  |
| Errors  | -                               |  |

| C++:    | HRESULT ceAsyncHandle::IsComplete(bool *pbComplete)                                                |                                           |  |  |
|---------|----------------------------------------------------------------------------------------------------|-------------------------------------------|--|--|
| NET:    | <pre>Int32 ceAsyncHandle.IsComplete(ref Boolean<br/>bComplete)</pre>                               |                                           |  |  |
| Info    | Check if the attached operation is completed. pbComplete / bComplete will be true if this is done. |                                           |  |  |
| Returns | Error Code.                                                                                        |                                           |  |  |
| Errors  | E_FAIL: ge                                                                                         | error<br>eneral error<br>Complete is NULL |  |  |

| C++:    | HRESULT ceAsyncHandle::Wait(uint uiTimeOutMs)                      |  |
|---------|--------------------------------------------------------------------|--|
| NET:    | <pre>Int32 ceAsyncHandle.Wait(uint uiTimeOutMs)</pre>              |  |
| Info    | Wait uiTimeOutMs milliseconds for transfer completion.             |  |
| Returns | Error Code.                                                        |  |
| Errors  | S_OK:no errorE_FAIL:general errorE_TIMEOUT:operation has timed out |  |

# D FPGA design

# FPGA source code copyright information

This source code is copyrighted by CESYS GmbH / GERMANY, unless otherwise noted.

### FPGA source code license

THIS SOURCECODE IS NOT FREE! IT IS FOR USE TOGETHER WITH THE CESYS USB3FPGA USB CARD (ARTICLE-NR.: C 1030-2805) ONLY! YOU ARE NOT ALLOWED TO MODIFY AND DISTRIBUTE OR USE IT WITH ANY OTHER HARDWARE, SOFTWARE OR ANY OTHER KIND OF ASIC OR PROGRAMMABLE LOGIC DESIGN WITHOUT THE EXPLICIT PERMISSION OF THE COPYRIGHT HOLDER!

### Disclaimer of warranty

THIS SOURCECODE IS DISTRIBUTED IN THE HOPE THAT IT WILL BE USEFUL, BUT THERE IS NO WARRANTY OR SUPPORT FOR THIS SOURCECODE. THE COPYRIGHT HOLDER PROVIDES THIS SOURCECODE "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THIS SOURCECODE IS WITH YOU. SHOULD THIS SOURCECODE PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.

IN NO EVENT WILL THE COPYRIGHT HOLDER BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THIS SOURCECODE (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THIS SOURCECODE TO OPERATE WITH ANY OTHER SOFTWARE-PROGRAMS, HARDWARE-CIRCUITS OR ANY OTHER KIND OF ASIC OR PROGRAMMABLE LOGIC DESIGN), EVEN IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.

# Files

| src/gpif_interface.vhd<br>src/sync_fifo_1Kx16.vhd<br>src/demo_application.vhd<br>demo_application.ucf<br>gpif_interface.ise<br>demo_application.bin | <ul> <li>: pinout and timing constraints needed for the GPIF</li> <li>: project-file for ISE vers. 9.1.03i</li> <li>: binary configuration file of demo_application.vhd after<br/>synthesis (check ISE</li> <li>=&gt;process "Generate Programming File"</li> <li>=&gt;"Properties"</li> <li>=&gt;"General Options"</li> <li>=&gt;"Create Binary Configuration File"</li> </ul> |
|-----------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|                                                                                                                                                     | for using *.bin-files!)                                                                                                                                                                                                                                                                                                                                                         |
|                                                                                                                                                     | , ,                                                                                                                                                                                                                                                                                                                                                                             |

### Modules

For user-applications with USB-transfer only the modules **gpif\_interface.vhd**, **sync\_fifo\_1Kx16.vhd** and the constraints from **demo\_application.ucf** are needed. Two types of transfers are supported:

- 1. Single read/write with 8-bit address
- 2. FIFO read/write

### Entity gpif\_interface

This entity encapsulates all functionality, which is needed for USB-transfers over FX-2-USB-Controller's GPIF. It is strongly recommended not to modify neither this module nor any other underlying entity!

There are two groups of port-signals.

One group contains the system-input-signals **IFCLK\_i**, **RESET\_i** and the GPIFcontrol-signals, which are labeled with the prefix **GPIF\_**. These signals must be connected directly to the appropriate FPGA-Pins in the top-level entity!

The other group contains the system-output-signals **clk\_o**, **rst\_o** and the usercontrol-signals, which are labeled with the prefix **app\_**. Only this one is needed for USB-transfers in user-applications! All user-control-signals are synchronous to **clk\_o**.

For every interface in this group you will find an appropriate function in the C++/C#-API:

| port(s) | function(s)    | notes                                                           |
|---------|----------------|-----------------------------------------------------------------|
| rst_o   | ResetFPGA()    | pulses <b>rst_o</b>                                             |
| clk_o   | SetGPIFSpeed() | changes <b>clk_o</b> -frequency<br>between 30 MHz and 48<br>MHz |

| port(s)                                                                                              | function(s)                       | notes                                                                        |
|------------------------------------------------------------------------------------------------------|-----------------------------------|------------------------------------------------------------------------------|
| app_we_o<br>app_adr_o(7 downto 0)<br>app_data_o(15 downto 0)<br>app_data_i(15 downto 0)              | ReadRegister()<br>WriteRegister() | single read/write with 8-bit<br>address, the LSBs of the<br>address are used |
| app_fifo_wr_i<br>app_fifo_data_i(15 downto 0)<br>app_fifo_full_o<br>app_fifo_wr_count_o(9 downto 0)  | ReadBulk()                        | FIFO transfer FPGA =><br>HOST                                                |
| app_fifo_rd_i<br>app_fifo_data_o(15 downto 0)<br>app_fifo_empty_o<br>app_fifo_rd_count_o(9 downto 0) | WriteBulk ()                      | FIFO trnasfer HOST =><br>FPGA                                                |

There is something special about the signal app\_usb\_short\_pkg\_inh\_i (USB short package inhibit) and the FIFO transfer from the FPGA to the HOST. USB-transfers are always package-oriented. If the FPGA-design fills the FIFO slower, than the USB-controller reads out the data, the maximum USB-package-size is not reached and the USB-controller sends a short package to the HOST. So in datastreaming-applications a lot of short packages could be send, which leads to a protocol-overhead. You can set the signal app\_usb\_short\_pkg\_inh\_i to 'logic 1' to avoid this effect and increase the transferrate. If app\_usb\_short\_pkg\_inh\_i = '1' then the USB-controller waits until the maximum USB-package-size is reached, before sending the data to the HOST. This feature should only be used for datastreams, which never end. Otherwise you could get a timeout in your ReadBulk()-function, because the USB-controller waits until the next package is complete, which possibly never happens.

Please take a look at the waveforms and the sourcecode examples at the end of this document to find out, how the other user-control-signals have to be used!

### Entity sync\_fifo\_1Kx16

This entity is a synchronous FIFO internally used in the module **gpif\_interface**. Two of these FIFOs are used. One for each direction. Each FIFO has 1023 entries for 16-bit-words. The FIFO-levels are reported by the signals **app\_fifo\_wr\_count\_o** and **app\_fifo\_rd\_count\_o**.

#### Entity demo\_application

This entity connects the external ports of the module **gpif\_interface** to the top level ports of the FPGA-design. Register-read/write, bulk-read/write and simple I/O-operations (LED on/off) are demonstrated here. There is a switch between data-loopback and infinite data-source/sink for bulk-transfers. The bulk-transfer from and to onboard SRAM over USB is demonstrated as well. Start- and stop-SRAM-addresses and data direction are defined by register-writes. Then a finite state machine copies the data from FIFOs to SRAM and vice versa.

### Waveforms



Figure 7: Single-Read/Write



Figure 8: FIFO-Transfer FPGA => HOST

This waveform demonstrates the behavior of **app\_fifo\_full\_o** and **app\_fifo\_wr\_count\_o** when there is no transaction on the USB-controller side of the FIFO. During simultaneous FIFO-read- and FIFO-write-transactions, the signals do not change. The signal **app\_fifo\_full\_o** will be cleared and **app\_fifo\_wr\_count\_o** will decrease, if there are read-transactions on the USB-controller side, but no write-transactions on the application side.



Figure 9: FIFO-Transfer HOST => FPGA

This waveform demonstrates the behavior of **app\_fifo\_empty\_o** and **app\_fifo\_rd\_count\_o** when there is no transaction on the USB-controller side of the FIFO. During simultaneous FIFO-read- and FIFO-write-transactions, the signals do not change. The signal **app\_fifo\_empty\_o** will be cleared and

**app\_fifo\_rd\_count\_o** will increase, if there are write-transactions on the USB-controller side, but no read-transactions on the application side. Please note the one clock-cycle delay between **app\_fifo\_rd\_i** and **app\_fifo\_data\_o**!

# Code samples

The following extracts of VHDL-code shows you some possible implementations of single-read/write and FIFO-transfer data communication:

```
type arr std16 is array(natural range <>) of
  std logic vector(15 downto 0);
  signal gp reg : arr std16(3 downto 0) := (others => (others => '0'));
      . . . .
 inst gpif interface : gpif interface
 port map
  (
      . . . .
   rst o => rst,
    clk o => clk,
    app we o => app we,
    app adr o => app adr,
    app data o => app datao,
    app data i => app datai,
    app fifo wr i => app fifo wr,
    app fifo data i => app fifo datai,
    app fifo full o => app fifo full,
    app fifo wr count o => app fifo wr count,
    app fifo rd i => app fifo rd,
    app fifo data o => app fifo datao,
    app fifo empty o => app fifo empty,
    app fifo rd count o => app fifo rd count,
    app usb short pkg inh i => app usb short pkg inh,
    debug => open
 );
      . . . .
 register read:
 process(app adr, gp reg)
 begin
   app datai <= (others => '0');
    for i in 0 to 3 loop
     if i = TO INTEGER (unsigned (app adr (1 downto 0))) then
        app datai <= gp reg(i);</pre>
      end if;
    end loop;
 end process;
```

```
-- or alternative implementation
register read:
process(app adr, gp reg)
begin
  app datai <= (others => '0');
  case app adr(1 downto 0) is
    when b<sup>"</sup>00" =>
      app datai <= gp reg(0);</pre>
    when b"01" =>
      app datai <= gp_reg(1);</pre>
    when b"10" =>
      app datai <= gp_reg(2);</pre>
    when b"11" =>
      app_datai <= gp_reg(3);</pre>
    when others => null;
  end case;
end process;
    . . . .
register write:
process(rst, clk)
begin
  if rst = '1' then
    gp reg <= (others => (others => '0'));
  elsif clk'event and clk = '1' then
    if app_we = '1' then
      for \overline{i} in 0 to 3 loop
        if i = TO INTEGER (unsigned (app adr (1 downto 0))) then
          gp reg(i) <= app datao;</pre>
        end if;
      end loop;
    end if;
  end if;
end process;
-- or alternative implementation
register_write:
process(rst, clk)
begin
  if rst = '1' then
    gp reg <= (others => (others => '0'));
  elsif clk'event and clk = '1' then
    if app_we = '1' then
      case app adr(1 downto 0) is
        when b"00" =>
          gp reg(0) <= app datao;</pre>
        when b"01" =>
          qp req(1) \leq app datao;
        when b"10" =>
          gp reg(2) <= app_datao;</pre>
        when b"11" =>
          gp reg(3) <= app datao;</pre>
        when others => null;
      end case;
    end if;
  end if;
end process;
```

. . . .

```
-- FIFO-transfer loopback example
signal app fifo almost full : std logic := '0';
signal app fifo rd delayed : std logic := '0';
    . . . .
app fifo wr <= app fifo rd delayed;</pre>
app fifo rd <= (not app fifo empty) and (not app fifo almost full);
app fifo datai <= app fifo datao;</pre>
process(clk)
begin
  if clk'event and clk = '1' then
    app fifo rd delayed <= app fifo rd;</pre>
  end if;
end process;
process (app fifo wr count)
begin
  if unsigned(app fifo wr count) >= 1022 then
    app_fifo_almost_full <= '1';</pre>
  else
    app fifo almost full <= '0';</pre>
  end if;
end process;
-- or alternative implementation
app fifo almost full <= '1'
                        when
                           app_fifo_wr_count(9 downto 1) = (x"FF" & '1')
                        else '0';
```

# **E** Additional information

# Newsgroups

There are several newsgroups that discuss FPGA and VHDL related themes. Two of them are:

comp.arch.fpga comp.lang.vhdl

# Links

For further information about FPGA and VHDL may be found through links located on CESYS website: <u>www.cesys.com.</u>

# Books

"VHDL Design, Representation and Synthesis" James R.Armstrong, F.Gail Gray Prentice Hall, ISBN 0-13-021670-4

# F Table of contents

|          | Copyright information                     |   |
|----------|-------------------------------------------|---|
| <u>A</u> | <u>Overview</u>                           | 1 |
|          | Summary of USB3FPGA                       | 1 |
|          | Feature list                              |   |
|          | Minimum requirements                      | 1 |
|          | FPGA Design Tools                         |   |
|          | Windows XP Quick-start installation guide |   |
| B        | Hardware                                  | 1 |
|          | SPARTAN-3E FPGA                           | 1 |
|          | Connector diagram                         |   |
|          | Board dimensions.                         |   |
|          | Power supply                              |   |
|          | USB 2.0 interface                         |   |
|          | <u>FPGA pin connections</u>               |   |
|          | LEDs.                                     |   |
|          | <u>FPGA Testpins</u>                      |   |
|          | Expansion port CON1                       |   |
|          | Clock signals and RESET                   |   |
|          | FPGA JTAG port                            |   |
|          | Memory interface                          | 8 |
|          | Undocumented IO.                          |   |
| <u>C</u> | <u>Software</u>                           | 1 |
|          | Files                                     | 1 |
|          | Driver installation                       |   |
|          | Diagnostics                               |   |
|          | <br>API                                   |   |
|          | General                                   |   |
|          | <u>ceUSB3 C++ API specs</u>               |   |
|          | ceUSB3 NET API specs                      |   |
|          | How to use the API                        |   |
|          | class ceDevice                            |   |
|          | <u>class celnfo</u>                       |   |
|          | class ceFPGA                              |   |
|          | class ceAsyncHandle.                      |   |
| D        | FPGA design                               | 1 |

|          | FPGA source code copyright information | .1 |
|----------|----------------------------------------|----|
|          | FPGA source code license               | 1  |
|          | Disclaimer of warranty                 | .1 |
|          | Files                                  | .2 |
|          | Modules                                | .2 |
|          | Entity gpif_interface                  | .2 |
|          | Entity sync_fifo_1Kx16                 | .3 |
|          | Entity demo application.               | .3 |
|          | Waveforms.                             | .4 |
|          | Code samples.                          | .5 |
|          |                                        |    |
| E        | Additional information                 | .1 |
|          | Newsgroups                             | .1 |
|          | Links                                  | .1 |
|          | Books.                                 | .1 |
|          |                                        |    |
| <u>F</u> | Table of contents                      | .1 |