nbus_app.c 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. #include "nbus_app.h"
  2. #include <string.h>
  3. nBus_TypeDef nBus;
  4. static uint8_t const crc8x_table[] = {
  5. 0x00,0x07,0x0E,0x09,0x1C,0x1B,0x12,0x15,0x38,0x3F,0x36,0x31,0x24,0x23,0x2A,0x2D,
  6. 0x70,0x77,0x7E,0x79,0x6C,0x6B,0x62,0x65,0x48,0x4F,0x46,0x41,0x54,0x53,0x5A,0x5D,
  7. 0xE0,0xE7,0xEE,0xE9,0xFC,0xFB,0xF2,0xF5,0xD8,0xDF,0xD6,0xD1,0xC4,0xC3,0xCA,0xCD,
  8. 0x90,0x97,0x9E,0x99,0x8C,0x8B,0x82,0x85,0xA8,0xAF,0xA6,0xA1,0xB4,0xB3,0xBA,0xBD,
  9. 0xC7,0xC0,0xC9,0xCE,0xDB,0xDC,0xD5,0xD2,0xFF,0xF8,0xF1,0xF6,0xE3,0xE4,0xED,0xEA,
  10. 0xB7,0xB0,0xB9,0xBE,0xAB,0xAC,0xA5,0xA2,0x8F,0x88,0x81,0x86,0x93,0x94,0x9D,0x9A,
  11. 0x27,0x20,0x29,0x2E,0x3B,0x3C,0x35,0x32,0x1F,0x18,0x11,0x16,0x03,0x04,0x0D,0x0A,
  12. 0x57,0x50,0x59,0x5E,0x4B,0x4C,0x45,0x42,0x6F,0x68,0x61,0x66,0x73,0x74,0x7D,0x7A,
  13. 0x89,0x8E,0x87,0x80,0x95,0x92,0x9B,0x9C,0xB1,0xB6,0xBF,0xB8,0xAD,0xAA,0xA3,0xA4,
  14. 0xF9,0xFE,0xF7,0xF0,0xE5,0xE2,0xEB,0xEC,0xC1,0xC6,0xCF,0xC8,0xDD,0xDA,0xD3,0xD4,
  15. 0x69,0x6E,0x67,0x60,0x75,0x72,0x7B,0x7C,0x51,0x56,0x5F,0x58,0x4D,0x4A,0x43,0x44,
  16. 0x19,0x1E,0x17,0x10,0x05,0x02,0x0B,0x0C,0x21,0x26,0x2F,0x28,0x3D,0x3A,0x33,0x34,
  17. 0x4E,0x49,0x40,0x47,0x52,0x55,0x5C,0x5B,0x76,0x71,0x78,0x7F,0x6A,0x6D,0x64,0x63,
  18. 0x3E,0x39,0x30,0x37,0x22,0x25,0x2C,0x2B,0x06,0x01,0x08,0x0F,0x1A,0x1D,0x14,0x13,
  19. 0xAE,0xA9,0xA0,0xA7,0xB2,0xB5,0xBC,0xBB,0x96,0x91,0x98,0x9F,0x8A,0x8D,0x84,0x83,
  20. 0xDE,0xD9,0xD0,0xD7,0xC2,0xC5,0xCC,0xCB,0xE6,0xE1,0xE8,0xEF,0xFA,0xFD,0xF4,0xF3
  21. };
  22. /* -------------------------------------------------------- */
  23. /* ------------------ PRIVATE FUNCTIONS-------------------- */
  24. /* -------------------------------------------------------- */
  25. static uint8_t crc8x_fast(void const *mem, uint16_t len) {
  26. uint8_t crc = CRC8_INIT_VALUE;
  27. uint8_t const *data = (uint8_t*)mem;
  28. if (data == NULL)
  29. return 0xff;
  30. crc &= 0xff;
  31. while (len--)
  32. crc = crc8x_table[crc ^ *data++];
  33. return crc;
  34. }
  35. inline static void receivePacket(void){
  36. nBus.hw_platform->uart_receive(nBus.rx_buffer, 64);
  37. }
  38. inline static void send_response(){
  39. if(nBus.send_response == SEND_RESPONSE) {
  40. nBus.tx_buffer[0] -= 1; // prvý bajt sa nepočíta
  41. nBus.hw_platform->uart_transmit(nBus.tx_buffer, nBus.tx_length);
  42. }
  43. nBus.hw_platform->led_off();
  44. }
  45. #if MODULE_MASTER == 1
  46. inline static void receive_slave_response(){
  47. }
  48. #endif
  49. static nBusCommandType_t get_request_type(){
  50. nBus.addressModule = nBus.rx_buffer[0];
  51. nBus.sensorInfo = *(const nBus_sensorByte_t*)&nBus.rx_buffer[1];
  52. nBus.function_code = *(const nBus_functionCode_t*)&nBus.rx_buffer[2];
  53. if(nBus.sensorInfo.address != 0 && nBus.addressModule !=0 ){
  54. nBus.request_type = UNICAST_TO_SENSOR;
  55. return UNICAST_TO_SENSOR;
  56. }
  57. if(nBus.sensorInfo.address == 0 && nBus.addressModule !=0 ){
  58. nBus.request_type = UNICAST_TO_MODULE;
  59. return UNICAST_TO_MODULE;
  60. }
  61. if(nBus.sensorInfo.address != 0 && nBus.addressModule == 0 ){
  62. nBus.request_type = BROADCAST_SPECIFIC_SENSORS;
  63. return BROADCAST_SPECIFIC_SENSORS;
  64. }
  65. nBus.request_type = BROADCAST_GLOBAL;
  66. return BROADCAST_GLOBAL;
  67. }
  68. static void process_request(){
  69. nBusCommandType_t request_type = get_request_type();
  70. nBus.send_response = SEND_RESPONSE;
  71. nBus.tx_buffer[0] = 0;
  72. nBus.tx_buffer[1] = nBus.rx_buffer[0]; // Module address
  73. nBus.tx_buffer[2] = nBus.rx_buffer[1]; // Sensor address
  74. uint8_t crcC = crc8x_fast(nBus.rx_buffer, nBus.rx_length-1);
  75. if(crcC != nBus.rx_buffer[nBus.rx_length-1]){
  76. nBus.send_response = NO_RESPONSE;
  77. return;
  78. }
  79. #if MODULE_SLAVE == 1
  80. // spracovanie broadcast komunikacie
  81. if((request_type == BROADCAST_SPECIFIC_SENSORS || request_type == BROADCAST_GLOBAL)) {
  82. nbus_slave_broadcast(&nBus, request_type);
  83. return;
  84. }
  85. #endif
  86. // paket nie je adresovany tomuto modulu
  87. if(nBus.addressModule != MODULE_ADDRESS) {
  88. nBus.send_response = NO_RESPONSE;
  89. return;
  90. }
  91. nBus.function_code.error = 0;
  92. nBus.tx_length = META_SIZE;
  93. #if MODULE_SLAVE == 1
  94. if(nBus.function_code.notReadWrite == REQUEST_GET) {
  95. if(request_type == UNICAST_TO_SENSOR) {
  96. nbus_slave_unicastToSensorGet(&nBus);
  97. }
  98. if(request_type == UNICAST_TO_MODULE) {
  99. nbus_slave_unicastToModuleGet(&nBus);
  100. }
  101. } else {
  102. // else request is REQUEST_SET
  103. if(request_type == UNICAST_TO_SENSOR) {
  104. nbus_slave_unicastToSensorSet(&nBus);
  105. }
  106. if(request_type == UNICAST_TO_MODULE) {
  107. nbus_slave_unicastToModuleSet(&nBus);
  108. }
  109. }
  110. nBus.tx_buffer[3] = *(uint8_t*)&nBus.function_code;
  111. nBus.tx_buffer[nBus.tx_length-1] = crc8x_fast(&nBus.tx_buffer[1], nBus.tx_length-2);
  112. nBus.tx_buffer[0] = nBus.tx_length;
  113. #endif
  114. #if MODULE_MASTER == 1
  115. if(nBus.function_code.notReadWrite == REQUEST_GET) {
  116. nbus_master_unicastToModuleGet(&nBus);
  117. } else {
  118. nbus_master_unicastToModuleSet(&nBus);
  119. }
  120. #endif
  121. }
  122. static void nbus_blink_LED(uint8_t delay) {
  123. nBus.hw_platform->led_on();
  124. nBus.hw_platform->delay_ms(delay);
  125. nBus.hw_platform->led_off();
  126. nBus.hw_platform->delay_ms(delay);
  127. nBus.hw_platform->led_on();
  128. nBus.hw_platform->delay_ms(delay);
  129. nBus.hw_platform->led_off();
  130. nBus.hw_platform->delay_ms(delay);
  131. }
  132. /* -------------------------------------------------------- */
  133. /* ----------------------- CALLBACKS----------------------- */
  134. /* -------------------------------------------------------- */
  135. /**
  136. * @brief UART receive complete.
  137. * This callback have to valled from application, when RX data is ready.
  138. * @param int size Size of received packet
  139. * Received packet is located in uBus.rx_buffer
  140. */
  141. void nbus_cb_UART_RX(int size){
  142. nBus.rx_length = size;
  143. nBus.uart_state = UART_RX_RECEIVED;
  144. }
  145. /* -------------------------------------------------------- */
  146. /* ------------------ PUBLIC FUNCTIONS-------------------- */
  147. /* -------------------------------------------------------- */
  148. /**
  149. * @brief Initialize of nBus stack
  150. * @param interface Driver for module application.
  151. * Every application have to implement base functions as Init, Reset, GetData, SetData, GetParam, SetParam, .... @see nBusAppInterface_t
  152. * These are application dependent functions.
  153. * @param hw Application level implementation of base functions (uart_receive, uart_transmit, led_on/off/toggle, ...).
  154. * These are MCU dependent functions.
  155. */
  156. void nbus_init(nBusAppInterface_t *interface, nBusPlatformInterface_t *hw){
  157. nBus.hw_platform = hw;
  158. nBus.rx_length = 0;
  159. nBus.data_timebase = 0;
  160. nBus.measure_active = MEASURE_STOPPED;
  161. nBus.uart_state = UART_RX_WAIT;
  162. nBus.interface = interface;
  163. receivePacket();
  164. }
  165. /**
  166. * @brief Initialize concrete application, if it is needed.
  167. * This function call "init()" function from nBusPlatformInterface_t, and this function is implemented in concrete application.
  168. * @param hw_interface Pointer to hardware definition structure. In STM32 it can be hSPI, hUART, hADC, ...
  169. * @param hw_config configuration for hardware structure object.
  170. */
  171. void nbus_init_app(void *hw_interface, void *hw_config){
  172. nBus.interface->init(hw_interface, hw_config);
  173. }
  174. /**
  175. * @brief Init memory driver for storing module/sensor parameters.
  176. * Implementation of memory driver is independent from nBus. To provide nonvolatile parameters store, is nneded implement this driver.
  177. * @param memDriver Memory driver for bas operation: read/write 1/2/4 byte from/to memory
  178. * @param capacity TBD
  179. * @todo imeplement capacity parameter
  180. */
  181. void nbus_init_memory_driver(nBus_MemoryDriver *memDriver, uint16_t capacity){
  182. nbus_memory_init(memDriver);
  183. nBus.memoryInterface = getnbusMemoryInterface();
  184. nbus_blink_LED(100);
  185. }
  186. /**
  187. * @brief Run protocol stack.
  188. * This is infinity protocol loop.
  189. * When stack is started (or reset) it blink as boot sign.
  190. *
  191. * Content of loop:
  192. * - if data was received: process request. Determine addressee and type of request
  193. * - execute request or discard request
  194. * - send response (only for unicast requests)
  195. *
  196. * In each iteration, the state of application flags are performed. The state of application is noticed from loop_callback() (if it is defined).
  197. * Meaning of return values:
  198. * - 0 - No event occurs,
  199. * - 1 - data from connected sensors are prepared, it will be read with read() function automatically.
  200. * - 2 - communication error/timeout. The reception of packed process will be reset.
  201. */
  202. void nbus_stack(void){
  203. nbus_blink_LED(50);
  204. while(1){
  205. if(nBus.uart_state == UART_RX_RECEIVED){
  206. process_request();
  207. nBus.uart_state = UART_RX_WAIT;
  208. send_response();
  209. #if MODULE_MASTER == 1
  210. receive_slave_response();
  211. #endif
  212. }
  213. if (nBus.hw_platform->loop_callback != NULL) {
  214. switch (nBus.hw_platform->loop_callback()){
  215. case 1:
  216. nBus.interface->read();
  217. break;
  218. case 2:
  219. nBus.uart_state = UART_RX_WAIT;
  220. break;
  221. }
  222. }
  223. }
  224. }