## I<sup>2</sup>C bus (Inter-Integrated Circuit) Designed for low-cost, medium data rate applications. (Phillips Semiconductor, 1980s) - ► Tutorial: <a href="http://www.esacademy.com/faq/i2c/">http://www.esacademy.com/faq/i2c/</a> - Characteristics: - serial, byte-oriented; - multiple-master; - fixed-priority arbitration; - moderate speeds: - standard mode: I00Kbits/s - fast mode: 400Kbits/s - high speed mode: 3.4 Mbits/s - ▶ Many microcontrollers come with built-in I<sup>2</sup>C controllers. Serial Buses Information Page: http://www.epanorama.net/links/serialbus.html ## I<sup>2</sup>C data link layer - Every device has an address - ▶ Set by device and/or system designer. - ▶ 7 bits in standard (10 bits in extension). - ▶ Bit 8 of address signals read (1) or write (0). - ▶ General call address (000000) for broadcast. - Bus transaction = series of one-byte transmissions - Master sends slave address followed by data to or from slave. - Good for "data-push" programming. ## I<sup>2</sup>C physical layer Uses only two wires (plus ground) ## I<sup>2</sup>C electrical interface (standard & fast speeds) - Open collector/drain drivers (default state high) - No global master for clock Source: I2C Specification ## I<sup>2</sup>C signaling - Bus = "wired-AND" configuration - Open collector/drain drivers on SDA & SCL - Resistor pulls bus up to logic 1. - Any sender can pull the bus down to 0, even if other senders are trying to drive the bus to 1. - Sender "releases" SDA by disabling its driver, allowing SDA to be pulled up to logic I - Data on SDA must be stable while SCL high - Data on SDA is sampled while SCL is high - SDA may change only while SCL low ### **Exceptions:** - ▶ SDA I->0 while SCL=I signals START condition - ▶ SDA 0-> I while SCL=I signals STOP condition ## I<sup>2</sup>C data format ## Clock synchronization - Master generates its own clock on SCL during data xfer - Clock synchronization uses wired-AND - Driving low pulls SCL low, resetting all clock counters - SCL remains low while any driver pulls it low - SCL low time = slowest clock (others in wait states) - First device to finishhigh state pulls SCL low start counting Source: I2C Specification ## Four I2C device operating modes #### Master-sender Module issues START and ADDRESS, and then transmits data to the addressed slave device #### Master-receiver Module issues START and ADDRESS, and receives data from the addressed slave device #### Slave-sender Another master issues START and the ADDRESS of this module, which then sends data to the master #### Slave-receiver Another master issues START and the ADDRESS of this module, which then receives data from the master. Some devices only support slave modes – sensors, memories, etc. ## I<sup>2</sup>C bus arbitration - Master may start sending if bus free - ▶ 2 or more may generate START at same time - Sender listens while sending. - Test SDA while SCL high - Sender stops transmitting if arbitration lost - Transmit I and hear 0 on SDA. - Arbitration continues through address & ack bits, and then data & ack bits if necessary ## Arbitration example Source: I2C Specification ### Data transfer - Send 8-bit byte (MSB first) - Each byte followed by acknowledge bit - master releases SDA line (high) during ack clock - slave must pull SDA low for proper acknowledge - if SDA left high, master may stop or repeat start - if master is receiving from slave, slave releases SDA to allow master to pull SDA low for ack - Slave can hold SCL low to force wait time between bytes ### Basic data formats #### Master transmitting data to slave #### Master receiving data from slave ## I<sup>2</sup>C transmissions (ACKs not shown) #### read from slave ### write, then read ## STM32 I<sup>2</sup>C Module (3 in STM32F407) - ▶ Standard I<sup>2</sup>C compliant bus interface. - ▶ All I<sup>2</sup>C bus-specific sequencing, protocol, arbitration, timing - 7-bit and 10-bit addressing - Standard (≤ I00KHz) or Fast (≤ 400KHz) speed modes - ▶ Multi-master capability use as master or slave - Also supports standards: - SMBus (System Management Bus) - PMBus (Power Management Bus) - ▶ DMA support between memory and data register - ▶ 2 interrupt vectors data transfer complete and errors ## STM32 I<sup>2</sup>C Module ## STM32 I<sup>2</sup>C registers #### I2C\_DR – I<sup>2</sup>C data register byte to be transmitted (start on DR write) byte received (RxNE=1) | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | |----|----|----|-------|------|----|---|---|----|----|----|----|--------|----|----|----| | | | | Dogo | nund | | | | | | | DF | R[7:0] | | | | | | | | Hesei | veu | | | | rw #### I2C\_OAR1 – I<sup>2</sup>C own address register 1 ADDMODE 0 = 7-bit ADD[7:1] (A second "own address" is also supported) # STM32 I<sup>2</sup>C – control register 1 I<sup>2</sup>C CR1 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | |-------|------|-------|-----|-----|-----|------|-------|---------------|------|-------|-------|-------------|------|-------|----| | SWRST | Res. | ALERT | PEC | POS | ACK | STOP | START | NO<br>STRETCH | ENGC | ENPEC | ENARP | SMB<br>TYPE | Res. | SMBUS | PE | | rw | | rw | rw | rw | PE Peripheral function **Enable** (1 enables the I<sup>2</sup>C module) STOP Generate after current byte xfer or after start condition sent START Master: repeated start generation, Slave: release bus after byte xfer ACK to be returned after byte received POS If ACK bit = 1: return ACK after current byte (0) or next byte (1) SWRST Software reset (or in reset state) NOSTRETCH Enable/disable clock stretch in slave mode when ADDR or BRG flag set, until reset by software ENGC Enable "general call" (ACK address 0x00) SMBUS 0 for I<sup>2</sup>C mode; 1 for SMBus mode (other bits for packet error checking (PEC) or SMBus setup) # SRM32 I<sup>2</sup>C – control register 2 ## I2C\_CR2 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | | | | |------|-----------|-------------|-------------|-------------|------|------|-----------|----|----|----|----|----|--|--|--| | LAST | DMA<br>EN | ITBUF<br>EN | ITEVT<br>EN | ITERR<br>EN | Rese | rved | FREQ[5:0] | | | | | | | | | | rw | rw | rw | rw | rw | | | rw | rw | rw | rw | rw | rw | | | | FREQ[5:0] = peripheral clock frequency (in MHz) allowed values [2MHz ... 42MHz] #### **DMA Control**: LAST: 1 = next DMA EOT is the last transfer DMAEN: 1 = DMA requests when TxE=1 or RxNE=1 #### Interrupt Control (interrupt generation events on next slide) **ITBUFEN**: 1 = TxE/RxNE event generates Event interrupt ITEVTEN: 1 = Event interrupt enabled ITERREN: 1 = Error interrupt enabled # STM32 I<sup>2</sup>C interrupts | Interrupt event | Event flag | Enable control bit | |--------------------------------------------------|------------|---------------------| | Start bit sent (Master) | SB | | | Address sent (Master) or Address matched (Slave) | ADDR | | | 10-bit header sent (Master) | ADD10 | ITEVFEN | | Stop received (Slave) | STOPF | | | Data byte transfer finished | BTF | | | Receive buffer not empty | RxNE | ITEVFEN and ITBUFEN | | Transmit buffer empty | TxE | TIEVEEN AND TIBOFEN | | Bus error | BERR | | | Arbitration loss (Master) | ARLO | | | Acknowledge failure | AF | | | Overrun/Underrun | OVR | ITERREN | | PEC error | PECERR | | | Timeout/Tlow error | TIMEOUT | | | SMBus Alert | SMBALERT | | # STM32 I<sup>2</sup>C – status register 1 (of two) I2C SR1 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | |--------------|-------------|------|------------|-------|-------|-------|-------|-----|------|------|-------|-------|-----|------|----| | SMB<br>ALERT | TIME<br>OUT | Res. | PEC<br>ERR | OVR | AF | ARLO | BERR | TxE | RxNE | Res. | STOPF | ADD10 | BTF | ADDR | SB | | rc_w0 | rc_w0 | | rc_w0 | rc_w0 | rc_w0 | rc_w0 | rc_w0 | r | r | | r | r | r | r | r | ADDR: Master: 1= address sent Slave: 1= received address matched OAR register or gen call SB: Master: 1= Start generated (clear by reading SR1 & DR) TxE: 1= transmitter buffer (DR) empty (can send a new byte) RxNE: 1= receiver buffer (DR) not empty (byte has been received) BTF: 1= data byte transfer finished successfully RxNE=1 & DR not read yet; TxE=1 & DR not written yet ARLO: 1= arbitration lost detected (this device lost to another) STOPF: 1= slave detected stop condition after ACK OVR: 1= DR register overrun/underrun (data lost) AF: 1= ACK failure (no ACK returned) BERR: 1= bus error (misplaced Start/Stop condition) ADD10: 1= master sent 1st byte of 10-bit address # STM32 I<sup>2</sup>C – status register 2 I2C SR2 | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | |----|----------|----|----|----|----|---|---|---|-------------|----------------|-------------|------|-----|------|-----| | | PEC[7:0] | | | | | | | | SMB<br>HOST | SMBDE<br>FAULT | GEN<br>CALL | Res. | TRA | BUSY | MSL | | r | r | r | r | r | r | r | r | r | r | r | r | | r | r | r | BUSY: 1= communication ongoing on the bus (cleared by Stop) MSL: 0= slave mode (default) 1= master mode (START has been sent) TRA: From R/W address bit: 1= data bytes to be TRAnsmitted 0= data bytes to be received DUALF: Received address matches OAR1 (0) or OAR2 (1) GENCALL: General call address (0x00) received when ENARP=1 (Other bits for PEC or SMBus) ## STM32 I2C bus "events" (from flags) - ▶ EV8: DR ready for new byte to transmit - ▶ BUSY MSL TXE (transmit buffer empty) - ▶ EV9: new byte received in the DR - ▶ BUSY MSL RXNE (receive buffer not empty) ## STM32 I2C bus "events" (from flags) #### Slave modes: - EVI: Own address received, data to be received from master - ▶ BUSY ADDR (MSL=0,TRA=0) - EVI: Own address received, data to be sent to master - ▶ BUSY ADDR TRA (MSL=0) - ▶ EV2: Slave byte received - ▶ BUSY RNXE (receive buffer not empty) - ▶ EV3: Slave byte transmitted - ▶ BUSY TRA TXE (transmit buffer empty) - ▶ BUSY TRA TXE BTF (transmit buffer empty and byte transfer finished) # I<sup>2</sup>C clock control register I<sub>2</sub>C\_CCR | 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 | | |-----|------|----------|----------|----|-----------|----|----|----|----|----|----|----|----|----|----|--| | F/S | DUTY | Boso | Deserved | | CCR[11:0] | | | | | | | | | | | | | rw | rw | Reserved | | rw | F/S 0=standard mode (≤ 100KHz), 1=fast mode (≤ 400KHz) #### Standard Mode: $$\begin{split} T_{high} &= T_{low} = CCR * T_{PCLK1} \\ \text{Fast Mode, DUTY} &= 0 \\ T_{high} &= T_{low} = CCR * T_{PCLK1} \\ \text{Fast Mode, DUTY} &= 1 \quad \text{(to reach 400KHz)} \\ T_{high} &= 9 * CCR * T_{PCLK1} \\ T_{low} &= 16 * CCR * T_{PCLK1} \end{split}$$ Ex: To generate 100KHz SCL in standard mode. If FREQR = 08, $$T_{PCLK1}$$ = 125ns Set CCR = 40 (0x28) $T_{high}$ = $T_{low}$ = 40\*125ns = 5000ns FREQR in CR2 ## Hierarchical/modular software design ## STM32 I2C peripheral driver functions - Configure control registers, etc. - ▶ I2C\_Init() initialize control registers, clock, etc. - ▶ I2C\_Cmd() enable the I2C module - Other functions to set/clear individual control bits - Bus management functions - ▶ I2C\_GenerateStart() signal START on the bus - ▶ I2C\_Send7bitAddress() send slave address - ▶ I2C\_GenerateStop() signal STOP on the bus - Data transfer functions - ▶ I2C\_SendData() send one byte to DR - ▶ I2C\_ReceiveData() get one byte from DR - Bus monitoring functions - I2C\_CheckEvent() test status flags for a bus "event" - ▶ I2C\_GetFlagStatus() test one flag in status register ## Typical master-to-slave transfer Codec\_WriteRegister(RegAddr,RegValue) - ▶ I2C\_GetFlagStatus() check flag BUSY=0 - ▶ I2C\_GenerateStart() signal START on the bus - ▶ I2C\_CheckEvent() test EV5 flags (start correct) - ▶ I2C\_Send7bitAddress() send slave address - ▶ I2C\_CheckEvent() test EV6 flags (slave address ACK) - ▶ I2C\_SendData() send first byte (register address) to DR - ▶ I2C\_CheckEvent() test EV8 flags (data sending, DR ready for byte) - ▶ I2C\_SendData() send second byte (register value) to DR - ▶ I2C\_GetFlagStatus() check flag BTF=I (byte transfer finished) - ▶ I2C\_GenerateStop() signal STOP on the bus ## Audio Code driver: key functions - Codec\_Init() all related device/module initialization: - Codec\_GPIO\_Init() - ▶ Enable clocks in RCC and all GPIO pins for I2C, I2S, DAC - Codec\_Reset() reset the Codec (RESET pin) - Codec\_CtrlInterface\_Init() - Calls I2C\_Init() with required parameters - Configure all Codec registers via I2C functions - Codec\_AudioInterfaceInit() - Initialize DAC and I2S modules - Codec\_WriteRegister() write value to a code register - Codec\_ReadRegister() read value from a code register ## STM32F4-Discovery Software - STM32F407VG peripheral drivers added to project from "Pack" (I2C, SPI, DAC, etc.) - stm32f4xx\_i2c.c => all I2C control/access functions - Discovery board chip drivers in - ..\stm32f4discovery\_fw\Utilities\STM32F4-Discovery\ stm32f4\_discovery\_audio\_codec.c - Initialize and control audio codec chip - ▶ Calls functions from I2C, I2S, GPIO, RCC module drivers # Cirrus Logic CS43L22 Portable Audio DAC with Integrated Class D Speaker Driver ## Discovery CS43L22 schematic