OneWireSensor.c 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  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. /**
  92. * @brief Чтение массива байт с шины One Wire
  93. *
  94. * @param instance Указатель на инстанс датчика
  95. * @param data Указатель на массив, куда будут записаны данные
  96. * @param len Количество байт
  97. */
  98. static void oneWire_readBytes(OneWireSensor* instance, uint8_t* data, size_t len) {
  99. for(size_t i = 0; i < len; i++) {
  100. data[i] = oneWire_read(instance);
  101. }
  102. }
  103. bool unitemp_OneWire_alloc(void* s, uint16_t* anotherValues) {
  104. Sensor* sensor = (Sensor*)s;
  105. OneWireSensor* instance = malloc(sizeof(OneWireSensor));
  106. if(instance == NULL) {
  107. FURI_LOG_E(APP_NAME, "Sensor %s instance allocation error", sensor->name);
  108. return false;
  109. }
  110. sensor->instance = instance;
  111. instance->gpio = unitemp_GPIO_getFromInt(anotherValues[0]);
  112. if(instance->gpio != NULL) {
  113. return true;
  114. }
  115. FURI_LOG_E(APP_NAME, "Sensor %s GPIO setting error", sensor->name);
  116. free(instance);
  117. return false;
  118. }
  119. static uint8_t onewire_CRC_update(uint8_t crc, uint8_t b) {
  120. // return pgm_read_byte(&onewire_crc_table[crc ^ b]);
  121. for(uint8_t p = 8; p; p--) {
  122. crc = ((crc ^ b) & 1) ? (crc >> 1) ^ 0b10001100 : (crc >> 1);
  123. b >>= 1;
  124. }
  125. return crc;
  126. }
  127. static bool onewire_CRC_check(uint8_t* data, size_t len) {
  128. uint8_t crc = 0;
  129. for(size_t i = 0; i < len; i++) {
  130. crc = onewire_CRC_update(crc, data[i]);
  131. }
  132. return !crc;
  133. }
  134. bool unitemp_OneWire_free(void* s) {
  135. Sensor* sensor = (Sensor*)s;
  136. free(sensor->instance);
  137. return true;
  138. }
  139. bool unitemp_OneWire_init(void* s) {
  140. OneWireSensor* instance = ((Sensor*)s)->instance;
  141. if(instance == NULL || instance->gpio == NULL) {
  142. FURI_LOG_E(APP_NAME, "Sensor pointer is null!");
  143. return false;
  144. }
  145. //Высокий уровень по умолчанию
  146. furi_hal_gpio_write(instance->gpio->pin, true);
  147. //Режим работы - OpenDrain, подтяжка включается на всякий случай
  148. furi_hal_gpio_init(
  149. instance->gpio->pin, //Порт FZ
  150. GpioModeOutputOpenDrain, //Режим работы - открытый сток
  151. GpioPullUp, //Принудительная подтяжка линии данных к питанию
  152. GpioSpeedVeryHigh); //Скорость работы - максимальная
  153. if(!oneWire_start(instance)) return false;
  154. //TODO: настройка разрядности преобразования
  155. return true;
  156. }
  157. bool unitemp_OneWire_deinit(void* s) {
  158. OneWireSensor* instance = ((Sensor*)s)->instance;
  159. if(instance == NULL || instance->gpio == NULL) return false;
  160. //Низкий уровень по умолчанию
  161. furi_hal_gpio_write(instance->gpio->pin, false);
  162. //Режим работы - аналог, подтяжка выключена
  163. furi_hal_gpio_init(
  164. instance->gpio->pin, //Порт FZ
  165. GpioModeAnalog, //Режим работы - аналог
  166. GpioPullNo, //Подтяжка выключена
  167. GpioSpeedLow); //Скорость работы - минимальная
  168. return true;
  169. }
  170. UnitempStatus unitemp_OneWire_update(void* s) {
  171. Sensor* sensor = (Sensor*)s;
  172. OneWireSensor* instance = ((Sensor*)s)->instance;
  173. if(sensor->status != UT_POLLING) {
  174. if(!oneWire_start(instance)) return UT_TIMEOUT;
  175. oneWire_write(instance, 0xCC); // skip ROM
  176. oneWire_write(instance, 0x44); // convert t
  177. return UT_POLLING;
  178. } else {
  179. if(!oneWire_start(instance)) return UT_TIMEOUT;
  180. oneWire_write(instance, 0xCC); // skip ROM
  181. oneWire_write(instance, 0xBE); // Read Scratch-pad
  182. uint8_t buff[9];
  183. oneWire_readBytes(instance, buff, 9);
  184. if(!onewire_CRC_check(buff, 9)) {
  185. return UT_BADCRC;
  186. }
  187. int16_t raw = buff[0] | ((int16_t)buff[1] << 8);
  188. sensor->temp = (float)raw / 16.0f;
  189. }
  190. return UT_OK;
  191. }