2015-08-12 10:53:41 +00:00
|
|
|
#include "mbed.h"
|
2015-09-11 15:39:38 +00:00
|
|
|
#include "MicroBit.h"
|
2015-08-12 10:53:41 +00:00
|
|
|
#include "twi_master.h"
|
|
|
|
#include "nrf_delay.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor.
|
|
|
|
* Create an instance of i2c
|
|
|
|
* @param sda the Pin to be used for SDA
|
|
|
|
* @param scl the Pin to be used for SCL
|
|
|
|
* Example:
|
|
|
|
* @code
|
|
|
|
* MicroBitI2C i2c(MICROBIT_PIN_SDA, MICROBIT_PIN_SCL);
|
|
|
|
* @endcode
|
|
|
|
* @note this should prevent i2c lockups as well.
|
|
|
|
*/
|
|
|
|
MicroBitI2C::MicroBitI2C(PinName sda, PinName scl) : I2C(sda,scl)
|
|
|
|
{
|
|
|
|
this->retries = 0;
|
|
|
|
}
|
|
|
|
|
2015-10-25 21:51:33 +00:00
|
|
|
/**
|
|
|
|
* Performs a complete read transaction. The bottom bit of the address is forced to 1 to indicate a read.
|
|
|
|
*
|
|
|
|
* @address 8-bit I2C slave address [ addr | 1 ]
|
|
|
|
* @data Pointer to the byte-array to read data in to
|
|
|
|
* @length Number of bytes to read
|
|
|
|
* @repeated Repeated start, true - don't send stop at end.
|
|
|
|
*
|
|
|
|
* @return MICROBIT_OK on success, MICROBIT_I2C_ERROR if an unresolved read failure is detected.
|
|
|
|
*/
|
2015-08-12 10:53:41 +00:00
|
|
|
int MicroBitI2C::read(int address, char *data, int length, bool repeated)
|
|
|
|
{
|
|
|
|
int result = I2C::read(address,data,length,repeated);
|
|
|
|
|
|
|
|
//0 indicates a success, presume failure
|
|
|
|
while(result != 0 && retries < MICROBIT_I2C_MAX_RETRIES)
|
|
|
|
{
|
|
|
|
_i2c.i2c->EVENTS_ERROR = 0;
|
|
|
|
_i2c.i2c->ENABLE = TWI_ENABLE_ENABLE_Disabled << TWI_ENABLE_ENABLE_Pos;
|
|
|
|
_i2c.i2c->POWER = 0;
|
|
|
|
nrf_delay_us(5);
|
|
|
|
_i2c.i2c->POWER = 1;
|
|
|
|
_i2c.i2c->ENABLE = TWI_ENABLE_ENABLE_Enabled << TWI_ENABLE_ENABLE_Pos;
|
|
|
|
twi_master_init_and_clear();
|
|
|
|
result = I2C::read(address,data,length,repeated);
|
|
|
|
retries++;
|
|
|
|
}
|
|
|
|
|
2015-10-25 21:51:33 +00:00
|
|
|
if(result != 0)
|
|
|
|
return MICROBIT_I2C_ERROR;
|
|
|
|
|
|
|
|
retries = 0;
|
|
|
|
return MICROBIT_OK;
|
2015-08-12 10:53:41 +00:00
|
|
|
}
|
|
|
|
|
2015-10-25 21:51:33 +00:00
|
|
|
/**
|
|
|
|
* Performs a complete write transaction. The bottom bit of the address is forced to 0 to indicate a write.
|
|
|
|
*
|
|
|
|
* @address 8-bit I2C slave address [ addr | 0 ]
|
|
|
|
* @data Pointer to the byte-arraycontaining the data to write
|
|
|
|
* @length Number of bytes to write
|
|
|
|
* @repeated Repeated start, true - don't send stop at end.
|
|
|
|
*
|
|
|
|
* @return MICROBIT_OK on success, MICROBIT_I2C_ERROR if an unresolved write failure is detected.
|
|
|
|
*/
|
2015-08-12 10:53:41 +00:00
|
|
|
int MicroBitI2C::write(int address, const char *data, int length, bool repeated)
|
|
|
|
{
|
|
|
|
int result = I2C::write(address,data,length,repeated);
|
|
|
|
|
|
|
|
//0 indicates a success, presume failure
|
|
|
|
while(result != 0 && retries < MICROBIT_I2C_MAX_RETRIES)
|
|
|
|
{
|
|
|
|
_i2c.i2c->EVENTS_ERROR = 0;
|
|
|
|
_i2c.i2c->ENABLE = TWI_ENABLE_ENABLE_Disabled << TWI_ENABLE_ENABLE_Pos;
|
|
|
|
_i2c.i2c->POWER = 0;
|
|
|
|
nrf_delay_us(5);
|
|
|
|
_i2c.i2c->POWER = 1;
|
|
|
|
_i2c.i2c->ENABLE = TWI_ENABLE_ENABLE_Enabled << TWI_ENABLE_ENABLE_Pos;
|
|
|
|
|
|
|
|
twi_master_init_and_clear();
|
|
|
|
result = I2C::write(address,data,length,repeated);
|
|
|
|
retries++;
|
|
|
|
}
|
|
|
|
|
2015-10-25 21:51:33 +00:00
|
|
|
if(result != 0)
|
|
|
|
return MICROBIT_I2C_ERROR;
|
2015-08-12 10:53:41 +00:00
|
|
|
|
2015-10-25 21:51:33 +00:00
|
|
|
retries = 0;
|
|
|
|
return MICROBIT_OK;
|
2015-08-12 10:53:41 +00:00
|
|
|
}
|