224 lines
3.9 KiB
C++
224 lines
3.9 KiB
C++
/**
|
|
* @file i2c_bb.cpp
|
|
* @brief Bit-banged I2C driver
|
|
*
|
|
* @date February 6, 2020
|
|
* @author Matthew Kennedy, (c) 2020
|
|
*/
|
|
|
|
#include "i2c_bb.h"
|
|
|
|
#include "io_pins.h"
|
|
#include "efi_gpio.h"
|
|
|
|
void BitbangI2c::sda_high() {
|
|
#if EFI_PROD_CODE
|
|
palSetPad(m_sdaPort, m_sdaPin);
|
|
#endif
|
|
}
|
|
|
|
void BitbangI2c::sda_low() {
|
|
#if EFI_PROD_CODE
|
|
palClearPad(m_sdaPort, m_sdaPin);
|
|
#endif
|
|
}
|
|
|
|
void BitbangI2c::scl_high() {
|
|
#if EFI_PROD_CODE
|
|
palSetPad(m_sclPort, m_sclPin);
|
|
#endif
|
|
}
|
|
|
|
void BitbangI2c::scl_low() {
|
|
#if EFI_PROD_CODE
|
|
palClearPad(m_sclPort, m_sclPin);
|
|
#endif
|
|
}
|
|
|
|
void BitbangI2c::init(brain_pin_e scl, brain_pin_e sda) {
|
|
#if EFI_PROD_CODE
|
|
if (m_sdaPort) return;
|
|
|
|
efiSetPadMode("i2c", scl, PAL_MODE_OUTPUT_OPENDRAIN); //PAL_STM32_OTYPE_OPENDRAIN
|
|
efiSetPadMode("i2c", sda, PAL_MODE_OUTPUT_OPENDRAIN);
|
|
|
|
m_sclPort = getHwPort("i2c", scl);
|
|
m_sclPin = getHwPin("i2c", scl);
|
|
|
|
m_sdaPort = getHwPort("i2c", sda);
|
|
m_sdaPin = getHwPin("i2c", sda);
|
|
#endif
|
|
|
|
// Both lines idle high
|
|
scl_high();
|
|
sda_high();
|
|
}
|
|
|
|
void BitbangI2c::start() {
|
|
// Start with both lines high (bus idle)
|
|
sda_high();
|
|
waitQuarterBit();
|
|
scl_high();
|
|
waitQuarterBit();
|
|
|
|
// SDA goes low while SCL is high
|
|
sda_low();
|
|
waitQuarterBit();
|
|
scl_low();
|
|
waitQuarterBit();
|
|
}
|
|
|
|
void BitbangI2c::stop() {
|
|
scl_low();
|
|
waitQuarterBit();
|
|
sda_low();
|
|
waitQuarterBit();
|
|
scl_high();
|
|
waitQuarterBit();
|
|
// SDA goes high while SCL is high
|
|
sda_high();
|
|
}
|
|
|
|
void BitbangI2c::sendBit(bool val) {
|
|
waitQuarterBit();
|
|
|
|
// Write the bit (write while SCL is low)
|
|
if (val) {
|
|
sda_high();
|
|
} else {
|
|
sda_low();
|
|
}
|
|
|
|
// Data setup time (~100ns min)
|
|
waitQuarterBit();
|
|
|
|
// Strobe the clock
|
|
scl_high();
|
|
waitQuarterBit();
|
|
scl_low();
|
|
waitQuarterBit();
|
|
}
|
|
|
|
bool BitbangI2c::readBit() {
|
|
waitQuarterBit();
|
|
|
|
scl_high();
|
|
|
|
waitQuarterBit();
|
|
waitQuarterBit();
|
|
|
|
#if EFI_PROD_CODE
|
|
// Read just before we set the clock low (ie, as late as possible)
|
|
bool val = palReadPad(m_sdaPort, m_sdaPin);
|
|
#else
|
|
bool val = false;
|
|
#endif
|
|
|
|
scl_low();
|
|
waitQuarterBit();
|
|
|
|
return val;
|
|
}
|
|
|
|
bool BitbangI2c::writeByte(uint8_t data) {
|
|
// write out 8 data bits
|
|
for (size_t i = 0; i < 8; i++) {
|
|
// Send the MSB
|
|
sendBit((data & 0x80) != 0);
|
|
|
|
data = data << 1;
|
|
}
|
|
|
|
// Force a release of the data line so the slave can ACK
|
|
sda_high();
|
|
|
|
// Read the ack bit
|
|
bool ackBit = readBit();
|
|
|
|
// 0 -> ack
|
|
// 1 -> nack
|
|
return !ackBit;
|
|
}
|
|
|
|
uint8_t BitbangI2c::readByte(bool ack) {
|
|
uint8_t result = 0;
|
|
|
|
// Read in 8 data bits
|
|
for (size_t i = 0; i < 8; i++) {
|
|
result = result << 1;
|
|
|
|
result |= readBit() ? 1 : 0;
|
|
}
|
|
|
|
// 0 -> ack
|
|
// 1 -> nack
|
|
sendBit(!ack);
|
|
|
|
return result;
|
|
}
|
|
|
|
void BitbangI2c::waitQuarterBit() {
|
|
// This yields a bitrate of about 320khz on a 168MHz F4
|
|
for (size_t i = 0; i < 30; i++) {
|
|
__asm__ volatile ("nop");
|
|
}
|
|
}
|
|
|
|
void BitbangI2c::write(uint8_t addr, const uint8_t* writeData, size_t writeSize) {
|
|
start();
|
|
|
|
// Address + write
|
|
writeByte(addr << 1 | 0);
|
|
|
|
// Write outbound bytes
|
|
for (size_t i = 0; i < writeSize; i++) {
|
|
writeByte(writeData[i]);
|
|
}
|
|
|
|
stop();
|
|
}
|
|
|
|
void BitbangI2c::writeRead(uint8_t addr, const uint8_t* writeData, size_t writeSize, uint8_t* readData, size_t readSize) {
|
|
start();
|
|
|
|
// Address + write
|
|
writeByte(addr << 1 | 0);
|
|
|
|
// Write outbound bytes
|
|
for (size_t i = 0; i < writeSize; i++) {
|
|
writeByte(writeData[i]);
|
|
}
|
|
|
|
// Send a repeated start bit to indicate transition to read
|
|
start();
|
|
|
|
// Address + read
|
|
writeByte(addr << 1 | 1);
|
|
|
|
for (size_t i = 0; i < readSize - 1; i++) {
|
|
// All but the last byte send ACK to indicate we're still reading
|
|
readData[i] = readByte(true);
|
|
}
|
|
|
|
// last byte sends NAK to indicate we're done reading
|
|
readData[readSize - 1] = readByte(false);
|
|
|
|
stop();
|
|
}
|
|
|
|
uint8_t BitbangI2c::readRegister(uint8_t addr, uint8_t reg) {
|
|
uint8_t retval;
|
|
|
|
writeRead(addr, ®, 1, &retval, 1);
|
|
|
|
return retval;
|
|
}
|
|
|
|
void BitbangI2c::writeRegister(uint8_t addr, uint8_t reg, uint8_t val) {
|
|
uint8_t buf[2];
|
|
buf[0] = reg;
|
|
buf[1] = val;
|
|
|
|
write(addr, buf, 2);
|
|
}
|