Ported from sparkfun library
Dependents: tc_agent Line_Trace_No_collision_Light_control_longer_dis
Diff: VL6180X.cpp
- Revision:
- 0:8aca38272614
diff -r 000000000000 -r 8aca38272614 VL6180X.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/VL6180X.cpp Tue Jul 19 03:49:54 2016 +0000 @@ -0,0 +1,254 @@ +/****************************************************************************** + * SparkFun_VL6180x.cpp + * Library for VL6180x time of flight range finder. + * Casey Kuhns @ SparkFun Electronics + * 10/29/2014 + * https://github.com/sparkfun/SparkFun_ToF_Range_Finder-VL6180_Arduino_Library + * + * The VL6180x by ST micro is a time of flight range finder that + * uses pulsed IR light to determine distances from object at close + * range. The average range of a sensor is between 0-200mm + * + * In this file are the functions in the VL6180x class + * + * Resources: + * This library uses the Arduino Wire.h to complete I2C transactions. + * + * Development environment specifics: + * IDE: Arduino 1.0.5 + * Hardware Platform: Arduino Pro 3.3V/8MHz + * VL6180x Breakout Version: 1.0 + * **Updated for Arduino 1.6.4 5/2015** + * + * This code is beerware. If you see me (or any other SparkFun employee) at the + * local pub, and you've found our code helpful, please buy us a round! + * + * Distributed as-is; no warranty is given. + ******************************************************************************/ + +#include "VL6180X.h" +#include "mbed.h" + +VL6180x::VL6180x(PinName sda, PinName scl) : _i2c(sda, scl) { +} + + +uint8_t VL6180x::VL6180xInit(void){ + uint8_t data; //for temp data storage + + data = VL6180x_getRegister(VL6180X_SYSTEM_FRESH_OUT_OF_RESET); + + if(data != 1) return VL6180x_FAILURE_RESET; + + //Required by datasheet + //http://www.st.com/st-web-ui/static/active/en/resource/technical/document/application_note/DM00122600.pdf + VL6180x_setRegister(0x0207, 0x01); + VL6180x_setRegister(0x0208, 0x01); + VL6180x_setRegister(0x0096, 0x00); + VL6180x_setRegister(0x0097, 0xfd); + VL6180x_setRegister(0x00e3, 0x00); + VL6180x_setRegister(0x00e4, 0x04); + VL6180x_setRegister(0x00e5, 0x02); + VL6180x_setRegister(0x00e6, 0x01); + VL6180x_setRegister(0x00e7, 0x03); + VL6180x_setRegister(0x00f5, 0x02); + VL6180x_setRegister(0x00d9, 0x05); + VL6180x_setRegister(0x00db, 0xce); + VL6180x_setRegister(0x00dc, 0x03); + VL6180x_setRegister(0x00dd, 0xf8); + VL6180x_setRegister(0x009f, 0x00); + VL6180x_setRegister(0x00a3, 0x3c); + VL6180x_setRegister(0x00b7, 0x00); + VL6180x_setRegister(0x00bb, 0x3c); + VL6180x_setRegister(0x00b2, 0x09); + VL6180x_setRegister(0x00ca, 0x09); + VL6180x_setRegister(0x0198, 0x01); + VL6180x_setRegister(0x01b0, 0x17); + VL6180x_setRegister(0x01ad, 0x00); + VL6180x_setRegister(0x00ff, 0x05); + VL6180x_setRegister(0x0100, 0x05); + VL6180x_setRegister(0x0199, 0x05); + VL6180x_setRegister(0x01a6, 0x1b); + VL6180x_setRegister(0x01ac, 0x3e); + VL6180x_setRegister(0x01a7, 0x1f); + VL6180x_setRegister(0x0030, 0x00); + + return 0; +} + +void VL6180x::VL6180xDefautSettings(void){ + //Recommended settings from datasheet + //http://www.st.com/st-web-ui/static/active/en/resource/technical/document/application_note/DM00122600.pdf + + //Enable Interrupts on Conversion Complete (any source) + VL6180x_setRegister(VL6180X_SYSTEM_INTERRUPT_CONFIG_GPIO, (4 << 3)|(4) ); // Set GPIO1 high when sample complete + + + VL6180x_setRegister(VL6180X_SYSTEM_MODE_GPIO1, 0x10); // Set GPIO1 high when sample complete + VL6180x_setRegister(VL6180X_READOUT_AVERAGING_SAMPLE_PERIOD, 0x30); //Set Avg sample period + VL6180x_setRegister(VL6180X_SYSALS_ANALOGUE_GAIN, 0x46); // Set the ALS gain + VL6180x_setRegister(VL6180X_SYSRANGE_VHV_REPEAT_RATE, 0xFF); // Set auto calibration period (Max = 255)/(OFF = 0) + VL6180x_setRegister(VL6180X_SYSALS_INTEGRATION_PERIOD, 0x63); // Set ALS integration time to 100ms + VL6180x_setRegister(VL6180X_SYSRANGE_VHV_RECALIBRATE, 0x01); // perform a single temperature calibration + //Optional settings from datasheet + //http://www.st.com/st-web-ui/static/active/en/resource/technical/document/application_note/DM00122600.pdf + VL6180x_setRegister(VL6180X_SYSRANGE_INTERMEASUREMENT_PERIOD, 0x09); // Set default ranging inter-measurement period to 100ms + VL6180x_setRegister(VL6180X_SYSALS_INTERMEASUREMENT_PERIOD, 0x0A); // Set default ALS inter-measurement period to 100ms + VL6180x_setRegister(VL6180X_SYSTEM_INTERRUPT_CONFIG_GPIO, 0x24); // Configures interrupt on ‘New Sample Ready threshold event’ + //Additional settings defaults from community + VL6180x_setRegister(VL6180X_SYSRANGE_MAX_CONVERGENCE_TIME, 0x32); + VL6180x_setRegister(VL6180X_SYSRANGE_RANGE_CHECK_ENABLES, 0x10 | 0x01); + VL6180x_setRegister16bit(VL6180X_SYSRANGE_EARLY_CONVERGENCE_ESTIMATE, 0x7B ); + VL6180x_setRegister16bit(VL6180X_SYSALS_INTEGRATION_PERIOD, 0x64); + + VL6180x_setRegister(VL6180X_READOUT_AVERAGING_SAMPLE_PERIOD,0x30); + VL6180x_setRegister(VL6180X_SYSALS_ANALOGUE_GAIN,0x40); + VL6180x_setRegister(VL6180X_FIRMWARE_RESULT_SCALER,0x01); +} +void VL6180x::getIdentification(struct VL6180xIdentification *temp){ + + temp->idModel = VL6180x_getRegister(VL6180X_IDENTIFICATION_MODEL_ID); + temp->idModelRevMajor = VL6180x_getRegister(VL6180X_IDENTIFICATION_MODEL_REV_MAJOR); + temp->idModelRevMinor = VL6180x_getRegister(VL6180X_IDENTIFICATION_MODEL_REV_MINOR); + temp->idModuleRevMajor = VL6180x_getRegister(VL6180X_IDENTIFICATION_MODULE_REV_MAJOR); + temp->idModuleRevMinor = VL6180x_getRegister(VL6180X_IDENTIFICATION_MODULE_REV_MINOR); + + temp->idDate = VL6180x_getRegister16bit(VL6180X_IDENTIFICATION_DATE); + temp->idTime = VL6180x_getRegister16bit(VL6180X_IDENTIFICATION_TIME); +} + + +uint8_t VL6180x::changeAddress(uint8_t old_address, uint8_t new_address){ + + //NOTICE: IT APPEARS THAT CHANGING THE ADDRESS IS NOT STORED IN NON-VOLATILE MEMORY + // POWER CYCLING THE DEVICE REVERTS ADDRESS BACK TO 0X29 + + if( old_address == new_address) return old_address; + if( new_address > 127) return old_address; + + VL6180x_setRegister(VL6180X_I2C_SLAVE_DEVICE_ADDRESS, new_address); + + return VL6180x_getRegister(VL6180X_I2C_SLAVE_DEVICE_ADDRESS); +} + + + +void VL6180x::triggerDistance() +{ + VL6180x_setRegister(VL6180X_SYSRANGE_START, 0x01); //Start Single shot mode +} + +uint8_t VL6180x::pollDistance() +{ + uint8_t status; + do{ + status = VL6180x_getRegister(VL6180X_RESULT_INTERRUPT_STATUS_GPIO); + }while((status & 0x7) != 4); + return VL6180x_getRegister(VL6180X_RESULT_RANGE_VAL); +} + +uint8_t VL6180x::getDistance() +{ + VL6180x_setRegister(VL6180X_SYSRANGE_START, 0x01); //Start Single shot mode + wait_ms(10); + return VL6180x_getRegister(VL6180X_RESULT_RANGE_VAL); +// VL6180x_setRegister(VL6180X_SYSTEM_INTERRUPT_CLEAR, 0x07); + // return distance; +} + +float VL6180x::getAmbientLight(vl6180x_als_gain VL6180X_ALS_GAIN) +{ + //First load in Gain we are using, do it everytime incase someone changes it on us. + //Note: Upper nibble shoudl be set to 0x4 i.e. for ALS gain of 1.0 write 0x46 + VL6180x_setRegister(VL6180X_SYSALS_ANALOGUE_GAIN, (0x40 | VL6180X_ALS_GAIN)); // Set the ALS gain + + //Start ALS Measurement + VL6180x_setRegister(VL6180X_SYSALS_START, 0x01); + + wait_ms(100); //give it time... + + VL6180x_setRegister(VL6180X_SYSTEM_INTERRUPT_CLEAR, 0x07); + + //Retrieve the Raw ALS value from the sensoe + unsigned int alsRaw = VL6180x_getRegister16bit(VL6180X_RESULT_ALS_VAL); + + //Get Integration Period for calculation, we do this everytime incase someone changes it on us. + unsigned int alsIntegrationPeriodRaw = VL6180x_getRegister16bit(VL6180X_SYSALS_INTEGRATION_PERIOD); + + float alsIntegrationPeriod = 100.0 / alsIntegrationPeriodRaw ; + + //Calculate actual LUX from Appnotes + + float alsGain = 0.0; + + switch (VL6180X_ALS_GAIN){ + case GAIN_20: alsGain = 20.0; break; + case GAIN_10: alsGain = 10.32; break; + case GAIN_5: alsGain = 5.21; break; + case GAIN_2_5: alsGain = 2.60; break; + case GAIN_1_67: alsGain = 1.72; break; + case GAIN_1_25: alsGain = 1.28; break; + case GAIN_1: alsGain = 1.01; break; + case GAIN_40: alsGain = 40.0; break; + } + +//Calculate LUX from formula in AppNotes + + float alsCalculated = (float)0.32 * ((float)alsRaw / alsGain) * alsIntegrationPeriod; + + return alsCalculated; +} + +// --- Private Functions --- // + +uint8_t VL6180x::VL6180x_getRegister(uint16_t reg) +{ + uint8_t dataout[2]; + uint8_t datain[1]; + + dataout[0] = (reg >> 8) & 0xFF; + dataout[1] = reg & 0xFF; + + _i2c.write(_addr, (char *) dataout, 2); + _i2c.read(_addr, (char *) datain, 1); + + return datain[0]; +} + +uint16_t VL6180x::VL6180x_getRegister16bit(uint16_t reg) +{ + uint8_t dataout[2],datain[2]; + + dataout[0] = (reg >> 8) & 0xFF; + dataout[1] = reg & 0xFF; + + uint16_t data; + + _i2c.write(_addr, (char *) dataout, 2); + _i2c.read(_addr, (char *) datain,2); + + data = (datain[0] << 8)|datain[1]; + + return data; +} + +void VL6180x::VL6180x_setRegister(uint16_t reg, uint8_t data) +{ + uint8_t dataout[3]; + + dataout[0] = (reg >> 8) & 0xFF; + dataout[1] = reg & 0xFF; + dataout[2] = data & 0xFF; + + _i2c.write(_addr, (char *) dataout, 3); +} + +void VL6180x::VL6180x_setRegister16bit(uint16_t reg, uint16_t data) +{ + uint8_t dataout[4]; + dataout[0] = (reg >> 8) & 0xFF; + dataout[1] = reg & 0xFF; + dataout[2] = (data >> 8) & 0xff; + dataout[3] = data & 0xFF; + _i2c.write(_addr, (char *) dataout, 4); +}