mini_gateway_master.c 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. //
  2. // Created by DELL on 2024/10/16.
  3. //
  4. #include "main.h"
  5. #include <stdint.h>
  6. #include "mini_gateway_master.h"
  7. #include "usart.h"
  8. #include "modbus.h"
  9. #include "cmsis_os2.h"
  10. #include "modbus_slave.h"
  11. uint8_t error_count = 0;
  12. static nmbs_t nmbs;
  13. extern uint8_t slaves[SLAVE_ADDRESS_END]; // 保存已经存在的从站号
  14. //static osMutexId_t mini_mutex;
  15. extern uint8_t is_slave_exist;
  16. static uint8_t coil[INNER_SLAVE_ADDRESS_END] = {0};
  17. static int32_t uart_read(uint8_t *buf, uint16_t count, int32_t byte_timeout_ms,
  18. void *arg) {
  19. HAL_StatusTypeDef status = HAL_UART_Receive(&huart4, buf, count, byte_timeout_ms);
  20. if (status == HAL_OK) {
  21. return count;
  22. } else {
  23. return 0;
  24. }
  25. }
  26. static int32_t uart_write(const uint8_t *buf, uint16_t count, int32_t byte_timeout_ms,
  27. void *arg) {
  28. HAL_GPIO_WritePin(GPIOA, GPIO_PIN_15, GPIO_PIN_SET);
  29. HAL_UART_Transmit(&huart4, buf, count, byte_timeout_ms);
  30. HAL_GPIO_WritePin(GPIOA, GPIO_PIN_15, GPIO_PIN_RESET);
  31. return count;
  32. }
  33. void sync_ac_status(ffx_status_t *status) {
  34. uint16_t power;
  35. uint16_t mode;
  36. uint16_t fan_speed;
  37. uint16_t set_temp;
  38. if (status->power == 0x01) {
  39. power = 0xAA;
  40. fan_speed = status->fan_speed + 1;
  41. set_temp = status->set_temp;
  42. } else
  43. power = 0x55;
  44. mode = status->mode + 1;
  45. switch (status->mode) {
  46. case 0://制冷模式
  47. mode = status->mode + 1;
  48. break;
  49. case 1://制热模式
  50. mode = status->mode + 3;
  51. break;
  52. case 2://除湿模式
  53. mode = status->mode;
  54. break;
  55. case 3://送风模式
  56. mode = status->mode;
  57. break;
  58. case 4://加湿模式、送风
  59. mode = 3;
  60. break;
  61. default:
  62. break;
  63. }
  64. uint16_t regs[5] = {power, mode, set_temp * 10, fan_speed, 0};
  65. nmbs_error err2 = nmbs_write_multiple_registers(&nmbs, 102 + 25 * (status->inner_num - 1), 5, regs);
  66. if (err2 == NMBS_ERROR_NONE) {
  67. set_reg_value(INNER_ERROR1_REG_ADDRESS, 0);
  68. error_count = 0;
  69. } else {
  70. set_reg_value(INNER_ERROR1_REG_ADDRESS, 1);
  71. error_count++;
  72. }
  73. }
  74. static bool check_comm() {
  75. uint16_t data;
  76. nmbs_error err = nmbs_read_holding_registers(&nmbs, 102, 1, &data);
  77. return err == NMBS_ERROR_NONE;
  78. }
  79. _Noreturn void mini_mater_task(void *pv) {
  80. uint8_t is_error_ffx[8] = {0};
  81. for (;;) {
  82. // if (osMutexAcquire(mini_mutex, osWaitForever) == osOK) {
  83. bool is_power_on = false;
  84. ffx_status_t status = {0};
  85. for (int i = 0; i < sizeof(coil); i++) {
  86. HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_13);
  87. if (coil[i] != 0) {
  88. uint16_t data[10];
  89. uint8_t inner_num = i + 1;
  90. nmbs_error err = nmbs_read_holding_registers(&nmbs,
  91. (MINI_GATEWAY_INNER_POWER_ADDRESS + 25 * i),
  92. 4, &data[0]);
  93. if (err == NMBS_ERROR_NONE) {
  94. if (data[0] == 0xAA) {
  95. status.power = 1;
  96. status.fan_speed = data[3] - 2;
  97. status.set_temp = data[2] / 10;
  98. } else
  99. status.power = 0;
  100. switch (data[1]) {
  101. case 1://制冷模式
  102. status.mode = 0;
  103. break;
  104. case 4://制热模式
  105. status.mode = 1;
  106. break;
  107. case 2://除湿模式
  108. status.mode = 2;
  109. break;
  110. case 3://送风模式
  111. status.mode = 3;
  112. break;
  113. default:
  114. break;
  115. }
  116. status.inner_num = inner_num;
  117. if ((status.power == INNER_POWER_ON) && (is_power_on == false)) {
  118. osDelay(100);
  119. is_power_on = true;
  120. set_reg_value(POWER_REG_ADDRESS, status.power);
  121. set_reg_value(WORK_MODE_REG_ADDRESS, status.mode);
  122. set_reg_value(FAN_SPEED_REG_ADDRESS, status.fan_speed);
  123. set_reg_value(SET_TEMP_REG_ADDRESS, status.set_temp);
  124. set_reg_value(INNER_ERROR1_REG_ADDRESS, 0);
  125. error_count = 0;
  126. DEBUG_PRINTF("Read that the slave is turned on.\n");
  127. }
  128. } else {
  129. error_count++;
  130. is_error_ffx[i]++;
  131. is_power_on = true;
  132. if ((status.power == INNER_POWER_OFF) && is_error_ffx[i] > 3) {
  133. set_reg_value(INNER_ERROR1_REG_ADDRESS, 1);
  134. set_reg_value(POWER_REG_ADDRESS, status.power);
  135. error_count = 0;
  136. is_error_ffx[i] = 0;
  137. }
  138. }
  139. }
  140. osDelay(1000);
  141. }
  142. if ((status.power == INNER_POWER_OFF) && (is_power_on == false)) {
  143. set_reg_value(INNER_ERROR1_REG_ADDRESS, 0);
  144. set_reg_value(POWER_REG_ADDRESS, status.power);
  145. error_count = 0;
  146. }
  147. osDelay(1000);
  148. // osMutexRelease(mini_mutex);
  149. // }
  150. }
  151. }
  152. // 检测从机是否存在
  153. bool check_inner_slave_exist(uint8_t slave_addr) {
  154. nmbs_set_destination_rtu_address(&nmbs, slave_addr); // 设置从机地址
  155. bool result;
  156. osDelay(100);
  157. nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL_ADDRESS + slave_addr - 1, 1, &result);
  158. if (result) {
  159. return true;
  160. } else {
  161. return false;
  162. }
  163. }
  164. _Noreturn void gateway_poll_task(void *pv) {
  165. uint8_t count = 0;
  166. for (;;) {
  167. while(1) {
  168. if(check_comm())
  169. break;
  170. osDelay(1500);
  171. DEBUG_PRINTF("Waiting for gateway to start.\n");
  172. }
  173. nmbs_bitfield regs = {0};
  174. // if (osMutexAcquire(mini_mutex, osWaitForever) == osOK) {
  175. for (int i = 0; i <= INNER_SLAVE_ADDRESS_END; i++) {
  176. coil[i] = 0;
  177. }
  178. nmbs_error err1 = nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL_ADDRESS, 8, &regs[0]);
  179. nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL2_ADDRESS, 8, &regs[1]);
  180. if (err1 == NMBS_ERROR_NONE) {
  181. set_reg_value(INNER_GROUP1_REG_ADDRESS, regs[0]);
  182. set_reg_value(INNER_GROUP2_REG_ADDRESS, regs[1]);
  183. set_reg_value(INNER_ERROR1_REG_ADDRESS, 0);
  184. // set_reg_value(INNER_ERROR3_REG_ADDRESS, regs[3] << 2);
  185. // set_reg_value(INNER_ERROR4_REG_ADDRESS, regs[4] << 3);
  186. error_count = 0;
  187. uint8_t buf;
  188. for (int i = 0; i <= INNER_SLAVE_ADDRESS_END; i++) {
  189. buf = regs[0] << (7 - i);
  190. coil[i] = buf >> 7;
  191. }
  192. } else {
  193. nmbs_error err2 = nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL_ADDRESS, 8, &regs[0]);
  194. osDelay(300);
  195. if (err2 == NMBS_ERROR_NONE) {
  196. uint8_t buf;
  197. for (int i = 0; i <= INNER_SLAVE_ADDRESS_END; i++) {
  198. buf = regs[0] << (7 - i);
  199. coil[i] = buf >> 7;
  200. }
  201. error_count = 0;
  202. set_reg_value(INNER_GROUP1_REG_ADDRESS, regs[0]);
  203. }
  204. }
  205. count--;
  206. // osMutexRelease(mini_mutex);
  207. // }
  208. osDelay(1000 * 60 * 1);
  209. }
  210. }
  211. void mini_gateway_master_init(void) {
  212. // mini_mutex = osMutexNew(NULL);
  213. nmbs_platform_conf platformConf;// 配置uart
  214. nmbs_platform_conf_create(&platformConf);
  215. platformConf.transport = NMBS_TRANSPORT_RTU;// RTU
  216. platformConf.read = &uart_read; // 读写函数1
  217. platformConf.write = &uart_write; // 读写函数
  218. nmbs_client_create(&nmbs, &platformConf);// 创建客户端
  219. nmbs_set_read_timeout(&nmbs, 300);
  220. nmbs_set_byte_timeout(&nmbs, 300);
  221. nmbs_set_destination_rtu_address(&nmbs, MINI_GATEWAY_SLAVE_ADDRESS);
  222. osThreadNew(gateway_poll_task, NULL, NULL);
  223. osThreadNew(mini_mater_task, NULL, NULL);
  224. }