

# IOb-UART16550

A NS16550A compatible IP Core

December 17, 2025

User Guide,

0.1 , Build

unknown



# **IOb-UART16550**

USER GUIDE,

0.1 , BUILD

UNKNOWN

---





## Document Version History

| Version | Date              | Person | Changes from previous version |
|---------|-------------------|--------|-------------------------------|
| 0.1     | December 17, 2025 | JTS    | Initial document version      |

# **IOb-UART16550**

USER GUIDE,

0.1 , BUILD

UNKNOWN

---





## Contents

|                                              |          |
|----------------------------------------------|----------|
| <b>1 Introduction</b>                        | <b>1</b> |
| 1.1 Features . . . . .                       | 1        |
| 1.2 Deliverables . . . . .                   | 1        |
| <b>2 Description</b>                         | <b>1</b> |
| 2.1 Block Diagram . . . . .                  | 2        |
| 2.2 Configuration . . . . .                  | 3        |
| 2.3 Interface Signals . . . . .              | 3        |
| 2.4 Control and Status Registers . . . . .   | 4        |
| 2.4.1 Configuration . . . . .                | 4        |
| <b>3 Usage</b>                               | <b>5</b> |
| 3.1 Instantiation . . . . .                  | 5        |
| 3.2 Simulation . . . . .                     | 5        |
| <b>4 Baremetal Drivers</b>                   | <b>6</b> |
| 4.1 iob_uart16550.h File Reference . . . . . | 6        |
| 4.1.1 Detailed Description . . . . .         | 7        |
| 4.1.2 Function Documentation . . . . .       | 8        |

## List of Tables

|   |                                                  |   |
|---|--------------------------------------------------|---|
| 1 | Table of subblocks in the core.                  | 2 |
| 2 | General operation group . . . . .                | 3 |
| 3 | Clock, clock enable and reset . . . . .          | 3 |
| 4 | Control and Status Registers interface . . . . . | 3 |
| 5 | RS232 interface . . . . .                        | 4 |
| 6 | UART16550 interrupt related signals . . . . .    | 4 |
| 7 | General Registers. . . . .                       | 5 |



## List of Figures

|   |                                                         |   |
|---|---------------------------------------------------------|---|
| 1 | IP Core Symbol . . . . .                                | 1 |
| 2 | High-Level Block Diagram . . . . .                      | 2 |
| 3 | Core Instance and Required Surrounding Blocks . . . . . | 5 |
| 4 | Testbench Block Diagram . . . . .                       | 6 |

# **IOb-UART16550**

USER GUIDE,

0.1 , BUILD

UNKNOWN

---



## 1 Introduction

The UART (Universal Asynchronous Receiver/Transmitter) core provides serial communication capabilities, which allow communication with modem or other external devices, like another computer using a serial cable and RS232 protocol. This core is designed to be maximally compatible with the industry standard National Semiconductors' 16550A device.



Figure 1: IP Core Symbol

### 1.1 Features

- Support for multiple interface types (selectable): IOb, AXI, AXI-Lite, WISHBONE, APB
- FIFO only operation
- Register level and functionality compatibility with NS16550A (but not 16450).
- Debug Interface in 32-bit data bus mode.

### 1.2 Deliverables

- Verilog RTL source code synthesizable for ASIC and FPGA
- Verilog testbench and simulation scripts for code coverage
- ASIC synthesis script and timing constraints
- FPGA synthesis scripts and timing constraints
- Bare-metal software driver and example user firmware
- Comprehensive user guide

## 2 Description

This section gives a detailed description of the IP core. The high-level block diagram is presented, along with a description of its subblocks. The parameters and macros that define the core configuration are listed and

explained. The interface signals are enumerated and described; if timing diagrams are needed, they are shown after the interface signals. Finally the Control and Status Registers (CSR) are outlined and explained.

## 2.1 Block Diagram

Figure 2 presents a high-level block diagram of the core, followed by a brief description of each block.



Figure 2: High-Level Block Diagram

The Verilog modules in the top-level entity of the core are described in the following tables. The table elements represent the subblocks in the Block Diagram.

| Module                         | Name                    | Description                                       |
|--------------------------------|-------------------------|---------------------------------------------------|
| iob_universal_converter_iob_wb | iob_universal_converter | Convert CSRs interface into internal wishbone bus |

Table 1: Table of subblocks in the core.



## 2.2 Configuration

The following tables describe the IP core configuration. The core may be configured using macros or parameters:

**'M'** Macro: a Verilog macro or `define` directive is used to include or exclude code segments, to create core configurations that are valid for all instances of the core.

**'P'** Parameter: a Verilog parameter is passed to each instance of the core and defines the configuration of that particular instance.

| Configuration | Type | Min | Typical | Max | Description       |
|---------------|------|-----|---------|-----|-------------------|
| ADDR_W        | P    | NA  | 5       | NA  | Address bus width |
| DATA_W        | P    | NA  | 32      | NA  | Data bus width    |

Table 2: General operation group

The macros not listed above are constants. They improve the code readability and should not be changed by the user. These constants are listed below:

**VERSION** Product version. This 16-bit macro uses nibbles to represent decimal numbers using their binary values. The two most significant nibbles represent the integral part of the version, and the two least significant nibbles represent the decimal part. Value: 16'h0001 = V0.01.

## 2.3 Interface Signals

The interface signals of the core are described in the following tables. Note that the output signals are registered in the core, while the input signals are not.

| Name   | Direction | Width | Description                    |
|--------|-----------|-------|--------------------------------|
| clk_i  | input     | 1     | Clock                          |
| cke_i  | input     | 1     | Clock enable                   |
| arst_i | input     | 1     | Asynchronous active-high reset |

Table 3: Clock, clock enable and reset

| Name         | Direction | Width  | Description               |
|--------------|-----------|--------|---------------------------|
| iob_valid_i  | input     | 1      | Request address is valid. |
| iob_addr_i   | input     | ADDR_W | Byte address.             |
| iob_wdata_i  | input     | 32     | Write data.               |
| iob_wstrb_i  | input     | 32/8   | Write strobe.             |
| iob_rvalid_o | output    | 1      | Read data valid.          |
| iob_rdata_o  | output    | 32     | Read data.                |
| iob_ready_o  | output    | 1      | Interface ready.          |

Table 4: Control and Status Registers interface

| Name        | Direction | Width | Description      |
|-------------|-----------|-------|------------------|
| rs232_rxd_i | input     | 1     | Receive data.    |
| rs232_txd_o | output    | 1     | Transmit data.   |
| rs232_rts_o | output    | 1     | Request to send. |
| rs232_cts_i | input     | 1     | Clear to send.   |

Table 5: RS232 interface

| Name        | Direction | Width | Description           |
|-------------|-----------|-------|-----------------------|
| interrupt_o | output    | 1     | UART interrupt source |

Table 6: UART16550 interrupt related signals

## 2.4 Control and Status Registers

The software accessible registers of the core are described in the following tables. Each subsection corresponds to a specific configuration of the core, since different configurations have different registers available. The tables give information on the name, read/write capability, address, hardware and software width, and a textual description. The addresses are byte aligned and given in hexadecimal format. The hardware width is the number of bits that the register occupies in the hardware, while the software width is the number of bits that the register occupies in the software. In each address, the right-justified field having "Hw width" bits conveys the relevant information. Each register has only one type of access, either read or write, meaning that reading from a write-only register will produce invalid data or writing to a read-only register will not have any effect.

### 2.4.1 Configuration

| Name        | R/W | Addr | Width |    | Default | Description                                                                                                                                                                                                                                                                                            |
|-------------|-----|------|-------|----|---------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|             |     |      | Hw    | Sw |         |                                                                                                                                                                                                                                                                                                        |
| RBR_THR_DLL | RW  | 0x0  | 8     | 8  | 0       | RBR (Receiver Buffer Register) when read, THR (Transmitter Holding Register) when written. When LCR.DLAB bit is set, this address accesses the Divisor Latch LSB (DLL).                                                                                                                                |
| IER_DLM     | RW  | 0x1  | 8     | 8  | 0       | Interrupt Enable Register. When LCR.DLAB bit is set, this address accesses the Divisor Latch MSB (DLM).                                                                                                                                                                                                |
| IIR_FCR     | RW  | 0x2  | 8     | 8  | 193     | IIR (Interrupt Identification Register) when read, FCR (FIFO Control Register) when written.                                                                                                                                                                                                           |
| LCR         | RW  | 0x3  | 8     | 8  | 3       | Line Control Register. The DLAB bit (MSB) controls access to the Divisor Latch registers.                                                                                                                                                                                                              |
| MCR         | W   | 0x4  | 8     | 8  | 0       | Modem Control Register.                                                                                                                                                                                                                                                                                |
| LSR         | R   | 0x5  | 8     | 8  | 96      | Line Status Register.                                                                                                                                                                                                                                                                                  |
| MSR         | R   | 0x6  | 8     | 8  | 0       | Modem Status Register.                                                                                                                                                                                                                                                                                 |
| VERSION     | R   | 0x8  | 16    | 16 | 0001    | Product version. This 16-bit register uses nibbles to represent decimal numbers using their binary values. The two most significant nibbles represent the integral part of the version, and the two least significant nibbles represent the decimal part. For example V12.34 is represented by 0x1234. |

Table 7: General Registers.

## 3 Usage

### 3.1 Instantiation

Figure 3 illustrates how to instantiate the IP core and, if applicable, the required external subblocks.



Figure 3: Core Instance and Required Surrounding Blocks

The RS232 interface that should be connected to an external UART (e.g. a USB-to-serial converter).

The CSRs bus (IOb native by default) should be connected to the desired manager component (e.g. a CPU).

The clock, clock enable, and reset ports can be connected to the desired clock and reset generator.

### 3.2 Simulation

The provided testbench uses the core instance described in Section 3.1. A high-level block diagram of the testbench is shown in Figure 4. The testbench is organized in a modular fashion, with each test described in a separate file. The test suite consists of all the test case files to make adding, modifying, or removing tests easy.

Add here any specific ip core description.

#### System-level Simulation

Upon request, simulation files to run the core embedded in a RISC-V system can be provided. The core is exercised in various modes by the RISC-V processor, using a bare-metal software program written in the C programming language.



Figure 4: Testbench Block Diagram

## 4 Baremetal Drivers

### 4.1 iob\_uart16550.h File Reference

High level iob\_uart16550 core functions.

```
#include <stdarg.h>
#include <stdint.h>
#include <stdlib.h>
```

#### Macros

- **#define UART\_PROGNAME "IOb-UART"**  
*Prefix to IOb-UART16550 specific prints.*
- **#define STX 2**  
*Start text. Signal start of data sequence to be printed.*
- **#define ETX 3**  
*End text. Signal end of data sequence to be printed.*
- **#define EOT 4**  
*End of transmission. Signal end of UART16550 connection.*
- **#define ENQ 5**  
*Enquiry. Signal start of UART16550 connection.*
- **#define ACK 6**  
*Acknowledge. Signal reception of incoming message.*
- **#define FTX 7**  
*File transfer. Signal file transfer request.*
- **#define FRX 8**  
*File reception. Signal file reception request.*



## Functions

- void uart16550\_init (int base\_address, uint16\_t div)  
*Initialize UART16550.*
- int uart16550\_base (int base\_address)  
*Change UART16550 base.*
- void uart16550\_finish ()  
*Close transmission.*
- char uart16550\_txready ()  
*Check if TX is ready.*
- void uart16550\_txwait ()  
*Wait for TX.*
- char uart16550\_rxready ()  
*Check if RX is ready.*
- void uart16550\_rxwait ()  
*Wait for RX Data.*
- void uart16550\_putc (char c)  
*Print char.*
- void uart16550\_puts (const char \*s)  
*Print string.*
- void uart16550\_sendfile (char \*file\_name, int file\_size, char \*mem)  
*Send file.*
- char uart16550\_getc ()  
*Get char.*
- int uart16550\_recvfile (char \*file\_name, char \*mem)  
*Receive file.*

### 4.1.1 Detailed Description

High level iob\_uart16550 core functions.

The present IOb-UART16550 software drivers implement a way to interface with the IOb-UART16550 peripheral for serial communication.

The present drivers provide base functionalities such as:

- initialization and setup
- basic control functions
- single character send and receive functions
- simple protocol for multi byte transfers



## 4.1.2 Function Documentation

### **uart16550\_base()**

```
int uart16550_base (
    int base_address)
```

Change UART16550 base.

Set a new IOb-UART16550 base address.

Returns

Previous base

### **uart16550\_finish()**

```
void uart16550_finish ()
```

Close transmission.

Send end of transmission (EOT) command via UART16550. Active wait until TX transfer is complete. Use this function to close console program.

Returns

void.

### **uart16550\_getc()**

```
char uart16550_getc ()
```

Get char.

Active wait and receive char/byte from UART16550.

Returns

received byte from UART16550.

### **uart16550\_init()**

```
void uart16550_init (
    int base_address,
    uint16_t div)
```

Initialize UART16550.



Reset UART16550, set IOb-UART16550 base address and set the division factor. The division factor is the number of clock cycles per symbol transferred.

For example, for a case with fclk = 100 Mhz for a baudrate of 115200 we should have  $\text{div} = (100 * 10^6 / 115200) = (868)$ .

#### Parameters

|                     |                                                    |
|---------------------|----------------------------------------------------|
| <i>base_address</i> | IOb-UART16550 instance base address in the system. |
| <i>div</i>          | Equal to round (fclk/baudrate).                    |

#### Returns

void.

### **uart16550\_putc()**

```
void uart16550_putc (
    char c)
```

Print char.

Send character via UART16550 to be printed by in console program.

#### Parameters

|          |                     |
|----------|---------------------|
| <i>c</i> | Character to print. |
|----------|---------------------|

#### Returns

void.

### **uart16550\_puts()**

```
void uart16550_puts (
    const char * s)
```

Print string.

Send string via UART16550 to be printed by in console program.

#### Parameters

|          |                                      |
|----------|--------------------------------------|
| <i>s</i> | Pointer to char array to be printed. |
|----------|--------------------------------------|

## IOb-UART16550

USER GUIDE,  
0.1 , BUILD  
UNKNOWN



### Returns

void.

## uart16550\_recvfile()

```
int uart16550_recvfile (
    char * file_name,
    char * mem)
```

Receive file.

Request variable size file via UART16550. Order of commands:

1. Send file receive (FRX) command.
2. Send file\_name.
3. Receive file\_size (in little endian format).
4. Send ACK command.
5. Receive file.

If memory pointer is not initialized, allocates memory for incomming file.

### Parameters

|                  |                                            |
|------------------|--------------------------------------------|
| <i>file_name</i> | Pointer to file name string.               |
| <i>mem</i>       | Pointer in memory to store incomming file. |

### Returns

Size of received file.

## uart16550\_rxready()

```
char uart16550_rxready ()
```

Check if RX is ready.

Check if UART16550 has received data



## Returns

RX ready flag

### **uart16550\_rxwait()**

```
void uart16550_rxwait ()
```

Wait for RX Data.

Active wait for RX incomming data.

## Returns

void.

### **uart16550\_sendfile()**

```
void uart16550_sendfile (
    char * file_name,
    int file_size,
    char * mem)
```

Send file.

Send variable size file via UART16550. Order of commands:

1. Send file transmit (FTX) command.
2. Send *file\_name*.
3. Send *file\_size* (in little endian format).
4. Send file.

## Parameters

|                  |                              |
|------------------|------------------------------|
| <i>file_name</i> | Pointer to file name string. |
| <i>file_size</i> | Size of file to be sent.     |
| <i>mem</i>       | Pointer to file.             |

## Returns

void.

### **uart16550\_txready()**

```
char uart16550_txready ()
```

## **IOb-UART16550**

USER GUIDE,  
0.1 , BUILD  
UNKNOWN



Check if TX is ready.

Check if UART16550 has data to send

Returns

TX ready flag

### **uart16550\_txwait()**

```
void uart16550_txwait ()
```

Wait for TX.

Active wait until TX is ready to process new byte to send.

Returns

void.