# Inter-Integrated Circuit (I<sup>2</sup>C) Interface

ECE 362

https://engineering.purdue.edu/ece362/



## Reading Assignment

- Textbook, Chapter 22, Serial Communication Protocols, pp. 527 –
   598
  - It's a long chapter.
  - 22.2 is about I2C. It's mostly pretty good. Read it and understand.
  - We'll next look at Section 22.1, UART, pp. 527–545.
  - Don't worry so much about the USB section.
    - Read that only if you're curious.
    - Not much we can do with that.
    - Other books are better for understanding USB.

## OLED LCD displays

- SPI lab this week...
- When you "smoke" your display...
- If you order a replacement OLED LCD from crystalfontz.com, make sure you get one configured for SPI. (CFAL1602CPB rather than CFAL1602CB or CFAL1602C-B)
- The two are the same hardware, but conversion requires some fine soldering.



# 12C Peripheral

- Two independent "channels".
  - Convert between an internal parallel word and an external serial stream.
  - Synchronous clock pulse for each bit.
  - Only "data" (SDA) and "clock" (SCL) signals.
  - No "slave select".
    - Multi-master bus.
    - All I2C endpoints send and receive on the same wires.
    - All I2C endpoints have an address.

# How can everything share the same wires?

- Normal digital logic outputs are connected to push-pull drivers.
- I2C signals are connected to "open-drain" drivers.
  - They can not pull up for a logic high.
  - Instead, a pull-up resistor is responsible for keeping each signal high unless it is pulled low by <u>any</u> I2C driver.





# The physical layer

- Two wires (SCL & SDA), plus shared ground.
- Voltages: 5v, 3.3v, 1.8v, ... etc
- You may use multiple devices that share a connection with different voltages by using bidirectional level shifters.
  - This has an impact on performance.

#### Baud rate

- 100 kbit/s (standard mode)
- 400 kbit/s (fast mode)
- 1 mbit/s (fast mode+ (fm+))
- 10 kbit/s (low speed mode)
- 3.4 mbit/s (high speed mode) [not popular]
- Speed also limited by
  - Bus capacitance (400pF)
  - Strength of the pull-up resistor.
  - Length of the network.

Contrast to SPI, which can easily run at multiple megabits per second.

Slowest we can clock a 48MHz STM32's SPI is ~187 kHz.

• Effective data rate is less than half of clock rate due to addressing, acknowledgements, etc.

## 12C is a *synchronous* protocol

- A clock pulse accompanies each data bit.
- A clock signal must be delivered to each data recipient.

 By comparison, an asynchronous protocol would require only a data line.

#### 12C is Multi-Master

- Any I2C device could transition between master and slave.
  - Normally, a CPU is a master, and a peripheral is a slave.
  - There could easily be multiple CPUs on a single I2C bus.
- Devices can be receivers, transmitters, or both:
  - Temperature sensor: only transmits
  - LCD display: only receives
  - Flash memory IC: receives and transmits

## Why use I2C if it's slower than SPI?

- It's convenient to easily connect multiple devices using only two wires.
- As long as speed is not critical, it doesn't matter that it's slower than SPI.

# What do the signals look like?

An idle I2C bus:
 (SDA and SCL high. Nothing pulling low.)

SDA SCL

#### What communication looks like

- An I2C device starts a transaction with a START (S) bit.
- Sends a 7-bit address. (10 bit addr? Not common.)
- Sends a 1-bit intent 0: write, 1: read.
- Listens for an ACK/NACK (sent by receiver).
- Sends a STOP (P) bit.

START and STOP bits are the only times that SDA changes when SCL is held high.



#### ACK/NACK

- low = ACK, high = NACK
- If no device on the I2C bus will respond to the particular address that was sent, then nothing will acknowledge.
  - If nothing acknowledges, there is nothing to pull the line low.
     This indicates failure.
- A long transmission ends with a NACK.
  - This indicates termination of a multi-byte transaction.

#### All data is sent MSB first

- Both addresses and data are sent MSB.
- I2C protocol makes no definitions for the contents of the data fields.
  - Whatever the devices agree on.

#### Data can follow the ACK.

A master writing a command and byte to a slave:

| S | slave addr | R/W | A   | command | A   | data   | A   | P |
|---|------------|-----|-----|---------|-----|--------|-----|---|
| 1 | 7 bits     | 1   | 1   | 8 bits  | 1   | 8 bits | 1   | 1 |
|   |            | (0) | (0) |         | (0) |        | (0) |   |

## Many transactions with one stop bit

A master reading a byte from a slave:

| S<br>1 | slave addr<br>7 bits | R/W<br>1 | A<br>1 | command<br>8 bits | A<br>1 |        | (request)            |  |
|--------|----------------------|----------|--------|-------------------|--------|--------|----------------------|--|
|        |                      | (0)      | (0)    |                   | (0)    |        |                      |  |
| S<br>1 | slave addr<br>7 bits | R/W<br>1 | A<br>1 | data<br>8 bits    | N<br>1 | P<br>1 | (reply)              |  |
|        |                      | (1)      | (0)    |                   | (1)    | V      | Who sends this NACK? |  |

## Exploring an I2C device

- Consider a device such as an EEPROM (24AA32AF)
  - 32kbit (4 kbyte) serial EEPROM.
  - How can we find the datasheet?
  - How can we interpret the datasheet?
  - How should we connect it?
  - What are the protocols to access it?
  - How should we program the STM32 to access it?

## Finding datasheets

- Go to major electronics supplier website.
   (Digikey, Newark, Jameco)
- Type in part name.
- Refine search.
- Click on the PDF icon.

## Interpreting the Datasheet

- Main things:
  - Features, Pinout
  - DC, AC Characteristics, timing (2.5 5.5V)
  - Pin descriptions. (A0...A2. WP. SDA resistor)
  - Functional description.
    - 24XX32A does not generate <u>any</u> ack bits if an internal programming cycle is in progress.
  - Device addressing.
  - Write and read protocols.

### **Device Addressing**

- I2C peripherals often have fixed addresses.
  - This one is 0x50.
- 24AA32A has 3 of 7 address bits configurable.
- Could put 8 of these EEPROMS on the same bus as 0x50, 0x51, 0x52, ..., 0x57.
- Some devices have an entirely configurable I2C addr.
  - Such as an STM32.

#### Write and Read Protocols

- Section 6 of datasheet explains <u>write</u>.
  - Control byte, high address, low address, data, ...

Could send zero bytes of data.

- Immediately after write, the EEPROM stops ACKing until the write cycle is complete.
- Section 8 of datasheet explains <u>read</u>.
  - Control byte, data. (Current address read)
  - Control byte, high address, low address, START, control byte, data, ... (random read and sequential read)

# A good question:

- Every I2C slave device on the same bus must have a unique address.
- Why doesn't the master device need an address?

## How to set up STM32?

- Textbook and FRM have decent examples.
  - With several mistakes or omissions.
- Textbook section 22.2 shows lots of calcuations for timing parameters. Whew.
  - Use table 83, page 642 of FRM for parameters for an 8MHz clock.
     (Uses HSI clock by default.)
- Textbook Example 22-11 shows clock enable.
  - Symbols for STM32 are different.

### RCC clock & I2C1 setup

```
// Initialize I2C1 to 400 kHz
void i2c init(void) {
                                                         See I2C I/O reg
    RCC->AHBENR |= RCC AHBENR GPIOBEN:
                                             description on FRM page 662.
   GPIOB->MODER |= 2 << (2*6) | 2 << (2*7):
   GPIOB -> AFR[0] \mid = 1 << (4*6) \mid 1 << (4*7);
   RCC->APB1ENR |= RCC APB1ENR I2C1EN;
   //RCC->CFGR3 |= RCC CFGR3 I2C1SW:
                                      // to set for 48MHz sysclk
                                       // default is 8MHz "HSI" clk
   // I2C CR1 Config
   I2C1->CR1 &= ~I2C CR1 PE;
                                       // Disable to perform reset.
                                       // 0: Analog noise filter on.
   I2C1->CR1 &= ~I2C CR1 ANF0FF:
   I2C1->CR1 &= ~I2C CR1 ERRIE;
                                       // Error interrupt disable
   I2C1->CR1 &= ~I2C CR1 NOSTRETCH;
                                       // Enable clock stretching
    // From table 83. p642 of FRM. Set for 400 kHz with 8MHz clock.
    I2C1->TIMINGR = 0:
   I2C1->TIMINGR &= ~I2C TIMINGR PRESC;// Clear prescaler
                                      // Set prescaler to 0
    I2C1->TIMINGR | = 0 << 28:
   I2C1->TIMINGR |= 3 << 20;
                                       // SCLDEL
    I2C1->TIMINGR |= 1 << 16:
                                       // SDADEL
   I2C1->TIMINGR |= 3 << 8;
                                       // SCLH
   I2C1->TIMINGR \mid = 9 << 0:
                                       // SCLL
   // I2C "Own address" 1 register (I2C OAR1)
   I2C1->OAR1 &= ~I2C OAR1 OA1EN;
                                       // Disable own address 1
   I2C1->OAR1 = I2C OAR1 OA1EN | Ox2:// Set 7-bit own address 1
   I2C1->OAR2 &= ~I2C OAR2 OA2EN;
                                       // Disable own address 2
   I2C1->CR2 &= ~I2C CR2 ADD10;
                                       // 0 = 7-bit mode; 1 = 10-bit
   I2C1->CR2 |= I2C CR2 AUTOEND;
                                       // Enable the auto end
   I2C1->CR2 \mid = I2C CR2 NACK;
                                       // For slave mode: set NACK
    I2C1->CR1 |= I2C CR1 PE;
                                       // Enable I2C1
```

## Why must it be so difficult?

- A prescaler divides down the selected clock. (PRESC)
- You must configure the data setup time.
  - Time between data transition and next SCL posedge.
  - This is the "delay" of SCL (SCLDEL).
- You must configure the data hold time.
  - Time between SCL negedge and data transition.
  - This is the "delay" of SDA (SDADEL).
- To set up a 400kHz clock, you're not just configuring a 400kHz square wave. Instead:
  - you must configure the clock's minimum low time (SCLL)
  - you must configure the clock's minimum high time (SCLH)... Why? Clock stretching!

## Recall what the signals look like

- An I2C device starts a transaction with a START (S) bit.
- Sends a 7-bit address. (10 bit addr? Not common.)
- Sends a 1-bit intent 0: write, 1: read.
- Listens for an ACK/NACK (sent by receiver).
- Sends a STOP (P) bit.

START and STOP bits are the only times that SDA changes when SCL is held high.



# SDA and SCL timing



## Clock stretching

- Both the data and clock lines of an I2C bus are open-drain.
- When clock stretching is enabled (is not disabled), any device on the I2C bus can lengthen the low time of a clock.
- The master drives the clock, but a slave device may not be able to keep up.
- Any slave device may lengthen the clock <u>only</u> after the ACK bit and before the MSB of the next byte.

# Clock stretch example



## Writing and Reading Data

- See the textbook:
  - Ex. 22-13: I2C Start()
  - Ex. 22-14: I2C\_Stop()
  - Ex. 22-15: I2C\_WaitLineIdle()
  - Ex. 22-16: I2C\_SendData()
  - Ex. 22-17: I2C\_ReceiveData()

These are almost usable "as is."

## Example Program

```
int main(void)
    i2c init();
    while(1) {
         i2c waitidle();
         i2c start(0x50, 0, 0);
         int x=0;
         while((I2C1->ISR & I2C ISR TC) == 0 &&
                 (I2C1->ISR \& I2\overline{C}\_IS\overline{R}\_STOPF) == 0 \&\&
                 (I2C1->ISR \& I2C ISR NACKF) == 0)
              x++; // Wait until TC flag is set
         if (I2C1->ISR & I2C ISR NACKF)
              I2C1->ICR |= I2C_ICR_NACKCF;
         if (I2C1->ISR & I2C ISR STOPF)
              I2C1 - > ICR \mid = I2\overline{C} IC\overline{R} STOPCF;
         else
             i2c stop();
         nano wait(1000000);
```

Try this.

Put the oscilloscope on it.

## More meaningful example

```
int main(void)
   init lcd():
   display1("___");
   i2c init();
   char addr1[] = "\0\0Hello, World!";
   i2c senddata(0x50, addr1, sizeof addr1);
   while(1) {
        if (i2c senddata(0x50, addr1, 2) < 0) {
            I2C1->ICR |= I2C ICR NACKCF;
            I2C1->ICR |= I2C ICR STOPCF;
       } else
            break:
   while(1) {
        uint8 t addr[] = \{0,0\};
        i2c senddata(0x50, addr, sizeof addr);
        char line[32]:
        i2c recvdata(0x50, line, sizeof line);
        display1(line);
       for(;;);
```

Write a string to I2C EEPROM.
Address 0x000

Try a 2-byte address update. Wait for an ACK instead of NACK.

Try a 2-byte address update.

Read 32 bytes from addr 0x000.

## Important I/O Registers

- I2Cx\_TIMINGR: set up the clock rate and setup/hold values
- I2Cx\_CR1: configure channel
- I2Cx\_CR2: set up operations
  - START, STOP
- I2Cx\_TXDR/RXDR: data registers
- I2Cx ISR: read status
  - NACKF, STOPF
- I2Cx\_ICR: clear status

## Send start and stop bits

```
void i2c start(uint32 t devaddr, uint8 t size, uint8 t dir) {
   // dir: 0 = master requests a write transfer
   // dir: 1 = master requests a read transfer
   uint32 t tmpreg = I2C1->CR2;
    tmpreg &= ~(I2C CR2 SADD | I2C CR2 NBYTES
                I2C CR2 RELOAD | I2C CR2 AUTOEND |
                I2C CR2 RD WRN | I2C CR2 START | I2C CR2 STOP);
    if (dir == 1)
        tmpreg |= I2C CR2 RD WRN; // Read from slave
    else
        tmpreg &= I2C CR2 RD WRN; // Write to slave
    tmpreq = ((devaddr << \overline{1}) \& I2C CR2 SADD) | ((size << 16) & I2C CR2 NBYTES);
    tmpreg |= I2C CR2 START;
    I2C1->CR2 = tmpreq;
void i2c stop(void) {
    if (I2C1->ISR & I2C ISR STOPF)
        return:
   // Master: Generate STOP bit after current byte has been transferred.
   I2C1->CR2 |= I2C CR2 STOP;
   // Wait until STOPF flag is reset
   while( (I2C1->ISR & I2C ISR STOPF) == 0);
   I2C1->ICR |= I2C ICR STOPCF; // Write to clear STOPF flag
void i2c waitidle(void) {
    while ((I2C1->ISR & I2C ISR BUSY) == I2C ISR BUSY); // while busy, wait.
```

 Start sends the slave addr, and intent, and configures the size.

# i2c\_senddata()

```
int8 t i2c senddata(uint8 t devaddr, void *pdata, uint8 t size) {
   int i:
   if (size <= 0 || pdata == 0) return -1;
    uint8 t *udata = (uint8 t*)pdata;
   i2c waitidle();
    // Last argument is dir: 0 =  sending data to the slave.
   i2c start(devaddr, size, 0);
    for(i=0: i<size: i++) {</pre>
        // TXIS bit is set by hardware when the TXDR register is empty and the
        // data to be transmitted must be written in the TXDR register. It is
        // cleared when the next data to be sent is written in the TXDR reg.
        // The TXIS flag is not set when a NACK is received.
        int count = 0;
        while( (I2C1->ISR & I2C ISR TXIS) == 0) {
            count += 1;
            if (count > 1000000) return -1:
            if (i2c checknack()) { i2c clearnack(); i2c stop(); return -1; }
        // TXIS is cleared by writing to the TXDR register.
        I2C1->TXDR = udata[i] & I2C TXDR TXDATA;
   // Wait until TC flag is set or the NACK flag is set.
   while((I2C1->ISR & I2C ISR TC) == 0 && (I2C1->ISR & I2C ISR NACKF) == 0);
   if ( (I2C1->ISR & I2C ISR NACKF) != 0)
        return -1;
   i2c stop();
    return 0;
```

#### Lab 9

- Using I2C devices (EEPROM and GPIO chip)
- Writing code to read/write the devices
- Looking at signals with the oscilloscope

## Oscilloscope interpretation



S50WaP

START 0x50 WRITE ACK STOP