onewire.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. /********************************* (C) РОТЕК ***********************************
  2. * @module onewire
  3. * @file onewire.c
  4. * @version 1.0.0
  5. * @date XX.XX.XXXX
  6. * $brief DS18B20 temperature sensor driver
  7. *******************************************************************************
  8. * @history Version Author Comment
  9. * XX.XX.XXXX 1.0.0 Telenkov D.A. First release.
  10. *******************************************************************************
  11. */
  12. #include "common_config.h"
  13. #include "stm32f2xx.h"
  14. #include "stm32f2xx_dma.h"
  15. #include "usart.h"
  16. #include "gpio.h"
  17. #include "onewire.h"
  18. /* Force remove debug output */
  19. #undef DBG
  20. #define DBG if(0)
  21. /**
  22. * @brief Размер буфера для отправки данных датчикам
  23. */
  24. #define DALLAS_BUF_SIZE 8
  25. #define DALLAS_TIME_OUT 1500000
  26. #define DALLAS_0 0x00
  27. #define DALLAS_1 0xff
  28. #define DALLAS_R_1 0xff
  29. typedef struct {
  30. gpio_t ow_pin;
  31. USART_TypeDef *uart_addr;
  32. DMA_InitTypeDef DMA_InitStructureTx;
  33. DMA_InitTypeDef DMA_InitStructureRx;
  34. sensorBuf[DALLAS_BUF_SIZE];
  35. }ow_sensor_t;
  36. ow_sensor_t ow_sensor[MAX_T_SENSORS];
  37. #define T_SENSOR_ADD(sensor_num, ow_line, uart_addr) \
  38. do { \
  39. ow_sensor[sensor_num].ow_pin = ow_line; \
  40. ow_sensor[sensor_num].uart_addr = uart_addr; \
  41. } while (0);
  42. /**
  43. * @brief Настройка USART
  44. * @retval
  45. */
  46. void DALLAS_Init()
  47. {
  48. for (uint8_t i = 0; i < MAX_T_SENSORS; i ++) {
  49. #define XTSENSOR(sensor_num, ow_line, uart_addrm) T_SENSOR_ADD(sensor_num, ow_line, uart_addr);
  50. T_SENSOR_TABLE
  51. #undef XTSENSOR
  52. uart_hw_init(ow_sensor[i].uart_addr, DALLAS_SENSOR_UART_BAUD, DALLAS_SENSOR_UART_WORD_LEN,
  53. DALLAS_SENSOR_UART_PARITY, DALLAS_SENSOR_UART_STOP_BIT);
  54. }
  55. DALLAS_DmaInit();
  56. }
  57. /**
  58. * @brief Настройка DMA
  59. * @retval
  60. */
  61. void DALLAS_DmaInit(void)
  62. {
  63. DMA_Stream_TypeDef *RxDMA_Stream = NULL;
  64. DMA_Stream_TypeDef *TxDMA_Stream = NULL;
  65. uint32_t RxDMA_Channel = 0;
  66. uint32_t TxDMA_Channel = 0;
  67. for (uint8_t i = 0; i < MAX_T_SENSORS; i ++) {
  68. if (ow_sensor[i].uart_addr == USART1) {
  69. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);
  70. RxDMA_Stream = DMA1_Stream2;
  71. TxDMA_Stream = DMA1_Stream7;
  72. RxDMA_Channel = DMA_Channel_4;
  73. TxDMA_Channel = DMA_Channel_4;
  74. } else if (ow_sensor[i].uart_addr == USART2) {
  75. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
  76. RxDMA_Stream = DMA1_Stream5;
  77. TxDMA_Stream = DMA1_Stream6;
  78. RxDMA_Channel = DMA_Channel_4;
  79. TxDMA_Channel = DMA_Channel_4;
  80. } else if (ow_sensor[i].uart_addr == USART3) {
  81. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
  82. RxDMA_Stream = DMA1_Stream1;
  83. TxDMA_Stream = DMA1_Stream3;
  84. RxDMA_Channel = DMA_Channel_4;
  85. TxDMA_Channel = DMA_Channel_4;
  86. } else if (ow_sensor[i].uart_addr == UART4) {
  87. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
  88. RxDMA_Stream = DMA1_Stream2;
  89. TxDMA_Stream = DMA1_Stream4;
  90. RxDMA_Channel = DMA_Channel_4;
  91. TxDMA_Channel = DMA_Channel_4;
  92. } else if (ow_sensor[i].uart_addr == UART5) {
  93. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
  94. RxDMA_Stream = DMA1_Stream0;
  95. TxDMA_Stream = DMA1_Stream7;
  96. RxDMA_Channel = DMA_Channel_4;
  97. TxDMA_Channel = DMA_Channel_4;
  98. } else if (ow_sensor[i].uart_addr == USART6) {
  99. RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);
  100. RxDMA_Stream = DMA1_Stream1;
  101. TxDMA_Stream = DMA1_Stream6;
  102. RxDMA_Channel = DMA_Channel_5;
  103. TxDMA_Channel = DMA_Channel_5;
  104. }
  105. /* DMA на чтение */
  106. DMA_DeInit(RxDMA_Stream);
  107. ow_sensor[i].DMA_InitStructureRx.DMA_Channel = RxDMA_Stream;
  108. ow_sensor[i].DMA_InitStructureRx.DMA_PeripheralBaseAddr = (uint32_t)&ow_sensor[i].uart_addr->DR;
  109. ow_sensor[i].DMA_InitStructureRx.DMA_Memory0BaseAddr = (uint32_t)&ow_sensor[i].sensorBuf;
  110. ow_sensor[i].DMA_InitStructureRx.DMA_DIR = DMA_DIR_PeripheralToMemory;
  111. ow_sensor[i].DMA_InitStructureRx.DMA_BufferSize = DALLAS_BUF_SIZE;
  112. ow_sensor[i].DMA_InitStructureRx.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  113. ow_sensor[i].DMA_InitStructureRx.DMA_MemoryInc = DMA_MemoryInc_Enable;
  114. ow_sensor[i].DMA_InitStructureRx.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
  115. ow_sensor[i].DMA_InitStructureRx.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
  116. ow_sensor[i].DMA_InitStructureRx.DMA_Mode = DMA_Mode_Normal;
  117. ow_sensor[i].DMA_InitStructureRx.DMA_Priority = DMA_Priority_Medium;
  118. DMA_Init(RxDMA_Stream, &ow_sensor[i].DMA_InitStructureRx);
  119. /* DMA на запись */
  120. DMA_DeInit(TxDMA_Stream);
  121. ow_sensor[i].DMA_InitStructureTx.DMA_Channel = TxDMA_Stream;
  122. ow_sensor[i].DMA_InitStructureTx.DMA_PeripheralBaseAddr = &ow_sensor[i].uart_addr->DR;
  123. ow_sensor[i].DMA_InitStructureTx.DMA_Memory0BaseAddr = (uint32_t)&Dow_sensor[i].sensorBuf;
  124. ow_sensor[i].DMA_InitStructureTx.DMA_DIR = DMA_DIR_MemoryToPeripheral;
  125. ow_sensor[i].DMA_InitStructureTx.DMA_BufferSize = DALLAS_BUF_SIZE;
  126. ow_sensor[i].DMA_InitStructureTx.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  127. ow_sensor[i].DMA_InitStructureTx.DMA_MemoryInc = DMA_MemoryInc_Enable;
  128. ow_sensor[i].DMA_InitStructureTx.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
  129. ow_sensor[i].DMA_InitStructureTx.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
  130. ow_sensor[i].DMA_InitStructureTx.DMA_Mode = DMA_Mode_Normal;
  131. ow_sensor[i].DMA_InitStructureTx.DMA_Priority = DMA_Priority_Medium;
  132. DMA_Init(TxDMA_Stream, &ow_sensor[i].DMA_InitStructureTx.);
  133. }
  134. }
  135. /**
  136. * @brief Настроить TX пин как питающий выход
  137. * @retval
  138. */
  139. void DALLAS_OutSetAsPower(t_sensor_list_t sensor)
  140. {
  141. gpio_pindef_t *pin = &gpio_pins[ow_sensor[sensor].ow_pin];
  142. gpio_hw_config_pin(pin->port, pin->pin, GPIO_MODE_OUT_CFG | GPIO_TYPE_PP_CFG) |
  143. GPIO_SPEED_HIGH_CFG));
  144. gpio_set(id, GPIO_SET);
  145. }
  146. /**
  147. * @brief Настроить TX пин как передатчик
  148. * @retval
  149. */
  150. void DALLAS_OutSetAsTX(t_sensor_list_t sensor)
  151. {
  152. gpio_pindef_t *pin = &gpio_pins[ow_sensor[sensor].ow_pin];
  153. uint8_t af_n = (uint8_t)(pin->flags >> _GPIO_AF_SHIFT);
  154. gpio_connect_af(id, af_n);
  155. gpio_hw_config_pin(pin->port, pin->pin, GPIO_MODE_AF_CFG |
  156. GPIO_SPEED_HIGH_CFG);
  157. }
  158. /**
  159. * @brief Сброс и проверка наличия устройства на шине 1-wire
  160. * @retval
  161. */
  162. uint8_t DALLAS_SensorReset(t_sensor_list_t sensor)
  163. {
  164. uint8_t presence;
  165. uart_hw_init(ow_sensor[sensor].uart_addr, DALLAS_SENSOR_RST_UART_BAUD,
  166. DALLAS_SENSOR_UART_WORD_LEN, DALLAS_SENSOR_UART_PARITY, DALLAS_SENSOR_UART_STOP_BIT);
  167. // отправляем 0xf0 на скорости 9600
  168. USART_ClearFlag(ow_sensor[sensor].uart_addr, USART_FLAG_TC);
  169. USART_SendData(ow_sensor[sensor].uart_addr, 0xf0);
  170. while (USART_GetFlagStatus(ow_sensor[sensor].uart_addr, USART_FLAG_TC) == RESET) { }
  171. presence = USART_ReceiveData(ow_sensor[sensor].uart_addr);
  172. uart_hw_init(ow_sensor[sensor].uart_addr, DALLAS_SENSOR_UART_BAUD, DALLAS_SENSOR_UART_WORD_LEN,
  173. DALLAS_SENSOR_UART_PARITY, DALLAS_SENSOR_UART_STOP_BIT);
  174. if (presence != 0xf0) {
  175. DBG printf("DALLAS_SensorReset: DALLAS_OK\r\n");
  176. return DALLAS_OK;
  177. }
  178. DBG printf("DALLAS_SensorReset: DALLAS_NO_DEVICE\r\n");
  179. return DALLAS_NO_DEVICE;
  180. }
  181. /**
  182. * @brief Отправка и прием данных
  183. * @param sendReset посылать RESET в начале общения.
  184. * OW_SEND_RESET
  185. * OW_NO_RESET
  186. * @param *command - массив байт, отсылаемых в шину.
  187. * Если нужно чтение - отправляем OW_READ_SLOTH
  188. * @param cLen - длина буфера команд, столько байт отошлется в шину
  189. * @param *data - если требуется чтение, то ссылка на буфер для чтения
  190. * @param dLen - длина буфера для чтения. Прочитается не более этой длины
  191. * @param readStart - с какого символа передачи начинать чтение (нумеруются с 0)
  192. * можно указать OW_NO_READ, тогда можно не задавать data и dLen
  193. * @retval
  194. */
  195. uint8_t DALLAS_Send(t_sensor_list_t sensor, uint8_t sendReset, uint8_t *command,
  196. uint8_t cLen, uint8_t *data, uint8_t dLen, uint8_t readStart)
  197. {
  198. uint32_t timeOutCounter = 0;
  199. DMA_Stream_TypeDef *RxDMA_Stream = NULL;
  200. DMA_Stream_TypeDef *TxDMA_Stream = NULL;
  201. if (ow_sensor[sensor].uart_addr == USART1) {
  202. RxDMA_Stream = DMA1_Stream2;
  203. TxDMA_Stream = DMA1_Stream7;
  204. } else if (ow_sensor[sensor].uart_addr == USART2) {
  205. RxDMA_Stream = DMA1_Stream5;
  206. TxDMA_Stream = DMA1_Stream6;
  207. } else if (ow_sensor[sensor].uart_addr == USART3) {
  208. RxDMA_Stream = DMA1_Stream1;
  209. TxDMA_Stream = DMA1_Stream3;
  210. } else if (ow_sensor[sensor].uart_addr == UART4) {
  211. RxDMA_Stream = DMA1_Stream2;
  212. TxDMA_Stream = DMA1_Stream4;
  213. } else if (ow_sensor[sensor].uart_addr == UART5) {
  214. RxDMA_Stream = DMA1_Stream0;
  215. TxDMA_Stream = DMA1_Stream7;
  216. } else if (ow_sensor[sensor].uart_addr == USART6) {
  217. RxDMA_Stream = DMA1_Stream1;
  218. TxDMA_Stream = DMA1_Stream6;
  219. }
  220. /* если требуется сброс - сбрасываем и проверяем на наличие устройств */
  221. if (sendReset == DALLAS_SEND_RESET) {
  222. DBG printf("_Send: DALLAS_SEND_RESET\r\n");
  223. if (DALLAS_SensorReset(sensor) == DALLAS_NO_DEVICE) {
  224. DBG printf("_Send: DALLAS_NO_DEVICE\r\n");
  225. return DALLAS_NO_DEVICE;
  226. }
  227. }
  228. while (cLen > 0) {
  229. DALLAS_ToBits(*command, ow_sensor[sensor].sensorBuf);
  230. command++;
  231. cLen--;
  232. /* DMA на чтение */
  233. DMA_DeInit(RxDMA_Stream);
  234. DMA_Init(RxDMA_Stream, &ow_sensor[sensor].DMA_InitStructureRx);
  235. /* DMA на запись */
  236. DMA_DeInit(TxDMA_Stream);
  237. DMA_Init(TxDMA_Stream, &ow_sensor[sensor].DMA_InitStructureTx);
  238. /* старт цикла отправки */
  239. USART_ClearFlag(ow_sensor[sensor].uart_addr, USART_FLAG_RXNE | USART_FLAG_TC );
  240. USART_DMACmd(ow_sensor[sensor].uart_addr, USART_DMAReq_Tx | USART_DMAReq_Rx, ENABLE);
  241. DMA_Cmd(RxDMA_Stream, ENABLE);
  242. DMA_Cmd(TxDMA_Stream, ENABLE);
  243. while (DMA_GetFlagStatus(RxDMA_Stream, DMA_FLAG_TCIF5) == RESET) {
  244. timeOutCounter++;
  245. if (timeOutCounter > DALLAS_TIME_OUT) {
  246. DBG printf("_Send: return DALLAS_ERROR\r\n");
  247. return DALLAS_ERROR;
  248. }
  249. }
  250. DBG printf("timeOutCounter: %u\r\n", (unsigned int)timeOutCounter);
  251. /* отключаем DMA */
  252. DMA_Cmd(RxDMA_Stream, DISABLE);
  253. DMA_Cmd(TxDMA_Stream, DISABLE);
  254. USART_DMACmd(ow_sensor[sensor].uart_addr, USART_DMAReq_Tx | USART_DMAReq_Rx, DISABLE);
  255. /* если прочитанные данные кому-то нужны - выкинем их в буфер */
  256. DBG printf("_Send: readStart = %d, dLen = %d\r\n", readStart, dLen);
  257. if (readStart == 0 && dLen > 0) {
  258. *data = DALLAS_ToByte(ow_sensor[sensor].sensorBuf);
  259. data++;
  260. dLen--;
  261. DBG printf("_Send: Copy\r\n");
  262. } else {
  263. if (readStart != DALLAS_NO_READ) {
  264. readStart--;
  265. DBG printf("_Send: readStart--\r\n");
  266. }
  267. DBG printf("_Send: else state\r\n");
  268. }
  269. }
  270. DBG printf("_Send: return DALLAS_OK\r\n");
  271. return DALLAS_OK;
  272. }
  273. /**
  274. * @brief Функция преобразует один байт в восемь, для передачи через USART
  275. * @param ow_byte - байт, который надо преобразовать
  276. * @param ow_bits - ссылка на буфер, размером не менее 8 байт
  277. * @retval
  278. */
  279. void DALLAS_ToBits(uint8_t ow_byte, uint8_t *ow_bits)
  280. {
  281. uint8_t i;
  282. for (i = 0; i < 8; i++) {
  283. if (ow_byte & 0x01) {
  284. *ow_bits = DALLAS_1;
  285. } else {
  286. *ow_bits = DALLAS_0;
  287. }
  288. ow_bits++;
  289. ow_byte = ow_byte >> 1;
  290. }
  291. }
  292. /**
  293. * @brief Обратное преобразование - из того, что получено через USART опять собирается байт
  294. * @param ow_bits - ссылка на буфер, размером не менее 8 байт
  295. * @param ow_byte - собранный байт данных
  296. * @retval
  297. */
  298. uint8_t DALLAS_ToByte(uint8_t *ow_bits)
  299. {
  300. uint8_t ow_byte, i;
  301. ow_byte = 0;
  302. for (i = 0; i < 8; i++) {
  303. ow_byte = ow_byte >> 1;
  304. if (*ow_bits == DALLAS_R_1) {
  305. ow_byte |= 0x80;
  306. }
  307. ow_bits++;
  308. }
  309. return ow_byte;
  310. }