# PULPissimo: Datasheet



The PULP team pulp-info@list.ee.ethz.ch

# **Contents**

| 1 | Ove  | rview   |                                             | 3  |
|---|------|---------|---------------------------------------------|----|
| 2 | Men  | nory Ma | ар                                          | 5  |
| 3 | CPU  | Core    |                                             | 6  |
| 4 | Adva | anced [ | Debug Unit                                  | 8  |
| 5 | Peri | pherals |                                             | 9  |
|   | 5.1  | FLL .   |                                             | 10 |
|   | 5.2  | GPIO    |                                             | 11 |
|   |      | 5.2.1   | PADDIR (Pad Direction)                      | 11 |
|   |      | 5.2.2   | PADIN (Input Values)                        | 11 |
|   |      | 5.2.3   | PADOUT (Output Values)                      | 11 |
|   |      | 5.2.4   | INTEN (Interrupt Enable)                    | 12 |
|   |      | 5.2.5   | INTTYPE0 (Interrupt Type 0)                 | 12 |
|   |      | 5.2.6   | INTTYPE1 (Interrupt Type 1)                 | 12 |
|   |      | 5.2.7   | INTSTATUS (Interrupt Status)                | 13 |
|   |      | 5.2.8   | GPIOEN (GPIO Enable)                        | 13 |
|   |      | 5.2.9   | PADCFG0-7 (Pad Configuration Registers 0-7) | 13 |
|   | 5.3  | SoC C   | ontrol                                      | 15 |
|   |      | 5.3.1   | Info                                        | 15 |
|   |      | 5.3.2   | Boot Address                                | 15 |
|   |      | 5.3.3   | Fetch Enable                                | 15 |
|   |      | 5.3.4   | PAD Mux                                     | 16 |
|   |      | 5.3.5   | PAD Configuration                           | 16 |
|   |      | 5.3.6   | JTAG Register                               | 16 |
|   |      | 5.3.7   | Core Status                                 | 17 |
|   |      | 5.3.8   | FLL Clock Select                            | 17 |
|   | 5.4  | Event/  | Interrupt Controller                        | 18 |
|   |      | 5.4.1   | Mask                                        | 18 |
|   |      | 5.4.2   | Interrupt                                   | 18 |
|   |      | 5.4.3   | Int Ack                                     | 18 |
|   |      | 5.4.4   | FIFO Content                                | 19 |
|   | 5.5  | Debug   |                                             | 20 |

# 1 Overview

PULPISSIMO is a 32 bit RI5CY single-core System-on-a-Chip. PULPISSIMO is the second version of the PULPINO system and it can be extended with the multi-core cluster of the PULP project.

Differently from the simpler PULPINO system, PULPISSIMO uses a more complex memory subsystem, an autonoumous I/O subsystem which uses the uDMA, new peripherals (eg the camera interface) and a new SDK.

Figure 1.1 shows a simplified block diagram of the SoC. As for PULPINO, PULPISSIMO can be configured at design time to use either the RISC-V or ZERO-RISCY. The peripherals are connected to the UDMA which transfers the date to the memory subsystem efficiently. The JTAG and the AXI plug have also access to the SoC. The AXI plug can be used to extend the microcontroller with a multi-core cluster or an accelerator. As for PULPINO, the advanced debug unit is used to access to system and core registers, memories and memory-mapped IO via JTAG. A logarithmic interconnect allows to link the core and the UDMA to the memory banks simultaneously.



Figure 1.1: PULPISSIMO Overview.

PULPISSIMO is mainly targeted at RTL simulation and ASICs. The FPGA versions has not yet been implemented.

# 2 Memory Map

Figure 2.1 shows the default memory-map of PULPISSIMO, whereas Please, consult the UDMA documentation for the peripherals attached to the UDMA memory-map of configuration.



Figure 2.1: PULPISSIMO memory-map.

# 3 CPU Core

PULPISSIMO supports both the RISC-V and the ZERO-RISCY RI5CY core. The two cores have the same external interfaces and are thus plug-compatible. Figure 3.1 and 3.2 show the two cores architectures.

For debugging purposes, all core registers have been memory mapped which allows to them to be accessed over the logaritmic-interconnect subsystem. The debug unit inside the core handles the request over this bus and reads/sets the core registers and/or halts the core.

The core supports performance counters. Those are mainly used for counting core internal events like stalls, but it is possible to count core-external events as well. For this purpose there is the <code>ext\_perf\_counters\_i</code> port where arbitrary events can be attached. The core then increases its internal performance counter for this event type every time a logic high is seen on this port.



Figure 3.1: RISCY core overview

Take a look at the cores documentation for more details.



Figure 3.2: zero-riscy core overview

# 4 Advanced Debug Unit

The advanced debug unit has now a logaritmic-interconnect master interface to access peripherals and memories. In contrast to PULPINO the adv. debug unit does no longer uses AXI to communicate with the system. All core registers are memory mapped which means that they can be read over the logaritmic-interconnect interface.

For more details please take a look at the documentation of the advanced debug unit.

# 5 Peripherals

Most of the peripherals in PULPISSIMO are connected to the uDMA subsystem which efficiently handles all the data-transfers autonoumsly. The uDMA must be programmed by the core via memory-mapped read and write operations to receive commands.

See the UDMA documentation for more details under the UDMA repository.

The GPIO, timers, event unit and event generator, debug and the FLLs are not connected to the uDMA instead but to the APB bus. Following a brief overview about these units is given.

# 5.1 FLL

PULPISSIMO containts 3 FLLs. One FLL is meant for generating the clock for the peripheral domain, one for the core domain (core, memories, event unit etc) and one is meant for the cluster. The latter is not used.

All the FLLs can be bypassed by writing to the JTAG register before the reset signal is asserted. See Section 5.3 for more details about the bypass register.

See the FLL documentation for more details under the FLL repository.

# **5.2 GPIO**

Table 5.1: GPIO Signals

| Signal                 | Direction | Description                       |
|------------------------|-----------|-----------------------------------|
| gpio_in[31:0]          | input     | Transmit Data                     |
| gpio_out[31:0]         | output    | Receive Data                      |
| gpio_dir[31:0]         | output    | Request to Send                   |
| gpio_padcfg[5:0][31:0] | output    | Pad Configuration                 |
| interrupt              | output    | Interrupt (Rise or Fall or Level) |

# 5.2.1 PADDIR (Pad Direction)

Address: 0x1A10\_1000 Reset Value: 0x0000\_0000

| 3 | 31 | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |        |
|---|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|--------|
|   | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D  | D | D | D | D | D | D | D | D | D | D | PADDIR |

Bit 31:0 PADDIR: Pad Direction.

Control the direction of each of the GPIO pads. A value of 1 means it is configured as an output, while 0 configures it as an input.

### 5.2.2 PADIN (Input Values)

Address: 0x1A10\_1004 Reset Value: 0x0000\_0000

| 31 | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |       |
|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|-------|
| I  | I  | I  | I  | I  | I  | I  | Ι  | Ι  | Ι  | Ι  | Ι  | Ι  | I  | I  | I  | I  | Ι  | Ι  | Ι  | I  | Ι  | I | Ι | Ι | I | I | I | I | I | Ι | Ι | PADIN |

Bit 31:0 PADIN: Input Values.

# 5.2.3 PADOUT (Output Values)

Address: 0x1A10\_1008 Reset Value: 0x0000\_0000



#### Bit 31:0 PADOUT: Output Values.

## 5.2.4 INTEN (Interrupt Enable)

Address: 0x1A10\_100C Reset Value: 0x0000\_0000

| 31 | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9  | 8  | 7  | 6  | 5  | 4  | 3  | 2  | 1  | 0  |       |
|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|-------|
| ΙТ | ΙТ | ΙТ | ІТ | ІТ | ΙТ | ΙТ | IT | IT | ІТ | ІТ | ІТ | ΙТ | ΙТ | ΙТ | ІТ | ΙТ | ΙΤ | ΙТ | ІТ | ΙТ | ІТ | ІТ | ΙТ | IΤ | ΙТ | ІТ | ІТ | ІТ | IΤ | ΙΤ | IΤ | INTEN |

## Bit 31:0 INTEN: Interrupt Enable.

Interrupt enable per input bit. INTTYPE0 and INTTYPE1 control the interrupt triggering behavior.

There are four triggers available

- INTTYPEO = 0, INTTYPE1 = 0: Level 1
- INTTYPEO = 1, INTTYPE1 = 0: Level 0
- INTTYPEO = 0, INTTYPE1 = 1: Rise
- INTTYPEO = 1, INTTYPE1 = 1: Fall

## 5.2.5 INTTYPE0 (Interrupt Type 0)

Address: 0x1A10\_1010 Reset Value: 0x0000\_0000



#### Bit 31:0 INTTYPE0: Interrupt Type 0.

Controls the interrupt trigger behavior together with INTTYPE1. Use INTEN to enable interrupts first.

## 5.2.6 INTTYPE1 (Interrupt Type 1)

Address: 0x1A10\_1014 Reset Value: 0x0000\_0000



#### Bit 31:0 INTTYPE1: Interrupt Type 1.

Controls the interrupt trigger behavior together with INTTYPE0. Use INTEN to enable interrupts first.

# 5.2.7 INTSTATUS (Interrupt Status)

Address: 0x1A10\_1018 Reset Value: 0x0000\_0000

| 3 | 1 3 | 80 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |           |
|---|-----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|-----------|
| 5 | 3 3 | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | s | S | S | S | S | S | S | S | S | S | INTSTATUS |

# Bit 31:0 INTSTATUS: Interrupt Status.

Contains interrupt status per GPIO line. The status register is cleared when read. Similarly the **interrupt** line is high while a bit is set in interrupt status and will be deasserted when the status register is read.

# 5.2.8 GPIOEN (GPIO Enable)

Address: 0x1A10\_101C Reset Value: 0x0000\_0000

| 31 | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |        |
|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|--------|
| S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  | S  |    | S  | S  | S  | S  | S  | S  | S  | S  | S  | S | S | S | S | S | S | S | S | S | S | GPIOEN |

#### Bit 31:0 GPIOEN: GPIO Enable.

Contains the enable bit per GPIO line.

## 5.2.9 PADCFG0-7 (Pad Configuration Registers 0-7)

Address: 0x1A10\_1020 - 0x1A10\_103C

Reset Value: 0x0000\_0000

| 3 | 1 3 | 0 : | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |           |
|---|-----|-----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|-----------|
| ] | P I | 0   | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р  | Р | Р | Р | Р | Р | Р | Р | Р | Р | Р | PADCFG0-7 |

#### Bit 31:0 PADCFG0-7: Pad Configuration Registers.

The pad configuration registers control various aspects of the pads that are typically used in ASICs, e.g. drive strength, Schmitt-Triggers, Slew Rate, etc. Since those configuration parameters depend on the exact pads used, each implementation is free to use the PADCFG0-7 registers in every way it wants and also leave them unconnected, if unneeded.

Writing to the PADOUTSET address (0x1A10 $_1040$ ), the content of the PADOUT register is updated with its content "ored" with the write data.

Writing to the PADOUTCLR address ( $0x1A10\_1044$ ), the content of the PADOUT register is updated with its content "anded" with the inverted write data.

# 5.3 SoC Control

PULPISSIMO features a small and simple APB peripheral which provides information about the platform and provides the means for pad muxing on the ASIC.

The following registers can be accessed.

#### 5.3.1 Info

Address: 0x1A10\_4000 Reset Value: 0x0000\_0000

| $31\ 30\ 29\ 28\ 27\ 26\ 25\ 24\ 23\ 22\ 21\ 20\ 19\ 18\ 17\ 16$ | $6\ 15\ 14\ 13\ 12\ 11\ 10\ 9\ 8\ 7\ 6\ 5\ 4\ 3\ 2\ 1\ 0$ | _    |
|------------------------------------------------------------------|-----------------------------------------------------------|------|
| Number of Cores                                                  | Number of Clusters                                        | INFO |

Bit 31:0 Info: This register holds the number of clusters and the number of cores in the each cluster. It is a read-only register.

#### 5.3.2 Boot Address

Address: 0x1A10\_4004 Reset Value: 0x1A10\_0000

| 31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 |          |
|-------------------------------------------------------------------------------------|----------|
| Boot Address                                                                        | BOOT_ADR |

Bit 31:0 Boot Address: This register holds the boot address.

#### 5.3.3 Fetch Enable

Address: 0x1A10\_4008 Reset Value: 0x0000\_0001



Bit 31:0 Fetch Enable: This register contains the value of the fetch enable signal of the core.

#### 5.3.4 PAD Mux

Address: 0x1A10\_4010 - 0x1A10\_401C

**Reset Value:** 0x0000\_0000

31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0

PADMUX

PAD\_MUX

Bit 31:0 PADMUX: The content of these registers can be used to multiplex pads when targeting an ASIC. The first register (0x1A10\_4010) can be used to sets the mux (2 bit select) from pin 0 (bits [1:0]) to 15 (bits [31:30]). The second register (0x1A10\_4014) can be used to sets the mux (2 bit select) from pin 16 (bits [1:0]) to 31 (bits [31:30]). The third register (0x1A10\_4018) can be used to sets the mux (2 bit select) from pin 32 (bits [1:0]) to 47 (bits [31:30]). The forth register (0x1A10\_401C) can be used to sets the mux (2 bit select) from pin 48 (bits [1:0]) to 63 (bits [31:30]).

## 5.3.5 PAD Configuration

Address: 0x1A10\_4020 - 0x1A10\_405C

Reset Value: 0x0000\_0000

PAD Configuration PAD CFG0-15

Bit 31:0 PAD CFG0-15: These 16 registers can be used for ASIC targets to configure pads, e.g. pull up, pull down values.

#### 5.3.6 JTAG Register

Address: 0x1A10\_4074 Reset Value: 0x0000\_0000

Bit 31:0 JTAG Register: This register contains the value of the input from the JTAG and can be used to write 8bit in the JTAG output register for system-to-JTAG communications.

#### 5.3.7 Core Status

 $Address: 0x1A10_40A0$  and  $0x1A10_40C0$ 

Reset Value: 0x0000\_0001

31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0

Core Status

CORE\_STATUS

Bit 31:0 Core Status: These 2 registers contain the status of the system for testing/verification purposes like End Of Computation. The 0x1A10\_40C0 register is read-only.

#### 5.3.8 FLL Clock Select

Address: 0x1A10\_40C8 Reset Value: 0x0000\_0000

Unused Settle Color of the Settle Color of the

Bit 31:0 FLL Clock Select: This register contains whether the system clock is coming from the FLL or the FLL is bypassed. It is a read-only register by the core but it can be written via JTAG.

# 5.4 Event/Interrupt Controller

PULPISSIMO features a lightweight event and interrupt controller which supports vectorized interrupts and events of up to 32 lines. It contains a FIFO of events from the peripherals or SW events. When an interrupt is ready and it is enabled (not masked), the unit sends the 5bit ID to the core and the interrupt request line is raised up. If the core takes the interrupt, it replies with the ID of the interrupt taken and the acknowledge signal. The communication between the interrupt controller and the core is completly asynchronous. Note that the interrupt controller can change the interrupt ID anytime but it must rely on the ID sent by the core to know which interrupt has been taken. This is an important feature that covers the situation where a higher priority interrupt request prevent another one that has been already sent to the core. Depending on the core state and core interrupt enable, the interrupt can be accepted within a couple of clock cycles.

#### 5.4.1 Mask

Address: 0x1A10\_9000 Reset Value: 0x0000\_0000



Bit 31:0 MASK: This register contains the MASK (interrupt enable) for each of the 32 interrupts or events. Writing to 0x1A10\_9004 sets the bits of the MASK register selected. Writing to 0x1A10\_9008 clears the bits of the MASK register selected.

#### 5.4.2 Interrupt

Address: 0x1A10\_900C Reset Value: 0x0000\_0000



Bit 31:0 INT: This register contains the pending interrupts or events. Writing to 0x1A10\_9010 sets the bits of the INT register selected. Writing to 0x1A10\_9014 clears the bits of the INT register selected.

#### 5.4.3 Int Ack

Address: 0x1A10\_9018 Reset Value: 0x0000\_0000

| 31 | 30 | 29 | 28 | 27 | 26 | 25 | 24 | 23 | 22 | 21 | 20 | 19 | 18 | 17 | 16 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |     |
|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|-----|
| Ι  | I  | Ι  | Ι  | I  | I  | I  | Ι  | I  | I  | I  | Ι  | Ι  | I  | Ι  | Ι  | Ι  | Ι  | Ι  | I  | Ι  | Ι  | I | I | I | I | Ι | I | I | I | I | I | ACK |

Bit 31:0 ACK: This register contains the ACK (interrupt enable) for each of the 32 interrupts or events. Writing to  $0x1A10\_901C$  sets the bits of the ACK register selected. Writing to  $0x1A10\_9020$  clears the bits of the ACK register selected.

#### 5.4.4 FIFO Content

Address: 0x1A10\_9024 Reset Value: 0x0000\_0000

| 31 30 29 28 27 26 25 24 23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0 | _         |
|---------------------------------------------------------------------------------------|-----------|
| Fifo Data                                                                             | FIFO_DATA |

Bit 31-0 FIFO\_DATA: Fifo Content.

This is a read-only register that contain the first valid value of the FIFO.

# 5.5 Debug Port

This block contains a apb2per bridge which allows to convert from APB commands to the debug bus. The debug bus directly connects the debug unit of the core to the APB bus and allows for memory mapped debugging. Since all core registers are memory mapped, it is possible to debug the core with a memory interface, rather than a highly specialized interface.

The debug bus signals are described as follows:

Table 5.2: Debug Bus Signals

| Signal          | Direction | Description    |
|-----------------|-----------|----------------|
| dbg_req         | Output    | Request signal |
| dbg_addr        | Output    | Address        |
| dbg_we          | Output    | Write enable   |
| dbg_wdata[31:0] | Output    | Data to write  |
| dbg_gnt         | Input     | Grant          |
| dbg_rvalid      | Input     | Response valid |
| dbg_rdata[31:0] | Input     | Read data      |

The protocol is very similar to the core data, and instruction interfaces. To read or write, the Master has to raise the request with the address, the write enable, and the data to write. As soon as the Slave sets the grant to 1, the transfer is granted. In case of a write operation the rvalid signal rises to one to inform the Master that the transaction was successful. In case of a read operation, the Slave returns the requested data on the rdata line, exactly one cycle after the grant.