

# 359 Final

## Review of C with pointers

- A pointer is a memory address

`int foo;`

- 声明变量

- 占用内存 (-一个整数型 int, - 故是32位或4个字节)

- 使用变量名 foo 来表示内存中的 bits.

`int *foo_ptr = &foo;`

- \* means `foo_ptr` is a pointer  
< its value is a memory address  
不容易一个整数

- & means "the address of"

< `&foo` → the address of foo  
not foo, but where foo exists in memory



- A variable is a box

< it has a size (in bytes)  
you put values in it

- The location of the box is its address
- `&foo`

- `foo_ptr` is a box with a variable
- it's type is "`int *`"

↳ means it is a pointer / address for a box with an int.

习题:

写两个指针

`int *ptr_a;`      /    `int *ptr_a, *ptr_b;`  
`int *ptr_b;`      "      "

通过指针赋值

- Suppose you want to assign a value to foo  
`foo = 42;`

- Do it with dereferencing  
`*foo_ptr = 42;`

↳ assign 42 to the box with the address of `foo_ptr`  
use \* on LHS, RHS (LHS - write to box; RHS read from box)

## Arrays

`int data[] = {45, 67, 89};`

- declare an array named "data"

- size is three

- determined by initial assignment (3 values)

- alternate `int data[3];` without initialization

- the variable "data" is a box with three int boxes init

· 三个连续的盒子

- So array variable in C are just pointers to the first item.

- In most cases

- `&data[0]` is the same thing as `data`.

· Decaying == `&data == &data[0]`

## Pointer arithmetic

```
int *data_ptr = data;
printf("first element: %d\n", *(data_ptr++));
printf("second element: %d\n", *(data_ptr++));
printf("third element: %d\n", *(data_ptr++));
```

输出: first element: 45

second element: 67

third element: 89

- note post-decrement operator

- increment after use `75++`

- for each increment

- actually added four to the pointer

- because `sizeof(int) == 4`

## Indexing 索引

`printf("%d\n", data[0]);`

- This prints

45

- [ ] denotes pointers with an offset to element value

· `data[1]` is the same as `*(data + 1)`

|                                         |              |
|-----------------------------------------|--------------|
| <code>int data[] = {45, 67, 89};</code> | prints<br>89 |
| <code>int *p = &amp;data[1]</code>      |              |

`printf("%d\n", p[1])`

## 指针

例一：

```
(1) int *ptr;
(2) char *ptr;
(3) int **ptr;
(4) int (*ptr)[3];
(5) int *(*ptr)[4];
```

### ① 指针的类型

(从语法的角度，只要把指针名字去掉，剩下的部分就是这个指针的类型。这是指针本身所具有的类型)

```
(1) int *ptr; // 指针类型: int*
(2) char *ptr; // 指针类型: char*
(3) int **ptr; // 指针类型: int**
(4) int (*ptr)[3]; // 指针类型: int(*)[3]
(5) int *(*ptr)[4]; // 指针类型: int*(*)[4]
```

### ② 指针所指向的类型

(当你通过指针来访问指针所指向的内存区时，指针所指向的类型决定了编译器将把那片内存区里的内容当做什么来看待。)(从语法上，只要把指针名字和指针左边的 \* 去掉，剩下的就是指针所指向的类型)

```
(1) int *ptr; // int
(2) char *ptr; // char
(3) int **ptr; // int*
(4) int (*ptr)[3]; // int[3]
(5) int *(*ptr)[4]; // int*[4]
```

### ③ 指针的值 — 或者叫指针所指向的内存区或地址。

指针的值是~~指针本身存储的数值~~，这个值将被编译器当作一个地址，而不是一个具体的数值。在32位程序里，所有类型的指针的值都是一个32位整数，因为32位程序里的存地址全都是32位长。指针所指向的内存区就是从指针的值所代表的那个内存地址开始，长度为 sizeof(指针所指向的类型)的一块内存区。以后，我们说一个指针的值是 xx，就相当于说该指针指向了以 xx 为首地址的一块内存区域；我们说一个指针指向某块内存区域，就相当于说该指针的值是这块内存区域的首地址。指针所指向的类型是两个完全不同的概念。在例一中，指针所指向的类型已经有了，但由于指针还未初始化，所以它所指向的内存区是不存在的，或者说它是无效的。

### ④ 指针本身所占据的内存区。

指针本身占多大内存？你只要用函数 sizeof(指针的类型) 测一下就知道了。在32位平台上，指针本身占据了4个字节的长度。指针本身占据的内存这个概念在判断一个指针表达式是否是左值时很有用。

## Structures and Unions

```
Struct foo {
    size_t size;
    char name[64];
    int answer-to-ultimate-question;
    unsigned shoe_size;
};

Struct foo my-foo;
my-foo.size = sizeof(struct foo);
Struct *foo_ptr = &my-foo;
foo_ptr->size = sizeof(struct foo);
```

- for access to members of struct
  - use ". "
- if we have a pointer to a struct use
  - " → "
  - does pointer arithmetic to member

## Multiple Indirection

```
int a = 3;
int *b = &a;
int **c = &b;
int ***d = &c;
****d = c;
***d = *c == b;
****d == ***c == *b == a == 3;
```

- & adds asterisks 
  - increase pointer level
- \* → [] remove asterisks
  - decrease pointer level

## Strings

```
char str[] = "I am the Walrus";
```

- C string is an array of char
  - terminated with null character, '\0000'
  - sentinel for end of string
- C string library
  - implemented on pointers to null-terminated arrays of char.

## C library strlen() 長度

```
int strlen(char *str) {
    int len = 0;
    while (*(str++)) ++len;
}
return len;
```

```
int strlen(char *str) {
    int i;
    for (i = 0; str[i]; ++i);
}
return i;
```

- two ways to implement strlen()
  - first uses pointer arithmetic and dereferencing.
  - second uses array indexing.

## C type casting

- Change one data type into another
  - int to float
  - float to int
  - ~~float~~ int to int\*
  - void\*\* to int\*

```
#define REG_ADDRESS 0x3f000000
int a = 5;
float b = 3.5;
// int to float
float f = (float)a;
// float to int
int i = (int)b;
// convert int to pointer
int *reg = (int*)REG_ADDRESS;
// convert pointer types
void *p = (void*)reg;
```

## Memory mapped I/O

```
int *gpio;
gpio = base address of Pi GPIO
*gpio = 5;
gpio[5] = 3;
int val = gpio[6];
```

## Allocating memory

- malloc allocates a block of memory of requested size, return pointer to that memory  
malloc 申请一个请求大小的内存块，返回指向该内存的指针。
- calloc
  - same but uses item size
- free returns the memory to OS when done.

```
void *malloc (size_t size);
void free (void *ptr);
void *calloc (size_t num, size_t size);
```

## Tut

### Input / Output

- Output      `printf("text")`
- Input        `scanf("%d", &n); // when n is a variable.`
- Concatenation
  - Signed integer: `printf("Age is %d", age)`
  - Decimal: `printf("Price is %f", price)`
  - String: `printf("Name is %s", fname)`

## Pointers

Pointers are used to map to a specific address of another variable.

## Review of basic electrical circuits.

### Voltage 电压

- electromotive force (EMF)      voltage  
electric pressure

- The electrical pressure that pushes current through an electrical system  
推动电流通过电气系统之压力。

### Current 电流

- The amount of charge flowing through a wire  
通过导线的电荷量。



### Resistance 阻值

- How a wire resists (or allows) current to flow given voltage (EMF) 电流在给定电压下流动
- Inverse of conductivity
  - might be easier to understand this way
  - but normally refer to resistance



### Nodes

- lines in the schematic 原理图
- roughly equivalent to a wire
- same node if no other component in connected lines
- voltage same everywhere in a node
  - \* lines with same color are the same nodes!



### Ideal voltage source

- ↳ does not exist as a real, physical device
- represent with a circle
  - ↳ +/- terminal (sometimes leave out -)
- keep voltage constant across two terminals
- units - Volts (V)



## Current conventions

- electrical current is (usually) the flow of electrons.
- electrons flow away from -ve terminal of voltage source
  - ↳ towards +ve
- But by convention we consider the current to be flow of positive charge
- Makes no difference to most analyses
- Summary
  - ↳ positive flow away from/out of plus terminal of voltage source
  - ↳ into negative terminal of voltage source.

## Resistor

- these can be real, physical devices
  - or integral part of real device
- resist the flow of current when voltage across them
- zigzag line .
- units - Ohms ( $\Omega$ )



## Ground

- Provides voltage reference
  - ↳ voltage must be across components
  - ↳ implies that when a node has a voltage it is across node to ground
- closes circuit



## Physical connections to RPi

### Wiring checklist

- Voltage level
    - 5V or 3.3
  - Polarity of devices
    - ↳ power source
    - ↳ semi-conductors / LCs
    - ↳ electrolytic capacitors
  - Output versus input versus bi-directional
  - pull-up resistors
    - ↳ and pull-down
- breadboard
- Separate voltages on Explorer Hat
- Two parts
    - 5V
    - 3.3V
  - 3.3V
    - ↳ connecting 5V to RPi
    - 3.3V can damage RPi
  - 5V side has buffers to convert 5V to 3.3V logic

### How to make connections

- Make connections with jumper wire
  - forms nodes in circuit
- Female pins connecting to RPi on bottom and right
  - labeled  RPi
  - 3 female pins on breadboard are connected

## Potentiometer

↳ 

Potentiometer forms a voltage divider

↳ Sweeping the middle contact varies  $V_{out}$  between  $V_{in}$  and 0V

↳ A/D connected via Explorer Hat
 

- allows RPi to read  $V_{out}$



## Analog to Digital on Explorer

- 4 A/D inputs on Explorer Hat Pro.
- 4 pins on left-hand side of 5V section
- Labeled 1, 2, 3, 4
- Connected to analog inputs of an ADS1015 IC

## Adafruit joystick 摆絲杆

- ↳ Has two potentiometers  
→ x axis / y axis
- ↳ one pushbutton

Inferred schematic (x, y)



### Pushbutton:

- Connect to digital input
- Circuit board says  
- "select shorts to ground when pressed"

## Floating inputs (浮动输入)

- Inputs that are not connected to anything are said to (没有连接到任何东西的输入被称为)  
- float
- That is, they have no specific voltage  
- Input is arbitrary
- The solution is to use either
  - pull-up resistor 上拉电阻
  - pull-down resistor 下拉电阻

## Pull-up resistors

- ↳ Pull-up connects floating input to logic high through the pull-up resistor 上拉通过上拉电阻将浮动输入连接到逻辑高电平。

↳ Note the ~~not~~ Thevenin equivalent MCU ???

↳ When button open - voltage divider

↳ When button closed - input goes to 0V - logic low



## Pull-down

- ↳ Connect floating input to ground through pull-down resistor
- ↳ Need to put other end of button to logic high
- ↳ When button open - 0V, logic low
- ↳ When button closes - 5V, logic high



- Uses pull-down for digital inputs  
so need other end of push button to go to 5V.

## Physical connections to RPi

### ADXL 345 - Accelerometer 加速度计

#### ↳ 3 axis accelerometer

- ↳ Adafruit mounts it on a breakout board
- provides essential peripheral circuitry 提供必要的外围电路
- pins for breadboard.

#### Breakout features

- ↳ 5V / 3.3V operation
- ADXL 1015 is 3.3V
- added voltage regulator

↳ x

## Interface to ADXL 345

- Two ways
  - ↳ I<sup>2</sup>C - Inter-Integrated circuit  
I-squared C or I-2-C
  - ↳ SPI - Serial Peripheral Interface bus.

- Same functionality
- Both in wide use
- Both supported well by RPi

→ ~~Not~~ I<sup>2</sup>C

很下面 →

## Memory mapped I/O

内存映射输入/输出

CPU怎样访问外设存储空间？

一种方式，CPU用单独的指令访问，这个单独的指令需要一个要操作的外设存储空间的地址，这就是I/O端口(I/O port)

另一种方式，CPU不用单独指令访问外设存储空间，用通常的访问存储器的方式访问外设存储空间，这就是I/O存储器(Memory-mapped)

memory port 好在它不占用空间，只要系统配置的常规存储器大小和系统地址空间大小的限制。

## Memory versus I/O

- conceptually
  - registers/memory are the same thing
- In practice
  - flip flop registers not efficient for main memory
    - too many transistors - space on silicon
    - too much power
  - but, flip-flop register can connect to other devices.

### key concept 关键概念

- use combinational logic to decode address bus
  - some addresses connect to memory
  - others connect to registers
- registers connect to electronics that interfaces with the physical world
  - digital  $\Rightarrow$  io pins
  - uart
  - dma ...

## I/O read / write from pointer ref

- write
  - $\hookrightarrow$  write contents of variable (CPU register) to memory or output register
- read
  - $\hookrightarrow$  read contents of memory or input register into variable (CPU register)

## GPIO

General Purpose Input / Output

通用型 I/O 输入/输出

\* 在嵌入式系统中，经常需要控制许多结构简单的外部设备或者电路，这些设备有的需要通过CPU控制，有的需要CPU提供驱动信号。

对设备的控制，使用传统的串口或者并口就显得比较复杂，所以，在嵌入式微处理器上通常提供了一种

“通用~~多~~引脚 I/O 端口”，也就是 GPIO port。计算机外置连接更简单

- Famous with microcontrollers
- $\hookrightarrow$  Provide extended functions without changing circuitry

- Has 54 lines

- Only a subset is available for the RPi
  - 26 lines (40 on RPi 3)

## Memory Mapped I/O



## Memory Mapped I/O with GPIO





## RPi GPIO pins

- The RPi GPIO has 54pins
  - not all brought out for users
  - header shows what you have to work with
  - green ones for general use
  - others have specific functions
- Since they are not plug-and-play, some work is needed to initialize, write, and read
- Each of these pins can have up to 6 functions
  - For example function 1 is write & 0 is read
  - From core's point of view



## GPIO Registers

| Address     | Field Name | Description             | Size | Read/Write |
|-------------|------------|-------------------------|------|------------|
| 0x7E20 0000 | GPFSEL0    | GPIO Function Select 0  | 32   | R/W        |
| 0x7E20 0004 | GPFSEL1    | GPIO Function Select 1  | 32   | R/W        |
| 0x7E20 0008 | GPFSEL2    | GPIO Function Select 2  | 32   | R/W        |
| 0x7E20 000C | GPFSEL3    | GPIO Function Select 3  | 32   | R/W        |
| 0x7E20 0010 | GPFSEL4    | GPIO Function Select 4  | 32   | R/W        |
| 0x7E20 0014 | GPFSEL5    | GPIO Function Select 5  | 32   | R/W        |
| 0x7E20 0018 |            | Reserved                |      |            |
| 0x7E20 001C | GPSET0     | GPIO Pin Output Set 0   | 32   | W          |
| 0x7E20 0020 | GPSET1     | GPIO Pin Output Set 1   | 32   | W          |
| 0x7E20 0024 |            | Reserved                |      |            |
| 0x7E20 0028 | GPCLR0     | GPIO Pin Output Clear 0 | 32   | W          |
| 0x7E20 002C | GPCLR1     | GPIO Pin Output Clear 1 | 32   | W          |
| 0x7E20 0030 |            | Reserved                |      |            |
| 0x7E20 0034 | GPIN0      | GPIO Pin Level 0        | 32   | R          |
| 0x7E20 0038 | GPIN1      | GPIO Pin Level 1        | 32   | R          |
| 0x7E20 003C |            | Reserved                |      |            |

More info refer to: BCM2835 ARM Peripherals (available on BB)



## GPIO Registers

- There are more registers
- Replace T in the virtual address by sf to obtain the physical address.

## Function Select Registers

|            |         |                   |
|------------|---------|-------------------|
| 0x3f200014 | GPFSEL5 | For pins 50 to 53 |
| 0x3f200010 | GPFSEL4 | For pins 40 to 49 |
| 0x3f20000C | GPFSEL3 | For pins 30 to 39 |
| 0x3f200008 | GPFSEL2 | For pins 20 to 29 |
| 0x3f200004 | GPFSEL1 | For pins 10 to 19 |
| 0x3f200000 | GPFSEL0 | For pins 0 to 9   |

Base GPIO address  $\rightarrow 0x3f200000$

地址 0x3f200000

## Accessing GPIO Registers

- GPIO uses "pseudo" memory-mapped I/O
- Accessed as if accessing memory locations

```

    ldr r0, =0x20200000 // Address of GPFSEL0
    // To read GPFSEL0
    ldr r1, [r0]
    // To write GPFSEL0
    str r1, [r0]
  
```

## Pin Offset from GPIO Base Address

- A register corresponds to many GPIO pins
- For example GPFSEL0 controls pin 0 to 9
- Each pin (0 to 9) is controlled by 3 bits in GPFEL0
- For example, bits 0-2 set the function for GPIO pin 0
- Each pin can be given a function (from up to 6)
- In our case, it is mostly input or output functions.

## Inside GPFSEL0

| Bit(s) | Field Name | Description                                                                                                                                                                                                                                                                                                                                                                          | Type | Reset |
|--------|------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------|-------|
| 31-30  | ---        | Reserved                                                                                                                                                                                                                                                                                                                                                                             | R    | 0     |
| 29-27  | FSEL9      | FSEL9 - Function Select 9<br>000 = GPIO Pin 9 is an input<br>001 = GPIO Pin 9 is an output<br>100 = GPIO Pin 9 takes alternate function 0<br>101 = GPIO Pin 9 takes alternate function 1<br>110 = GPIO Pin 9 takes alternate function 2<br>111 = GPIO Pin 9 takes alternate function 3<br>011 = GPIO Pin 9 takes alternate function 4<br>010 = GPIO Pin 9 takes alternate function 5 | R/W  | 0     |
| 26-24  | FSEL8      | FSEL8 - Function Select 8                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 23-21  | FSEL7      | FSEL7 - Function Select 7                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 20-18  | FSEL6      | FSEL6 - Function Select 6                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 17-15  | FSEL5      | FSEL5 - Function Select 5                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 14-12  | FSEL4      | FSEL4 - Function Select 4                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 11-9   | FSEL3      | FSEL3 - Function Select 3                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 8-6    | FSEL2      | FSEL2 - Function Select 2                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 5-3    | FSEL1      | FSEL1 - Function Select 1                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |
| 2-0    | FSEL0      | FSEL0 - Function Select 0                                                                                                                                                                                                                                                                                                                                                            | R/W  | 0     |

※ 寄存器 GPFSEL0 - GPFSEL5 → 功能寄存器  
指定管脚为输入/输出，每3位决定一个管脚。

## Calculating the Pin Offset

- If pin# is single-digit
 
$$\text{offset} = \text{address for GPFSEL0} + (\text{base GPIO address}, 0x3f200000) \times \text{pin\#}$$
- If pin# > 9 (more than one digit)
 
$$\text{Offset} = \text{base GPIO address} + 4 \times (\text{pin\#} \text{ div } 10)$$

↳ Address of GPFSEL1 to GPFSEL5
- Example (pins 10 to 19):
 
$$\text{offset} = 0x3f200000 + 4 \times (1) = 0x3f200004$$

↳ address for GPFSEL1

## Getting to Appropriate Pins

- Given some pin#
- Let  $GPFSEL\{n\}$  be the register that controls this pin# (i.e.,  $n = \text{pin\#} \text{ div } 10$ )
- Let  $d$  be the least significant digit of pin#  
 $\rightarrow d = \text{pin\# mod } 10$  余数.
- The 3 bits of  $GPFSEL\{n\}$  that select pin#'s function start at  $d+3$

### Example

Given Pin 32

- Register  $GPFSEL3$ , offset  $0x3f20000c$   
 $\rightarrow = 4 \times (32 \text{ div } 10) + 0x3f200000$
- Least significant bit of pin# is  $2 = 32 \text{ mod } 10$
- The three bits of  $GPFSEL3$  that control pin 32 start at bit  $3 \times 2 = 6$   
 i.e. bits 6-8 (6,7,8) select the function of pin 32

### $GPFSEL\{n\}$

|          |       |       |       |       |       |       |      |     |     |     |
|----------|-------|-------|-------|-------|-------|-------|------|-----|-----|-----|
| 30-31    | 27-29 | 24-26 | 21-23 | 18-20 | 15-17 | 12-14 | 9-11 | 6-8 | 3-5 | 0-2 |
| reserved | Pin   | Pin   | Pin   | Pin   | Pin   | Pin   | Pin  | Pin | Pin | Pin |

fn31 fn38 fn37 fn36 fn35 fn34 fn33 fn32 fn31 fn30

### Setting the bits in $GPFSEL\{n\}$

- Clear the appropriate bits in  $GPFSEL\{n\}$   
 $\rightarrow$  Same as setting up for input
- Set the appropriate bits in  $GPFSEL\{n\}$

### Bit Bashing

$x$  is a logical variable  
 ↪ also a bit  
 ↪ also a GPIO pin state

|             |                   |
|-------------|-------------------|
| $x = 0$     | clear bit         |
| $x = 1$     | keep bit the same |
| $x + 0 = x$ | keep bit the same |
| $x + 1 = 1$ | set bit           |

### $\rightarrow$ Back to setting the bits in $GPFSEL\{n\}$

#### ① Clear the appropriate bits in $GPFSEL\{n\}$

$\rightarrow$  Same as setting up for input

$\rightarrow$  Clear means AND with 0 = 0

$\rightarrow$  must AND the bits for GPIO pin with 0

$\rightarrow$  But must preserve the other bits 13 12

. with AND, preserve with 1

$\rightarrow$  Solution - AND with integer that has 0 where I want to clear, and 1 everywhere else

#### ② Set the appropriate bits in $GPFSEL\{n\}$

$\rightarrow$  assume starting with the bits as 0

should previously have cleared the bits

$\rightarrow$  setting appropriate bits mean OR with bit pattern  $\Rightarrow$

$\rightarrow$  but must preserve the other bits

with OR, preserve with 0

$\rightarrow$  solution - OR with integer that has bit pattern where I want it, 0 everywhere else

Clear the appropriate bits in  $GPFSEL\{n\}$

```

ldr r0, =<address of GPFSEL\{n\}>
ldr r1, [r0] // copy GPFSEL\{n\} into r1
mov r2, #7 // (b0111)
lsl r2, #9 // index of 1st bit for pin3
// r2 = 0 111 000 000 000
bic r1, r2 // clear pin3 bits
mov r3, #1 // output function code
lsl r3, #9 // r3 = 0 001 000 000 000
orr r1, r3 // set pin3 function in r1
str r1, [r0] // write back to GPFSEL0

```

Set the appropriate bits in  $GPFSEL\{n\}$

```

r0 = address of GPFSEL\{n\}
r1 = copy of GPFSEL\{n\}
mov r2, #7 // (b0111)
lsl r2, <index of 1st bit of the corresponding 3>
Example: pin#1, 1st bit is 3
        lsl r2, #3
        orr r1, r2
        str r1, [r0]

```

### Example - Setting pin 3 to output

```

ldr r0, =0x20200000 // address for GPFSEL0
ldr r1, [r0] // copy GPFSEL0 into r1
mov r2, #7 // (b0111)
lsl r2, #9 // index of 1st bit for pin3
// r2 = 0 111 000 000 000
bic r1, r2 // clear pin3 bits
mov r3, #1 // output function code
lsl r3, #9 // r3 = 0 001 000 000 000
orr r1, r3 // set pin3 function in r1
str r1, [r0] // write back to GPFSEL0

```

### Using C

```

#define GPIO_BASE 0x3f200000 // = b11...
unsigned *gpio = mmap(...); // GPIO base in virtual memory
// GPIO setup macros. Always use INP_GPIO(x)
// before using OUT_GPIO(x) or SET_GPIO_ALT(x,y)
#define INP_GPIO(g) *(gpio+(g)/10) & -(1<<((g)%10)*3))
#define OUT_GPIO(g) *(gpio+(g)/10) |= (1<<((g)%10)*3))
bo01 - means
        configure for output

```

found on the internet in many many places  
original source unknown

### Using C alternate

```

#define GPIO_BASE 0x3f200000
unsigned *gpio = mmap(...); // GPIO base in virtual memory
// GPIO setup macros. Always use INP_GPIO(x)
// before using OUT_GPIO(x) or SET_GPIO_ALT(x,y)
#define INP_GPIO(g) gpio[(g)/10] = gpio[(g)/10] & -(1<<((g)%10)*3))
#define OUT_GPIO(g) gpio[(g)/10] = gpio[(g)/10] | (1<<((g)%10)*3))

```

### Performing I/O

$\rightarrow$  Once a pin function is set, then pin can be used for I/O

$\rightarrow$  To write to a pin, use  $GPIOSET\{0,1\}$  or  $GPIOCLR\{0,1\}$

$\rightarrow$  To read from a pin, use  $GPIOEV\{0,1\}$

### GPIO Pin Output Set Registers (GPSETn)

**SYNOPSIS** The output set registers are used to set a GPIO pin. The SET(n) field defines the respective GPIO pin to set, writing a "0" to the field has no effect. If the GPIO pin is being used as an input (by default) then the value in the SET(n) field is ignored. However, if the pin is subsequently defined as an output then the bit will be set according to the last set/clear operation. Separating the set and clear functions removes the need for read-modify-write operations.

| Bit(s) | Field Name     | Description                         | Type | Reset |
|--------|----------------|-------------------------------------|------|-------|
| 31-0   | SETn (n=0..31) | 0 = No effect<br>1 = Set GPIO pin n | R/W  | 0     |

Table 6-8 – GPIO Output Set Register 0

Dividing a pin# by 32 determines which register must be used

| Bit(s) | Field Name      | Description                         | Type | Reset |
|--------|-----------------|-------------------------------------|------|-------|
| 31-22  | -               | Reserved                            | R    | 0     |
| 21-0   | SETn (n=32..53) | 0 = No effect<br>1 = Set GPIO pin n | R/W  | 0     |

Table 6-9 – GPIO Output Set Register 1 From: BCM2835 ARM Peripherals

### Setting (Writing to) a Pin

- ↳ If writing 1, then use GPSET{n}, n=0..1  
· gpioAddr + 28 + f4\*n}
- ↳ If writing 0, use GPCLR{n}
  - A write of 0 to GPSET{n} is ignored
  - gpioAddr + 40 + f4\*n

### GPIO Pin Output Clear Registers (GPCLRn)

**SYNOPSIS** The output clear registers are used to clear a GPIO pin. The CLR(n) field defines the respective GPIO pin to clear, writing a "0" to the field has no effect. If the GPIO pin is being used as an input (by default) then the value in the CLR(n) field is ignored. However, if the pin is subsequently defined as an output then the bit will be set according to the last set/clear operation. Separating the set and clear functions removes the need for read-modify-write operations.

| Bit(s) | Field Name     | Description                           | Type | Reset |
|--------|----------------|---------------------------------------|------|-------|
| 31-0   | CLRn (n=0..31) | 0 = No effect<br>1 = Clear GPIO pin n | R/W  | 0     |

Table 6-10 – GPIO Output Clear Register 0

| Bit(s) | Field Name      | Description                         | Type | Reset |
|--------|-----------------|-------------------------------------|------|-------|
| 31-22  | -               | Reserved                            | R    | 0     |
| 21-0   | CLRn (n=32..53) | 0 = No effect<br>1 = Set GPIO pin n | R/W  | 0     |

Table 6-11 – GPIO Output Clear Register 1 From: BCM2835 ARM Peripherals

### Example, Writing to pin# 3

• r1 = value to write {0,13}  
 r0 = #3 //pin#  
 ldr r2, = 0x20200000 //base GPZO reg  
 mov r2, #1  
 ls1 r2, r0 //align bit for pin#3  
 tgeq r1, #0  
 streq r2, [r2, #40] //GPCLR0  
 streq r2, [r2, #28] //GPSET0

### Using C (for pins 0-31)

```
unsigned int *gpio = mmap(...);
#define GPSET0 13
#define GPCLR0 10
...
if (want to set the pin)
  gpio[GPSET0] = 1 << pinNumber;
else
  gpio[GPCLR0] = 1 << pinNumber;
```

### Reading a Pin

↳ Register GPLEVn is used to read a bit from a pin

↳ In the RPi, only GPLEV0 is used

- Bit n corresponds to the value of pin n

↳ Address of GPLEV0 is 0xf200000 + 52

### Inside GPLEV{n}

#### GPIO Pin Level Registers (GPLEVn)

**SYNOPSIS** The pin level registers return the actual value of the pin. The LEV(n) field gives the value of the respective GPIO pin.

| Bit(s) | Field Name     | Description                                     | Type | Reset |
|--------|----------------|-------------------------------------------------|------|-------|
| 31-0   | LEVn (n=0..31) | 0 = GPIO pin n is low<br>1 = GPIO pin n is high | R/W  | 0     |

Table 6-12 – GPIO Level Register 0

| Bit(s) | Field Name      | Description                                     | Type | Reset |
|--------|-----------------|-------------------------------------------------|------|-------|
| 31-22  | -               | Reserved                                        | R    | 0     |
| 21-0   | LEVn (n=32..53) | 0 = GPIO pin n is low<br>1 = GPIO pin n is high | R/W  | 0     |

Table 6-13 – GPIO Level Register 1

### Example, Reading from pin #3

```
r0 = #3 //pin#
ldr r2, = 0x20200000 //base GPZO reg
ldr r1, [r2, #52] //GPLEV0
mov r3, #1
ls1 r3, r0 //align pin3 bit
and r1, r3 //mask everything else
tgeq r1, #0
moveq r4, #0 //return 0
movne r4, #1 //return 1
```

### Using C (for pins 0-31)

```
unsigned int *gpio = mmap(...);
#define GPLEV0 13
...
int v;
// v = pin 3 value
v = (gpio[GPLEV0] >> 3) & 1;
```

### Other GPZO Functions

- Clock signal
- UART
- JTAG
- I2C
- Etc ...

## UART

### Protocol

→ Universal Asynchronous Receiver/Transmitter Protocol

通用异步收发传输器

\* 它将要传输的数据在串行通信与并行通信之间进行转换

\* 作为把并行输入信号转成串行输出信号的芯片，UART通常被集成于其他通讯接口的连接上。

## RPi GPIO

↳ General Purpose I/O

↳ Programmable

↳ 26 header pins

↳ Provide UART and 5V power

↳ Provide others two e.g. CSI (camera serial interface) and DS1 (Display serial interface)

• Noe plug and play 不是即插即用



## JTAG Interface

↳ Joint Test Action Group 联合测试工作组

\* 是一种国际标准测试协议 (IEEE 1149.1兼容)  
主要用于芯片内部测试。

## UART

↳ Low speed serial I/O protocol

↳ USB and Firewire are much faster

• But more complex

↳ Ubiquitous and highly in use 着遍和高度使用

↳ Especially useful when low-speed transfer is sufficient

↳ Inexpensive

↳ Converts between parallel signal and serial signal 在并行信号和串行信号之间进行转换。



↳ Connects a core to an I/O Device

↳ 16550 UART is a famous IC Integrated Circuit chip 芯片

### UART Protocol

- 1 means no data (legacy from telegraphy)
- Start bit is signal 0

~~stop stop~~ Bit 0 ... Bit 3 Bit 2 Bit 1 Bit 0 ~~start~~

• One or more stop bits are signal 1

• Timing is important to guarantee proper communication

\* UART 作为异步串行通信协议 in - 中，实际  
是将传输数据的每个字符一位接一位地转换传输。

### UART Functions

• TX (Transmit to I/O device)

• RX (Receive from I/O device)

### RTS & CTS

• CTS (Can Transmit) impacts the transmitter: it does not send new bits if CTS line is 0

• RTS (Ready To Receive) impacts the receiver



## RPi AUX Peripherals 外围设备

- RPi has three auxiliary peripherals:
  - ↳ Mini UART and 2 SPI masters  
(Serial Interface Peripheral 外设接口)
  - ↳ All three share the same area of registers
  - ↳ They share a common interrupt
  - ↳ They are controlled by the same register(ENB)

### AUXENB Register

| AUXENB Register (0x7E21 5004) |                  |                                                                                                                                                                                                                          |      |       |
|-------------------------------|------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------|-------|
| Bit(s)                        | Field Name       | Description                                                                                                                                                                                                              | Type | Reset |
| 31:3                          |                  | Reserved, write zero, read as don't care                                                                                                                                                                                 |      |       |
| 2                             | SPI2 enable      | If set the SPI 2 module is enabled.<br>If clear the SPI 2 module is disabled. That also disables any SPI 2 module register access                                                                                        | R/W  | 0     |
| 1                             | SPI 1 enable     | If set the SPI 1 module is enabled.<br>If clear the SPI 1 module is disabled. That also disables any SPI 1 module register access                                                                                        | R/W  | 0     |
| 0                             | Mini UART enable | If set the mini UART is enabled. The UART will immediately start receiving data, especially if the UART1_RX line is <i>low</i> .<br>If clear the mini UART is disabled. That also disables any mini UART register access | R/W  | 0     |

### RPi Mini UART

- ↳ Is a low throughput UART intended to be used as a console
  - does not have all UART functions

- ↳ Some of its features:

7 or 8 bit operation  
 ↳ 1 start bit and 1 stop bit  
 ↳ No parity bit  
 ↳ 8-symbol deep FIFO to receive and transmit  
 ↳ 先进先出?  
 RTS and CTS can be controlled by S/W  
 16550-like registers (but not 16550 compatible)

### Band Rate 速率

- ↳ Measurement of asynchronous transmission speed
- ↳ Represents the number of bits sent (including control bits: start, stop and parity)
- ↳ Measures bits per second
- ↳ measured in baud
- ↳ In digital system 1 baud = 1 bit/sec

※ 速率即调制速率，指的是有效数据信号调制载波的速率，即单位时间内载波调制状态变化的次数。

### Mini UART Baud Rate

$$\hookrightarrow \text{Baud rate} = \frac{\text{system\_clock\_frequency}}{8 * \text{baudrate register} + 1}$$

$$\hookrightarrow \text{Baudrate register} = \left( \frac{\text{system\_clock\_frequency}}{\text{baudrate} / 8} \right) - 1$$

#### Example

$$\text{Clock} = 250 \text{ MHz}$$

$$\text{Baudrate\_register} = 0$$

$$\hookrightarrow \text{Baud rate} = 31.25 \text{ Mega baud (mbps)}$$

### Mini UART Essential Registers

| Register | Function                  | Access | Address (RPi) |
|----------|---------------------------|--------|---------------|
| ENB      | UART Enable               | R/W    | 0x20215004    |
| IO       | Receive Data<br>Send Data | R<br>W | 0x20215040    |
| IER      | Interrupt Enable          | R/W    | 0x20215044    |
| IIR      | Interrupt Identification  | R      | 0x20215048    |
| FCR      | FIFO Control              | W      |               |
| LCR      | Line Control              | R/W    | 0x2021504C    |
| MCR      | Modem Control             | R/W    | 0x20215050    |
| LSR      | Line Status               | R      | 0x20215054    |
| MSR      | Modem Status              | R      | 0x20215058    |
| SCRATCH  | Scratch Pad               | R/W    | 0x2021505C    |

### Other Registers

| Register | Function                                | Access | Address    |
|----------|-----------------------------------------|--------|------------|
| CONTROL  | Extra control features                  | R/W    | 0x20215060 |
| STATUS   | Info about internal status of mini UART | R      | 0x20215064 |
| BAUD     | Access to baud counter                  | R/W    | 0x20215068 |

### Initializing UART

- ① ENB = 0x00000001 ; enable UART
- ② IER = 0x00000000 ; enable interrupts
- ③ CONTROL = 0x00000000 ; [0] receive enable  
 ↳ [1] send enable
- ④ LCR = 0x00000001 ; [0]=1 => 8-bit communication
- ⑤ MCR = 0x00000000 ; [1]=0 => Request-to-(RTS)  
 ↳ Transmit Signal is high!
- ⑥ FCR (IIR = 0x000000C6 ; [1]=0, [0]=1 => reset RCV FIFO  
 ↳ [2]=0 => reset SND FIFO  
 ↳ [7:6] enable FIFO)
- ⑦ BAUD = 270 ; system clock freq/baud rate / 8 = 250M / 115.2K - 1 = 270
- ⑧ Set GPIO line 14 for transmission (TXD)
- ⑨ Set GPIO line 15 for receiving (RXD)

- ⑩ Disable GPIO pull-up/down
- ⑪ Wait (150 cycles)
- ⑫ Assert clock lines (14 & 15)
- ⑬ Wait (50 cycles)
- ⑭ Clear clock lines
- ⑮ Enable bits 0 (recv) and 1 (send) in CONTROL.

| AUX_MU_LSR_REG Register (0x7E21 5054)                        |                   |                                                                                                                                                                                                                                                                                                                             |      |       |
|--------------------------------------------------------------|-------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------|-------|
| Synopsis: The AUX_MU_LSR_REG register shows the data status. |                   |                                                                                                                                                                                                                                                                                                                             |      |       |
| Bit(s)                                                       | Field Name        | Description                                                                                                                                                                                                                                                                                                                 | Type | Reset |
| 31:8                                                         |                   | Reserved, write zero, read as don't care                                                                                                                                                                                                                                                                                    |      |       |
| 7                                                            |                   | Reserved, write zero, read as don't care<br>This bit has a function in a 16550 compatible UART but is ignored here                                                                                                                                                                                                          | 0    |       |
| 6                                                            | Transmitter idle  | This bit is set if the transmit FIFO is empty and the transmitter is idle. (Finished shifting out the last bit).                                                                                                                                                                                                            | R    | 1     |
| 5                                                            | Transmitter empty | This bit is set if the transmit FIFO can accept at least one byte.                                                                                                                                                                                                                                                          | R    | 0     |
| 4:2                                                          |                   | Reserved, write zero, read as don't care<br>Some of these bits have functions in a 16550 compatible UART but are ignored here                                                                                                                                                                                               | 0    |       |
| 1                                                            | Receiver Overrun  | This bit is set if there was a receiver overrun. That is, one or more characters arrived whilst the receive FIFO was full. The newly arrived characters have been discarded. This bit is cleared each time this register is read. To do a non-destructive read of this overrun bit use the Mini Uart Extra Status register. | R/C  | 0     |
| 0                                                            | Data ready        | This bit is set if the receive FIFO holds at least 1 symbol.                                                                                                                                                                                                                                                                | R    | 0     |

### Writing a Character

1. Test the LSR register if the transmitter is empty  
can write if bit #5 in LSR is set  
wait in a loop until bit #5 is set
2. Write one byte to the ZO register
3. Recall
  - LSR is 0x2021504C
  - ZO 0x20215040

### Reading a Character

1. Test the LSR register if there is something to read  
can read if bit #0 in LSR is set  
wait in a loop until bit #0 is set
2. Read one byte from the ZO register.

### Reading a Memory Mapped Register

```
.equ AUX-MU-LSR-REG, 0x20215054
ldr r2, =AUX-MU-LSR-REG
ldr r1, [r2]
tst r1, #0x20 // 0x20 = 100000
```

### Writing a Memory Mapped Register

```
.equ AUX-MU-ZO-REG, 0x20215040
ldr r2, =AUX-MU-ZO-REG
ldr r0, #41 // ASCII A
str r0, [r2] // write ASCII A
```

### Example

```
bl InitUART // initialize UART
mov r0, #0x41 // ASCII code for A
loop:
  bl PutCharUART // write char
  add r0, #1 // next char
  cmp r0, #0x5A // ASCII for Z
  bne loop // loop until z.
```

```
void putCharuart(register char c) {
    while ((aux[AUX-MU-LSR] & 0x20) == 0);
    // loop until space in mini uart fifo
    aux[AUX-MU-ZO] = c;
}

char getCharuart() {
    while ((aux[AUX-MU-LSR] & 0x01) == 0);
    // loop until something in input fifo
    return (char)(aux[AUX-MU-ZO] & 0x7f);
}
```

### Running on UART Terminal

```
screen /dev/ttys0 225200.
```

## Exceptions Interrupts 异常中断

### Basic Concepts

- ↳ An exception is a system event or condition that requires action by privileged software to deal with the event. 特殊软件
- ↳ This software is called an exception handler or interrupt handler or interrupt service routine. 异常处理程序
- ↳ Polling 抢权输入输出
  - Program waits in a loop until ready for input or output
  - cannot respond/move on while in loop
  - easy
  - bad
- ↳ interrupt driven 中断驱动
  - program operates without wait loops
  - handles I/O by responding to interrupts
  - harder
  - better
- ↳ Some exceptions are synchronous
  - the result from the execution (or attempted execution) of an instruction
  - once handled, control returns to the next instruction.
- ↳ Some exceptions are asynchronous
  - called interrupts
  - may occur at any time
  - before handling save program state
  - once handled, restore program state and continue with next instruction
- ↳ Interrupts are usually triggered by external hardware
  - an electrical signal applied to an input pin on the CPU
  - ARM has 2
    - ↳  $IRQ$  - normal interrupt requests
    - ↳  $FIRQ$  - fast interrupt requests
- ↳ There ~~are~~ may be many source of an interrupt
  - CPU has one  $IRQ$
  - an interrupt controller coordinates many sources.



- ↳ The controller serializes and prioritizes contending interrupts 控制器对冲突的中断进行序列化和优先级分配

- ↳ Normally one handler for each type of exception
  - CPU selects the exception handler to use with exception vector table
  - or interrupt vector table

### Anatomy of an interrupt



### Interrupt sources

- ↳ most peripherals can be configured to generate interrupt 大多数外围设备可以配置生成中断
  - e.g., when edge is detected on a gpio pin
  - \*  $GPRENO = 0x1 \ll 23$
- ↳ Also have registers to record which peripherals have generated interrupts 还有寄存器来记录哪些外围设备产生了中断
  - Interrupt controller
  - \*  $GPEDSO \& (0x1 \ll 23)$

### Interrupt controller

- ↳ has 3 registers to enable specific IRQs

$$* IRQ\_ENABLE\_IRQ\_2 = (0x1 \ll 20)$$

- ↳ has 3 registers that record which specific IRQ is pending.

The base address for the ARM interrupt register is 0x7E008000.

#### Registers overview

| Address offset | Name               | Notes |
|----------------|--------------------|-------|
| 0x200          | IRQ basic pending  |       |
| 0x204          | IRQ pending 1      |       |
| 0x208          | IRQ pending 2      |       |
| 0x20C          | FIQ control        |       |
| 0x210          | Enable IRQs 1      |       |
| 0x214          | Enable IRQs 2      |       |
| 0x218          | Enable Basic IRQs  |       |
| 0x21C          | Disable IRQs 1     |       |
| 0x220          | Disable IRQs 2     |       |
| 0x224          | Disable Basic IRQs |       |

VG

The following is a table which lists all interrupts which can come from the peripherals which can be handled by the ARM.

ARM peripherals interrupts table.

| #  | IRQ 0-15             | #  | IRQ 16-31 | #  | IRQ 32-47       | #  | IRQ 48-63   |
|----|----------------------|----|-----------|----|-----------------|----|-------------|
| 0  |                      | 16 |           | 32 |                 | 48 |             |
| 1  | system timer match 1 | 17 |           | 33 |                 | 49 | gpio int[0] |
| 2  |                      | 18 |           | 34 |                 | 50 | gpio int[1] |
| 3  | system timer match 3 | 19 |           | 35 |                 | 51 | gpio int[2] |
| 4  |                      | 20 |           | 36 |                 | 52 | gpio int[3] |
| 5  |                      | 21 |           | 37 |                 | 53 | i2c int     |
| 6  |                      | 22 |           | 38 |                 | 54 | spi int     |
| 7  |                      | 23 |           | 39 |                 | 55 | pcm int     |
| 8  |                      | 24 |           | 40 |                 | 56 |             |
| 9  | USB controller       | 25 |           | 41 |                 | 57 | uart int    |
| 10 |                      | 26 |           | 42 |                 | 58 |             |
| 11 |                      | 27 |           | 43 | i2c_spi_slv_int | 59 |             |
| 12 |                      | 28 |           | 44 |                 | 60 |             |
| 13 |                      | 29 | Aux int   | 45 | pwa0            | 61 |             |
| 14 |                      | 30 |           | 46 | pwa1            | 62 |             |
| 15 |                      | 31 |           | 47 |                 | 63 |             |

The table above has many empty entries. These should not be enabled as they will interfere with the GPU operation.

p 112

```
// Reference to the global shared value
extern unsigned int sharedValue;

void IRQ_Handler()
{
    // Handle GPIO interrupts in general
    if (*IRQ_PENDING_3 == (0x1 << 20)) { // was there an interrupt from gpio<17>
        if (*GPIOIN == (0x1 << 17)) { // was the interrupt from pin 17
            // Clear the interrupt by writing a 1 to the GPIO Data Register
            *GPIOIN = (0x1 << 17);
            // Handle the interrupt
            // We do this by incrementing the shared global variable
            sharedValue++;
        }
    }

    // Return to the IRQ exception handler stub
    return;
}

// Declares a global shared variable
unsigned int sharedValue;
void main()
{
    unsigned int localValue;

    // Init shared and local value
    localValue = sharedValue = 0;

    init_GPIO17_to_existingInputInterrupt(); // init input pin with interrupt enabled
    enableIRQ17(); // enable interrupts for the CPU

    // Loop forever, waiting for interrupts to change the shared value
    while (1)
    {
        if (localValue != sharedValue) { // has sharedValue changed - interrupt happened
            localValue = sharedValue; // update localValue
            // Print out the shared value
            uart_putchar((sharedValue) | 0x20); // uart_putchar(sharedValue); uart_puts("a");
        }
    }
}
```

## Interrupt controller

L> Note: GPIO IRQs are grouped into 3 banks

- IRQ 49: pins 0-27
- IRQ 50: pins 28-45
- IRQ 51: pins 46-53
- IRQ 52: all pins

## Examples



## How to write a handler

L> should be minimal amount of code

- needs to be fast
- don't want to be in handler for long

L> needs to sort out where the interrupt is coming from

L> use global variable to share data

L> clear the interrupt before returning.

# SNES

↳ Super Nintendo Entertainment System Controller

GPIO SNES Pin - P1:2/3

DAT 19 GPIO10 (SPZ-MOSI)  
LAT 21 GPIO9 (SPZ-MISO)  
CLK 23 GPIO11 (SPZ-CLK)

| Button Number/<br>Clock Cycle | Button                                |
|-------------------------------|---------------------------------------|
| 1                             | B                                     |
| 2                             | Y                                     |
| 3                             | Select                                |
| 4                             | Start                                 |
| 5                             | Joy-pad UP                            |
| 6                             | Joy-pad DOWN                          |
| 7                             | Joy-pad LEFT                          |
| 8                             | Joy-pad RIGHT                         |
| 9                             | A                                     |
| 10                            | X                                     |
| 11                            | Left                                  |
| 12                            | Right                                 |
| 13-16                         | Unused - for future use (always high) |

SNES Lines

↳ LATCH: latch line

- ↳ used by the core to instruct the controller to latch the state of the buttons internally
- which buttons on the controller are pressed

↳ CLOCK: clock line

- ↳ The core sends a clock signal of 16 intervals
- one for each button
- Interval 1 is for Button B; 2 for Y etc...

↳ DATA: data line

- ↳ Controller sends serial signal to core indicating which buttons are pressed.

Initializing SNES

- Set GPIO pin 11 (CLK) to output
- Set GPIO pin 9 (LATCH) to output
- Set GPIO pin 10 (DATA) to input (code 0)

Communication Protocol 通信协议

↳ LATCH is used to instruct the SNES to sample and latch buttons

- Which buttons are pressed
- Latched in a register

↳ CLOCK line is used to instruct the SNES to synchronize serial signal sent from SNES to the core

时钟线用于同步从SNES发送到核心的串行信号。

↳ DATA line is used to read the serialized latched

register representing button sampling.

数据线用于读取代表按钮采样的串行锁存寄存器。

Communication Protocol (1)

Core Asks for a report on buttons



Communication Protocol (2)

SNES Samples Buttons

↳ Sampling buttons = creating a register representation of which buttons are pressed

↳ One bit for each button is sufficient

• Example: 11110111100

• Three buttons are pressed (buttons 0, 1, & 7)

Communication Protocol (3)

Core pulses clock; SNES to sends report



Communication Protocol (3)

Core pulses clock; SNES to sends report

↳ Pulsing continues for 16 times

↳ SNES has 12 buttons

• First 12 pulses sample the buttons

↳ Last 4 pulses are not used (always 0)

• Reserved for future extensions

Communication Protocol (3)



※ 通信协议，开放系统互连协议中最早的一种之一，它为连接不同操作系统和不同硬件体系结构的互联网络提供通信支持。

Again - ABM in C

|                |    |                                               |
|----------------|----|-----------------------------------------------|
| #define CLK    | 11 | ← GPIO pins                                   |
| #define LAT    | 9  |                                               |
| #define DAT    | 10 |                                               |
| #define GPSET0 | 7  |                                               |
| #define GPCLR0 | 10 | ← register offsets from<br>GPIO base in words |
| #define GPLDVO | 13 |                                               |
| #define HCYC   | 6  | ← cycle times in micro<br>seconds             |
| #define FCYC   | 12 |                                               |

## Raspberry Pi System Timer

### System timer

- The Broadcom system on chip has a free running 64 bit counter
- The Counter is driven by a clock with frequency of 1 MHz
  - 1 million ticks per second
  - 1 microsecond per tick

### Registers for the time

↳ base address is 0x3f003000

### Polling the timer

↳ Use the two 32-bit registers

- CLO - low order bits
- CHI - high order bits

↳ Not aligned for 64-bit read

↳ Do two 32-bit reads

### How to read

- read CHI
- read CLO
- read CHI again
  - if CHI changed we know the counter turned over between bits 31/32
- if CHI changed, then read CLO again
- assemble a 64-bit count from the two 32-bit parts

```
unsigned long getSystemTimerCounter() {
    unsigned int h = -1, l;
    // we must read MMIO area as 2 separate 32
    // bit reads
    h = *(CHI);
    l = *(CLO);
    // we have to repeat it if high word changed
    // during read
    if (h != CHI) {
        h = *(CHI);
        l = *(CLO);
    }
    // compose long int value
    return (unsigned long) h < 32 | l;
```

### System Timer

```
0x3f003000
12.1 System Timer Registers
ST Address Map
Address Register Name Description
0x0 System Timer Control/Status
0x4 System Timer Counter Lower 32 Bits
0x8 System Timer Counter Higher 32 Bits
0xc System Timer Compare 0
0x10 System Timer Compare 1
0x14 System Timer Compare 2
0x18 System Timer Compare 3

systock = (unsigned int *) iomap(..., CLOCK_BASE);
clock = (long *) iamsysclock(1);

int readTimeController() {
    int conWord = 0;
    int bitMask;
    int bit;
    gpio[GPSET0] = 1 << CLE; // CLK high
    gpio[GPCLR0] = 1 << LAT; // Latch high
    timeWait(HCYC); // wait for 1usec
    gpio[GPSET0] = 1 << CLE; // CLK low
    gpio[GPCLR0] = 1 << LAT; // Latch low
    bitMask = 1;
    while (bitMask & 0x10000) { // wait for 0usec
        timeWait(HCYC); // wait for 1usec
        bit = (gpio[GPLDVO] >> DAT) & 1; // read DAT
        if (bit) conWord |= bitMask; // set bit in conWord
        gpio[GPSET0] = 1 << CLE; // CLK high
        bitMask <<= 1;
    }
    return conWord;
}

One iteration
LATCH
CLOCK
DATA
```

```
16 iterations of pulseLoop

#define _GPIOD_
#define _GPIOL_

int readTimeController();
int readTimeController();
void timedWait( int delay );

#define GPIO_A 0
#define GPIO_B 0
#define GPIO_RIGHT 11
#define GPIO_IS_BUTTON_DOWN( bottom ) ((bottom >> bottom) & 1 == 0)

char *buttonString[] = {
    "B", "A", "Select", "Start",
    "D", "Up", "Right", "Left",
    "X", "Y", "Left", "Right"
};

#endif
```

## Polled timed wait

- ↳ read start time from the timer
- ↳ wait in busy loop until timer value reaches start time + desired delay.

```
void timedWait(int delay) {
    unsigned long endTime;
    endTime = getSystemTimerCounter() + delay;
    while (endTime > getSystemTimerCounter())
        asm volatile("nop");
}
```

## Timer interrupts



## Timer compare registers

- ↳ 4 timer compare registers
  - C0, C1, C2, C3
- ↳ timer status register (CS)
  - bits 0-3 indicate timer match
  - writing 1 to bits 0-3 clears corresponding interrupt

## Interrupt control

- ↳ use bit 1 of
  - IRQ\_Pending-1
  - Enable\_IRQS-1
  - Disable\_IRQS-1
- ↳ after that looks like GPIO interrupts.

## To get an interrupt in the future

- ↳ read the timer
- ↳ add desired delay in microsecond
- ↳ write least sig 32 bits to C1 register
- ↳ enable C1 interrupts in Enable\_IRQS-1
- ↳ enable interrupts in CPU

## To handle timer interrupt

- ↳ Verify that interrupt came from C1
  - Check bit 1 in pending-1 register
- ↳ clear interrupt
  - write 1 to bit 1 of CS

↳ do what needs doing

↳ set up a future interrupt if needed

## Example handler

```
void IRQ_handler() {
    // Handle GPIO interrupts in general
    if (*IRQ_Pending-1 & 0x2) {
        // clear timer interrupt
        *CS = 0x2; // clear system timer match 1 (C1)
        // do some stuff
        // set C1 for interrupt 2s into future
        unsigned long t = getSystemTimerCounter();
        *C1 = (unsigned int)(t + 2000000) & 0xfffff;
    }
}
```

\*: 树莓派的计时器

→ GPIO控制器类似，计时器也有一个地址。  
在这里计时器的地址是 20003000 (+十六进制)

树莓派没有板载电源所以不加电源时候时间就不准，因此我们只能依靠这个计时器来计时。计时器本身有64位，与其搭配的还有32位的控制寄存器、4个32位的比较寄存器，计时器是双端口的。

计时器每一秒/μs加1，每次加完都会将低32位与比较寄存器进行比较，如果相等仍一个匹配上了，就根据所匹配的寄存器对控制/或者叫状态寄存器进行更新

## Video Programming

### 2D Arrays

↳ Two-dimensional arrays must be mapped onto 1D memory ↳ can use row-major ordering

- Store each element of row 0, then row 1, etc.
- Used in most high-level languages, including C, C++.

↳ or column-major ordering

- Store each element of column 0, then column 1, ...

#### Example

- Logical arrangement.

|    |    |    |
|----|----|----|
| 10 | 20 | 30 |
| 40 | 50 | 60 |

- Mapping in row-major order

|    |    |    |    |    |    |
|----|----|----|----|----|----|
| 10 | 20 | 30 | 40 | 50 | 60 |
|----|----|----|----|----|----|

- Mapping in column-major order

|    |    |    |    |    |    |
|----|----|----|----|----|----|
| 10 | 40 | 20 | 50 | 30 | 60 |
|----|----|----|----|----|----|

## Indexing 2D Arrays

↳ The indices for a multidimensional array must be converted into an offset

- Added to array starting address

↳ For a 2D array list [0..n-1][0..m-1], row-major order, list[i][j] is:

$$((m \cdot i) + j) \cdot \text{Esize}$$

#### Example C Code declaration of

char array  $\text{list}[2][3]$  → 24 bytes

array  $\text{list}[2][3]$  maps to an offset of  $(3 \cdot (1+2)) \cdot 1$

$$(3 \cdot (1+2)) \cdot \text{Esize}$$



## In C



```
int a[6] = {10, 20, 30, 40, 50, 60};
int *p;
int b;

a[3] = 5; // assign element 3 of a
p = a; // p refers to elements of a
*(p+3) = 5; // does the same assignment
b = a[5]; // assign b value of a[5] - 60
b = *(p+4); // assign b value of a[4] - 50
// note - with C pointer arithmetic
// C accounts for no. bytes in int.
```

```
int a[6] = {10, 20, 30, 40, 50, 60};
int nr=2, nc=3; // n rows, n columns
int *p;
int b;

a[1*nc+2] = 5; // assigns a[1,2]
p = a; // p refers to elements of a
*(p+1*nc+2) = 5; // does the same assignment
b = a[0*nc+2]; // assign b value of a[0,2] - 30
b = *(p+4); // assign b value of a[4] - 50
```

## Frame-Buffers 帧缓存 (显存)

※ 显屏所显示画面的一个直接映射，又称位映射图 (Bit Map) 或光栅。

帧缓存的每一个存储单元对应屏幕上一个像素，整个帧缓存对应一幅图像。

↳ A frame-buffer is a memory mapped arrangement for monitors 帧缓存区是用于显示器的内存映射安排

↳ Each pixel is mapped to a memory address

↳ To write a pixel, the user simply set the corresponding value in main memory

↳ Can be implemented in a separate Video RAM or in a reserved section of RAM.



```

unsigned char *frameBuffer = (unsigned char *) 0x12345678;
int nr = 768;
int nc = 1024;
int pitch = 1024;
int i, j;

frameBuffer[i * nc + j] = 0xff // assuming nc = pitch
frameBuffer[i * pitch + j] = 0xff; // best!

```

## Drawing a Pixel

→ To draw a single pixel, set its value in the frame buffer

- Must map from 2D logical space → 1D Physical RAM
- Must know pitch and element size for particular resolution
- Use formula: row offset = y \* pitch
  - X, y are pixel's coordinates
  - Origin is upper left-hand corner
  - x range: 0 to width-1 ✓
  - y range: 0 to height-1 ✓
  - Pitch is number of bytes in row
    - ↳ not necessarily equal to number of elements \* size
  - physical offset = row offset + x \* element size

## Frame Buffer Coordinates



### Examples

- E.g. XGA (3 bytes per pixel, 24-bit color)
- pitch given by frame buffer info
  - happens to be  $3 \times 1024 = 3072$
- element offset =  $(y * \text{pitch}) + x * 3$ 
  - Pixel (0,0)
  $0 * \text{pitch} + 0 * 0 = 0$
  - Pixel (1023, 767) (lower right-hand corner)
  $(767 * \text{pitch}) + 1023 * 3 = 2359293$
  - check
  $(1024 * 768 - 1) * 3 = 2359293$

```

static FrameBufferInfo
fbi_attribute_((aligned(16)));
void fillBoxVideo(
    unsigned char r,
    unsigned char g, unsigned char b,
    unsigned int xo, unsigned int yo,
    unsigned int w, unsigned int h) {
    unsigned char *fb;
    int x, y;

    for(y = yo; y < yo+h; y++) {
        fb = (unsigned char *) ((unsigned int)fbi.fb +
                               y * fbi.pitch + 3 * xo);
        for(x = xo; x < xo+w; x++) {
            *fb++ = r;
            *fb++ = g;
            *fb++ = b;
        }
    }
}

```

## Plot a point

• extrapolate from box fill

```

// NB - untested - probably needs
// some type casting
// for 24-bit color
static FrameBufferInfo
fbi_attribute_((aligned(16)));
void point(
    unsigned char r,
    unsigned char g,
    unsigned char b,
    unsigned int x,
    unsigned int y) {
    unsigned char *fb;

    // add more code to check bounds if
    // you like
    fb = fbi.fb + y * fbi.pitch + 3 * x;
    *fb++ = r;
    *fb++ = g;
    *fb++ = b;
}

```

## DDA

→ Digital Difference Analyzer

→ a well-known algorithm for drawing lines on raster.

→ Bresenham's algorithm is best-known example

???

```

void line(int xo, int yo, int xi, int yi) {
    int dx = abs(xi-xo), sx = xo<xi ? 1 : -1;
    int dy = abs(yi-yo), sy = yo<yi ? 1 : -1;
    int err = (dx>dy ? dx : -dy)/2, e2;

    for(; ;){
        setPixel(xo,yo);
        if (xo==xi && yo==yi) break;
        e2 = err;
        if (e2 > dx) { err -= dy; xo += sx; }
        if (e2 < dy) { err += dx; yo += sy; }
    }
}

```

## P: 3 Video Control Mailbox ???

## I<sup>2</sup>C - Inter-Integrated Circuit bus

\* I<sup>2</sup>C 是一种简单、双向二线制同步串行总线。  
它只需要两根线即可在连接于总线上的器件之间传递信息。

- ↳ Sometimes called a two-wire interface (TWI)
- ↳ Multiple devices chained on 2 wires
  - does not include power
  - SDA = serial data
  - SCL = serial clock



- ↳ Each device must have a unique address
- ↳ Devices contain a set of register that control their behaviour
  - these are specific to each device
- ↳ Use the device to writing to and reading from these registers.

- ↳ Write to a device register
  - send device address
    - address on the bus
  - send register address
    - address within the device
  - send data.

- ↳ Read from a device register
  - send device address
    - address on the bus
  - send register address
    - address within the device
  - to listen for data to come back

### Important methods

- read\_byte\_data(addr, cmd)
  - returns byte from register
- read\_word\_data(addr, cmd)
  - returns word (16-bits) from register
- read\_i2c\_block\_data(addr, cmd, len=32)
  - returns list of len bytes from registers starting at cmd
- write\_byte\_data(addr, cmd, val)
  - writes val (8-bit) to register
- write\_word\_data(addr, cmd, val)
  - writes val (16-bits) to register
- write\_i2c\_block\_data(addr, cmd, [vals])
  - writes list of b-bit [vals] to registers starting at cmd

## ABM I<sup>2</sup>C

← 前面

### On RPi

- ↳ On RPi: I<sup>2</sup>C happens through the GPIO
  - requires specific pin selection
  - uses alternate function.

## BSC (I<sup>2</sup>C)

Broadcom Serial controller

- ↳ RPi has 3 BSC MASTERS

- BSC0      0x7E20-5000
- BSC1      0x7E80-4000
- BSC2      0x7E80-5000

- ↳ Use BSC1

- very old Pis may use BSC0
- BSC0 now used for EEPROM ID on hat

### Two things to do

#### GPIO

- configure GPIO pins for BSC1

pins 2 and 3

- required at initialization before you can use BSC1

#### BSC1 registers

- use these registers to perform I<sup>2</sup>C transfers.

## GPIO Registers

| Address     | Field Name | Description             | Size | Read/Write |                 |
|-------------|------------|-------------------------|------|------------|-----------------|
| 0x7E20_0000 | GPISEL0    | GPIO Function Select 0  | 32   | R/W        | Function Select |
| 0x7E20_0004 | GPISEL1    | GPIO Function Select 1  | 32   | R/W        |                 |
| 0x7E20_0008 | GPISEL2    | GPIO Function Select 2  | 32   | R/W        |                 |
| 0x7E20_000C | GPISEL3    | GPIO Function Select 3  | 32   | R/W        |                 |
| 0x7E20_0010 | GPISEL4    | GPIO Function Select 4  | 32   | R/W        |                 |
| 0x7E20_0014 | GPISEL5    | GPIO Function Select 5  | 32   | R/W        |                 |
| 0x7E20_0018 |            | Reserved                |      |            |                 |
| 0x7E20_001C | GPISET0    | GPIO Pin Output Set 0   | 32   | W          | Set (Write 1)   |
| 0x7E20_0020 | GPISET1    | GPIO Pin Output Set 1   | 32   | W          |                 |
| 0x7E20_0024 |            | Reserved                |      |            |                 |
| 0x7E20_0028 | GPICLR0    | GPIO Pin Output Clear 0 | 32   | W          | Clear (Write 0) |
| 0x7E20_002C | GPICLR1    | GPIO Pin Output Clear 1 | 32   | W          |                 |
| 0x7E20_0030 |            | Reserved                |      |            |                 |
| 0x7E20_0034 | GPIOEV0    | GPIO Pin Level 0        | 32   | R          | Level (Read)    |
| 0x7E20_0038 | GPIOEV1    | GPIO Pin Level 1        | 32   | R          |                 |
| 0x7E20_003C |            | Reserved                |      |            |                 |

### GPIO alt functions

Every GPIO pin can carry an alternate function. Up to 6 alternate functions are available but not every pin has that many alternate functions. The table below gives a quick overview.

|       | Pull | ALT0      | ALT1      | ALT2      | ALT3 | ALT4 | ALT5 |
|-------|------|-----------|-----------|-----------|------|------|------|
| GPIO0 | High | S0A0      | S0A0      | S0A0      |      |      |      |
| GPIO1 | High | S0C0      | S0C0      | S0C0      |      |      |      |
| GPIO2 | High | S0A1      | S0A1      | S0A1      |      |      |      |
| GPIO3 | High | S0L1      | S0L1      | S0L1      |      |      |      |
| GPIO4 | High | S0L0      | S0L0      | S0L0      |      |      |      |
| GPIO5 | High | S0L1      | S0L1      | S0L1      |      |      |      |
| GPIO6 | High | S0L0      | S0L0      | S0L0      |      |      |      |
| GPIO7 | High | S0L0_C0_N | S0L0_C0_N | S0L0_C0_N |      |      |      |
| GPIO8 | High | S0L0_C0_N | S0L0_C0_N | S0L0_C0_N |      |      |      |

## GPIO initialization

↳ Need GPIO pins 2,3 initialized for Alt0 function

↳ Review how you did this previously

- clear all config bits for the pin
  - equivalent to configuring for input
- write bits for desired functions
  - output or Alt.

|       |       |                                                                                                                                                                                                                                                                                                                                                                                      |     |   |
|-------|-------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----|---|
| 29-27 | FSEL9 | FSEL9 - Function Select 9<br>000 = GPIO Pin 9 is an input<br>001 = GPIO Pin 9 is an output<br>100 = GPIO Pin 9 takes alternate function 0<br>101 = GPIO Pin 9 takes alternate function 1<br>110 = GPIO Pin 9 takes alternate function 2<br>111 = GPIO Pin 9 takes alternate function 3<br>011 = GPIO Pin 9 takes alternate function 4<br>010 = GPIO Pin 9 takes alternate function 5 | R/W | 0 |
| 26-24 | FSEL8 | FSEL8 - Function Select 8                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 23-21 | FSEL7 | FSEL7 - Function Select 7                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 20-18 | FSEL6 | FSEL6 - Function Select 6                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 17-15 | FSEL5 | FSEL5 - Function Select 5                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 14-12 | FSEL4 | FSEL4 - Function Select 4                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 11-9  | FSEL3 | FSEL3 - Function Select 3                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 8-6   | FSEL2 | FSEL2 - Function Select 2                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 5-3   | FSEL1 | FSEL1 - Function Select 1                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |
| 2-0   | FSEL0 | FSEL0 - Function Select 0                                                                                                                                                                                                                                                                                                                                                            | R/W | 0 |

## In C - map GPIO registers

```
#include ...
#define GPIO_BASE 0x3f200000
#define BLOCK_SIZE 4096

int main()
{
    // Open /dev/mem
    int fd = open("/dev/mem", O_RDWR | O_SYNC);
    unsigned int *gpio = (unsigned int *) mmap(
        NULL, BLOCK_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd,
        GPIO_BASE);
    // Define gpio0/1 as sda0/scl0 (alt 0)

    // clear gpio2
    gpio[0] = gpio[0] & ~(0b111 << (SDA1 * 3));
    // alto in gpio2
    gpio[0] = gpio[0] | (0b100 << (SDA1 * 3));
    // clear gpio3
    gpio[0] = gpio[0] & ~(0b111 << (SCL1 * 3));
    // alto in gpio3
    gpio[0] = gpio[0] | (0b100 << (SCL1 * 3));
    ...
}
```

Use BSC1 registers for transfers

↳ Registers listed on p28-29 ...

↳ There are 8 registers for each BSC

↳ They're all the same, but correspond to different pins / buses.

offset in bytes  
remember pointer arithmetic

| I2C Address Map |               |                              |      |
|-----------------|---------------|------------------------------|------|
| Address Offset  | Register Name | Description                  | Size |
| 0x0             | C             | Control                      | 32   |
| 0x4             | S             | Status                       | 32   |
| 0x8             | DL            | Data Length                  | 32   |
| 0xc             | SA            | Slave Address                | 32   |
| 0x10            | DIF           | Data FIFO first in first out | 32   |
| 0x14            | CD            | Clock Divider                | 32   |
| 0x18            | DO            | Data Delay                   | 32   |
| 0x1c            | CST           | Clock Stretch Timeout        | 32   |

## C - Control

- Used to
  - enable interrupts
  - clear the FIFO
  - define R/W operation

| Bit(s) | Field Name | Description                                                                                                                                                                                                                                                                                                    | Type | Reset |
|--------|------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------|-------|
| 31:18  |            | Reserved - Write as 0, read as don't care                                                                                                                                                                                                                                                                      |      |       |
| 15     | I2CEN      | I2C Enable<br>0 = BSC controller disabled<br>1 = BSC controller is enabled                                                                                                                                                                                                                                     | RW   | 0x0   |
| 14:11  |            | Reserved - Write as 0, read as don't care                                                                                                                                                                                                                                                                      |      |       |
| 10     | INTR       | INTR interrupt on RX<br>0 = Don't generate interrupt on RXR condition.<br>1 = Generate interrupt while RXR = 1.                                                                                                                                                                                                | RW   | 0x0   |
| 9      | INTT       | INTT interrupt on TX<br>0 = Don't generate interrupt on TXW condition.<br>1 = Generate interrupt while TXW = 1.                                                                                                                                                                                                | RW   | 0x0   |
| 8      | INTD       | INTD interrupt on DONE<br>0 = Don't generate interrupt on DONE condition.<br>1 = Generate interrupt while DONE = 1.                                                                                                                                                                                            | RW   | 0x0   |
| 7      | ST         | ST Start Transfer<br>0 = No action. 1 = Start a new transfer. One shot operation. Read back as 0.                                                                                                                                                                                                              | RW   | 0x0   |
| 6      |            | Reserved - Write as 0, read as don't care                                                                                                                                                                                                                                                                      |      |       |
| 5:4    | CLEAR      | CLEAR FIFO Clear<br>00 = No action. 1 = Clear FIFO. One shot operation. 1 = Clear FIFO. One shot operation.<br>If CLEAR and ST are written in the same operation, the FIFO is cleared before the new frame is started. Read back as 0.<br>Note: 2 bits are used to maintain compatibility to previous version. | RW   | 0x0   |
| 3:1    |            | Reserved - Write as 0, read as don't care                                                                                                                                                                                                                                                                      |      |       |
| 0      | READ       | READ Read Transfer<br>0 = Write Packet Transfer. 1 = Read Packet Transfer.                                                                                                                                                                                                                                     | RW   | 0x0   |

## S - Status

### • Status of BSC

| Bit(s) | Field Name | Description                                                                                                                                                                              | Type | Reset |
|--------|------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------|-------|
| 31:10  |            | Reserved - Write as 0, read as don't care                                                                                                                                                |      |       |
| 9      | CLKT       | CLKT Clock Stretch Timeout<br>0 = No errors detected. 1 = Slave has held the clock signal longer than specified by writing 1 to the field.                                               | RW   | 0x0   |
| 8      | ERR        | ERR ACK Error<br>0 = No errors detected. 1 = Slave has not acknowledged its address. Cleared by writing 1 to the field.                                                                  | RW   | 0x0   |
| 7      | RXF        | RXE - FIFO Full<br>0 = FIFO is not full. 1 = FIFO is full. If a read is underway, no further serial data will be received until data is read from FIFO.                                  | RO   | 0x0   |
| 6      | TXE        | TDX - FIFO Empty<br>0 = FIFO is empty. 1 = FIFO is empty. If a write is underway, no further serial data can be transmitted until data is written to the FIFO.                           | RO   | 0x1   |
| 5      | RXD        | RDX - FIFO contains Data<br>0 = FIFO is empty. 1 = FIFO contains at least 1 byte. Cleared by reading sufficient data from FIFO.                                                          | RO   | 0x0   |
| 4      | TXD        | TDX - FIFO can accept Data<br>0 = FIFO is full. The FIFO cannot accept more data. 1 = FIFO has space for at least 1 byte.                                                                | RO   | 0x1   |
| 3      | RXR        | RXR - FIFO needs Reading (Full)<br>0 = FIFO is less than full and a read is underway. 1 = FIFO is or more full and a read is underway. Cleared by reading sufficient data from the FIFO. | RO   | 0x0   |
| 2      | TXW        | TDX - FIFO needs Writing (full)<br>0 = FIFO is at least full and a write is underway. 1 = FIFO is more full than a write is underway. Cleared by writing sufficient data to the FIFO.    | RO   | 0x0   |
| 1      | DONE       | DONE Transfer Done<br>0 = Transfer not completed. 1 = Transfer complete. Cleared by writing 1 to the field.                                                                              | RW   | 0x0   |
| 0      | TA         | TA Transfer Active<br>0 = Transfer not active. 1 = Transfer active.                                                                                                                      | RO   | 0x0   |

## DLEN - Data Length

- specifies length of data transfer

|      |      |                                                                                                                                                                                                                                                                                                                                |    |     |
|------|------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----|-----|
| 15:0 | DLEN | Data Length:<br>Writing to DLEN specifies the number of bytes to be transmitted/received. Reading from DLEN when TA = 1 or DONE = 1, returns the number of bytes still to be transmitted or received. Reading from DLEN when TA = 0 and DONE = 0, returns the last DLEN value written. DLEN can be left over multiple packets. | RW | 0x0 |
|------|------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----|-----|

higher order bits – write as 0, read as don't care

## A – slave address

| A Register                                                                                                                                                                                       |            |                                           |      |       |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------|-------------------------------------------|------|-------|
| Synopsis                                                                                                                                                                                         |            |                                           |      |       |
| The slave address register specifies the slave address and cycle type. The address register can be left across multiple transfers. The ADDR field specifies the slave address of the I2C device. |            |                                           |      |       |
| Bit(s)                                                                                                                                                                                           | Field Name | Description                               | Type | Reset |
| 31:7                                                                                                                                                                                             |            | Reserved - Write as 0, read as don't care |      |       |
| 6:0                                                                                                                                                                                              | ADDR       | Slave Address                             | RW   | 0x0   |

## FIFO

| FIFO Register                                                                                                                                                                                                                                                                                                                                                                                                                   |            |                                                                                                                |      |       |
|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------|----------------------------------------------------------------------------------------------------------------|------|-------|
| Synopsis                                                                                                                                                                                                                                                                                                                                                                                                                        |            |                                                                                                                |      |       |
| The Data FIFO register is used to access the FIFO. Write cycles to this address place data in the 16-byte FIFO, ready to transmit on the BSC bus. Read cycles access data received from the bus. Data writes to a full FIFO will be ignored and data reads from an empty FIFO will result in invalid data. The FIFO can be cleared using the I2CC.CLEAR field. The DATA field specifies the data to be transmitted or received. |            |                                                                                                                |      |       |
| Bit(s)                                                                                                                                                                                                                                                                                                                                                                                                                          | Field Name | Description                                                                                                    | Type | Reset |
| 31:8                                                                                                                                                                                                                                                                                                                                                                                                                            |            | Reserved - Write as 0, read as don't care                                                                      |      |       |
| 7:0                                                                                                                                                                                                                                                                                                                                                                                                                             | DATA       | Writes to the register write transmit data to the FIFO. Reads from register reads received data from the FIFO. | RW   | 0x0   |

## In C - map the registers

```
#include ...
#define BSC1_BASE 0x3804000
#define BLOCK_SIZE 4096

int main()
{
    // Open /dev/mem
    int fd = open("/dev/mem", O_RDWR | O_SYNC);

    ... 

    unsigned int *bsci = (unsigned int *)mmap(
        NULL, BLOCK_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd,
        BSC1_BASE);
}
```

## Do read-byte-data

A DLEN FIFO

```
// read device id from ADXL345
// start with writing the address register, then
start write cycle

bsci[3] = 0x53; // ADXL345 address
bsci[2] = 1; // data length 1 (register address)
bsci[4] = 0x00; // register address in FIFO to write
bsci[1] = (1 << 9) | (1 << 8) | (1 << 1), STATUS
//clearCLKT,ERR,DONE
bsci[0] = (1 << 15) | (1 << 7), //12CEN,ST

// wait till done
while( !(bsci[1] & 0x02) ) { // 0x02 masks DONE
bit
    usleep(1000);
}

// STATUS
```

DLEN STATUS

```
// get the response from the register
bsci[2] = 1;
bsci[1] = (1 << 9) | (1 << 8) | (1 << 1);
//clearCLKT,ERR,DONE
bsci[0] = (1 << 15) | (1 << 7) | (1 << 4) | 1;
// 12CEN, ST, CLEAR, READ

// wait till done
while( !(bsci[1] & 0x02) ) { // 0x02 masks
DONE bit
    usleep(1000);
}

unsigned int devId = bsci[4];
printf("device ID: 0x%02x\n", devId);
munmap(bsci, BLOCK_SIZE);
munmap(gpio, BLOCK_SIZE);
close(fd);
```

## Serial Peripheral Interface Bus

### Some I<sub>2</sub>C examples

↳ environmental breakout

↳ GPS

↳ Spectral sensor

...

## Serial Peripheral Interface Bus

↳ SPI bus

串行外围接口总线

↳ very common

- next only to I<sub>2</sub>C

↳ compared to I<sub>2</sub>C

- faster

- more wires required

- less flexible hardware interface

### Some SPI Examples

↳ ~~some~~ TFT display

↳ accelerometer

↳ real time clock

### Basics



↳ Defines Master and Slave

- master is microcontroller (CPU)

- slave is the peripheral (sensor, display, ...)

↳ Two serial lines

- MISO - master in slave out

- MOSI - master out slave in

- tri-state

• high impedance when SS is high

↳ SCLK - serial clock

- master provides clock signal to time serial communications

↳ SS

- enables ~~one~~ slave device

- active low

- separately controlled SS lines allow multiple slave devices on a single master.

控制多条SS线，许许多设备上有多台设备。

### The protocol

Master



Slave



## Actuators 电动机.

↳ An actuator is a component of a machine that is responsible for moving or controlling a mechanism or system, for example by actuating (opening or closing) a valve; in simple terms, it is a "mover".

↳ Need actuators in any computer system that makes things move

↳ 3D printer

---

## Current / Voltage / Power

↳ Actuators require a lot of power when compared to a logic gate

↳ Means more  
- current / voltage / power

↳ Almost always require external circuits to handle power.

## Motors 电动机

↳ Focus on motors

- most (not all) actuators have a motor

↳ - most of what I have to show you as example

→ Brushed DC Motor 直流电机.

操作 ...