



SpiNNaker Application Programming Interface version 0.0 14 August 2013 (1)

ಹ



## About this Document

### Background

SpiNNaker was designed at the University of Manchester within an EPSRC-funded project in collaboration with the University of Southampton, ARM Limited and Silistix Limited. Subsequent development took place within a second EPSRC-funded project which added the universities of Cambridge and Sheffield to the collaboration. The work would not have been possible without EPSRC funding, and the support of the EPSRC and the industrial partners is gratefully acknowledged.

### Intellectual Property rights

All rights to the SpiNNaker design and its associated software are the property of the University of Manchester with the exception of those rights that accrue to the project partners in accordance with the contract terms.

#### Disclaimer

The details in this design document are presented in good faith but no liability can be accepted for errors or inaccuracies. The design of a complex chip multiprocessor and its associated software is a research activity where there are many uncertainties to be faced, and there is no guarantee that a SpiNNaker system will perform in accordance with the specifications presented here.

The APT group in the School of Computer Science at the University of Manchester was responsible for all of the architectural and logic design of the SpiNNaker chip, with the exception of synthesizable components supplied by ARM Limited and interconnect components supplied by Silistix Limited. All design verification was also carried out by the APT group. As such the industrial project partners bear no responsibility for the correct functioning of the device.

#### Error notification and feedback

Please email details of any errors, omissions, or suggestions for improvement to Steve Furber <steve.furber@manchester.ac.uk>

#### Change history

| version | date       | changes       |
|---------|------------|---------------|
| 0.0     | 20/05/2011 | Initial draft |

# Contents

| this L    | Oocument                                                                                                                            | <b>2</b>                                                                                                                                                                                                                                                                                                                                 |
|-----------|-------------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| kground   | d                                                                                                                                   | 2                                                                                                                                                                                                                                                                                                                                        |
| ellectual | Property rights                                                                                                                     | 2                                                                                                                                                                                                                                                                                                                                        |
| claimer   |                                                                                                                                     | 2                                                                                                                                                                                                                                                                                                                                        |
| or notifi | cation and feedback                                                                                                                 | 2                                                                                                                                                                                                                                                                                                                                        |
| ange his  | tory                                                                                                                                | 2                                                                                                                                                                                                                                                                                                                                        |
|           |                                                                                                                                     |                                                                                                                                                                                                                                                                                                                                          |
| _         |                                                                                                                                     | 4                                                                                                                                                                                                                                                                                                                                        |
| Event-    | -driven programming model                                                                                                           | 4                                                                                                                                                                                                                                                                                                                                        |
| 0.1.1     | Design considerations                                                                                                               | 5                                                                                                                                                                                                                                                                                                                                        |
| Progra    | amming interface                                                                                                                    | 5                                                                                                                                                                                                                                                                                                                                        |
| 0.2.1     | Events                                                                                                                              | 5                                                                                                                                                                                                                                                                                                                                        |
| 0.2.2     | Callback arguments                                                                                                                  | 6                                                                                                                                                                                                                                                                                                                                        |
| 0.2.3     | Pre-defined constants                                                                                                               | 6                                                                                                                                                                                                                                                                                                                                        |
| 0.2.4     | Pre-defined types                                                                                                                   | 7                                                                                                                                                                                                                                                                                                                                        |
| 0.2.5     | Pre-declared variables                                                                                                              | 8                                                                                                                                                                                                                                                                                                                                        |
| 0.2.6     | Kernel services                                                                                                                     | 8                                                                                                                                                                                                                                                                                                                                        |
|           | 12011101 801 (1008 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1                                                                              |                                                                                                                                                                                                                                                                                                                                          |
|           | ekground<br>claimer<br>or notifi-<br>ange his<br>eation p<br>Event<br>0.1.1<br>Program<br>0.2.1<br>0.2.2<br>0.2.3<br>0.2.4<br>0.2.5 | ckground cllectual Property rights claimer or notification and feedback ange history  cation programming interface (API)  Event-driven programming model  0.1.1 Design considerations  Programming interface  0.2.1 Events  0.2.2 Callback arguments  0.2.3 Pre-defined constants  0.2.4 Pre-defined types  0.2.5 Pre-declared variables |



**(**)





# Application programming interface (API)

#### 0.1Event-driven programming model

The SpiNNaker Programming Model (PM) is a simple, event-driven model. Applications do not control execution flow, they can only indicate the functions, referred to as callbacks, to be executed when specific events occur, such as the arrival of a packet, the completion of a Direct Memory Access (DMA) transfer or the lapse of a periodic time interval. An Application Run-time Kernel (ARK) controls the flow of execution and schedules/dispatches application callback functions when appropriate.



Figure 1: SpiNNaker event-driven programming framework.

Fig. 1 shows the basic architecture of the event-driven framework. Application developers write callback routines that are associated with events of interest and register them at a certain priority with the kernel. When the corresponding event occurs the scheduler either executes the callback immediately and atomically (in the case of a non-queueable callback) or places it into a scheduling queue at a position according to its priority (in case of a queueable callback). When control is returned to the dispatcher (following the completion of a callback) the highest-priority queueable callback is executed. Queueable callbacks do not necessarily execute atomically: they may be pre-empted by non-queueable callbacks if a corresponding event occurs during their execution. The dispatcher goes to sleep (low-power consumption state) if the pending callback queues are empty and will be awakened by an event. Application developers can designate one non-queueable callback as the preeminent callback, which has the highest priority and can pre-empt other non-queueable callbacks as well as all queueable ones.

The preeminent callback is associated with a FIQ interrupt while other non-queueable callbacks are associated with IRQ interrupts. The API provides different functions to



Φ

ന

disable interrupts: spin1\_irq\_disable disables IRQs, spin1\_fiq\_disable disables FIQs while spin1\_int\_disable disables both FIQs and IRQs. The use of spin1\_fiq\_disable may lead to priority inversion.

#### Design considerations 0.1.1

**MANCHESTE** 

- Non-queueable callbacks are available as a method of pre-empting long running tasks with short, high priority tasks. The allocation of application tasks to non-queueable callbacks must be carefully considered. The selection of the preeminent callback can be particularly important. Long-running operations should not be executed in nonqueueable callbacks for fear of starving queueable callbacks.
- Queueable callbacks may require critical sections (i.e., sections that are completed atomically) to prevent pre-emption during access to shared resources. Critical sections may be achieved by disabling interrupts before accessing the shared resource and reenabling them afterwards. Applications are executed in a privileged mode to allow the callback programmer to insert these critical sections. This approach has the risk that it allows the programmer to modify peripherals -such as the system controllerunchecked.
- Non-queueable callbacks may also require critical sections, as they can be pre-empted by the preeminent callback.
- Events –usually triggered by interrupts have priority determined by the programming of the Vectored Interrupt Controller (VIC). This allows priority to be determined when multiple events corresponding to different non-queueable callbacks occur concurrently. It also affects the order in which queueable callbacks of the same priority are queued.

#### 0.2Programming interface

The following sections introduce the events and functions supported by the API.

#### 0.2.1**Events**

The SpiNNaker PM is event-driven: all computation follows from some event. The following events are available to the application:

| event               | trigger                                           |
|---------------------|---------------------------------------------------|
| MC packet received  | reception of a multicast packet                   |
| DMA transfer done   | successful completion of a DMA transfer           |
| Timer tick          | passage of specified period of time               |
| SDP packet received | reception of a SpiNNaker Datagram Protocol packet |
| User event          | software-triggered interrupt                      |

In addition, errors can also generate events:

| — events not yet supported — |                                             |  |
|------------------------------|---------------------------------------------|--|
| event                        | trigger                                     |  |
| MCP parity error             | multicast packet received with wrong parity |  |
| MCP framing error            | wrongly framed multicast packet received    |  |
| DMA transfer error           | unsuccessful completion of a DMA transfer   |  |
| DMA transfer timeout         | DMA transfer is taking too long             |  |



Each of these events is handled by a kernel routine which may schedule or execute an application callback, if one is registered by the application.

### 0.2.2 Callback arguments

MANCHESTER

Callbacks are functions with two unsigned integer arguments (which may be NULL) and no return value. The arguments may be cast into the appropriate types by the callback. The arguments provided to callbacks (where 'none' denotes a superfluous argument) by each event are:

| event               | first argument                  | second argument       |
|---------------------|---------------------------------|-----------------------|
| MC packet received  | uint key                        | uint payload          |
| DMA transfer done   | uint transfer_ID                | uint tag              |
| Timer tick          | uint simulation $_{	ext{time}}$ | uint none             |
| SDP packet received | uint *mailbox                   | uint destination_port |
| User event          | uint arg0                       | uint arg1             |

#### 0.2.3 Pre-defined constants

| logic value | value    | keyword |
|-------------|----------|---------|
| true        | (0 == 0) | TRUE    |
| false       | (0!=0)   | FALSE   |

| function result | value | keyword |
|-----------------|-------|---------|
| failure         | 0     | FAILURE |
| success         | 1     | SUCCESS |

| transfer direction    | value | keyword   |
|-----------------------|-------|-----------|
| read (system to TCM)  | 0     | DMA_READ  |
| write (TCM to system) | 1     | DMA_WRITE |

| packet payload  | value | keyword      |
|-----------------|-------|--------------|
| no payload      | 0     | NO_PAYLOAD   |
| payload present | 1     | WITH_PAYLOAD |

| event               | value | keyword            |
|---------------------|-------|--------------------|
| MC packet received  | 0     | MC_PACKET_RECEIVED |
| DMA transfer done   | 1     | DMA_TRANSFER_DONE  |
| Timer tick          | 2     | TIMER_TICK         |
| SDP packet received | 3     | SDP_PACKET_RX      |
| User event          | 4     | USER_EVENT         |

(1)

ന

(C)

#### 0.2.4 Pre-defined types

MANCHESTER

| type             | value                                                         | size      |
|------------------|---------------------------------------------------------------|-----------|
| uint             | unsigned int                                                  | 32 bits   |
| ushort           | unsigned short                                                | 16 bits   |
| uchar            | unsigned char                                                 | 8 bits    |
| callback_t       | ${\rm void}\ ({\rm *callback\_t})\ ({\rm uint},\ {\rm uint})$ | 32 bits   |
| $sdp\_msg\_t$    | struct (see below)                                            | 292 bytes |
| $diagnostics\_t$ | struct (see below)                                            | 44 bytes  |

#### SDP message structure

```
// SDP message (=292 bytes)
{\bf typedef\ struct\ sdp\_msg}
  struct sdp_msg *next;
                             // Next in free list
  ushort length;
                             // length
  ushort checksum;
                             // checksum (if used)
  // sdp_hdr_t
  uchar flags;
                             // SDP flag byte
  uchar tag;
                             // SDP IPtag
                             // SDP destination port
  uchar dest_port;
  uchar srce_port;
                             // SDP source port
                             // SDP destination address
  ushort dest_addr;
                             // SDP source address
  ushort srce_addr;
  // cmd_hdr_t (optional)
  ushort cmd_rc;
                             // Command/Return Code
                             // Sequence number
  ushort seq;
  uint arg1;
                             // Arg 1
                             // Arg 2
  uint arg2;
                             // Arg 3
  uint arg3;
  // user data (optional)
  uchar data [SDP_BUF_SIZE]; // User data (256 bytes)
  uint _PAD;
                             // Private padding
} sdp_msg_t;
```

### diagnostics variable structure

```
typedef struct
  uint exit_code;
                                    // simulation exit code
  uint warnings;
                                    // warnings type bit map
                                    // total routed MC packets during simulation
  uint total_mc_packets;
                                    // total dumped MC packets by the router
// total discarded MC packets by API
  uint dumped_mc_packets;
  uint discarded_mc_packets;
  uint dma_transfers;
                                    // total DMA transfers requested
                                    // total DMA bursts completed
// dma queue full count
  uint dma_bursts;
  uint dma_queue_full;
  uint task_queue_full;
                                    // task queue full count
  uint tx_packet_queue_full;
                                    // transmitter packet queue full count
                                    // write-back buffer errror count
  uint writeBack_errors;
} diagnostics_t;
```





### 0.2.5 Pre-declared variables

| variable    | type                   | function                                                     |
|-------------|------------------------|--------------------------------------------------------------|
| leadAp      | uchar                  | TRUE if appointed chip-wise application leader               |
| diagnostics | ${\bf diagnostics\_t}$ | returns diagnostic information (if turned on in compilation) |

#### 0.2.6 Kernel services

The kernel provides a number of services to the application programmer:

### Simulation control functions

|                                                             |           |              | Start simulation |  |
|-------------------------------------------------------------|-----------|--------------|------------------|--|
| function                                                    | arguments | description  |                  |  |
| uint spin1_start                                            | void      | no arguments |                  |  |
| returns: EXIT_CODE $(0 = NO ERRORS)$                        |           |              |                  |  |
| notes: • transfers control from the application to the ARK. |           |              |                  |  |
| • use spin1_kill to indicate a non-zero EXIT_CODE.          |           |              |                  |  |

|                                                                  |             |              | Stop simulation |
|------------------------------------------------------------------|-------------|--------------|-----------------|
| function                                                         | arguments   | description  |                 |
| void spin1_stop                                                  | void        | no arguments |                 |
| returns:                                                         | no return v | alue         |                 |
| notes: • transfers control from the ARK back to the application. |             |              |                 |

|                                                                  | Stop simulation and report error |  |  |
|------------------------------------------------------------------|----------------------------------|--|--|
| function                                                         | arguments description            |  |  |
| void spin1_kill uint error error code to report                  |                                  |  |  |
| returns: no return value                                         |                                  |  |  |
| notes: • transfers control from the ARK back to the application. |                                  |  |  |
| • The argument is used as the return value for spin1_start.      |                                  |  |  |

|                           |              | Set the timer tick period           |
|---------------------------|--------------|-------------------------------------|
| function                  | arguments    | description                         |
| void spin1_set_timer_tick | uint period  | timer tick period (in microseconds) |
| returns:                  | no return va | lue                                 |

|                                |                                            | Request simulation time |
|--------------------------------|--------------------------------------------|-------------------------|
| function                       | arguments                                  | description             |
| uint spin1_get_simulation_time | void                                       | no arguments            |
| returns:                       | timer ticks since the start of simulation. |                         |

version 0.0

returns: no return value

• sets the map of the cores that need to synchronise to start the simulation. notes:

• the numbers of chips & cores default to 1, thus no synchronisation is attempted.

#### Core Map Examples

```
// chips are identified using Cartesian coordinates.
// Note that the core map is a uni-dimensional array but
// describes a bi-dimensional array of chips in x-major format
// i.e., the order is (0, 0), (0, 1), \ldots, (1, 0), (1, 1), \ldots
// 2 x 2 core map on SpiNN-2, SpiNN-3 and SpiNN-4 boards - 2 cores on each chip
uint const NUMBER_OF_CHIPS = 4;
                                       // virtual 2 x 2 array of chips
uint core_map[NUMBER_OF_CHIPS] =
  0x6,
              0x6,
                       // (0, 0), (0, 1)
                       // (1, 0), (1, 1)
  0x6.
              0x6
};
// "hexagonal" 8 x 8 core map on SpiNN-4 board - 16 cores on each chip
uint const NUMBER_OF_CHIPS = 64;
                                        // virtual 8 x 8 array of chips
uint core_map[NUMBER_OF_CHIPS] =
  0x1fffe,
              0 \times 1  ff e ,
                          0x1fffe,
                                      0 \times 1  ff e ,
                                                 0,
                                                                                     0.
  0x1fffe,
              0 \times 1  ffe ,
                                      0x1fffe,
                          0 \times 1  ffe ,
                                                 0x1fffe,
                                                              0,
                                                                          0,
                                                                                     0,
  0 \times 1  ffe ,
              0 \times 1  ff e ,
                          0 \times 1  ffe ,
                                      0 \times 1 \, \text{fffe},
                                                 0x1fffe,
                                                             0x1fffe,
                                                                                     0,
                                                                          0,
  0x1fffe,
              0 \times 1  ffe ,
                          0x1fffe,
                                      0 \times 1  ff e ,
                                                 0x1fffe,
                                                              0x1fffe,
                                                                         0x1fffe,
              0 \times 1  ffe ,
                          0x1fffe,
                                      0x1fffe,
                                                 0x1fffe,
                                                             0x1fffe,
                                                                         0x1fffe,
  0x1fffe.
                                                                                     0x1fffe.
                          0x1fffe,
                                                 0 \times 1  ff e ,
  0,
              0x1fffe,
                                      0x1fffe,
                                                             0x1fffe,
                                                                         0x1fffe,
                                                                                     0x1fffe.
                                      0x1fffe,
                                                 0x1fffe,
                                                             0x1fffe,
  0,
              0,
                          0x1fffe,
                                                                         0x1fffe,
                                                                                     0x1fffe.
  0,
                          0,
              0,
                                      0x1fffe,
                                                 0x1fffe, 0x1fffe,
                                                                         0x1fffe,
                                                                                     0 \times 1  fffe
// "notched" 5 x 5 core map on SpiNN-4 board - variable number of cores
uint const NUMBER_OF_CHIPS = 64;
                                         // virtual 8 x 8 array of chips
uint core_map [NUMBER_OF_CHIPS] =
{
  6,
              6.
                                                 O,
                                                              0,
                                                                          0,
                                                                                      0.
              6,
                          2,
  6,
                                      2,
                                                 2,
                                                             0,
                                                                         0,
                                                                                     0,
                                                 2,
                          2,
                                                                         0,
  6,
                                      2,
                                                             0,
                                                                                     0.
  2,
              2,
                          6,
                                      2.
                                                             0,
                                                                          0,
                                                                                     0.
  2,
              2,
                                                 2,
                                      2,
                                                                                     0.
                          2.
                                                             0,
                                                                         0,
  0,
              0,
                          0,
                                      0,
                                                 0,
                                                                                     0,
                                                              0,
  0,
              0,
                          0,
                                      0,
                                                 0,
                                                              0,
                                                                                     0,
                                                                          0.
                                      0,
                                                                                     0
  0.
              0,
                          0,
                                                 0.
                                                              0,
                                                                         0,
};
// NOTE: core maps with "holes" may not synchronise in the current version.
// INCORRECT 8 x 8 core map on SpiNN-4 board - 7 cores on each chip
uint const NUMBER_OF_CHIPS = 64;
                                          // virtual 8 x 8 array of chips
uint core_map [NUMBER_OF_CHIPS] =
  0 \, \mathrm{xfe} ,
                                                                         0,
                                                                                     0,
              0xfe,
                          0xfe,
                                      0xfe,
                                                 0.
                                                              0,
                                                                                     0,
  0xfe,
              0xfe,
                          0xfe,
                                      0xfe,
                                                 0xfe,
                                                              0,
                                                                          0,
              0xfe,
                                                 0 \, \mathrm{xfe},
                                                             0 \, \mathrm{xfe},
                          0 \, \mathrm{xfe} ,
                                                                         0,
                                                                                     0,
  Ο.
                                      0xfe,
  0xfe,
              0xfe,
                          0xfe,
                                      0xfe,
                                                 0xfe,
                                                              0xfe,
                                                                         0xfe,
                                                                                     0,
  0xfe,
              0xfe,
                          0xfe,
                                      0xfe,
                                                 0xfe,
                                                              0xfe,
                                                                         0xfe,
                                                                                     0xfe,
              0xfe,
                          0xfe,
                                                             0xfe,
                                                                         0xfe,
  0,
                                      0xfe,
                                                 0xfe,
                                                                                     0xfe.
  0,
                                      0xfe,
                                                 0xfe,
                                                              0xfe,
                                                                         0xfe,
                                                                                     0xfe,
              0,
                          0xfe,
  0,
              0,
                          0,
                                      0xfe.
                                                 0xfe,
                                                              0xfe.
                                                                         0xfe.
                                                                                     0 \, \mathrm{xfe}
};
```

**(** 

Y

**(**)



returns: no return value

**notes:** • sets the map of the cores that need to synchronise to start the simulation.

• the numbers of chips & cores default to 1, thus no synchronisation is attempted.

#### Core Map Examples

```
// chips are identified using Cartesian coordinates.
// 2 x 2 core map on SpiNN-2, SpiNN-3 and SpiNN-4 boards - 2 cores on each chip
uint const NUMBER_OF_XCHIPS = 2;
                                          // virtual 2 x 2 array of chips
uint const NUMBER_OF_YCHIPS = 2;
uint core_map [NUMBER_OF_XCHIPS] [NUMBER_OF_YCHIPS] =
  \{0x6,
             0x6},
                       // (0, 0), (0, 1)
  \{0x6,
             0x6
                       // (1, 0), (1, 1)
// "hexagonal" 8 x 8 core map on SpiNN-4 board - 16 cores on each chip
uint const NUMBER_OF_XCHIPS = 8;
                                         // virtual 8 x 8 array of chips
uint const NUMBER_OF_YCHIPS = 8;
uint core_map [NUMBER_OF_XCHIPS] [NUMBER_OF_YCHIPS] =
{
  \{0\,x\,1\,fffe\;,\;\;0\,x\,1\,fffe\;,\;\;0\,x\,1\,fffe\;,\;\;
                                                                        0,
                                     0x1fffe,
                                                                                    0},
   \{0\,x\,1\,fffe\;,\;\;0\,x\,1\,fffe\;,\;\;0\,x\,1\,fffe\;,\;\;
                                     0x1fffe,
                                                 0x1fffe,
                                                                                    0},
  \{0x1fffe, 0x1fffe,
                         0x1fffe,
                                     0 \times 1 \, \text{fffe},
                                                 0x1fffe,
                                                             0x1fffe,
                                                                                    0\}\,,
                                                                        0,
  {0x1fffe, 0x1fffe,
                         0x1fffe,
                                     0x1fffe,
                                                 0x1fffe,
                                                             0x1fffe,
                                                                        0x1fffe,
                                                                                    0},
  {0x1fffe, 0x1fffe,
                         0x1fffe,
                                     0x1fffe,
                                                 0x1fffe,
                                                             0x1fffe,
                                                                        0x1fffe,
                                                                                    0x1fffe},
             0 \times 1  ffe,
                         0x1fffe,
                                                                        0 \times 1  ffe ,
                                     0x1fffe,
                                                 0x1fffe,
                                                             0x1fffe,
                                                                                    0 \times 1  ffee \},
  {0,
                                                 0x1fffe,
  {0,
             0,
                         0x1fffe,
                                     0x1fffe,
                                                             0 \times 1  ffe ,
                                                                        0x1fffe,
                                                                                    0x1fffe},
  {0,
                                     0x1fffe,
                                                 0 \times 1  ff fe,
                                                            0 \times 1  ffe ,
             0.
                                                                        0x1fffe,
                                                                                    0x1fffe}
};
// "notched" 4 x 5 core map on SpiNN-4 board - variable number of cores
uint const NUMBER_OF_XCHIPS = 4;
                                         // virtual 4 x 5 array of chips
uint const NUMBER_OF_XCHIPS = 5;
uint core_map [NUMBER_OF_XCHIPS] [NUMBER_OF_YCHIPS] =
  {6,
                                                 \mathbf{O} ,
                         2,
                                                 2\},
  \{6,
                                     2,
              6,
  {6,
                         2,
                                     2,
                                                 2},
              6,
                                                 2}
  \{2,
              2,
                                     2,
                         2,
// NOTE: core maps with "holes" may not synchronise in the current version.
// INCORRECT 6 x 7 core map on SpiNN-4 board - 7 cores on each chip
uint const NUMBER_OF_XCHIPS = 6;
                                         // virtual 6 x 7 array of chips
uint const NUMBER_OF_YCHIPS = 7;
uint core_map [NUMBER_OF_XCHIPS] [NUMBER_OF_YCHIPS] =
                                                               0,
  \{0xfe,
                0xfe,
                           0xfe,
                                       0xfe,
                                                                           0},
                                                                           0),
  \{0xfe,
                0xfe,
                           0xfe,
                                       0xfe,
                                                   0xfe,
                                                               0,
                                       0\,\mathrm{xfe} ,
                0xfe,
                                                                           0,
  {O,
                           0xfe,
                                                   0xfe,
                                                               0xfe.
  {0xfe,
                0xfe,
                           0\,\mathrm{xfe} ,
                                       0xfe,
                                                   0xfe,
                                                               0xfe,
                                                                           0 x fe },
                0xfe,
  \{0xfe,
                           0xfe,
                                       0xfe,
                                                   0xfe,
                                                               0xfe,
                                                                           0 x fe },
  \{0,
                0xfe,
                           0xfe,
                                       0xfe.
                                                   0xfe,
                                                               0xfe.
                                                                           0 \, x \, fe \}
};
```

### Event management functions

MANCHESTER

|                        | Register callback to be executed when event_id occurs |                                  |  |
|------------------------|-------------------------------------------------------|----------------------------------|--|
| function               | arguments                                             | description                      |  |
| void spin1_callback_on | $uint event\_id$                                      | event that triggers callback     |  |
|                        | $callback\_t\ callback$                               | callback function pointer        |  |
|                        | uint priority                                         | priority <0 denotes preeminent   |  |
|                        |                                                       | priority 0 denotes non-queueable |  |
|                        |                                                       | priorities >0 denote queueable   |  |
| 1                      |                                                       |                                  |  |

returns: no return value

notes:

- $\bullet$  a callback registration over rides any previous ones for the same event.
- only one callback can be registered as preeminent.
- a second preeminent registration is demoted to non-queueable.

|                         |                | Deregister callback from event_id |
|-------------------------|----------------|-----------------------------------|
| function                | arguments      | description                       |
| void spin1_callback_off | uint event_id  | event that triggers callback      |
| returns:                | no return valu | ie e                              |

|                                                                                               | Schedule a callback for execution with given priority |                           |  |
|-----------------------------------------------------------------------------------------------|-------------------------------------------------------|---------------------------|--|
| function                                                                                      | arguments                                             | description               |  |
| uint spin1_schedule_callback                                                                  | callback_t callback                                   | callback function pointer |  |
|                                                                                               | uint arg0                                             | callback argument         |  |
|                                                                                               | uint arg1                                             | callback argument         |  |
|                                                                                               | uint priority                                         | callback priority         |  |
| returns:                                                                                      | SUCCESS (=1) / FAILURE (=0)                           |                           |  |
| <b>notes:</b> • this function allows the application to schedule a callback without an event. |                                                       |                           |  |
| • priority <= 0 must not be used (unpredictable results).                                     |                                                       |                           |  |

| • | function | arguments | are not | validated |
|---|----------|-----------|---------|-----------|
| • | Tuncuon  | arguments | are not | vanuateu. |

|                                                                                       |                  | Trigger a user event |
|---------------------------------------------------------------------------------------|------------------|----------------------|
| function                                                                              | arguments        | description          |
| uint spin1_trigger_user_event                                                         | uint arg0        | callback argument    |
|                                                                                       | uint arg1        | callback argument    |
| returns                                                                               | s: SUCCESS       | (=1) / FAILURE (=0)  |
| notes: • FAILURE indicates a trigger attempt before a previous one has been serviced. |                  |                      |
| • arg0 and arg1 will be passed as arguments to the registered callback.               |                  |                      |
| • function arguments                                                                  | are not validate | ed.                  |





# Data transfer functions

|                                                                                 |                         | Request a DMA transfer     |  |
|---------------------------------------------------------------------------------|-------------------------|----------------------------|--|
| function                                                                        | arguments               | description                |  |
| uint spin1_dma_transfer                                                         | uint tag                | for application use        |  |
|                                                                                 | void $*system\_address$ | address in system NoC      |  |
|                                                                                 | void $*tcm\_address$    | address in TCM             |  |
|                                                                                 | uint direction          | DMA_READ / DMA_WRITE       |  |
|                                                                                 | uint length             | transfer length (in bytes) |  |
| returns: unique transfer identification number (TID)                            |                         |                            |  |
| <b>notes:</b> • completion of the transfer generates a DMA transfer done event. |                         |                            |  |
| • a registered callback can use TID and tag to identify the completed request.  |                         |                            |  |
| • DMA transfers are completed in the order in which they are requested.         |                         |                            |  |
| $\bullet$ TID = FAILURE (= 0) indicates failure to schedule the transfer.       |                         |                            |  |
| • function arguments are not validated.                                         |                         |                            |  |
| • may cause DMA error or DMA timeout events.                                    |                         |                            |  |

|                                                |                 | Copy a block of memory     |
|------------------------------------------------|-----------------|----------------------------|
|                                                |                 | copy at block of memory    |
| function                                       | arguments       | description                |
| void spin1_memcpy                              | void *dst       | destination address        |
|                                                | void const *src | source address             |
|                                                | uint len        | transfer length (in bytes) |
| returns:                                       | no return value |                            |
| notes: • function arguments are not validated. |                 |                            |
| • may cause a data abor                        | t.              |                            |



# Communications functions

|                           |                             | Send a multicast packet              |
|---------------------------|-----------------------------|--------------------------------------|
| function                  | arguments                   | description                          |
| uint spin1_send_mc_packet | uint key                    | packet key                           |
|                           | uint data                   | packet payload                       |
|                           | uint load                   | 1 = payload present / 0 = no payload |
| returns:                  | SUCCESS (=1) / FAILURE (=0) |                                      |

|                                                     | Flush s   | software outgoing multicast packet queue |
|-----------------------------------------------------|-----------|------------------------------------------|
| function                                            | arguments | description                              |
| uint spin1_flush_tx_packet_queue                    | void      | no arguments                             |
| returns:                                            | SUCCESS   | (=1) / FAILURE (=0)                      |
| notes: • queued packets are thrown away (not sent). |           |                                          |

|                                          | Flush software incoming multicast packet queue |  |  |
|------------------------------------------|------------------------------------------------|--|--|
| function                                 | arguments description                          |  |  |
| uint spin1_flush_rx_packet_queue         | void no arguments                              |  |  |
| returns:                                 | SUCCESS (=1) / FAILURE (=0)                    |  |  |
| notes: • queued packets are thrown away. |                                                |  |  |

SpiNNaker





|                                 |              | Request a free SDP message container |
|---------------------------------|--------------|--------------------------------------|
| function                        | arguments    | description                          |
| $sdp\_msg\_t * spin1\_msg\_get$ | void         | no arguments                         |
| returns:                        | pointer to n | nessage (NULL if unsuccessful)       |

|                     |                    | Free an SDP message container |
|---------------------|--------------------|-------------------------------|
| function            | arguments          | description                   |
| void spin1_msg_free | $sdp\_msg\_t *msg$ | pointer to message            |
| returns:            | no return value    |                               |



Φ

# Critical section support functions

|                        |             |                  | Disable IRQ interrupts |
|------------------------|-------------|------------------|------------------------|
| function               | arguments   | description      |                        |
| uint spin1_irq_disable | void        | no arguments     |                        |
| returns:               | contents of | CPSR before inte | rrupt flags altered.   |

|                        |             | Disable FIQ interrupts               |
|------------------------|-------------|--------------------------------------|
| function               | arguments   | description                          |
| uint spin1_fiq_disable | void        | no arguments                         |
| returns:               | contents of | CPSR before interrupt flags altered. |

|                        |             |                  | Disable ALL interrupts |
|------------------------|-------------|------------------|------------------------|
| function               | arguments   | description      |                        |
| uint spin1_int_disable | void        | no arguments     |                        |
| returns:               | contents of | CPSR before inte | rrupt flags altered.   |

|                         |              | Restore core mode and interrupt state |
|-------------------------|--------------|---------------------------------------|
| function                | arguments    | description                           |
| void spin1_mode_restore | uint status  | CPSR state to be restored             |
| returns:                | no return va | alue.                                 |





## System resources access functions

|                        |              |              | Get core ID |
|------------------------|--------------|--------------|-------------|
| function               | arguments    | description  |             |
| uint spin1_get_core_id | void         | no arguments |             |
| returns:               | core ID in b | oits [4:0].  |             |

|                                                                                           |              | Get chip ID  |  |
|-------------------------------------------------------------------------------------------|--------------|--------------|--|
| function                                                                                  | arguments    | description  |  |
| $uint spin1\_get\_chip\_id$                                                               | void         | no arguments |  |
| returns:                                                                                  | chip ID in b | oits [15:0]. |  |
| <b>notes:</b> • chip ID contains x coordinate in bits [15:8], y coordinate in bits [7:0]. |              |              |  |

|                   |          |              |                                      | Get ID |
|-------------------|----------|--------------|--------------------------------------|--------|
| function          |          | arguments    | description                          |        |
| uint spin1_get_id |          | void         | no arguments                         |        |
|                   | returns: | chip ID in b | oits [20:5] / core ID in bits [4:0]. |        |

|                                                    |                                                                                 |                   | Control state of board LEDs                   |  |
|----------------------------------------------------|---------------------------------------------------------------------------------|-------------------|-----------------------------------------------|--|
| functio                                            | n                                                                               | arguments         | description                                   |  |
| void sp                                            | in1_led_control                                                                 | uint p            | new state for board LEDs                      |  |
| returns: no return value.                          |                                                                                 |                   |                                               |  |
| notes:                                             | notes: • the number of LEDs and their colour varies according to board version. |                   |                                               |  |
|                                                    | • to turn LEDs 0 and                                                            | l 1 on: spin1_led | $_{\text{control}}$ (LED_ON (0) + LED_ON (1)) |  |
| • to invert LED 2: spin1_led_control (LED_INV (2)) |                                                                                 |                   |                                               |  |
|                                                    | • to turn LED 0 off:                                                            | spin1_led_contro  | l (LED_OFF (0))                               |  |

|                                                                   | Set up a multicast routing table entry      |  |
|-------------------------------------------------------------------|---------------------------------------------|--|
| function                                                          | arguments description                       |  |
| uint spin1_set_mc_table_entry                                     | uint entry table entry                      |  |
|                                                                   | uint key entry routing key field            |  |
|                                                                   | uint mask entry mask field                  |  |
|                                                                   | uint route entry route field                |  |
| returns:                                                          | SUCCESS (=1) / FAILURE (=0).                |  |
| notes: • see SpiNNaker datashe                                    | neet for details of the MC table operation. |  |
| • entries 0 to 999 are available to the application.              |                                             |  |
| • routing keys with $bit[15] = 1$ and $bit[10] = 0$ are reserved. |                                             |  |
| • function arguments are not validated.                           |                                             |  |



|                                          |                                                   | Allocate a new block of DTCM      |
|------------------------------------------|---------------------------------------------------|-----------------------------------|
| function                                 | arguments                                         | description                       |
| void * spin1_malloc                      | uint bytes                                        | size of the memory block in bytes |
| returns:                                 | pointer to the new memory block.                  |                                   |
| notes: • memory blocks are word-aligned. |                                                   |                                   |
| • memory is allocated in                 | • memory is allocated in DTCM.                    |                                   |
| • there is no support for                | • there is no support for freeing a memory block. |                                   |







# Miscellaneous

|                     |                                                                    |             | Wait for a given time       |
|---------------------|--------------------------------------------------------------------|-------------|-----------------------------|
| functio             | n                                                                  | arguments   | description                 |
| void spin1_delay_us |                                                                    | uint time   | wait time (in microseconds) |
|                     | returns:                                                           | no return v | alue                        |
| notes:              | • the function busy waits for the given time (in microseconds).    |             |                             |
|                     | • prevents any queueable callbacks from executing (use with care). |             |                             |

|                                                                                   |             | Generate a 32-bit pseudo-random number |
|-----------------------------------------------------------------------------------|-------------|----------------------------------------|
| function                                                                          | arguments   | description                            |
| void spin1_rand                                                                   | void        | no arguments                           |
| returns:                                                                          | 32-bit pseu | do-random number                       |
| notes: • Function based on example function in:                                   |             |                                        |
| • "Programming Techniques", ARM document ARM DUI 0021A.                           |             |                                        |
| • Uses a 33-bit shift register with exclusive-or feedback taps at bits 33 and 20. |             |                                        |

|                  | Provide a seed to the pseudo-random number generator |
|------------------|------------------------------------------------------|
| function         | arguments description                                |
| void spin1_srand | uint seed 32-bit seed                                |
|                  | returns: no return value                             |



MANCHESTER

#### 0.2.7**Application Programme Structure**

In general, an application programme contains three basic sections:

- Application Functions: General application functions to support the callbacks.
- Application Callbacks: Functions to be associated with run-time events.
- Application Main Function: Variable initialisation, callback registration and transfer of control to main loop.

The structure of a simple application programme is shown on the next page. Many details are left out for brevity.

MANCH**EST**ER

Y



```
// declare application types and variables
neuron_state state[1000];
spike_bin bins[1000][16];
                         ---- application functions
/* -
void izhikevich_update(neuron_state *state){
    \label{eq:spin1_send_mc_packet} \textbf{spin1\_send\_mc\_packet} \, (\, \ker \, , \  \, 0 \, , \, \, \textbf{NO\_PAYLOAD}) \, ;
}
syn_row_addr lookup_synapse_row(neuron_key key)
void bin_spike(neuron_key key, axn_delay delay, syn_weigth weight)
                          — application callbacks
void update_neurons()
    if (spin1_get_simulation_time() > 1000) // simulation time in "ticks"
        spin1\_stop();
        for (i=0; i < 1000; i++) izhikevich_update(state[i]);
}
void process_spike(uint key, uint payload)
    row_addr = lookup_synapses(key);
    tid = spin1_dma_transfer(tag, row_addr, syn_buffer, READ, row_len);
void schedule_spike()
    bin_spike(key, delay, weight);

    application main

void c_main()
    // initialise variables and timer tick
    spin1_set_timer_tick(1000); // timer tick period in microseconds
    // register callbacks
    spin1_callback_on(TIMER_TICK, update_neurons, 1);
    spin1_callback_on(MCPACKET_RECEIVED, process_spike, 0);
    spin1_callback_on(DMA_TRANSFER_DONE, schedule_spike, 0);
    // transfer control to the run-time kernel
    spin1_start();
    // control returns here on execution of spin1_stop()
}
```