OneWireSensor.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. #include "OneWireSensor.h"
  2. #include <furi.h>
  3. #include <furi_hal.h>
  4. #include <one_wire/one_wire_host.h>
  5. const SensorType DS18B20 = {
  6. .typename = "DS18B20",
  7. .interface = &ONE_WIRE,
  8. .pollingInterval = 500,
  9. .allocator = unitemp_OneWire_alloc,
  10. .mem_releaser = unitemp_OneWire_free,
  11. .initializer = unitemp_OneWire_init,
  12. .deinitializer = unitemp_OneWire_deinit,
  13. .updater = unitemp_OneWire_update};
  14. /**
  15. * @brief Запуск общения с датчиком
  16. *
  17. * @param sensor Указатель на датчик
  18. * @return Истина если датчик отозвался
  19. */
  20. static bool oneWire_start(OneWireSensor* instance) {
  21. furi_hal_gpio_write(instance->gpio->pin, false);
  22. furi_delay_us(500);
  23. furi_hal_gpio_write(instance->gpio->pin, true);
  24. //Ожидание подъёма шины
  25. uint32_t t = furi_get_tick();
  26. while(!furi_hal_gpio_read(instance->gpio->pin)) {
  27. //Выход если шина не поднялась
  28. if(furi_get_tick() - t > 100) return false;
  29. }
  30. furi_delay_us(100);
  31. bool status = !furi_hal_gpio_read(instance->gpio->pin);
  32. furi_delay_us(400);
  33. return status;
  34. }
  35. /**
  36. * @brief Запись байта на шину One Wire
  37. *
  38. * @param instance Указатель на инстанс датчика
  39. * @param data Записываемый байт
  40. */
  41. static void oneWire_write(OneWireSensor* instance, uint8_t data) {
  42. for(int i = 0; i < 8; i++) {
  43. if((data & (1 << i)) != 0) {
  44. // write 1
  45. furi_hal_gpio_write(instance->gpio->pin, false);
  46. furi_delay_us(1);
  47. furi_hal_gpio_write(instance->gpio->pin, true);
  48. furi_delay_us(90);
  49. } else {
  50. furi_hal_gpio_write(instance->gpio->pin, false);
  51. furi_delay_us(90);
  52. furi_hal_gpio_write(instance->gpio->pin, true);
  53. //Ожидание подъёма шины
  54. uint32_t t = furi_get_tick();
  55. while(!furi_hal_gpio_read(instance->gpio->pin)) {
  56. //Выход если шина не поднялась
  57. if(furi_get_tick() - t > 100) return;
  58. }
  59. }
  60. }
  61. }
  62. /**
  63. * @brief Чтение бита на шине One Wire
  64. *
  65. * @param instance Указатель на инстанс датчика
  66. * @return Логический уровень бита
  67. */
  68. static bool oneWire_read_bit(OneWireSensor* instance) {
  69. furi_hal_gpio_write(instance->gpio->pin, false);
  70. furi_delay_us(2); // Длительность низкого уровня, минимум 1 мкс
  71. furi_hal_gpio_write(instance->gpio->pin, true);
  72. furi_delay_us(8); // Пауза до момента сэмплирования, всего не более 15 мкс
  73. bool r = furi_hal_gpio_read(instance->gpio->pin);
  74. furi_delay_us(80); // Ожидание до следующего тайм-слота, минимум 60 мкс с начала низкого уровня
  75. return r;
  76. }
  77. /**
  78. * @brief Чтение байта с шины One Wire
  79. *
  80. * @param instance Указатель на инстанс датчика
  81. * @return
  82. */
  83. static uint8_t oneWire_read(OneWireSensor* instance) {
  84. uint8_t r = 0;
  85. for(uint8_t p = 8; p; p--) {
  86. r >>= 1;
  87. if(oneWire_read_bit(instance)) r |= 0x80;
  88. }
  89. return r;
  90. }
  91. bool unitemp_OneWire_alloc(void* s, uint16_t* anotherValues) {
  92. Sensor* sensor = (Sensor*)s;
  93. OneWireSensor* instance = malloc(sizeof(OneWireSensor));
  94. if(instance == NULL) {
  95. FURI_LOG_E(APP_NAME, "Sensor %s instance allocation error", sensor->name);
  96. return false;
  97. }
  98. sensor->instance = instance;
  99. instance->gpio = unitemp_GPIO_getFromInt(anotherValues[0]);
  100. if(instance->gpio != NULL) {
  101. return true;
  102. }
  103. FURI_LOG_E(APP_NAME, "Sensor %s GPIO setting error", sensor->name);
  104. free(instance);
  105. return false;
  106. }
  107. bool unitemp_OneWire_free(void* s) {
  108. Sensor* sensor = (Sensor*)s;
  109. free(sensor->instance);
  110. return true;
  111. }
  112. bool unitemp_OneWire_init(void* s) {
  113. OneWireSensor* instance = ((Sensor*)s)->instance;
  114. if(instance == NULL || instance->gpio == NULL) {
  115. FURI_LOG_E(APP_NAME, "Sensor pointer is null!");
  116. return false;
  117. }
  118. //Высокий уровень по умолчанию
  119. furi_hal_gpio_write(instance->gpio->pin, true);
  120. //Режим работы - OpenDrain, подтяжка включается на всякий случай
  121. furi_hal_gpio_init(
  122. instance->gpio->pin, //Порт FZ
  123. GpioModeOutputOpenDrain, //Режим работы - открытый сток
  124. GpioPullUp, //Принудительная подтяжка линии данных к питанию
  125. GpioSpeedVeryHigh); //Скорость работы - максимальная
  126. //TODO: настройка разрядности преобразования
  127. return true;
  128. }
  129. bool unitemp_OneWire_deinit(void* s) {
  130. OneWireSensor* instance = ((Sensor*)s)->instance;
  131. if(instance == NULL || instance->gpio == NULL) return false;
  132. //Низкий уровень по умолчанию
  133. furi_hal_gpio_write(instance->gpio->pin, false);
  134. //Режим работы - аналог, подтяжка выключена
  135. furi_hal_gpio_init(
  136. instance->gpio->pin, //Порт FZ
  137. GpioModeAnalog, //Режим работы - аналог
  138. GpioPullNo, //Подтяжка выключена
  139. GpioSpeedLow); //Скорость работы - минимальная
  140. return true;
  141. }
  142. UnitempStatus unitemp_OneWire_update(void* s) {
  143. Sensor* sensor = (Sensor*)s;
  144. OneWireSensor* instance = ((Sensor*)s)->instance;
  145. oneWire_start(instance);
  146. oneWire_write(instance, 0xCC); // skip ROM
  147. oneWire_write(instance, 0x44); // convert t
  148. furi_delay_ms(800);
  149. oneWire_start(instance);
  150. oneWire_write(instance, 0xCC); // skip ROM
  151. oneWire_write(instance, 0xBE); // Read Scratch-pad
  152. uint8_t Temp_byte1 = oneWire_read(instance);
  153. uint8_t Temp_byte2 = oneWire_read(instance);
  154. uint16_t TEMP = ((uint16_t)Temp_byte2 << 8) | Temp_byte1;
  155. sensor->temp = (float)TEMP / 16;
  156. return UT_OK;
  157. }