| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311 |
- #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;
- 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 {
- __disable_irq();
- pin_init_opendrain_in_isr_ctx();
- error = OneWireSlaveError::NO_ERROR;
- if(show_presence()) {
- // TODO think about multiple command cycles
- receive_and_process_cmd();
- result =
- (error == OneWireSlaveError::NO_ERROR ||
- error == OneWireSlaveError::INCORRECT_ONEWIRE_CMD);
- } else {
- result = false;
- }
- pin_init_interrupt_in_isr_ctx();
- __enable_irq();
- }
- 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;
- }
- }
- }
|