| View previous topic :: View next topic | 
	
	
		| Author | Message | 
	
		| beaker404 
 
 
 Joined: 24 Jul 2012
 Posts: 163
 
 
 
			    
 
 | 
			
				| I2C troubles. |  
				|  Posted: Mon Aug 06, 2018 2:36 pm |   |  
				| 
 |  
				| Continuing on trying to use the VL53L0X chip to do some distance measuring.  I found this product:  https://www.robotshop.com/media/files/pdf/tinylidar-tof-range-finder-sensor-datasheet.pdf 
 it claims to handle the complex API and 'dumb it all down'.  Got one, giving it a go.  Clock and Data lines look good on the scope, going to ground and up to about 4.3V I only read -1 for the MSB and LSB off the chip.  Perhaps someone else can spy what I am missing.
 
 
  	  | Code: |  	  | #USE I2C(MASTER,I2C1, SLOW, FORCE_HW, SMBUS) | 
 
 
  	  | Code: |  	  | i2c_start(); i2c_write(0x00);                   // reset to 0x10 address
 i2c_write(0x06);
 i2c_stop();
 
 delay_ms(100);
 
 i2c_start();
 i2c_write(0x10);
 i2c_write(0x4d);                   // set up for single read mode
 i2c_write(0x53);
 i2c_stop();
 
 delay_ms(100);
 
 i2c_start();
 i2c_write(0x10);
 i2c_write(0x54);                   // disable WDT
 i2c_write(0x30);
 i2c_stop();
 
 delay_ms(100);
 
 while(1) {                        // endless loop
 
 i2c_start();
 i2c_write(DISTANCE_ADDRESS);
 i2c_write(0x44);                   // single read
 msb = i2c_read();
 lsb = i2c_read(0);
 i2c_stop();
 
 led_toggle();
 
 distance_reading = msb << 8;
 distance_reading = distance_reading + lsb;
 
 distance_sum = distance_sum + distance_reading;
 distance_index = distance_index + 1;
 
 
 
 if(seconds >= 1) {  // one second pasted, average data and send string via BT
 
 seconds = 0;
 
 if(distance_index > 0) {
 distance_value = (float)((float)(distance_sum)/((float)(distance_index)));
 distance_sum = 0;
 distance_index = 0;
 }
 else {
 distance_value = distance_reading;
 distance_sum = 0;
 distance_index = 0;
 }
 
 
 fprintf(USB_SERIAL,"**\t%.3f\t#\n\r",distance_value);
 
 fprintf(USB_SERIAL,"**\t%d\t%d\t#\n\r",msb,lsb);
 }
 
 }  // end endless while
 | 
 
 not the prettiest of code I know.  I am using the default address of 0x10 for the device.  That is the value of DISTANCE_ADDRESS
 
 MPLAB 8.91
 CCS PCM C Compiler, Version 5.064, xxxxx
 Windows 10
 |  | 
	
		|  | 
	
		| temtronic 
 
 
 Joined: 01 Jul 2010
 Posts: 9587
 Location: Greensville,Ontario
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Aug 06, 2018 2:52 pm |   |  
				| 
 |  
				| OK, I know nothing about that device but... 
 i2c_write(0x00);                   // reset to 0x10 address
 
 hmm, is this correct ?
 
 Whenever coding, it helps to add comments to every line ,especially drivers. Costs nothing in codespace, but might help others now or you later, when you ask 'WHY did I do THAT"!??
 
 Also run PCM P's  'I2C scanner' program ,located in the code library. It'll confirm basic, low level access to any I2C device. If it reports 'device found at 0x23' and there's only one device, you now KNOW it's true address.
 Some 'camps' use the 7 bit address +R/W while CCS uses 8 bit addresses.
 
 Jay
 |  | 
	
		|  | 
	
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Aug 06, 2018 3:05 pm |   |  
				| 
 |  
				| The tinyLiDAR manual says this: 
  	  | Quote: |  	  | D Read Distance 0x44
 
 Read distance in mm from the most recent measurement
 Host -> Sensor
 Send Command
 0x44 {D}
 Sensor -> Host
 Read Response
 2 bytes with MSB first
 Example:
 Read from distance from tinyLiDAR at default address of 0x10.
 Value read was 0x0690 or 1680mm.
 <10:w> 44
 <10:r> 06 90
 
 | 
 The 'r' and 'w' refers to the i2c r/w bit.  If it's a read operation, this bit is
 set = 1.
 
 I have edited your main loop code as shown below, so that it works
 according to the manual's description.  I have indicated the changes
 as shown in bold:
 
  	  | Quote: |  	  | i2c_start(); i2c_write(0x10);
 i2c_write(0x44);
 i2c_write(0x11);
 msb = i2c_read();
 lsb = i2c_read(0);
 i2c_stop();
 | 
 
 However, I don't actually know for certain whether they are using
 7-bit or 8-bit i2c address mode.   So follow Temtronic's suggestion about
 checking the i2c slave address.
 
 Last edited by PCM programmer on Mon Aug 06, 2018 3:21 pm; edited 1 time in total
 |  | 
	
		|  | 
	
		| elcrcp 
 
 
 Joined: 11 Mar 2016
 Posts: 62
 Location: izmir / Turkey
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Aug 06, 2018 3:19 pm |   |  
				| 
 |  
				|  	  | Code: |  	  | i2c_start(); i2c_write(0x00);                   // reset to 0x10 address
 i2c_write(0x06);
 i2c_stop();
 
 | 
 This probably is wrong, you need to communicate with default address.
 
 And PCM's correction on code is probably right but I think write/read adresses are 0x20 and 0x21 since arduino uses 7 bit adressing while CCS uses 8
 _________________
 There is nothing you can't do if you try
 |  | 
	
		|  | 
	
		| beaker404 
 
 
 Joined: 24 Jul 2012
 Posts: 163
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Aug 06, 2018 3:34 pm |   |  
				| 
 |  
				| Got a device found at 0x20. I will report more tomorrow.
 |  | 
	
		|  | 
	
		| beaker404 
 
 
 Joined: 24 Jul 2012
 Posts: 163
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Aug 06, 2018 3:56 pm |   |  
				| 
 |  
				| tried with 0x20 for the base address, and using 0x21 for the read.  still -1 reported back.  I have commented allot of the code out.  Done with it for today. will try again tomorrow.
 |  | 
	
		|  | 
	
		| temtronic 
 
 
 Joined: 01 Jul 2010
 Posts: 9587
 Location: Greensville,Ontario
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Aug 06, 2018 4:14 pm |   |  
				| 
 |  
				| OK, silly question. What is the value of the I2C pullup resistors ?
 
 Also does PCM P's program work and return with the address ?
 
 Jay
 |  | 
	
		|  | 
	
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Aug 06, 2018 4:19 pm |   |  
				| 
 |  
				| In the link that you gave for documentation, it says: 
  	  | Quote: |  	  | I2C Pull-up resistors: 2x 4.7K SMD | 
 So it has two 4.7K pullups built-in.
 
 When you start working on the project again, post your revised
 code, so we can check if you're using the correct slave address
 in all sections of the program.
 |  | 
	
		|  | 
	
		| Ttelmah 
 
 
 Joined: 11 Mar 2010
 Posts: 19962
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Aug 07, 2018 12:54 am |   |  
				| 
 |  
				| 0x10, is the Arduino address. 7bit. The PIC address will be 0x20, so this is right. You are not reversing the I2C bus correctly. You need to issue a read command after a bus restart to reverse the bus...
 Personally, simplify. Make a connection between the ground and clear pins for a short while, to reset the chip to factory defaults. This sets it to address 0x20, and the single step mode. Then you can use the basic code from the top of page 6 of the data sheet:
 
  	  | Code: |  	  | #define LIDAR 0x20
 #define WR 0
 #define RD 1
 
 while(1)
 {                        // endless loop
 delay_ms(14); //max single read speed 75Hz
 i2c_start();
 i2c_write(LIDAR | WR); //we are writing
 i2c_write('D');  // issue single 'distance read' command
 i2c_start();      //restart
 i2c_write(LIDAR | RD); //select the read address
 msb = i2c_read(); //now read
 lsb = i2c_read(0);
 i2c_stop();
 
 led_toggle();
 
 //then your code to deal with this.
 }  // end endless while
 
 | 
 
 Note how the I2C bus reversal is done.
 |  | 
	
		|  | 
	
		| beaker404 
 
 
 Joined: 24 Jul 2012
 Posts: 163
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Aug 07, 2018 7:34 am |   |  
				| 
 |  
				| Still not working.  As for pullups I am only using the 4.7K that are on the LIDAR, nothing on my control PCB.  I still get -1 for MSB and LSB after doing the suggestions in this thread. 
 I think my hardware is good because the scanner program reported a device found at 0X20.
 
 
  	  | Code: |  	  | #use i2c(Master, sda=PIN_C4, scl=PIN_C3) /*
 *********************************************************************************************
 *  DEFINES
 *********************************************************************************************
 */
 
 #DEFINE LIDAR 0x20
 #DEFINE RD 1
 #DEFINE WR 0
 | 
 
 
  	  | Code: |  	  | int msb = 0; int lsb = 0;
 | 
 
 
 
  	  | Code: |  	  | delay_ms(14); //max single read speed 75Hz i2c_start();
 i2c_write(LIDAR | WR); //we are writing
 i2c_write('D');  // issue single 'distance read' command
 i2c_start();      //restart
 i2c_write(LIDAR | RD); //select the read address
 msb = i2c_read(); //now read
 lsb = i2c_read(0);
 i2c_stop();
 | 
 
 not sure where to go now, I think it is close, but no way to really know.
 |  | 
	
		|  | 
	
		| beaker404 
 
 
 Joined: 24 Jul 2012
 Posts: 163
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Aug 07, 2018 7:45 am |   |  
				| 
 |  
				| I have simplified the code down to do a simple on board LED turn on/off command 0x47. This does not work either.  Not sure why the scanner program would detect the device, and report the correct base address and simple LED control would not work.
 |  | 
	
		|  | 
	
		| Ttelmah 
 
 
 Joined: 11 Mar 2010
 Posts: 19962
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Aug 07, 2018 9:02 am |   |  
				| 
 |  
				| Are you sure the internal resistors are enabled?. There are two links on the board that can be cut or made to enable the resistors. You haven't mentioned what PIC?. On some PIC's the SMBUS option does not work. How did you have the bus setup when you were running the I2C scanner program?. Setting should be the same.
 The write should work even if the PIC bus levels were set wrong.
 Double check the settings you actually used for the bus scanner program.
 |  | 
	
		|  | 
	
		| beaker404 
 
 
 Joined: 24 Jul 2012
 Posts: 163
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Aug 07, 2018 9:19 am |   |  
				| 
 |  
				| Yes the internal resistors are enabled.  I am configured like the scanner program, that is the USE statement that is in the last post of my code.  I am using 16F876 for a PIC, sorry thought I included that with the other specifics. |  | 
	
		|  | 
	
		| beaker404 
 
 
 Joined: 24 Jul 2012
 Posts: 163
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Aug 07, 2018 12:47 pm |   |  
				| 
 |  
				| Just a follow up for the thread.  I got the LIDAR to work.  A couple of things, one there is a WDT on the board that resets after 3 seconds of no I2C bus activity.  I had a really slow loop running to toggle LEDS so, disabled the WDT first thing, then I had to put the chip in single step mode before calling for a measurement.  Lastly, Ttelmah's comment about needing to reset the buss is correct, had to put a i2c_stop() in front of it to keep it from hanging after a few reads.   After the i2c_stop() was added, correct data flowed and no lock ups. Isolated it down by focusing on controlling the on board LED first then expanding out.  The i2c scanner code was really nice to have to verify that hardware wise, things were good.  Thanks everyone.  I will post some working code for the group in the next couple days and will post a new thread for this hardware in the CODE section because the datasheet does not do any PIC or CCS code examples.  Here is the basic loop that worked:
 
 
  	  | Code: |  	  | delay_ms(14); //max single read speed 75Hz i2c_start();
 i2c_write(LIDAR | WR); //we are writing
 i2c_write('D');  // issue single 'distance read' command
 i2c_stop();
 i2c_start();      //restart
 i2c_write(LIDAR | RD); //select the read address
 msb = i2c_read(); //now read
 lsb = i2c_read(0);
 i2c_stop();
 | 
 |  | 
	
		|  | 
	
		| Ttelmah 
 
 
 Joined: 11 Mar 2010
 Posts: 19962
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Wed Aug 08, 2018 3:23 am |   |  
				| 
 |  
				| That is quite interesting. They don't give any actual bus timing specifications in the data. I've seen the need for a stop/start, rather than a simple restart on a few devices, but they all have delays specified between the last transmitted byte and the start (typically something like 1.5uSec), so:
 
  	  | Code: |  	  | delay_ms(14); //max single read speed 75Hz
 i2c_start();
 i2c_write(LIDAR | WR); //we are writing
 i2c_write('D');  // issue single 'distance read' command
 //i2c_stop();
 delay_us(2); //may also work
 i2c_start();      //restart
 i2c_write(LIDAR | RD); //select the read address
 msb = i2c_read(); //now read
 lsb = i2c_read(0);
 i2c_stop();
 
 | 
 May work as well.
 
 On these it is also worth (if doing rapid sequential I2C transfers), adding a 2uSec delay after the I2C stop, since again there are problems if you do a second transaction immediately after the first.
 
 Glad you have it working now.
  |  | 
	
		|  | 
	
		|  |