* Refactor I2C_SOFT to new #define * Stitch in some of TS101 Update ShowStartupWarnings.cpp Update OLED.hpp Update stm32f1xx_hal_msp.c Update Setup.cpp Update Power.cpp Update Pins.h Update configuration.h Power Muxing Working dual input Voltage handler Scan mode required for differing injected channels Inject both dc readings Update configuration.h Update configuration.h Use htim4 for adc control on TS101 Refactor htim names Add ADC_TRIGGER Speed up BB I2C a lil Update configuration.h Update startup_stm32f103t8ux.S Update configuration.h Add LIS2DH clone LIS2DH gains another clone Create tooling to allow mapping accelerometers onto different buses Update startup_stm32f103t8ux.S Ensure PD IRQ is pulled up * Stitch in some of TS101 Update ShowStartupWarnings.cpp Update OLED.hpp Update stm32f1xx_hal_msp.c Update Setup.cpp Update Power.cpp Update Pins.h Update configuration.h Power Muxing Working dual input Voltage handler Scan mode required for differing injected channels Inject both dc readings Update configuration.h Update configuration.h Use htim4 for adc control on TS101 Refactor htim names Add ADC_TRIGGER Speed up BB I2C a lil Update configuration.h Update startup_stm32f103t8ux.S Update configuration.h Add LIS2DH clone LIS2DH gains another clone Create tooling to allow mapping accelerometers onto different buses Update startup_stm32f103t8ux.S Ensure PD IRQ is pulled up Allow toggle which button enters PD debug * Update Pins.h * Fix hard coded IRQ Pin Update stm32f1xx_it.c * Enable EPR * Tip resistance measurement * TS101 is a direct drive tip Update BSP.cpp * Add S60 and TS101 to builds Update push.yml * Update MOVThread.cpp * Refactor power menu handler * Correct prescaler Forgot to update since I changed the period * Tune in the timer divider for tip control to make PWM less audible --------- Co-authored-by: discip <53649486+discip@users.noreply.github.com>
318 lines
6.3 KiB
C++
318 lines
6.3 KiB
C++
/*
|
|
* I2CBB1.cpp
|
|
*
|
|
* Created on: 12 Jun 2020
|
|
* Author: Ralim
|
|
*/
|
|
#include "configuration.h"
|
|
#ifdef I2C_SOFT_BUS_1
|
|
#include "FreeRTOS.h"
|
|
#include <I2CBB1.hpp>
|
|
SemaphoreHandle_t I2CBB1::I2CSemaphore = NULL;
|
|
StaticSemaphore_t I2CBB1::xSemaphoreBuffer;
|
|
void I2CBB1::init() {
|
|
// Set GPIO's to output open drain
|
|
GPIO_InitTypeDef GPIO_InitStruct;
|
|
__HAL_RCC_GPIOA_CLK_ENABLE();
|
|
__HAL_RCC_GPIOB_CLK_ENABLE();
|
|
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_MEDIUM;
|
|
GPIO_InitStruct.Pin = SDA_Pin;
|
|
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD;
|
|
GPIO_InitStruct.Pull = GPIO_PULLUP;
|
|
HAL_GPIO_Init(SDA_GPIO_Port, &GPIO_InitStruct);
|
|
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_MEDIUM;
|
|
GPIO_InitStruct.Pin = SCL_Pin;
|
|
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD;
|
|
GPIO_InitStruct.Pull = GPIO_PULLUP;
|
|
HAL_GPIO_Init(SCL_GPIO_Port, &GPIO_InitStruct);
|
|
SOFT_SDA1_HIGH();
|
|
SOFT_SCL1_HIGH();
|
|
// To ensure bus is unlocked; we toggle the Clock a bunch of times to make things error out
|
|
for (int i = 0; i < 128; i++) {
|
|
SOFT_SCL1_LOW();
|
|
asm("nop");
|
|
asm("nop");
|
|
asm("nop");
|
|
asm("nop");
|
|
SOFT_SCL1_HIGH();
|
|
asm("nop");
|
|
asm("nop");
|
|
asm("nop");
|
|
asm("nop");
|
|
}
|
|
I2CSemaphore = xSemaphoreCreateMutexStatic(&xSemaphoreBuffer);
|
|
unlock();
|
|
}
|
|
|
|
bool I2CBB1::probe(uint8_t address) {
|
|
if (!lock())
|
|
return false;
|
|
start();
|
|
bool ack = send(address);
|
|
stop();
|
|
unlock();
|
|
return ack;
|
|
}
|
|
|
|
bool I2CBB1::Mem_Read(uint16_t DevAddress, uint16_t MemAddress, uint8_t *pData, uint16_t Size) {
|
|
if (!lock())
|
|
return false;
|
|
start();
|
|
bool ack = send(DevAddress);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return false;
|
|
}
|
|
ack = send(MemAddress);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return false;
|
|
}
|
|
SOFT_SCL1_LOW();
|
|
SOFT_I2C_DELAY();
|
|
// stop();
|
|
start();
|
|
ack = send(DevAddress | 1);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return false;
|
|
}
|
|
while (Size) {
|
|
pData[0] = read(Size > 1);
|
|
pData++;
|
|
Size--;
|
|
}
|
|
stop();
|
|
unlock();
|
|
return true;
|
|
}
|
|
|
|
bool I2CBB1::Mem_Write(uint16_t DevAddress, uint16_t MemAddress, const uint8_t *pData, uint16_t Size) {
|
|
if (!lock())
|
|
return false;
|
|
start();
|
|
bool ack = send(DevAddress);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return false;
|
|
}
|
|
ack = send(MemAddress);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return false;
|
|
}
|
|
while (Size) {
|
|
resetWatchdog();
|
|
ack = send(pData[0]);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return false;
|
|
}
|
|
pData++;
|
|
Size--;
|
|
}
|
|
stop();
|
|
unlock();
|
|
return true;
|
|
}
|
|
|
|
void I2CBB1::Transmit(uint16_t DevAddress, uint8_t *pData, uint16_t Size) {
|
|
if (!lock())
|
|
return;
|
|
start();
|
|
bool ack = send(DevAddress);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return;
|
|
}
|
|
while (Size) {
|
|
ack = send(pData[0]);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return;
|
|
}
|
|
pData++;
|
|
Size--;
|
|
}
|
|
stop();
|
|
unlock();
|
|
}
|
|
|
|
void I2CBB1::Receive(uint16_t DevAddress, uint8_t *pData, uint16_t Size) {
|
|
if (!lock())
|
|
return;
|
|
start();
|
|
bool ack = send(DevAddress | 1);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return;
|
|
}
|
|
while (Size) {
|
|
pData[0] = read(Size > 1);
|
|
pData++;
|
|
Size--;
|
|
}
|
|
stop();
|
|
unlock();
|
|
}
|
|
|
|
void I2CBB1::TransmitReceive(uint16_t DevAddress, uint8_t *pData_tx, uint16_t Size_tx, uint8_t *pData_rx, uint16_t Size_rx) {
|
|
if (Size_tx == 0 && Size_rx == 0)
|
|
return;
|
|
if (lock() == false)
|
|
return;
|
|
if (Size_tx) {
|
|
start();
|
|
bool ack = send(DevAddress);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return;
|
|
}
|
|
while (Size_tx) {
|
|
ack = send(pData_tx[0]);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return;
|
|
}
|
|
pData_tx++;
|
|
Size_tx--;
|
|
}
|
|
}
|
|
if (Size_rx) {
|
|
start();
|
|
bool ack = send(DevAddress | 1);
|
|
if (!ack) {
|
|
stop();
|
|
unlock();
|
|
return;
|
|
}
|
|
while (Size_rx) {
|
|
pData_rx[0] = read(Size_rx > 1);
|
|
pData_rx++;
|
|
Size_rx--;
|
|
}
|
|
}
|
|
stop();
|
|
unlock();
|
|
}
|
|
|
|
void I2CBB1::start() {
|
|
/* I2C Start condition, data line goes low when clock is high */
|
|
SOFT_SCL1_HIGH();
|
|
SOFT_SDA1_HIGH();
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SDA1_LOW();
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SCL1_LOW();
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SDA1_HIGH();
|
|
}
|
|
|
|
void I2CBB1::stop() {
|
|
/* I2C Stop condition, clock goes high when data is low */
|
|
SOFT_SDA1_LOW();
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SCL1_HIGH();
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SDA1_HIGH();
|
|
SOFT_I2C_DELAY();
|
|
}
|
|
|
|
bool I2CBB1::send(uint8_t value) {
|
|
|
|
for (uint8_t i = 0; i < 8; i++) {
|
|
write_bit(value & 0x80); // write the most-significant bit
|
|
value <<= 1;
|
|
}
|
|
|
|
SOFT_SDA1_HIGH();
|
|
bool ack = (read_bit() == 0);
|
|
return ack;
|
|
}
|
|
|
|
uint8_t I2CBB1::read(bool ack) {
|
|
uint8_t B = 0;
|
|
|
|
uint8_t i;
|
|
for (i = 0; i < 8; i++) {
|
|
B <<= 1;
|
|
B |= read_bit();
|
|
}
|
|
|
|
SOFT_SDA1_HIGH();
|
|
if (ack)
|
|
write_bit(0);
|
|
else
|
|
write_bit(1);
|
|
return B;
|
|
}
|
|
|
|
uint8_t I2CBB1::read_bit() {
|
|
uint8_t b;
|
|
|
|
SOFT_SDA1_HIGH();
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SCL1_HIGH();
|
|
SOFT_I2C_DELAY();
|
|
|
|
if (SOFT_SDA1_READ())
|
|
b = 1;
|
|
else
|
|
b = 0;
|
|
|
|
SOFT_SCL1_LOW();
|
|
return b;
|
|
}
|
|
|
|
void I2CBB1::unlock() { xSemaphoreGive(I2CSemaphore); }
|
|
|
|
bool I2CBB1::lock() {
|
|
if (I2CSemaphore == NULL) {}
|
|
bool a = xSemaphoreTake(I2CSemaphore, (TickType_t)100) == pdTRUE;
|
|
return a;
|
|
}
|
|
|
|
bool I2CBB1::I2C_RegisterWrite(uint8_t address, uint8_t reg, uint8_t data) { return Mem_Write(address, reg, &data, 1); }
|
|
|
|
uint8_t I2CBB1::I2C_RegisterRead(uint8_t address, uint8_t reg) {
|
|
uint8_t temp = 0;
|
|
Mem_Read(address, reg, &temp, 1);
|
|
return temp;
|
|
}
|
|
|
|
void I2CBB1::write_bit(uint8_t val) {
|
|
if (val) {
|
|
SOFT_SDA1_HIGH();
|
|
} else {
|
|
SOFT_SDA1_LOW();
|
|
}
|
|
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SCL1_HIGH();
|
|
SOFT_I2C_DELAY();
|
|
SOFT_SCL1_LOW();
|
|
}
|
|
|
|
bool I2CBB1::writeRegistersBulk(const uint8_t address, const I2C_REG *registers, const uint8_t registersLength) {
|
|
for (int index = 0; index < registersLength; index++) {
|
|
if (!I2C_RegisterWrite(address, registers[index].reg, registers[index].val)) {
|
|
return false;
|
|
}
|
|
if (registers[index].pause_ms)
|
|
delay_ms(registers[index].pause_ms);
|
|
}
|
|
return true;
|
|
}
|
|
#endif
|