mini_gateway_master.c 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  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. ffx_status_t status = {0};
  82. for (;;) {
  83. // if (osMutexAcquire(mini_mutex, osWaitForever) == osOK) {
  84. bool is_power_on = false;
  85. for (int i = 0; i < sizeof(coil); i++) {
  86. if (coil[i] != 0) {
  87. HAL_GPIO_TogglePin(GPIOB, GPIO_PIN_13);
  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. if ((status.fan_speed == 0) || (status.fan_speed >= 6))
  101. status.power = 0;
  102. if((get_reg_value(POWER_REG_ADDRESS) == 0)){
  103. DEBUG_PRINTF("The INER_POWER_REG_ADDRESS_1 is 0.\r\n");
  104. }
  105. switch (data[1]) {
  106. case 1://制冷模式
  107. status.mode = 0;
  108. break;
  109. case 4://制热模式
  110. status.mode = 1;
  111. break;
  112. case 2://除湿模式
  113. status.mode = 2;
  114. break;
  115. case 3://送风模式
  116. status.mode = 3;
  117. break;
  118. default:
  119. break;
  120. }
  121. status.inner_num = inner_num;
  122. if ((status.power == INNER_POWER_ON) && (is_power_on == false)) {
  123. osDelay(100);
  124. is_power_on = true;
  125. set_reg_value(POWER_REG_ADDRESS, status.power);
  126. set_reg_value(WORK_MODE_REG_ADDRESS, status.mode);
  127. set_reg_value(FAN_SPEED_REG_ADDRESS, status.fan_speed);
  128. set_reg_value(SET_TEMP_REG_ADDRESS, status.set_temp);
  129. set_reg_value(INNER_ERROR1_REG_ADDRESS, 0);
  130. error_count = 0;
  131. }
  132. } else {
  133. error_count++;
  134. is_error_ffx[i]++;
  135. is_power_on = true;
  136. if ((status.power == INNER_POWER_OFF) && is_error_ffx[i] > 10) {
  137. DEBUG_PRINTF("The INER_POWER_REG_ADDRESS is_2 0.\r\n");
  138. set_reg_value(INNER_ERROR1_REG_ADDRESS, 1);
  139. set_reg_value(POWER_REG_ADDRESS, status.power);
  140. error_count = 0;
  141. is_error_ffx[i] = 0;
  142. }
  143. }
  144. // DEBUG_PRINTF("Read that the slave is turned on: %d.\r\n", inner_num);
  145. } else HAL_GPIO_WritePin(GPIOB, GPIO_PIN_13, GPIO_PIN_SET);
  146. osDelay(1000);
  147. }
  148. if ((status.power == INNER_POWER_OFF) && (is_power_on == false)) {
  149. DEBUG_PRINTF("The INER_POWER_REG_ADDRESS is_3 0.\r\n");
  150. set_reg_value(INNER_ERROR1_REG_ADDRESS, 0);
  151. set_reg_value(POWER_REG_ADDRESS, status.power);
  152. error_count = 0;
  153. }
  154. osDelay(1000);
  155. // osMutexRelease(mini_mutex);
  156. // }
  157. }
  158. }
  159. // 检测从机是否存在
  160. bool check_inner_slave_exist(uint8_t slave_addr) {
  161. nmbs_set_destination_rtu_address(&nmbs, slave_addr); // 设置从机地址
  162. bool result;
  163. osDelay(100);
  164. nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL_ADDRESS + slave_addr - 1, 1, &result);
  165. if (result) {
  166. return true;
  167. } else {
  168. return false;
  169. }
  170. }
  171. _Noreturn void gateway_poll_task(void *pv) {
  172. uint8_t count = 0;
  173. for (;;) {
  174. while(1) {
  175. if(check_comm())
  176. break;
  177. osDelay(1500);
  178. DEBUG_PRINTF("Waiting for gateway to start.\r\n");
  179. }
  180. nmbs_bitfield regs = {0};
  181. // if (osMutexAcquire(mini_mutex, osWaitForever) == osOK) {
  182. for (int i = 0; i <= INNER_SLAVE_ADDRESS_END; i++) {
  183. coil[i] = 0;
  184. }
  185. nmbs_error err1 = nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL_ADDRESS, 8, &regs[0]);
  186. // nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL2_ADDRESS, 8, &regs[1]);
  187. if (err1 == NMBS_ERROR_NONE) {
  188. set_reg_value(INNER_GROUP1_REG_ADDRESS, regs[0]);
  189. // set_reg_value(INNER_GROUP2_REG_ADDRESS, regs[1]);
  190. set_reg_value(INNER_ERROR1_REG_ADDRESS, 0);
  191. // set_reg_value(INNER_ERROR3_REG_ADDRESS, regs[3] << 2);
  192. // set_reg_value(INNER_ERROR4_REG_ADDRESS, regs[4] << 3);
  193. error_count = 0;
  194. uint8_t buf;
  195. for (int i = 0; i <= INNER_SLAVE_ADDRESS_END; i++) {
  196. buf = regs[0] << (7 - i);
  197. coil[i] = buf >> 7;
  198. }
  199. } else {
  200. nmbs_error err2 = nmbs_read_coils(&nmbs, MINI_GATEWAY_INNER_COIL_ADDRESS, 8, &regs[0]);
  201. osDelay(300);
  202. if (err2 == NMBS_ERROR_NONE) {
  203. uint8_t buf;
  204. for (int i = 0; i <= INNER_SLAVE_ADDRESS_END; i++) {
  205. buf = regs[0] << (7 - i);
  206. coil[i] = buf >> 7;
  207. }
  208. error_count = 0;
  209. set_reg_value(INNER_GROUP1_REG_ADDRESS, regs[0]);
  210. }
  211. }
  212. count--;
  213. // osMutexRelease(mini_mutex);
  214. // }
  215. osDelay(1000 * 60 * 1);
  216. }
  217. }
  218. void mini_gateway_master_init(void) {
  219. // mini_mutex = osMutexNew(NULL);
  220. nmbs_platform_conf platformConf;// 配置uart
  221. nmbs_platform_conf_create(&platformConf);
  222. platformConf.transport = NMBS_TRANSPORT_RTU;// RTU
  223. platformConf.read = &uart_read; // 读写函数1
  224. platformConf.write = &uart_write; // 读写函数
  225. nmbs_client_create(&nmbs, &platformConf);// 创建客户端
  226. nmbs_set_read_timeout(&nmbs, 300);
  227. nmbs_set_byte_timeout(&nmbs, 300);
  228. nmbs_set_destination_rtu_address(&nmbs, MINI_GATEWAY_SLAVE_ADDRESS);
  229. osThreadNew(gateway_poll_task, NULL, NULL);
  230. osThreadNew(mini_mater_task, NULL, NULL);
  231. }