J.D. Nicoud, Mouette 5 CH-1092 Belmont, Switzerland Tel +41 21 728-6156, Fax 728-6157 Email info@didel.com, www.didel.com This file: www.didel.com/doc/DopicI2C.pdf # I<sup>2</sup>C / SM-Bus PIC routines for PIC 16F84, 16F870, ... #### Table of content Introduction 7. Appendix (Web names are case sensitive) 2. ICF 2.1 Delays 2 Start I<sup>2</sup>C principle and macros 22233334455556688 7.1 I2C macros, routines and test programs Start and Stop List: www.didel.com/doc/Xi2c.html 2.3 Bit selection Variables www.didel.com/doc/Xi2cV.asi 2.4 Caution Macros www.didel.com/doc/Xi2cM.asi 8-bit write Routines www.didel.com/doc/Xi2cR.asi 2.6 8-bit read Test simple 2.7 Acknowledge2.8 Basic I<sup>2</sup>C protocle www.didel.com/doc/Xi2ct0.asm **PCF** 8574 Test 2.9 Imortant note on variables www.didel.com/doc/Xi2c8574.asm 2.10 Testing the write routine 3.1 Transactions 3.2 Addressing 7.2 Short delays macros I<sup>2</sup>C main features List: www.didel.com/doc/Xdel.html Macro and www.didel.com/doc/XdelR.asi Test program www.didel.com/doc/Xdel.asm I/O expander PCF 8574 EEPROM 24C01 7.3 Long delays routines List: www.didel.com/doc/Xdelai.html Macro and www.didel.com/doc/XdelaiR.asi program Test #### 1. Introduction $I^2C$ has been proposed by Philips as an efficient way of having a processor controlling a set of I/O devices over 2 signal lines. It is used inside PCs mostly for reading temperature sensors and power controller, and has been renamed by Intel as the SM-Bus (www.sbs-forum.org/smbus/specs/). www.didel.com/doc/Xdelai.asm Multi-master transfers are possible, but our objective here is only to learn how to control existing $I^2C$ circuits, and program a PIC as a slave. We will not explain all the features of $I^2C$ Those having undersood this document will be ready to read Philips or SM-Bus documentation. This note shows how to use $I^2C$ transfers on Microchip PIC processors which do not have dedicated hardware for this. From the examples I have seen, it is indeed easier to use our routines than to understand and program the $I^2C$ control registers of a 16F76 or 16F877. So, unless you have to be multitask and are ready to spend a lot of time learning to handle correctly a set of interrupts, consider using our simple routines. We first explain the I2C timings and how to implement them with macros and routines. We show how to use them with two $I^2C$ circuits: the PCF8584 parallel port interface and the 24C01 EEPROM. A simple PIC as a slave (a low cost 12C508 for instance) adds some timing constraints and requires to slow down the transfer down to 10 kBit/s. We use CALM assembly language notations. CALM (Common Assembly Language for Microprocessors) has been developed at the EPFL since 1976 and supports 15 processors with a simple explicit orthogonal syntax. CALM is close to Motorola notations, with more explicit addressing modes. For programmers with Intel or AVR experience, several differences (order of operands, mnemonics) make the transitions more difficult. For beginners, CALM has proven its efficiency, due to its explicit syntax: when Microchip writes btfsc Reg,bit one should remember that this mean "Test in register eg" the bit "bit" and Skip if this bit is set. CALM writes TestSkip,BS Port:#bit which means exacly this. CALM uses the # sign for immediate addressing instead if a "i" letter in the mnemonic. The number of instruction mnemonics is greatly reduced, the addressing mode being very explicit. Going from one processor to another is easy, since only the specificity of the architecure and a few special instructions have to be learned. For those not familiar with CALM, but having some experience with the PIC, the reference card available at www.didel.com/doc/Pic84Calm.pdef lists the instructions in both notations. For those without experience with PIC, but with a good understanding on microcontrolers and assembly language, a detailed document in english is available at www.didel.com/doc/Pic84E.html. For absolute beginners, we have the **PIC**Génial documentation in french (www.didel.com/PicGenial.html) ## 2. I<sup>2</sup>C principle and macros When sending an 8-bit word in serial, one needs to know when the data starts and when it stops. The great idea of $I^2C$ is to use 2 lines, one for the clock (so there is no precise bit rate to select) and one for the data. Valid data must be stable when the clock is at level one. Start and stop bits result from a violation of that rule, easy duci2c1 to generate and decode. In order to simplify, we take all the long timings to 5 $\mu$ s. Fig. 1 $l^2C$ principle and timings ## 2.1. Delays At 4 MHz, the SDA to SCL delay of 250 ns will result from two consecutive instructions for processors up to 16MHz. Short delays are generated by Nop instructions, which takes 1 $\mu$ s at 4 MHz. The instruction Jump APC+1 takes 2 $\mu$ s since it jumps to the next instruction (APC is the assembler program counter value), and takes more time due to the instruction pipeline being broken. The following macros generate the delays we will need. | . Macro | Nop2 | | .Macro Nop3 | | . Macro | Nop3 | | |----------|------|-------|-------------|-------|---------|------|-------| | | Jump | APC+1 | Nop | | | Nop | | | . Endmad | cro | | Jump | APC+1 | | Jump | APC+1 | | | | | Endmacro | | Endma | icro | | If you have a faster processor, the xDel macro and routine in appendix (www.didel.com/XDelM.asm and www.didel.com/XDelR.asm) is an elegant way to generate delays between 1 and 20 instruction cycles (or more if you add instructions in the XDly routine). The macro inserts a maximum of two instructions and the associated XDly routine is quite short and does not use any register or modify W. ### 2.2. Start and stop Now we can control the delays, it is easy to generate the sequence (these are not the definitive macros, see later). ``` .Macro FullSCK .Macro Start Stop . Macro SetSCK ; SCK .../''\.. ClrSDA ; SDA ´´`\..... CIrSDA Nop4 Nop4 Nop ; SCK ''''\. CIrSCK SetSCK ; SDA ...../' CIrSCK . Endmacro Nop2 SetSDA ; SCK ../''' . Endmacro . Endmacro ``` The function of CIrSDA and the other macros referred inside these macros is evident. How to implement them? The naive way would be to initialize two ports bits as output, and set or clear these bits. A special handling would be necessary for read transfer and acknowledge. I<sup>2</sup>C bus is specified as open-collector and we have to do it for the SDA line at least in order to avoid from time to time a short circuit between the master and slave SCL outputs. Open collector means that the zero are active, but not the ones; pull-ups will assert the one. This is easily implemented on portA for instance by initializing the port to zero, and playing on the direction to get an output at level zero and ones: When a port line is initialized as output, a zero is active. When the port line is initialized as input, the pull-up resistor sets a one (if no other output on the bus forces a zero). #### 2.3. Bit selection Let us define bSCL and bSDA the bit number of the port, and PortI2C the port on which the two lines are wired (the same port is highly preferable). For instance, if SCL and SDA are connected on PortC bits 3 and 4 of the 16F870 (to be compatible with an hardware implementation on the 16F876), one should declare: ``` bSCK = 3 bSDA = 4 DirC = 2'10011001 ; bits 3 and 4 are important Port I 2C = 7 ; Port C Tris I 2C = 7 ; Bank 1 Tris C ``` #### 2.4. Caution Before any call of a $I^2C$ macro or routine, one should be sure that the PortI2C bits bSCL and bSDA are at zero. If there is no interaction with another I/O using the same port, the bits are initialized in the beginning and will stay, but pay attention with the Set bit and Clr bit of other bits on the same port! We can write now the basic macros for changing the lines. These macros have to be executed in register bank 1. This will be under the control of the Start and Stop conditions, hence the above routines will be modified. It is important that wireing constraints do not appear in any macro or routine. The macro and routine files must be applicable to any PIC processor and any wireing option, with the only constraint that SCL and SDA have to be on the same port. ``` .Macro SetSDA ; SDA= 1 and Inp .Macro SetSCK SCK = 1 FullSCK . Macro Set Tris I 2c:#bSDA Set Tris I 2c:#bSCK Nop . EndMacro . EndMacro Tris I 2c:#bSCK Set ; SDA = 0 if outp .Macro CIrSDA .Macro CIrSCK SCK = \emptyset Nop4 Tris I 2c: #bSDA Tris I 2c:#bSCK Tris I 2c: #bSCK Clr Clr Clr . EndMacro . EndMacro Nop . EndMacro ``` We repeat that one should be careful with open-collector programmed lines. A Set or Clr on another bit of the same port will read the byte on PortI2C, modify the assigned bit, and write the byte. This mean that if SCL for instance is one on the line, it will be copied in the port register, and the line will go to one when selected as output. The clock line will stay at one and no more clock will appear! #### 2.5. 8-bit write The module to write 8 bits prepared in the ADi2c register is quite similar to all serial transfer modules (fig 2). Register AdI2C will contain addresses or data, hence its name. #### 2.6. 8-bit read The module to read a byte is quite similar: the data line is tested when the clock is active and copied into the AdI2C register via the Carry. In order to test the SDA bit, one have to switch back to bank 0 for the time of the test. Macro Bank1toO correspond to the instruction Clr Status:#RPO and the other is a Set. ``` ; Read a byte TestSkip, BC Port I 2C: #bSDA; SCK à 0 out: AdI2C = W Move #8,W RLC ADi2c Move W,CBit Bank@to1 SetSDA FullSCK DecSkip, EQ CBit CIrC L$ Jump Bank1to0 Ad12c.W Move ``` ### 2.7. Acknowledge An interesting feature of $I^2C$ is the unit receiving the data has to send a positive or negative acknowledge. This occurs immediately after the 8 data clock. The sender leaves the data line at one and the slave generates a zero (Ack) or leaves the line at one (NAck). The sender generates a clock, during which it reads the data line and sees if a unit is replying (Ack) or not (NAck), which may mean that the receiver is not here, or can not/will not accept any more data for this transaction. ## 2.8. Basic I<sup>2</sup>C protocol An I<sup>2</sup>C write transfer consists of a start condition, 8 data bits, and acknowledge and a stop bit, as shown in Fig 3. The acknowledge is activated at the negative edge of the last data clock, before the sender has deactivated its possible zero. Hence the importance of an open collector, to avoid a "short circuit" visible on the scope as a medium level, without consequence on the correct transfer operation, as long the collision has desappeared when the next clock is active. The sender takes then again the control to generate the stop condition. doci2c3 Fig. 3 Basic write transfer With our macros, all the transfer must be executed on register bank 1, and the best is to include the bank switching in the start and stop macros. The main program will stay in bank 0. The corrected Sart and Stop macros are ``` .Macro Bank@to1 .Macro Start . Macro Stop Set Status:#RP0 Bank@to1 CIrSDA ; SDA '''\.... CIrSDA . Endmacro Nop ; SDA .Macro Bank1to0 SetSCK Non4 . . . . . . / ; SCK ..... Status:#RP0 Clr CIrSCK Nop2 SetSDA ; SCK . Endmacro Nop2 . Endmacro Bank1to0 . Fndmacro ``` The routine to write a byte on a I2C slave uses ADi2c as input and the carry bit as output. ``` Routine Wrl2C | Write a byte and test acknowledge ; An Ack is expected ADi2c address or data Carry = 0 if Ack, Carry = 1 if NAck SetSDA out: Wrl2C: ClrC Bank1to0 ; Acknowledge test Start TestSkip, BC Port I 2C: #bSDA #8,W Move : Transmit ADi2c SetC Move W.CBit Bank@to1 L$: RLC ADi2c FullSCK Skip,CC Nop3 SetSDA ; Carry = 1 CIrSDA Skip,CS Ret CIrSDA ; Carry = 0 FullSCk DecSkip, EQ CBit L$ Jump ``` I<sup>2</sup>C PIC routines - 5 - DIDEL April 2001 ### 2.9. Important note on variables The two variables we use, ADi2c and CBit are accessed inside the routine at a time bank 1 is active. It is of no importance on the 16F84, since the access to variables does not depend on the bank. But on the 74C76 or 76C87x, there are different variables on bank 0, 1, 2, 3. However, variables at addresses 16´70 to 16´FF are identical on all banks, and this is the place we have to define our I2C variables. ## 2.10. Testing the write routine A scope or any I<sup>2</sup>C slave circuit can be used to test our routine. Let us suppose that this slave is 16´70. Let us also have a LED on e.g bit 0 of PortB to show if there is an acknowledge or not. The program that writes continuously on the slave chip is given below, and available on www.didel.com/doc/Xi2cT0.asm. This program insert the I2C macros and the write routine Xi2cWR.asi, explained in previous section. In the future, all our I<sup>2</sup>C routines will be put inside a Xi2cR.asi file, and what will be important to check every time a routine is called are the parameters in, out and modified, and of course its functionnality. ``` Program Xi2cT0 | Simple write test Programme ; JDN 280301 .Loc Ø ; program start Deb: Move #DirA,W .Proc 16F84 W, TrisA ; bit on Status register that control Ba Move RB0 = 5 Move #DirB,W PortA Move W, TrisB bSCK = 0 : 12C bits Clr ; important for SCK SDA PortA bSDA = 1 #SlaveAddress.W DirA = 2'00011 ; Bits 0 and 1 as input Loop: Move = 5 ; I2C bits on Call Write I 2 C Port I 2C Skip, CS Tris I 2C = Porti2c ; Ack LedOn ; 8574 I/O expander SlaveAdress = 16'70 Skip,CC . Ins Xi2cM.asi ; Inserts I2C macros LedOff ; NAck PortB ; space on the scope Jump Wait ; Led bright if output=0 bLed = 0 ; to ease synchronization DirB =2'00000000 ; Bit Ø as output Jump Loop . Macro LedOn . Ins Xi2c0.asi PortB:#bLamp Clr . Endmacro Routine Wait 1ms wait . Macro LedOff Wait: Clr Set PortB:#bLamp . Endmacro W$: Add #-1,W ; 4 us loop Variables Skip, EQ ; repeated 256 times .Loc 16'0C ; begin of variables Jump Ret CBit: .Blk.16 ADi2c: ``` If the $I^2C$ slave (here the 8574 described later, but any I2C chip can be used it the SlaveAddress is correctly declared) is correctly connected and powered, the LED will bright. By disconnecting SCL or SDA, or will not. Have also a look with a scope. For adjusting the timings and for any real time programming a scope, even an old one, is essential. #### 3. I2C main features As mentionned above, we will not detail all the $I^2C$ features. Most applications just control several $I^2C$ slaves and we want to do it in a simple and efficient way. #### 3.1. Transactions The write transfer we have seen is drawn in a simplified way on Fig 4 a). It selects a slave, and has no other application than an existence test. The simplest $I^2C$ slaves (Fig 4 b) accept an 8-bit data word and can provide an 8-bit information back. This is the case of the PCF8574 parallel port we will see later. Reading needs to repeat the address, with the least significant bit (the $\overline{R}/W$ bit) set to one. All $I^2C$ addresses are even. The read transfer sequence is double: a write transfer to write the selected register, and then a write cycle with the read address, followed by a read cycle. Start/Stop can be used for both transfers, but a slightly shorter Restart condition can be used inbetween. More complex $I^2C$ slaves have a set of 8-bit registers. The register is selected before sending the data (Fig 4 c). Read transfers are terminated by a NAck, to mention the master the transfer is complet. The reason for this is multiple data transfers can be done in the same transaction (Fig 4 d). An EEPROM or a RAM for instance has an address register which points to a data byte that can be read or written. A data transfer increments automatically the address counter, and consecutive locations can be transferred (bloc transfer). There may be limitations on the length of the bloc, and delays after writing, in the case of an EEPROM. Other $I^2C$ circuits, e.g. a real clock, are indeed similar; they have many registers, which can be addressed independantly, or in sequence as a block transfer to consecutive addresses. doci2c4 Fig. 4 I<sup>2</sup>C transfer sequences ### 3.2. Addressing $I^2C$ addresses are 7 bits on bits 7..0. Bit 0 is a Read/Write indication: read addresses are odd. Many $I^2C$ circuits take the 2 or 3 low address bit on circuit pins, in order to hardwire these address bits and support several identical chips in parallel (or try to avoid an address used by another slave). #### 4. I2C routines In order to be compatible with different addressed devices and different transaction formats, we have defined a set of routines which make the code as flexible and short as possible. One needs to define 6 registers at bank-independent addresses: CBit: .Blk.16 1 Ad | 2C: .Blk.16 1 AdSlave: .Blk.16 1 Reg | 2C: .Blk.16 1 Data | 2C: .Blk.16 1 AdSlave must be loaded with the slave address. It has to be reloaded by the 2 instructions below only when it is necessary to select a new slave. Move #AdNextSlave, W Move W, AdSlave Regl2c is the first byte written, usually the register address, Datal2c is the data. The write low level routine uses ADi2c register. The source code of our routines (www.didel.com/doc/i2c.html) is not detailed here; we recommend their reading, but they can be used without their understanding if the parameters they need are correctly prepared. ``` Routine StartWrite Send the slave addresse, the register internal address and the data ; A block of data may follow. A Stop must terminate the transfer. in: AdSlave slave address in: Regi2C register address in: Regi2C adts written mod: W, ADi2c ``` The StartR, StopR routines have the same effect as the Start and Stop macros, but they insert only one byte of code. Transaction of Fig 4 b) are not supported, since our routines have been optimized for complex $I^2C$ circuits. For this case, new routines have to be written, as shown in next section. Transaction of Fig 4 c) is programmed as below. ### For writing: AdSlave OK, load RegI2C and DataI2C Call StartWrite Call StopR #### For reading: Load AdSlave, load RegI2C Call StartRead Call GiveNAckStop The data read is both in ADi2c and W Transaction of Fig 4 d) is programmed as below. #### For writing: AdSlave OK, load Reg12C and first data in Data12C Call StartWrite Load second data in Data12C Call Write12C Load third data in Data12C Call Write12C Call StopR #### For reading: AdSlave OK, load Reg I 2 CC Call StartRead Handle first data in WCC Read I 2 CC Handle second data in WCC Read I 2 CC Handle third data in WCC All GiveNAckStop The data read is both in ADi2c and WCC III Start Read I SiveNAckStop The data read is both in ADi2c and WCC III Start Read I SiveNAckStop The data read is both in ADi2c and WCC III Start Read I SiveNAckStop The data read I SiveNAckStop The data read I SiveNAckStop The data read I SiveNAckStop The data read I SiveNAckStop The Material ## 5. Example 1: 8-bit I/O expander PCF 8574 The I<sup>2</sup>C I/O expander PCF 8574 is easy to implement and convenient to get more inputs and outputs from a microcontroller system. The Max 1608/Max1609 circuis have a quite similar functionnality. There is no direction to assign, since the outputs are open-collector: an input is an output at state one. with below, program available its complete www.didel.com/doc/Xi2c8574.asm, reads 4 switches on the low 4 bits (pull-up are required) and copies the content on the 4 high bits. Simpler write and read routines could be written if there is only one 8574 circuit to be controlled; the 8574 is the only circuit not requiring internal sub-addresses. | | Move | #Ad8574,W | ; Declared as 16´40 to | WrSingle I 2C: | | |-------|------|----------------|-----------------------------|----------------|-------------| | | Move | W,AdSlave | | Move | AdSlave, W | | | Move | #2'00001111 | ,W | Move | W,ADi2c | | | Move | W,Datai2c | ; Data direction initialize | Start | | | | Call | WrSingle I 2C | | Call | Write I 2 C | | Loop: | | | | Stop | | | | Call | RdSingle I 2C | ; Result in W and Datal | Ret | | | | Swap | Data I 2C, W | | RdSingle I 2C: | | | | Or | #2'00001111 | | Move | AdSlave, W | | | Call | WrSingle I 2 C | | Move | W,ADi2c | | | Jump | Loop | | Start | | | | | | | Call | Write I 2 C | | | | | | Call | Read I 2 C | | | | | | Call | GiveNAckR | | | | | | Ret | | ## 6. Example 2: EEPROM 24LC01 128x8 The 24LC01 from Microchip is an 8-pin circuit (also available in a miniature 5-pin package) that stores up to 128 bytes of data. There a single control register, the address pointer that defines where the next data will be read or written. This address pointer is incremented after every read, and the reading is immediate. For writing one location at a time, a programming time of 10ms must be respected (the circuit will not acknowledge a transfer during this time). Block write are efficient, but one needs to understand their limitation: The EEPROM block is structured as 16 8-byte blocks and has an 8-byte input buffer. Within the same block, one can do a block transfer of 8 byte maximum, if one stays inside the same block (the autoincrement concerns only the 8 low bit of the address pointer). If the complete memory has to be written, it is hence necessary to reload the address pointer every 8 bytes. The interest of that feature is to program up to 8 bytes with a 10 ms programming time. The routine we have defined are ideally suited for this case. A single position write and a read are programmed as below. ``` Write: Read: #Ad24LC01,W ; If not #Ad24LC01,W Move Move Move W, AdSlave Move W,AdSlave done before #eeAddr1,W Move ; Selected eeprom addres Move #eeAddr1,W ; Selected eeprom addres W.RegI2C Move Move W.Real2C Move #Data1,W ; Usually a variable Call Start Read W, Data I 2C GiveNAckStop Move Call Call StartWrite result in W and ADi2c Call StopR ``` Reading or writing a block can be done by calling the routine ReadI2C or WriteI2C before the StopR. For instance, the module to read NN bytes from Addr1 is given below. One notices that intermediate read gets an Ack, but the last read gets a NAck. ``` BlockRead: #Ad24LC01,W ; If not done W, AdSlave; done before Move #eeAddr1,W Move ; Selected eeprom address Move W, Reg I 2C Call Move #NN-1,W ; NN is the block lengh W, CBlock; CBlock variable to be defined Move L$: data available in W and ADi2c GiveAck Call Read L2C Call ``` Jump L\$ last data available in W and ADi2c Call GiveNAckStop ## 7. Appendix ## 7.1. I<sup>2</sup>C macros and routines Following files, referred in www.didel.com/doc/Xi2c.html will help you to execute the test programs and use our .asi files for you own applications. - Xi2cV.asi includes definitions and variables. Definitions will be copied in your program, or inside an imported file that defines your hardware. - Xi2cM.asi are the macros. - Xi2cR.asi are the routines ### 7.2. Short delays macro and routine File XDelR.asi (www.didel.com/doc/XdelR.asi) has to be inserted after the beginning of the program if macros are called by the program, which we do not recommend in applications, but it is of course the case in the test program Xdel.asm. For instance, Del 3 produce the code: Jump APC+1 Nop Nop Call Del 7 produces the code Nop Call Del 7 ### 7.3. Long delays routines File XDelaiR.asi (www.didel.com/doc/XdelaiR.asi) includes two routines for short delays (unit 100 $\mu$ s) and long delais (unit 20ms, max 5 sec). Test program Xdelai.asm blinks a Led.