[FL-85][FL-446][FL-720] Dallas key blanks and OneWire lib rework (#313)
* sepate one wire class * TM2004 writer * app mode write ds1990 * test another blanks protocol * new ibutton slave * one wire states * tim1 capture compare and update interrupts * interrupt mgr, new timers IRQ * discard HAL_TIM_PeriodElapsedCallback from main * add exti_14 line * add external interrupt callback * use int mgr in input * better interrupt managment * add interrupt callback enable and disable fns * properly init app * changed timings * rename one wire classes * use new owb classes * properly remove interrupts * new blanks writer * remove unused tests * new core includes * extern c guard * fix api_interrupt_remove usage * remove debug info, new way to detect blanks writing * remove copy constructor * change keys template * fix app sources recipe
This commit is contained in:
312
lib/onewire/one_wire_slave.cpp
Normal file
312
lib/onewire/one_wire_slave.cpp
Normal file
@@ -0,0 +1,312 @@
|
||||
#include "one_wire_slave.h"
|
||||
#include "callback-connector.h"
|
||||
#include "main.h"
|
||||
#include "one_wire_device.h"
|
||||
|
||||
#define OWET OneWireEmulateTiming
|
||||
|
||||
void OneWireSlave::start(void) {
|
||||
// add exti interrupt
|
||||
api_interrupt_add(exti_cb, InterruptTypeExternalInterrupt, this);
|
||||
|
||||
// init gpio
|
||||
gpio_init(one_wire_pin_record, GpioModeInterruptRiseFall);
|
||||
pin_set_float();
|
||||
|
||||
// init instructions per us count
|
||||
__instructions_per_us = (SystemCoreClock / 1000000.0f);
|
||||
}
|
||||
|
||||
void OneWireSlave::stop(void) {
|
||||
// deinit gpio
|
||||
gpio_init_ex(one_wire_pin_record, GpioModeInput, GpioPullNo, GpioSpeedLow);
|
||||
|
||||
// remove exti interrupt
|
||||
api_interrupt_remove(exti_cb, InterruptTypeExternalInterrupt);
|
||||
|
||||
// deattach devices
|
||||
deattach();
|
||||
}
|
||||
|
||||
OneWireSlave::OneWireSlave(const GpioPin* pin) {
|
||||
one_wire_pin_record = pin;
|
||||
exti_cb = cbc::obtain_connector(this, &OneWireSlave::exti_callback);
|
||||
}
|
||||
|
||||
OneWireSlave::~OneWireSlave() {
|
||||
stop();
|
||||
}
|
||||
|
||||
void OneWireSlave::attach(OneWireDevice* attached_device) {
|
||||
device = attached_device;
|
||||
device->attach(this);
|
||||
}
|
||||
|
||||
void OneWireSlave::deattach(void) {
|
||||
device = nullptr;
|
||||
device->deattach();
|
||||
}
|
||||
|
||||
void OneWireSlave::set_result_callback(OneWireSlaveResultCallback result_cb, void* ctx) {
|
||||
this->result_cb = result_cb;
|
||||
this->result_cb_ctx = ctx;
|
||||
}
|
||||
|
||||
void OneWireSlave::pin_set_float() {
|
||||
gpio_write(one_wire_pin_record, true);
|
||||
}
|
||||
|
||||
void OneWireSlave::pin_set_low() {
|
||||
gpio_write(one_wire_pin_record, false);
|
||||
}
|
||||
|
||||
void OneWireSlave::pin_init_interrupt_in_isr_ctx(void) {
|
||||
hal_gpio_init(one_wire_pin_record, GpioModeInterruptRiseFall, GpioPullNo, GpioSpeedLow);
|
||||
__HAL_GPIO_EXTI_CLEAR_IT(one_wire_pin_record->pin);
|
||||
}
|
||||
|
||||
void OneWireSlave::pin_init_opendrain_in_isr_ctx(void) {
|
||||
hal_gpio_init(one_wire_pin_record, GpioModeOutputOpenDrain, GpioPullNo, GpioSpeedLow);
|
||||
__HAL_GPIO_EXTI_CLEAR_IT(one_wire_pin_record->pin);
|
||||
}
|
||||
|
||||
OneWiteTimeType OneWireSlave::wait_while_gpio_is(OneWiteTimeType time, const bool pin_value) {
|
||||
uint32_t start = DWT->CYCCNT;
|
||||
uint32_t time_ticks = time * __instructions_per_us;
|
||||
uint32_t time_captured;
|
||||
|
||||
do {
|
||||
time_captured = DWT->CYCCNT;
|
||||
if(gpio_read(one_wire_pin_record) != pin_value) {
|
||||
OneWiteTimeType remaining_time = time_ticks - (time_captured - start);
|
||||
remaining_time /= __instructions_per_us;
|
||||
return remaining_time;
|
||||
}
|
||||
} while((time_captured - start) < time_ticks);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool OneWireSlave::show_presence(void) {
|
||||
// wait while master delay presence check
|
||||
wait_while_gpio_is(OWET::PRESENCE_TIMEOUT, true);
|
||||
|
||||
// show presence
|
||||
pin_set_low();
|
||||
delay_us(OWET::PRESENCE_MIN);
|
||||
pin_set_float();
|
||||
|
||||
// somebody also can show presence
|
||||
const OneWiteTimeType wait_low_time = OWET::PRESENCE_MAX - OWET::PRESENCE_MIN;
|
||||
|
||||
// so we will wait
|
||||
if(wait_while_gpio_is(wait_low_time, false) == 0) {
|
||||
error = OneWireSlaveError::PRESENCE_LOW_ON_LINE;
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool OneWireSlave::receive_bit(void) {
|
||||
// wait while bus is low
|
||||
OneWiteTimeType time = OWET::SLOT_MAX;
|
||||
time = wait_while_gpio_is(time, false);
|
||||
if(time == 0) {
|
||||
error = OneWireSlaveError::RESET_IN_PROGRESS;
|
||||
return false;
|
||||
}
|
||||
|
||||
// wait while bus is high
|
||||
time = OWET::MSG_HIGH_TIMEOUT;
|
||||
time = wait_while_gpio_is(time, true);
|
||||
if(time == 0) {
|
||||
error = OneWireSlaveError::AWAIT_TIMESLOT_TIMEOUT_HIGH;
|
||||
return false;
|
||||
}
|
||||
|
||||
// wait a time of zero
|
||||
time = OWET::READ_MIN;
|
||||
time = wait_while_gpio_is(time, false);
|
||||
|
||||
return (time > 0);
|
||||
}
|
||||
|
||||
bool OneWireSlave::send_bit(bool value) {
|
||||
const bool write_zero = !value;
|
||||
|
||||
// wait while bus is low
|
||||
OneWiteTimeType time = OWET::SLOT_MAX;
|
||||
time = wait_while_gpio_is(time, false);
|
||||
if(time == 0) {
|
||||
error = OneWireSlaveError::RESET_IN_PROGRESS;
|
||||
return false;
|
||||
}
|
||||
|
||||
// wait while bus is high
|
||||
time = OWET::MSG_HIGH_TIMEOUT;
|
||||
time = wait_while_gpio_is(time, true);
|
||||
if(time == 0) {
|
||||
error = OneWireSlaveError::AWAIT_TIMESLOT_TIMEOUT_HIGH;
|
||||
return false;
|
||||
}
|
||||
|
||||
// choose write time
|
||||
if(write_zero) {
|
||||
pin_set_low();
|
||||
time = OWET::WRITE_ZERO;
|
||||
} else {
|
||||
time = OWET::READ_MAX;
|
||||
}
|
||||
|
||||
// hold line for ZERO or ONE time
|
||||
delay_us(time);
|
||||
pin_set_float();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool OneWireSlave::send(const uint8_t* address, const uint8_t data_length) {
|
||||
uint8_t bytes_sent = 0;
|
||||
|
||||
pin_set_float();
|
||||
|
||||
// bytes loop
|
||||
for(; bytes_sent < data_length; ++bytes_sent) {
|
||||
const uint8_t data_byte = address[bytes_sent];
|
||||
|
||||
// bit loop
|
||||
for(uint8_t bit_mask = 0x01; bit_mask != 0; bit_mask <<= 1) {
|
||||
if(!send_bit(static_cast<bool>(bit_mask & data_byte))) {
|
||||
// if we cannot send first bit
|
||||
if((bit_mask == 0x01) && (error == OneWireSlaveError::AWAIT_TIMESLOT_TIMEOUT_HIGH))
|
||||
error = OneWireSlaveError::FIRST_BIT_OF_BYTE_TIMEOUT;
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
bool OneWireSlave::receive(uint8_t* data, const uint8_t data_length) {
|
||||
uint8_t bytes_received = 0;
|
||||
|
||||
pin_set_float();
|
||||
|
||||
for(; bytes_received < data_length; ++bytes_received) {
|
||||
uint8_t value = 0;
|
||||
|
||||
for(uint8_t bit_mask = 0x01; bit_mask != 0; bit_mask <<= 1) {
|
||||
if(receive_bit()) value |= bit_mask;
|
||||
}
|
||||
|
||||
data[bytes_received] = value;
|
||||
}
|
||||
return (bytes_received != data_length);
|
||||
}
|
||||
|
||||
void OneWireSlave::cmd_search_rom(void) {
|
||||
const uint8_t key_bytes = 8;
|
||||
uint8_t* key = device->id_storage;
|
||||
|
||||
for(uint8_t i = 0; i < key_bytes; i++) {
|
||||
uint8_t key_byte = key[i];
|
||||
|
||||
for(uint8_t j = 0; j < 8; j++) {
|
||||
bool bit = (key_byte >> j) & 0x01;
|
||||
|
||||
if(!send_bit(bit)) return;
|
||||
if(!send_bit(!bit)) return;
|
||||
|
||||
const bool bit_recv = receive_bit();
|
||||
if(error != OneWireSlaveError::NO_ERROR) return;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool OneWireSlave::receive_and_process_cmd(void) {
|
||||
uint8_t cmd;
|
||||
receive(&cmd, 1);
|
||||
|
||||
if(error == OneWireSlaveError::RESET_IN_PROGRESS) return true;
|
||||
if(error != OneWireSlaveError::NO_ERROR) return false;
|
||||
|
||||
switch(cmd) {
|
||||
case 0xF0:
|
||||
// SEARCH ROM
|
||||
cmd_search_rom();
|
||||
return true;
|
||||
|
||||
case 0x33:
|
||||
// READ ROM
|
||||
device->send_id();
|
||||
return false;
|
||||
|
||||
default: // Unknown command
|
||||
error = OneWireSlaveError::INCORRECT_ONEWIRE_CMD;
|
||||
}
|
||||
|
||||
if(error == OneWireSlaveError::RESET_IN_PROGRESS) return true;
|
||||
return (error == OneWireSlaveError::NO_ERROR);
|
||||
}
|
||||
|
||||
bool OneWireSlave::bus_start(void) {
|
||||
bool result = true;
|
||||
|
||||
if(device == nullptr) {
|
||||
result = false;
|
||||
} else {
|
||||
pin_init_opendrain_in_isr_ctx();
|
||||
error = OneWireSlaveError::NO_ERROR;
|
||||
|
||||
if(show_presence()) {
|
||||
__disable_irq();
|
||||
|
||||
// TODO think about multiple command cycles
|
||||
bool return_to_reset = receive_and_process_cmd();
|
||||
result =
|
||||
(error == OneWireSlaveError::NO_ERROR ||
|
||||
error == OneWireSlaveError::INCORRECT_ONEWIRE_CMD);
|
||||
|
||||
__enable_irq();
|
||||
} else {
|
||||
result = false;
|
||||
}
|
||||
|
||||
pin_init_interrupt_in_isr_ctx();
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
void OneWireSlave::exti_callback(void* _pin, void* _ctx) {
|
||||
// interrupt manager get us pin constant, so...
|
||||
uint32_t pin = (uint32_t)_pin;
|
||||
OneWireSlave* _this = static_cast<OneWireSlave*>(_ctx);
|
||||
|
||||
if(pin == _this->one_wire_pin_record->pin) {
|
||||
volatile bool input_state = gpio_read(_this->one_wire_pin_record);
|
||||
static uint32_t pulse_start = 0;
|
||||
|
||||
if(input_state) {
|
||||
uint32_t pulse_length = (DWT->CYCCNT - pulse_start) / __instructions_per_us;
|
||||
if(pulse_length >= OWET::RESET_MIN) {
|
||||
if(pulse_length <= OWET::RESET_MAX) {
|
||||
// reset cycle ok
|
||||
bool result = _this->bus_start();
|
||||
|
||||
if(_this->result_cb != nullptr) {
|
||||
_this->result_cb(result, _this->result_cb_ctx);
|
||||
}
|
||||
} else {
|
||||
error = OneWireSlaveError::VERY_LONG_RESET;
|
||||
}
|
||||
} else {
|
||||
error = OneWireSlaveError::VERY_SHORT_RESET;
|
||||
}
|
||||
} else {
|
||||
//FALL event
|
||||
pulse_start = DWT->CYCCNT;
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user