I2C Tutorial
I2C Tutorial
I2C Tutorial
I2C (Inter-Integrated Circuit) is a short distance serial interface that requires only 2 bus lines for data
transfer. It was invented by Philips in 1980′s, originally to provide easy on-board communications
between a CPU and various peripheral chips in a TV set. Today, it is widely used in varieties of
embedded systems to connect low speed peripherals (external EEPROMs, digital sensors, LCD
drivers, etc) to the main controller. In this experiment, we will cover an overview of I2C protocol, its
implementation in PIC microcontrollers, and the method of connecting single and multiple devices on a
common I2C bus. We will demonstrate the technique by connecting two I2C EEPROM chips
(24LC512) and an I2C compatible temperature sensor (DS1631) with PIC18F2550 microcontroller.
THEORY
I2C bus has two lines: a serial data line (SDA) and a serial clock line (SCL). Any data sent from one
device to another goes through the SDA line, whereas the SCL line provides the necessary
synchronization clock for the data transfer. The devices on an I2C bus are either Masters or Slaves.
Only a Master can initiate a data transfer and Slaves respond to the Master. It is possible to have
multiple Masters on a common bus, but only one could be active at a time. The SCL clock line is
always driven by the master. In this tutorial, we will discuss a single master case, and our master is the
PIC18F2550 microcontroller. The figure below shows an I2C bus with a single master and three
slaves. Slaves can never initiate a data transfer but they can transfer data over the I2C bus, and that is
always controlled by the Master.
Both SCL and SDA lines are open drain drivers, and are therefore, connected to a positive supply
voltage through pull-up resistors. This means the I2C devices can only pull the line low, but they
cannot drive it high. When no device is pulling on the line, it will float high through the pull-up resistor.
This is why pull-up resistors are important in I2C. The open-drain outputs of I2C devices helps to
perform the wired-AND function on the bus. Data on the I2C bus can be transferred at a rate up to 100
Kbps (in standard mode), 400 Kbps (in fast mode), or up to 3.4 Mbps (in high-speed mode).
Prior to any transaction on the bus, a Start condition is issued by the Master device to inform all the
slave devices that something is about to be transmitted on the bus. As a result, all connected slave
devices will listen to the serial data line for instructions. The Start condition is issued by pulling the
SDA line low followed by the SCL line. Once the data transfer is finished, the bus Master sends a Stop
condition to inform other devices that it would like to release the bus. The signaling used for a Stop
condition is a release of the SCL line followed by a release of the SDA line. Remember that when the
lines are released, they float high because of the pull-up resistors. Thus, the Start and Stop sequences
mark the beginning and end of a transaction with the slave device.
Each device connected to the bus is software addressable by a unique 7-bit or 10-bit address. The
use of 10-bit address is not very common and therefore, is not discussed here. The first byte sent after
the Start condition is known as Control byte. The first seven bits of the control byte make up the slave
address, whereas the eighth bit (LSB) is a data direction bit (R/W): a ‘zero’ in the LSB of the first byte
indicates that the Master will write information to a selected slave. A ‘one’ in this position indicates that
the Master will read data from the slave. For 7-bit devices, typically the first four bits are fixed, the next
three bits are set by hardware address pins (A0, A1, and A2) that allow the user to modify the
I2C address allowing up to eight of the same devices to operate on the I2C bus. These pins are held
high to VCC or held low to GND. We will discuss more in the circuit section on how to setup the device
address on the bus.
The control byte (first byte after the Start condition) holds the slave address
DATA TRANSFER
Every byte put on the SDA line must be 8-bits long. The data is sent on the SDA line starting with the
most significant bit (MSB) first and the SCL line produces a synchronization clock. The data on the
SDA line is considered valid when SCL is high, and therefore the data must be stable during the HIGH
period of the clock. The HIGH or LOW state of the data line can only change when the clock signal on
the SCL line is LOW. This is how the timing of each bit works.
If a slave is not in a position to receive or transmit another complete byte of data until it has performed
some other function, for example servicing an internal interrupt, it can hold the SCL line low to force
the master into a wait state. Data transfer continues when the slave is ready for another byte of data
and releases the clock line.
A data transfer is always terminated by a Stop condition generated by the master. However, if a
master still wishes to communicate on the bus, it can generate a repeated-Start condition and address
another slave without first generating a Stop condition.
ACKNOWLEDGMENT
Usually, a receiver which has been addressed is obliged to generate an acknowledge (ACK) after
each byte has been received. The acknowledge takes place after the 8th data bit has been transferred
in any transaction. During this state the transmitter should release the SDA bus to allow the receiver to
drive it. The receiver drives the SDA signal low to acknowledge receipt of the byte. If the receiver does
not drive SDA low, the condition is a no-acknowledge (NACK) and the operation is aborted. If the byte
sent is the control byte (slave address + R/W bit), then only that slave which has the matching address
will respond with an acknowledge.
Now lets discuss how these various sequences of operation takes place while the PIC18F2550
microcontroller communicates with 24LC512 EEPROM and DS1631 temperature sensor over an I2C
bus. The MSSP (Master Synchronous Serial Port) module in PIC18F2550 allows I2C communication
through two of its I/O pins: RB0/SDA (21) and RB1/SCL (22). The functioning details of this module is
not discussed here because of the use of mikroC compiler that provides library routines for I2C
communication.
24LC512 is an I2C compatible serial EEPROM from Microchip Technology with a capacity of 64K x 8
(512 Kbits). The pin diagram of this IC chip is shown in the circuit section. The first four bits of the 7-bit
address for this device is set as ’1010′. The next three bits are, however, configurable through its A0,
A1, and A2 pins. For example, a 24LC512 device with A0 at logic high and A1-A2 pins grounded will
have its 7-bit address as ’1010001′ . This scheme allows a maximum of 8 similar devices addressable
on the same I2C bus.
For byte write operation, two bytes of addresses are required to select one out of 65536 locations in
the EEPROM. Master provides these two address bytes after the control byte has been sent. 24LC512
responds with an acknowledge pulse after receiving each address byte. Master then sends a byte of
data to be written in to the memory. Upon receipt of this data, 24LC512 sends an acknowledge pulse.
The Master then terminates the data transfer by issuing a Stop condition.
To minimize write cycle time, 24LC512 offers Page write feature, which allows simultaneous writing of
up to 128 contiguous bytes. A page write is initiated in the same way as a byte write, but instead of
generating a Stop condition, the master transmits up to 127 additional bytes, which are temporarily
stored in the on-chip page buffer and will be written into memory after the master has transmitted a
Stop condition. Please read the datasheet of 24LC512 to learn more about the Page write operation.
READ OPERATION
Read operations are initiated in the same way as write operations with the exception that the R/W bit
of the control byte is set to ‘1’. 24LC512 allows three basic types of read operations: current
address read, random read and sequential read.
Internally, the EEPROM contains an address counter that maintains the address of the last word
accessed, incremented by ‘1’. Therefore, if the previous read access was to address ‘n’, the next
current address read operation would access data from address n + 1. Upon receipt of the control byte
with R/W bit set to ‘1’, the EEPROM issues an acknowledge and transmits the current address data
byte. The master will not acknowledge the transfer but does generate a following Stop condition and
the EEPROM discontinues transmission.
Random read operations allow the master to access any memory location in a random manner. To
perform this type of read operation, first the word address must be set. This is done by sending the
word address to 24LC512 as part of a write operation (R/W bit set to ‘0’). After the word address is
sent, the master generates a repeated-Start condition following the acknowledge. This terminates the
write operation. Then, the master issues the control byte again but with the R/W bit set to a one. The
EEPROM will then issue an acknowledge and transmit the 8-bit data word. The master will
not acknowledge the transfer but does generate a Stop condition which causes the EEPROM to
discontinue transmission.
Sequential reads are initiated by either a current address read or a random address read. After
receiving the first byte from 24LC512, the master issues an acknowledge instead of the Stop condition
used in a current address or random read. This acknowledge directs 24LC512 to transmit the next
sequentially addressed 8-bit word. Following the final byte transmitted to the master, the master will
NOT generate an acknowledge, but will generate a Stop condition.
Pin number 7 of 24LC512 is a hardware Write Protect input. If this pin is tied to Vcc, write operations
are inhibited but read operations are not affected. In our experiment, we will ground this pin.
DS1631 is a digital thermometer manufactured by Dallas Semiconductor (now MAXIM) that provides
9, 10, 11, or 12-bit (user selectable) temperature measurements over a -55 °C to 125 °C. The default
resolution at power-up is 12-bit, corresponding to the temperature increment of 0.0625 °C. The
communication with DS1631 is achieved through I2C interface and three address pins (A0, A1, and
A2) allow up to 8 devices to be multi-dropped on the same 2-wire bus (see the pin diagram in the
circuit section). The 7-bit address of each slave is 1 0 0 1 A2 A1 A0, where A0, A1, and A2 are user
selectable through the corresponding input pins.
TEMPERATURE MEASUREMENT
I recommend to read the datasheet of DS1631 for details on its architecture and temperature
conversion process. Here, I am describing only the one-shot mode of temperature conversion.
Suppose, the device is just powered up, and the resolution of conversion is set to 12-bit. In one-shot
mode, the DS1631 sensor starts converting temperature into 12-bit digital word after receiving a
command byte, 51h, from the Master. This is known as the Start Convert T command. After the
conversion, the digital temperature is stored as a 16-bit two’s complement number in to its two-byte
temperature register: TH and TL (see below). The Sign bit (S) indicates whether the temperature is
positive (S=0) or negative (S=1).
The master can read the temperature data from DS1631 by sending a Read Temperature (AAh)
command. After receiving an ACK in response to the command, the master must generate a repeated
Start followed by a control byte with the same slave address as the first control byte. However, this
time the R/W bit must be a 1, which tells DS1631 that a “read” is being performed. DS1631 sends an
ACK in response to this control byte, and it begins transmitting the requested data on the next clock
cycle. For two byte reads (TH and TL registers), the master must respond to the first data byte with an
ACK and to the second byte with a NACK followed by a STOP. If only the most significant byte of data
is needed, the master can issue a NACK followed by a STOP after reading the first data byte.
CIRCUIT SETUP
The circuit diagram for I2C experiment is shown below. Two serial EEPROMs (24LC512) and a
DS1631 temperature sensor are connected on a common I2C bus. The address conflict between the
two EEPROMs is avoided by connecting one’s A0 pin to Gnd and others to Vcc. The 7-bit addresses
of all three I2C devices are shown in the diagram. The two 10 K resistors are the pull-up resistors for
the I2C bus lines. The SDA and SCL lines go to RB0 and RB1 pins of PIC18F2550, respectively. I am
using the Start USB for PIC board again for PIC18F2550.
A standard 16×2 character LCD is also used in the circuit to display the I2C Read/Write operations.
The microcontroller pins used to drive the LCD are shown below.
SOFTWARE
We will be writing an application software for testing the I2C communication between the PIC18F2550
microcontroller and the three devices that are multi-dropped on a common I2C bus. The program will
first check the presence of all three devices on the bus, and then sends a temperature conversion
command to DS1631. The two-byte temperature will be then read and stored in to the two EEPROMs
(higher byte in one EEPROM and lower byte in the other, starting at location 0). After five temperature
samples are recorded, they will be read sequentially from EEPROM locations, converted in to actual
temperature units (°C) and displayed on LCD screen.
CONCLUSION
In this tutorial, we briefly looked at what I2C is and how it is used to communicate data to and from a
PIC Microcontroller and I2C compatible slave devices. We explored the Read, Write, and addressing
schemes for multiple devices on a common I2C bus, and its implementation using mikroC Pro for PIC
compiler. I2C has become so popular in embedded world that there are hundreds of I2C compatible
devices available in market in the form of ADC, DAC, SRAM, I/O expander, humidity sensor, etc. Now
we understand how this protocol works, it’s time to play with these other devices.