| 
	
	|  |  |  
	
		| View previous topic :: View next topic |  
		| Author | Message |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				| I2C Master & Slave with 2 PIC's SOLVED |  
				|  Posted: Sun Jun 23, 2019 7:26 pm |   |  
				| 
 |  
				| Hi 
 I finally have connected two boards to make a master and slave communication with PIC16F1847
 1. The master connected to PC terminal
 2. Master & Slave boards connected with 6" cables for I2C
 3. 2K2 resistors on SDA and SCL
 4. Both boards powered from same 5VDC source
 
 I am sending from the PC to the master:
 
  	  | Quote: |  	  | 55 AA CC CB or 55 AA DD DC or
 55 AA EE ED
 | 
 and suppose to get the same back twice, ones report from the master and ones report from the slave I2C to master and to PC.
 
 The communication of master with PC is OK but the data supposed to be from the slave is the default value in the main, for example:
 
  	  | Quote: |  	  | 55 AA CC CB 55 AA 00 FF
 | 
 My conclusion is the I2C don't work.
 
  	  | Code: |  	  | ///////////////////////////////////////////////////////////////// //Master I2C test program
 //TM1847.c
 /////////////////////////////////////////////////////////////////
 #include <16F1847.h>
 
 #FUSES HS,NOIESO,PLL,WDT,MCLR,PUT
 #FUSES NOPROTECT,NOCPD,WRT,STVREN
 #FUSES BROWNOUT,BORV25,NODEBUG,NOLVP
 #use delay(clock=32MHz,crystal=8MHz,restart_wdt)
 #use rs232(baud=9600,parity=N,xmit=PIN_B5,rcv=PIN_B2,bits=8,restart_wdt,errors)//alternative
 #use i2c(Master,Slow,I2C1,restart_wdt)//I2C1 module, automatically hardware
 //////////////////////////////////////////////////////////////
 #define flashled PIN_A3
 int flashcnt=0;
 int scomtxwords=0;
 int scomtxw2=0;
 int scomtxwchs=0;
 short rxnewmsgF=0;
 int rxwords=0;
 int rxdata0=0;
 int rxdata1=0;
 int rxdata2=0;
 int rxdata3=0;
 int rxwchs=0;
 short I2CstartF=0;
 int slave01addr=0x80;
 int slave01dataout=0x00;
 int slave01datain=0x00;
 /////////////////////////////////////////////////////////////
 #INT_TIMER1
 void  TIMER1_isr(void)
 {
 flashcnt++;
 set_timer1(25536);//timer1 overflow 40ms
 }
 //FUNCTIONS////////////////////////////////////////////////////////
 void FUNCTIONS(void)
 {
 //LED FLASH////////////////////////////////////////////////////////
 if(flashcnt>=25)
 {
 flashcnt=0;
 output_toggle(flashled);
 }
 //RX NEW MESSAGE///////////////////////////////////////////////////
 if(rxnewmsgF==1)
 {
 scomtxw2=rxdata2;
 slave01dataout=rxdata2;//updates data received to send to slave
 enable_interrupts(INT_TBE);//sends back to PC the rx data received
 delay_ms(2000);//delay before sending data to slave
 I2CstartF=1;
 }
 //I2C & PC communication///////////////////////////////////////////
 if(I2CstartF==1)
 {
 i2c_start();//Start condition
 i2c_write(slave01addr);//R/W bit low for a write
 i2c_write(slave01dataout);//Data to the slave
 i2c_start();//Restart the bus
 i2c_write((slave01addr)+1);//R/W bit high for a read
 slave01datain = i2c_read();//Read the data from the slave
 i2c_stop();
 scomtxw2=slave01datain;//sends slave data to PC
 enable_interrupts(INT_TBE);
 }
 }
 //rs232 TX/////////////////////////////////////////////////////////
 #int_TBE
 void TBE_isr(void)
 {
 switch(scomtxwords)
 {
 case 0:
 {
 putc(85);
 scomtxwchs=85;
 scomtxwords++;
 }
 break;
 case 1:
 {
 putc(170);
 scomtxwchs=255;
 scomtxwords++;
 }
 break;
 case 2:
 {
 putc(scomtxw2);
 scomtxwchs=scomtxwchs+scomtxw2;
 scomtxwords++;
 }
 break;
 case 3:
 {
 putc(scomtxwchs);
 scomtxwords=0;
 disable_interrupts(INT_TBE);
 }
 break;
 }
 }
 //rs232 RX/////////////////////////////////////////////////////////
 #INT_RDA
 void  RDA_isr(void)
 {
 switch(rxwords)
 {
 case 0:
 {
 rxdata0=getc();
 if (rxdata0==85)
 {
 rxwchs=85;
 rxwords=1;
 }
 else
 {
 rxwords=0;
 }
 }
 break;
 case 1:
 {
 rxdata1=getc();
 if (rxdata1==170)
 {
 rxwchs=255;
 rxwords++;
 }
 else
 {
 rxwords=0;
 }
 }
 break;
 case 2:
 {
 rxdata2=getc();
 rxwchs=rxwchs+rxdata2;
 rxwords++;
 }
 break;
 case 3:
 {
 rxdata3=getc();
 if (rxwchs==rxdata3)
 {
 rxnewmsgF=1;
 }
 rxwords=0;
 }
 break;
 }
 }
 
 #ZERO_RAM
 
 void main()
 {
 setup_adc_ports(NO_ANALOGS);
 setup_wdt(WDT_256MS);//~256 ms reset
 setup_timer_1(T1_INTERNAL|T1_DIV_BY_8);//65 ms overflow
 enable_interrupts(INT_TIMER1);
 enable_interrupts(INT_TBE);
 enable_interrupts(INT_RDA);
 enable_interrupts(GLOBAL);
 
 while(TRUE)
 {
 FUNCTIONS();
 restart_wdt();
 }
 
 }
 
 | 
 
  	  | Code: |  	  | //////////////////////////////////////////////////////////////// //Slave I2C test program
 //TS1847.c
 /////////////////////////////////////////////////////////////////
 #include <16F1847.h>
 
 #FUSES HS,NOIESO,PLL,WDT,MCLR,PUT
 #FUSES NOPROTECT,NOCPD,WRT,STVREN
 #FUSES BROWNOUT,BORV25,NODEBUG,NOLVP
 #use delay(clock=32MHz,crystal=8MHz,restart_wdt)
 #use i2c(Slave,I2C1,restart_wdt)//I2C1 module, automatically hardware
 //////////////////////////////////////////////////////////////
 #define flashled PIN_A3
 int flashcnt=0;
 int slave01addr=0x80;
 int slave01dataout=0xAA;
 int slave01datain=0x00;
 /////////////////////////////////////////////////////////////
 #INT_TIMER1
 void  TIMER1_isr(void)
 {
 flashcnt++;
 set_timer1(25536);//timer1 overflow 40ms
 }
 //////////////////////////////////////////////////////////////////
 #INT_SSP
 void  SSP_isr(void)
 {
 slave01addr = i2c_isr_state();
 if(slave01addr==0x80)//Master is sending data
 {
 slave01datain = i2c_read();
 slave01dataout=slave01datain;
 }
 if(slave01addr==0x81)//Master is requesting data from slave
 {
 i2c_write(slave01dataout);
 }
 }
 ///////////////////////////////////////////////////////////////////
 void FUNCTIONS(void)
 {
 
 if(flashcnt>=25)
 {
 flashcnt=0;
 output_toggle(flashled);
 }
 }
 ///////////////////////////////////////////////////////////////////
 #ZERO_RAM
 
 void main()
 {
 setup_adc(NO_ANALOGS);
 setup_wdt(WDT_256MS);
 setup_timer_1(T1_INTERNAL|T1_DIV_BY_8);//65 ms overflow
 enable_interrupts(INT_TIMER1);
 enable_interrupts(INT_SSP);
 enable_interrupts(GLOBAL);
 
 while(TRUE)
 {
 FUNCTIONS();
 restart_wdt();
 }
 
 }
 | 
 
 First time I am trying to make an I2C using CCS, what I am doing wrong?
 
   (made in the far past I2C in assembler and worked)
 
 Best wishes
 Joe
 
 Last edited by gjs_rsdi on Mon Feb 03, 2020 5:56 pm; edited 3 times in total
 |  |  
		|  |  
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Sun Jun 23, 2019 7:54 pm |   |  
				| 
 |  
				|  	  | Quote: |  	  | if(I2CstartF==1) {
 i2c_start();//Start condition
 i2c_write(slave01addr);//R/W bit low for a write
 i2c_write(slave01dataout);//Data to the slave
 i2c_start();//Restart the bus
 i2c_write((slave01addr)+1);//R/W bit high for a read
 slave01datain = i2c_read();//Read the data from the slave
 i2c_stop();
 scomtxw2=slave01datain;//sends slave data to PC
 enable_interrupts(INT_TBE);
 }
 | 
 You are missing the NACK on the last i2c_read().
 It should be:
 
  	  | Code: |  	  | slave01datain = i2c_read(0); | 
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Sun Jun 23, 2019 11:37 pm |   |  
				| 
 |  
				| Thank you PCM Programmer 
 I change:
 
  	  | Code: |  	  | slave01datain = i2c_read(0); | 
 Sill the same.
 I checked again the boards and connections to be sure that they are OK, no problems there.
 I must have some additional mistake
   
 Best wishes
 Joe
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Sun Jun 23, 2019 11:44 pm |   |  
				| 
 |  
				| Sorry, forget to mention: 
 CCS PCM C Compiler, Version 5.062, 31220
 
 Best wishes
 Joe
 |  |  
		|  |  
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 |  |  
		|  |  
		| Ttelmah 
 
 
 Joined: 11 Mar 2010
 Posts: 19962
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Jun 24, 2019 12:29 am |   |  
				| 
 |  
				| Your slave handling is wrong. 
 You are misunderstanding what has to happen, and when.
 
 State = 0x00 - this is the address byte being received for a _write_
 0x01 - this is the data byte then written (state increments for each
 following byte.
 State = 0x80 - this is the address byte being received for a _read_.
 In this state the address byte must be read, and then immediately the
 reply written.
 On many chips the 'standard' is to follow the address byte, with a second
 'register address' byte. This is what is done in the CCS examples, so
 on these the first actual data byte is state==2. However for what you
 show state==1 can be used.
 
 Now you are not handling the write state numbers. State 0x0, and 0x1
 will not result in your slave doing an I2C read. Result there will be a
 peripheral overflow....
 
 Now also understand that the first byte that arrives is always the address.
 Not the data.
 
 For your system, you need:
 
  	  | Code: |  	  | //////////////////////////////////////////////////////////////////
 #INT_SSP
 void  SSP_isr(void)
 {
 int8 incoming;
 slave01addr = i2c_isr_state();
 if (slave01address==0x0)
 incoming=i2c_read(); //dummy read of the device address
 if (slave01address==0x1) {
 //This is where the data _arrives_
 slave01datain = i2c_read();
 slave01dataout=slave01datain;
 }
 if(slave01addr==0x80)//Master is requesting data from slave
 {
 dummy=i2c_read(2); //here we need to read the address byte
 //without releasing the clock line, ready to load the data
 i2c_write(slave01dataout);
 //Now load the data
 
 //Now on some chips, the clock does not correctly release.
 #BIT CKP=getenv("BIT:CKP")
 CKP=TRUE; //This ensures it does release.
 }
 }
 
 | 
 
 I've also added the patch for CKP. This is not needed on most chips
 but it's safer to include it.
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Mon Jun 24, 2019 3:19 am |   |  
				| 
 |  
				| Thank you for the answers   
 I read the link PCM Programmer wrote and the post of Ttelmah.
 I will learn the subject, implement accordingly and test.
 Will be back with the results.
 
 Best wishes
 Joe
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Wed Jun 26, 2019 3:39 am |   |  
				| 
 |  
				| Hi 
 I am back after tried a few things but with no success to make a communication between I2C master and slave
   
 Whatever I am doing I am getting in the terminal:
 
  	  | Quote: |  	  | 55,AA,CC,CB 55,AA,CC,CB
 55,AA,00,FF
 | 
 The 55,AA,00,FF supposed to be the slave answer.
 
 My last slave is the ex_slave.c
 
  	  | Code: |  	  | #INT_SSP void  SSP_isr(void)
 {
 unsigned int8 incoming, state;
 state = i2c_isr_state();
 
 if(state <= 0x80)                      //Master is sending data
 {
 if(state == 0x80)
 incoming = i2c_read(2);          //Passing 2 as parameter, causes the function to read the SSPBUF without releasing the clock
 else
 incoming = i2c_read();
 
 if(state == 1)                      //First received byte is address
 address = incoming;
 else if(state >= 2 && state != 0x80)   //Received byte is data
 buffer[address++] = incoming;
 }
 
 if(state >= 0x80)                      //Master is requesting data
 {
 i2c_write(buffer[address++]);
 }
 }
 | 
 My last main:
 
  	  | Code: |  	  | if(I2CstartF==1) {
 //Master write to Slave////////////////////////////////////////////
 i2c_start();
 i2c_write(0xA0);//slave address
 i2c_write(i2cWaddr);//slave location address
 i2c_write(i2cWdata);//CC; DD; FF
 i2c_stop();
 delay_cycles(60);
 //Master read from Slave///////////////////////////////////////////
 i2c_start();
 i2c_write(0xA0);//device address
 i2c_write(i2cWaddr);//write location address
 i2c_start();
 i2c_write(0xA1);//write command to slave
 i2cRdata=i2c_read(0);
 i2c_stop();
 scomtxw2=i2cRdata;
 I2CstartF=0;
 enable_interrupts(INT_TBE);//rpt to PC data from slave
 }
 | 
 I made int i2cRdata=0xFF; in the master.
 If the slave not sending data, the i2cRdata should be 0xFF, but I have 0x00 so I think the slave sends 0x00
 I made the slave to send 0xAA instead of  buffer[address++] but still have 0x00.
 
 For sure I am making mistakes.
 Please help
 
 Best wishes
 Joe
 |  |  
		|  |  
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Thu Jun 27, 2019 1:36 am |   |  
				| 
 |  
				| Here is a bug.  You don't specify the slave address in your #use i2c() statement:
 
  	  | Quote: |  	  | //Slave I2C test program //TS1847.c
 /////////////////////////////////////////////////////////////////
 #include <16F1847.h>
 
 #FUSES HS,NOIESO,PLL,WDT,MCLR,PUT
 #FUSES NOPROTECT,NOCPD,WRT,STVREN
 #FUSES BROWNOUT,BORV25,NODEBUG,NOLVP
 #use delay(clock=32MHz,crystal=8MHz,restart_wdt)
 #use i2c(Slave,I2C1,restart_wdt) // I2C1 module, automatically hardware
 //////////////////////////////////////////////////////////////
 | 
 
 Example of how to specify the slave address:
 
  	  | Code: |  	  | #use i2c(Slave, I2C1, address=0x80, restart_wdt) | 
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Mon Jul 01, 2019 7:55 pm |   |  
				| 
 |  
				| Thank you PCM Programmer 
 I corrected the slave bug (added slave address) still not working.
 I have already many versions tested without success.
 I found a program I wrote in 2006 for PIC18F252 to read & write for EEPROM 24C512.
 
 I will buy a 24C512 and will start from there, I don't remember if the program was working.
 
 Best wishes
 Joe
 |  |  
		|  |  
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Jul 01, 2019 9:19 pm |   |  
				| 
 |  
				| Another useful test is to run the slave at 32 MHz, and run the master at 4 MHz.  This gives the slave plenty
 of time to respond.  If there is a problem with the slave
 responding in time, and this test makes it work, then
 you know what to do. ie., insert larger delay(s) in the
 master code, if both are running at 32 MHz.
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Tue Jul 02, 2019 2:06 am |   |  
				| 
 |  
				| Thank you again PCM Programmer. 
 I will do as you advised, just right now my brain stuck
   So I will start first to talk with the 24C512.
 
 Best wishes
 Joe
 |  |  
		|  |  
		| temtronic 
 
 
 Joined: 01 Jul 2010
 Posts: 9587
 Location: Greensville,Ontario
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Jul 02, 2019 4:59 am |   |  
				| 
 |  
				| I suggest you stop using the WDT until you get your program running 100%. 
 All too often I see code here with WDT enabled and wonder why. Unless properly coded a WDT can easily lead you 'down the wrong path'... thinking 'this' is wrong when in reality it's 'that'.
 WDT should only be installed AFTER 'main()' is truly working properly.
 
 Jay
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Tue Jul 02, 2019 4:22 pm |   |  
				| 
 |  
				| Thanks Jay, will disable it. 
 Best wishes
 Joe
 |  |  
		|  |  
		| gjs_rsdi 
 
 
 Joined: 06 Feb 2006
 Posts: 468
 Location: Bali
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Wed Jul 10, 2019 12:52 am |   |  
				| 
 |  
				| Finally I have a working master with 24C512   Thanks everyone with the help
   Understanding how I2C works (or any other subject) is the key.
 I added back the wdt (eliminated it per Jay advice) because in some stage the PIC stuck so I wanted to learn why.
 
 The working program:
 
  	  | Code: |  	  | ///////////////////////////////////////////////////////////////// //PIC16F1847 Master I2C with 24C512 eeprom test program
 //TM1847.c
 /////////////////////////////////////////////////////////////////
 #include <16F1847.h>
 
 #FUSES INTRC_IO,NOIESO,PLL,WDT,MCLR,PUT
 #FUSES NOPROTECT,NOCPD,WRT,STVREN
 #FUSES BROWNOUT,BORV25,NODEBUG,NOLVP
 #use delay(clock=32MHz,INTERNAL=8MHz)
 #use rs232(baud=9600,parity=N,xmit=PIN_B5,rcv=PIN_B2,bits=8,restart_wdt,errors)//V1
 #use i2c(Master,Slow,I2C1)//I2C1 module, automatically hardware
 
 #define flashled PIN_A2
 short powerupF=1;
 int flashcnt=0;
 short I2CstartF=0;
 int C512rpt=0;
 int i2cStartTimer=0;
 int scomtxwords=0;//V1
 int scomtxw2=0;//V1
 int scomtxwchs=0;//V1
 
 #INT_TIMER1
 void  TIMER1_isr(void)
 {
 flashcnt++;
 set_timer1(25536);//timer1 overflow 40ms
 }
 
 #int_TBE//V1
 void TBE_isr(void)
 {
 switch(scomtxwords)
 {
 case 0:
 {
 putc(85);
 scomtxwchs=85;
 scomtxwords++;
 }
 break;
 case 1:
 {
 putc(170);
 scomtxwchs=255;
 scomtxwords++;
 }
 break;
 case 2:
 {
 putc(scomtxw2);
 scomtxwchs=scomtxwchs+scomtxw2;
 scomtxwords++;
 }
 break;
 case 3:
 {
 putc(scomtxwchs);
 scomtxwords=0;
 disable_interrupts(INT_TBE);
 }
 break;
 }
 }
 
 void COUNTERS(void)
 {
 if(flashcnt>=25)
 {
 flashcnt=0;
 if(powerupF==1)
 {
 powerupF=0;
 enable_interrupts(INT_TBE);//should send 0x00
 }
 output_toggle(flashled);//LED flash every sec
 i2cStartTimer++;
 if(i2cStartTimer==4)
 {
 I2CstartF=1;//start i2c every 4 sec
 i2cStartTimer=0;
 }
 }
 }
 
 void I2C (void)
 {
 if(I2CstartF==1)
 {
 //write 0xAA to the 24C512
 I2CstartF=0;
 i2c_start();
 i2c_write(0xA0);//24C512 address with write
 i2c_write(0x00);//24C512 write address high
 i2c_write(0x00);//24C512 write address low
 i2c_write(0xAA);//data byte
 i2c_stop();
 //read back 0xAA from the 24C512
 delay_ms(60); //ensure the 24C512 has had time to complete the write
 i2c_start();
 i2c_write(0xA0);//24C512 address with write
 i2c_write(0x00);//24C512 read address high
 i2c_write(0x00);//24C512 read address low
 i2c_start();
 i2c_write(0xA1);//24C512 address with read~
 C512rpt = i2c_read(0);//24C512 rpt to master
 i2c_stop();
 scomtxw2 = C512rpt;
 enable_interrupts(INT_TBE);//rpt to PC data from 24C512
 }
 }
 
 #ZERO_RAM
 
 void main()
 {
 setup_adc_ports(NO_ANALOGS);
 setup_wdt(WDT_128MS);
 setup_timer_1(T1_INTERNAL|T1_DIV_BY_8);//65 ms overflow
 enable_interrupts(INT_TIMER1);
 disable_interrupts(INT_TBE);//V1
 enable_interrupts(GLOBAL);
 
 while(TRUE)
 {
 COUNTERS();
 I2C();
 restart_wdt();
 }
 }
 
 | 
 The program works also without sending address, then you get the data from location 0x01 that is 0xFF for empty chip
 
 
  	  | Code: |  	  | i2c_start(); i2c_write(0xA1);//24C512 address with read~
 C512rpt = i2c_read(0);//24C512 rpt to master
 i2c_stop();
 | 
 
 Thank you again PCM Programmer, Ttelmah and Jay
   
 Will start now with the slave PIC
 
 Best wishes
 Joe
 |  |  
		|  |  
		|  |  
  
	| 
 
 | You cannot post new topics in this forum You cannot reply to topics in this forum
 You cannot edit your posts in this forum
 You cannot delete your posts in this forum
 You cannot vote in polls in this forum
 
 |  
 Powered by phpBB © 2001, 2005 phpBB Group
 
 |