ua_connection.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. #include "ua_util.h"
  2. #include "ua_connection.h"
  3. #include "ua_types_encoding_binary.h"
  4. #include "ua_securechannel.h"
  5. // max message size is 64k
  6. const UA_ConnectionConfig UA_ConnectionConfig_standard =
  7. {.protocolVersion = 0, .sendBufferSize = 65536, .recvBufferSize = 65536,
  8. .maxMessageSize = 65536, .maxChunkCount = 1};
  9. void UA_Connection_init(UA_Connection *connection) {
  10. connection->state = UA_CONNECTION_CLOSED;
  11. connection->localConf = (UA_ConnectionConfig){0,0,0,0,0};
  12. connection->remoteConf = (UA_ConnectionConfig){0,0,0,0,0};
  13. connection->channel = UA_NULL;
  14. connection->sockfd = 0;
  15. connection->handle = UA_NULL;
  16. UA_ByteString_init(&connection->incompleteMessage);
  17. connection->write = UA_NULL;
  18. connection->close = UA_NULL;
  19. connection->recv = UA_NULL;
  20. connection->getBuffer = UA_NULL;
  21. connection->releaseBuffer = UA_NULL;
  22. }
  23. void UA_Connection_deleteMembers(UA_Connection *connection) {
  24. UA_ByteString_deleteMembers(&connection->incompleteMessage);
  25. }
  26. UA_ByteString UA_Connection_completeMessages(UA_Connection *connection, UA_ByteString received)
  27. {
  28. if(received.length == -1)
  29. return received;
  30. /* concat the existing incomplete message with the new message */
  31. UA_ByteString current;
  32. if(connection->incompleteMessage.length < 0)
  33. current = received;
  34. else {
  35. current.data = UA_realloc(connection->incompleteMessage.data,
  36. connection->incompleteMessage.length + received.length);
  37. if(!current.data) {
  38. /* not enough memory */
  39. UA_ByteString_deleteMembers(&connection->incompleteMessage);
  40. connection->incompleteMessage.length = -1;
  41. UA_ByteString_deleteMembers(&received);
  42. received.length = -1;
  43. return received;
  44. }
  45. UA_memcpy(current.data + connection->incompleteMessage.length, received.data, received.length);
  46. current.length = connection->incompleteMessage.length + received.length;
  47. UA_ByteString_deleteMembers(&received);
  48. UA_ByteString_init(&connection->incompleteMessage);
  49. }
  50. /* find the first non-complete message */
  51. size_t end_pos = 0; // the end of the last complete message
  52. while(current.length - end_pos >= 16) {
  53. if(!(current.data[0] == 'M' && current.data[1] == 'S' && current.data[2] == 'G') &&
  54. !(current.data[0] == 'O' && current.data[1] == 'P' && current.data[2] == 'N') &&
  55. !(current.data[0] == 'H' && current.data[1] == 'E' && current.data[2] == 'L') &&
  56. !(current.data[0] == 'A' && current.data[1] == 'C' && current.data[2] == 'K') &&
  57. !(current.data[0] == 'C' && current.data[1] == 'L' && current.data[2] == 'O')) {
  58. current.length = end_pos; // throw the remaining bytestring away
  59. break;
  60. }
  61. UA_Int32 length = 0;
  62. size_t pos = end_pos + 4;
  63. UA_StatusCode retval = UA_Int32_decodeBinary(&current, &pos, &length);
  64. if(retval != UA_STATUSCODE_GOOD || length < 16 ||
  65. length > (UA_Int32)connection->localConf.maxMessageSize) {
  66. current.length = end_pos; // throw the remaining bytestring away
  67. break;
  68. }
  69. if(length + (UA_Int32)end_pos > current.length)
  70. break; // the message is incomplete
  71. end_pos += length;
  72. }
  73. if(current.length == 0) {
  74. /* throw everything away */
  75. UA_ByteString_deleteMembers(&current);
  76. current.length = -1;
  77. return current;
  78. }
  79. if(end_pos == 0) {
  80. /* no complete message in current */
  81. connection->incompleteMessage = current;
  82. UA_ByteString_init(&current);
  83. } else if(current.length != (UA_Int32)end_pos) {
  84. /* there is an incomplete message at the end of current */
  85. connection->incompleteMessage.data = UA_malloc(current.length - end_pos);
  86. if(connection->incompleteMessage.data) {
  87. UA_memcpy(connection->incompleteMessage.data, &current.data[end_pos], current.length - end_pos);
  88. connection->incompleteMessage.length = current.length - end_pos;
  89. }
  90. current.length = end_pos;
  91. }
  92. return current;
  93. }
  94. void UA_Connection_detachSecureChannel(UA_Connection *connection) {
  95. #ifdef UA_MULTITHREADING
  96. UA_SecureChannel *channel = connection->channel;
  97. if(channel)
  98. uatomic_cmpxchg(&channel->connection, connection, UA_NULL);
  99. uatomic_set(&connection->channel, UA_NULL);
  100. #else
  101. if(connection->channel)
  102. connection->channel->connection = UA_NULL;
  103. connection->channel = UA_NULL;
  104. #endif
  105. }
  106. void UA_Connection_attachSecureChannel(UA_Connection *connection, UA_SecureChannel *channel) {
  107. #ifdef UA_MULTITHREADING
  108. if(uatomic_cmpxchg(&channel->connection, UA_NULL, connection) == UA_NULL)
  109. uatomic_set(&connection->channel, channel);
  110. #else
  111. if(channel->connection != UA_NULL)
  112. return;
  113. channel->connection = connection;
  114. connection->channel = channel;
  115. #endif
  116. }