ua_securechannel.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. #include "ua_util.h"
  2. #include "ua_securechannel.h"
  3. #include "ua_statuscodes.h"
  4. #include "ua_types_encoding_binary.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. UA_ByteString UA_Connection_completeMessages(UA_Connection *connection, UA_ByteString received)
  10. {
  11. if(received.length == -1){
  12. return received;
  13. }
  14. /* concat received to the incomplete message we have */
  15. UA_ByteString current;
  16. if(connection->incompleteMessage.length < 0)
  17. current = received;
  18. else {
  19. UA_Byte *longer = UA_realloc(connection->incompleteMessage.data,
  20. connection->incompleteMessage.length + received.length);
  21. if(!longer) {
  22. UA_ByteString_deleteMembers(&received);
  23. UA_ByteString_deleteMembers(&connection->incompleteMessage);
  24. connection->incompleteMessage.length = -1;
  25. received.length = -1;
  26. return received;
  27. }
  28. UA_memcpy(longer+connection->incompleteMessage.length, received.data, received.length);
  29. UA_ByteString_deleteMembers(&received);
  30. connection->incompleteMessage.data = longer;
  31. connection->incompleteMessage.length = connection->incompleteMessage.length + received.length;
  32. current.data = connection->incompleteMessage.data;
  33. current.length = connection->incompleteMessage.length;
  34. }
  35. /* find the first non-complete message */
  36. size_t end_pos = 0;
  37. while(current.length - end_pos >= 32) {
  38. if(!(current.data[0] == 'M' && current.data[1] == 'S' && current.data[2] == 'G') &&
  39. !(current.data[0] == 'O' && current.data[1] == 'P' && current.data[2] == 'N') &&
  40. !(current.data[0] == 'H' && current.data[1] == 'E' && current.data[2] == 'L') &&
  41. !(current.data[0] == 'A' && current.data[1] == 'C' && current.data[2] == 'K') &&
  42. !(current.data[0] == 'C' && current.data[1] == 'L' && current.data[2] == 'O')) {
  43. current.length = end_pos; // throw the remaining bytestring away
  44. break;
  45. }
  46. UA_Int32 length = 0;
  47. size_t pos = end_pos + 4;
  48. if((UA_Int32)pos<=current.length)
  49. UA_Int32_decodeBinary(&current, &pos, &length);
  50. if(length < 32 || length > (UA_Int32)connection->localConf.maxMessageSize) {
  51. current.length = end_pos; // throw the remaining bytestring away
  52. break;
  53. }
  54. if(length + (UA_Int32)end_pos > current.length)
  55. break; // the message is not complete
  56. end_pos += length;
  57. }
  58. if(current.length == 0) { //flag was set to through everything away
  59. UA_String_deleteMembers(&current);
  60. current.length = -1;
  61. return current;
  62. }
  63. /* return all complete messages, retain the (last) incomplete one */
  64. //the message is not ready - retain it
  65. if(current.length - end_pos > 0) {
  66. if(connection->incompleteMessage.length < 0){
  67. connection->incompleteMessage.data = UA_malloc(current.length - end_pos);
  68. if(!connection->incompleteMessage.data)
  69. UA_ByteString_init(&connection->incompleteMessage);
  70. UA_memcpy(connection->incompleteMessage.data, current.data, current.length - end_pos);
  71. connection->incompleteMessage.length = current.length - end_pos;
  72. UA_ByteString_deleteMembers(&current);
  73. current.length = -1;
  74. }
  75. }
  76. if(end_pos == 0 && current.length - end_pos > 0){
  77. //incomplete - do not forget to destroy the buffer
  78. if(current.data==received.data)
  79. UA_ByteString_deleteMembers(&received);
  80. current.length = -1; //no complete message yet
  81. }else{
  82. //a prefix is a complete message
  83. //we have some incomplete message saved
  84. if(connection->incompleteMessage.data){
  85. //prepare the output buffer - reuse the one from the socket
  86. current.data = UA_realloc(received.data, end_pos);
  87. if(!current.data){
  88. UA_String_deleteMembers(&current);
  89. current.length = -1;
  90. return current;
  91. }
  92. memcpy(current.data, connection->incompleteMessage.data, end_pos);
  93. //is there something left in the incomplete buffer?
  94. if((UA_Int32)end_pos <= connection->incompleteMessage.length){
  95. //yes - save the suffix
  96. memcpy(connection->incompleteMessage.data, connection->incompleteMessage.data+end_pos, connection->incompleteMessage.length - end_pos);
  97. //shrink the buffer
  98. connection->incompleteMessage.data = UA_realloc(connection->incompleteMessage.data, connection->incompleteMessage.length - end_pos);
  99. connection->incompleteMessage.length = connection->incompleteMessage.length - end_pos;
  100. }else{
  101. //no - cleanup
  102. UA_String_deleteMembers(&connection->incompleteMessage);
  103. connection->incompleteMessage.length = -1;
  104. }
  105. }
  106. current.length = end_pos;
  107. }
  108. return current;
  109. }
  110. void UA_SecureChannel_init(UA_SecureChannel *channel) {
  111. UA_MessageSecurityMode_init(&channel->securityMode);
  112. UA_ChannelSecurityToken_init(&channel->securityToken);
  113. UA_AsymmetricAlgorithmSecurityHeader_init(&channel->clientAsymAlgSettings);
  114. UA_AsymmetricAlgorithmSecurityHeader_init(&channel->serverAsymAlgSettings);
  115. UA_ByteString_init(&channel->clientNonce);
  116. UA_ByteString_init(&channel->serverNonce);
  117. channel->requestId = 0;
  118. channel->sequenceNumber = 0;
  119. channel->connection = UA_NULL;
  120. channel->session = UA_NULL;
  121. }
  122. void UA_SecureChannel_deleteMembers(UA_SecureChannel *channel) {
  123. UA_AsymmetricAlgorithmSecurityHeader_deleteMembers(&channel->serverAsymAlgSettings);
  124. UA_ByteString_deleteMembers(&channel->serverNonce);
  125. UA_AsymmetricAlgorithmSecurityHeader_deleteMembers(&channel->clientAsymAlgSettings);
  126. UA_ByteString_deleteMembers(&channel->clientNonce);
  127. UA_ChannelSecurityToken_deleteMembers(&channel->securityToken);
  128. }
  129. void UA_SecureChannel_delete(UA_SecureChannel *channel) {
  130. UA_SecureChannel_deleteMembers(channel);
  131. UA_free(channel);
  132. }
  133. UA_Boolean UA_SecureChannel_compare(UA_SecureChannel *sc1, UA_SecureChannel *sc2) {
  134. return (sc1->securityToken.channelId == sc2->securityToken.channelId);
  135. }
  136. //TODO implement real nonce generator - DUMMY function
  137. UA_StatusCode UA_SecureChannel_generateNonce(UA_ByteString *nonce) {
  138. if(!(nonce->data = UA_malloc(1)))
  139. return UA_STATUSCODE_BADOUTOFMEMORY;
  140. nonce->length = 1;
  141. nonce->data[0] = 'a';
  142. return UA_STATUSCODE_GOOD;
  143. }
  144. UA_StatusCode UA_SecureChannel_updateRequestId(UA_SecureChannel *channel, UA_UInt32 requestId) {
  145. //TODO review checking of request id
  146. if(channel->requestId+1 != requestId)
  147. return UA_STATUSCODE_BADINTERNALERROR;
  148. channel->requestId++;
  149. return UA_STATUSCODE_GOOD;
  150. }
  151. UA_StatusCode UA_SecureChannel_updateSequenceNumber(UA_SecureChannel *channel, UA_UInt32 sequenceNumber) {
  152. //TODO review checking of sequence
  153. if(channel->sequenceNumber+1 != sequenceNumber)
  154. return UA_STATUSCODE_BADINTERNALERROR;
  155. channel->sequenceNumber++;
  156. return UA_STATUSCODE_GOOD;
  157. }
  158. void UA_Connection_detachSecureChannel(UA_Connection *connection) {
  159. #ifdef UA_MULTITHREADING
  160. UA_SecureChannel *channel = connection->channel;
  161. if(channel)
  162. uatomic_cmpxchg(&channel->connection, connection, UA_NULL);
  163. uatomic_set(&connection->channel, UA_NULL);
  164. #else
  165. if(connection->channel)
  166. connection->channel->connection = UA_NULL;
  167. connection->channel = UA_NULL;
  168. #endif
  169. }
  170. void UA_Connection_attachSecureChannel(UA_Connection *connection, UA_SecureChannel *channel) {
  171. #ifdef UA_MULTITHREADING
  172. if(uatomic_cmpxchg(&channel->connection, UA_NULL, connection) == UA_NULL)
  173. uatomic_set(&connection->channel, channel);
  174. #else
  175. if(channel->connection != UA_NULL)
  176. return;
  177. channel->connection = connection;
  178. connection->channel = channel;
  179. #endif
  180. }