DatagramPeerIO.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. /**
  2. * @file DatagramPeerIO.c
  3. * @author Ambroz Bizjak <ambrop7@gmail.com>
  4. *
  5. * @section LICENSE
  6. *
  7. * This file is part of BadVPN.
  8. *
  9. * BadVPN is free software: you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License version 2
  11. * as published by the Free Software Foundation.
  12. *
  13. * BadVPN is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License along
  19. * with this program; if not, write to the Free Software Foundation, Inc.,
  20. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  21. */
  22. #include <stdint.h>
  23. #include <misc/debug.h>
  24. #include <system/BLog.h>
  25. #include <client/DatagramPeerIO.h>
  26. #include <generated/blog_channel_DatagramPeerIO.h>
  27. #define DATAGRAMPEERIO_MODE_NONE 0
  28. #define DATAGRAMPEERIO_MODE_CONNECT 1
  29. #define DATAGRAMPEERIO_MODE_BIND 2
  30. #define DATAGRAMPEERIO_COMPONENT_SINK 1
  31. #define DATAGRAMPEERIO_COMPONENT_SOURCE 2
  32. static int init_persistent_io (DatagramPeerIO *o, btime_t latency, PacketPassInterface *recv_userif);
  33. static void free_persistent_io (DatagramPeerIO *o);
  34. static void init_sending (DatagramPeerIO *o, BAddr addr, BIPAddr local_addr);
  35. static void free_sending (DatagramPeerIO *o);
  36. static void init_receiving (DatagramPeerIO *o);
  37. static void free_receiving (DatagramPeerIO *o);
  38. static void error_handler (DatagramPeerIO *o, int component, const void *data);
  39. static void reset_mode (DatagramPeerIO *o);
  40. static void recv_decoder_notifier_handler (DatagramPeerIO *o, uint8_t *data, int data_len);
  41. static void send_encoder_notifier_handler (DatagramPeerIO *o, uint8_t *data, int data_len);
  42. int init_persistent_io (DatagramPeerIO *o, btime_t latency, PacketPassInterface *recv_userif)
  43. {
  44. // init error domain
  45. FlowErrorDomain_Init(&o->domain, (FlowErrorDomain_handler)error_handler, o);
  46. // init receiving
  47. // init assembler
  48. if (!FragmentProtoAssembler_Init(&o->recv_assembler, o->spproto_payload_mtu, recv_userif, 1, fragmentproto_max_chunks_for_frame(o->spproto_payload_mtu, o->payload_mtu), BReactor_PendingGroup(o->reactor))) {
  49. goto fail0;
  50. }
  51. // init notifier
  52. PacketPassNotifier_Init(&o->recv_notifier, FragmentProtoAssembler_GetInput(&o->recv_assembler), BReactor_PendingGroup(o->reactor));
  53. // init decoder
  54. if (!SPProtoDecoder_Init(&o->recv_decoder, PacketPassNotifier_GetInput(&o->recv_notifier), o->sp_params, 2, BReactor_PendingGroup(o->reactor))) {
  55. goto fail1;
  56. }
  57. // init connector
  58. PacketRecvConnector_Init(&o->recv_connector, o->effective_socket_mtu, BReactor_PendingGroup(o->reactor));
  59. // init buffer
  60. if (!SinglePacketBuffer_Init(&o->recv_buffer, PacketRecvConnector_GetOutput(&o->recv_connector), SPProtoDecoder_GetInput(&o->recv_decoder), BReactor_PendingGroup(o->reactor))) {
  61. goto fail2;
  62. }
  63. // init sending base
  64. // init disassembler
  65. FragmentProtoDisassembler_Init(&o->send_disassembler, o->reactor, o->payload_mtu, o->spproto_payload_mtu, -1, latency);
  66. // init encoder
  67. if (!SPProtoEncoder_Init(&o->send_encoder, o->sp_params, FragmentProtoDisassembler_GetOutput(&o->send_disassembler), BReactor_PendingGroup(o->reactor))) {
  68. BLog(BLOG_ERROR, "SPProtoEncoder_Init failed");
  69. goto fail3;
  70. }
  71. // init notifier
  72. PacketRecvNotifier_Init(&o->send_notifier, SPProtoEncoder_GetOutput(&o->send_encoder), BReactor_PendingGroup(o->reactor));
  73. if (SPPROTO_HAVE_OTP(o->sp_params)) {
  74. PacketRecvNotifier_SetHandler(&o->send_notifier, (PacketRecvNotifier_handler_notify)send_encoder_notifier_handler, o);
  75. }
  76. // init connector
  77. PacketPassConnector_Init(&o->send_connector, o->effective_socket_mtu, BReactor_PendingGroup(o->reactor));
  78. // init buffer
  79. if (!SinglePacketBuffer_Init(&o->send_buffer, PacketRecvNotifier_GetOutput(&o->send_notifier), PacketPassConnector_GetInput(&o->send_connector), BReactor_PendingGroup(o->reactor))) {
  80. BLog(BLOG_ERROR, "SinglePacketBuffer_Init failed");
  81. goto fail4;
  82. }
  83. return 1;
  84. fail4:
  85. PacketPassConnector_Free(&o->send_connector);
  86. PacketRecvNotifier_Free(&o->send_notifier);
  87. SPProtoEncoder_Free(&o->send_encoder);
  88. fail3:
  89. FragmentProtoDisassembler_Free(&o->send_disassembler);
  90. SinglePacketBuffer_Free(&o->recv_buffer);
  91. fail2:
  92. PacketRecvConnector_Free(&o->recv_connector);
  93. SPProtoDecoder_Free(&o->recv_decoder);
  94. fail1:
  95. PacketPassNotifier_Free(&o->recv_notifier);
  96. FragmentProtoAssembler_Free(&o->recv_assembler);
  97. fail0:
  98. return 0;
  99. }
  100. void free_persistent_io (DatagramPeerIO *o)
  101. {
  102. // free sending base
  103. SinglePacketBuffer_Free(&o->send_buffer);
  104. PacketPassConnector_Free(&o->send_connector);
  105. PacketRecvNotifier_Free(&o->send_notifier);
  106. SPProtoEncoder_Free(&o->send_encoder);
  107. FragmentProtoDisassembler_Free(&o->send_disassembler);
  108. // free receiving
  109. SinglePacketBuffer_Free(&o->recv_buffer);
  110. PacketRecvConnector_Free(&o->recv_connector);
  111. SPProtoDecoder_Free(&o->recv_decoder);
  112. PacketPassNotifier_Free(&o->recv_notifier);
  113. FragmentProtoAssembler_Free(&o->recv_assembler);
  114. }
  115. void init_sending (DatagramPeerIO *o, BAddr addr, BIPAddr local_addr)
  116. {
  117. // init sink
  118. DatagramSocketSink_Init(&o->send_sink, FlowErrorReporter_Create(&o->domain, DATAGRAMPEERIO_COMPONENT_SINK), &o->sock, o->effective_socket_mtu, addr, local_addr, BReactor_PendingGroup(o->reactor));
  119. // connect sink
  120. PacketPassConnector_ConnectOutput(&o->send_connector, DatagramSocketSink_GetInput(&o->send_sink));
  121. }
  122. void free_sending (DatagramPeerIO *o)
  123. {
  124. // disconnect sink
  125. PacketPassConnector_DisconnectOutput(&o->send_connector);
  126. // free sink
  127. DatagramSocketSink_Free(&o->send_sink);
  128. }
  129. void init_receiving (DatagramPeerIO *o)
  130. {
  131. // init source
  132. DatagramSocketSource_Init(&o->recv_source, FlowErrorReporter_Create(&o->domain, DATAGRAMPEERIO_COMPONENT_SOURCE), &o->sock, o->effective_socket_mtu, BReactor_PendingGroup(o->reactor));
  133. // connect source
  134. PacketRecvConnector_ConnectInput(&o->recv_connector, DatagramSocketSource_GetOutput(&o->recv_source));
  135. }
  136. void free_receiving (DatagramPeerIO *o)
  137. {
  138. // disconnect source
  139. PacketRecvConnector_DisconnectInput(&o->recv_connector);
  140. // free source
  141. DatagramSocketSource_Free(&o->recv_source);
  142. }
  143. void error_handler (DatagramPeerIO *o, int component, const void *data)
  144. {
  145. ASSERT(o->mode == DATAGRAMPEERIO_MODE_CONNECT || o->mode == DATAGRAMPEERIO_MODE_BIND)
  146. DebugObject_Access(&o->d_obj);
  147. int error = *((int *)data);
  148. switch (component) {
  149. case DATAGRAMPEERIO_COMPONENT_SINK:
  150. switch (error) {
  151. case DATAGRAMSOCKETSINK_ERROR_BSOCKET:
  152. BLog(BLOG_NOTICE, "sink BSocket error %d", BSocket_GetError(&o->sock));
  153. break;
  154. case DATAGRAMSOCKETSINK_ERROR_WRONGSIZE:
  155. BLog(BLOG_NOTICE, "sink wrong size error");
  156. break;
  157. default:
  158. ASSERT(0);
  159. }
  160. break;
  161. case DATAGRAMPEERIO_COMPONENT_SOURCE:
  162. switch (error) {
  163. case DATAGRAMSOCKETSOURCE_ERROR_BSOCKET:
  164. BLog(BLOG_NOTICE, "source BSocket error %d", BSocket_GetError(&o->sock));
  165. break;
  166. default:
  167. ASSERT(0);
  168. }
  169. break;
  170. default:
  171. ASSERT(0);
  172. }
  173. }
  174. void reset_mode (DatagramPeerIO *o)
  175. {
  176. switch (o->mode) {
  177. case DATAGRAMPEERIO_MODE_NONE:
  178. break;
  179. case DATAGRAMPEERIO_MODE_CONNECT:
  180. // set default mode
  181. o->mode = DATAGRAMPEERIO_MODE_NONE;
  182. // free receiving
  183. free_receiving(o);
  184. // free sending
  185. free_sending(o);
  186. // free socket
  187. BSocket_Free(&o->sock);
  188. break;
  189. case DATAGRAMPEERIO_MODE_BIND:
  190. // set default mode
  191. o->mode = DATAGRAMPEERIO_MODE_NONE;
  192. // remove recv notifier handler
  193. PacketPassNotifier_SetHandler(&o->recv_notifier, NULL, NULL);
  194. // free receiving
  195. free_receiving(o);
  196. // free sending
  197. if (o->bind_sending_up) {
  198. free_sending(o);
  199. }
  200. // free socket
  201. BSocket_Free(&o->sock);
  202. break;
  203. default:
  204. ASSERT(0);
  205. }
  206. }
  207. void recv_decoder_notifier_handler (DatagramPeerIO *o, uint8_t *data, int data_len)
  208. {
  209. ASSERT(o->mode == DATAGRAMPEERIO_MODE_BIND)
  210. DebugObject_Access(&o->d_obj);
  211. // obtain addresses from last received packet
  212. BAddr addr;
  213. BIPAddr local_addr;
  214. DatagramSocketSource_GetLastAddresses(&o->recv_source, &addr, &local_addr);
  215. if (!o->bind_sending_up) {
  216. // init sending
  217. init_sending(o, addr, local_addr);
  218. // set sending up
  219. o->bind_sending_up = 1;
  220. } else {
  221. // update addresses
  222. DatagramSocketSink_SetAddresses(&o->send_sink, addr, local_addr);
  223. }
  224. }
  225. void send_encoder_notifier_handler (DatagramPeerIO *o, uint8_t *data, int data_len)
  226. {
  227. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  228. DebugObject_Access(&o->d_obj);
  229. if (o->handler_otp_warning && SPProtoEncoder_GetOTPPosition(&o->send_encoder) == o->handler_otp_warning_num_used) {
  230. o->handler_otp_warning(o->handler_otp_warning_user);
  231. return;
  232. }
  233. }
  234. int DatagramPeerIO_Init (DatagramPeerIO *o, BReactor *reactor, int payload_mtu, int socket_mtu, struct spproto_security_params sp_params, btime_t latency, PacketPassInterface *recv_userif)
  235. {
  236. ASSERT(payload_mtu >= 0)
  237. ASSERT(payload_mtu <= UINT16_MAX)
  238. ASSERT(socket_mtu >= 0)
  239. ASSERT(spproto_validate_security_params(sp_params))
  240. ASSERT(spproto_payload_mtu_for_carrier_mtu(sp_params, socket_mtu) > sizeof(struct fragmentproto_chunk_header))
  241. ASSERT(PacketPassInterface_GetMTU(recv_userif) >= payload_mtu)
  242. // set parameters
  243. o->reactor = reactor;
  244. o->payload_mtu = payload_mtu;
  245. o->sp_params = sp_params;
  246. // calculate SPProto payload MTU
  247. o->spproto_payload_mtu = spproto_payload_mtu_for_carrier_mtu(o->sp_params, socket_mtu);
  248. // calculate effective socket MTU
  249. o->effective_socket_mtu = spproto_carrier_mtu_for_payload_mtu(o->sp_params, o->spproto_payload_mtu);
  250. // set no OTP warning handler
  251. if (SPPROTO_HAVE_OTP(o->sp_params)) {
  252. o->handler_otp_warning = NULL;
  253. }
  254. // set mode none
  255. o->mode = DATAGRAMPEERIO_MODE_NONE;
  256. // init persistent I/O objects
  257. if (!init_persistent_io(o, latency, recv_userif)) {
  258. goto fail1;
  259. }
  260. DebugObject_Init(&o->d_obj);
  261. return 1;
  262. fail1:
  263. return 0;
  264. }
  265. void DatagramPeerIO_Free (DatagramPeerIO *o)
  266. {
  267. DebugObject_Free(&o->d_obj);
  268. // reset mode
  269. reset_mode(o);
  270. // free persistent I/O objects
  271. free_persistent_io(o);
  272. }
  273. PacketPassInterface * DatagramPeerIO_GetSendInput (DatagramPeerIO *o)
  274. {
  275. DebugObject_Access(&o->d_obj);
  276. return FragmentProtoDisassembler_GetInput(&o->send_disassembler);
  277. }
  278. int DatagramPeerIO_Connect (DatagramPeerIO *o, BAddr addr)
  279. {
  280. ASSERT(BAddr_IsRecognized(&addr) && !BAddr_IsInvalid(&addr))
  281. DebugObject_Access(&o->d_obj);
  282. // reset mode
  283. reset_mode(o);
  284. // init socket
  285. if (BSocket_Init(&o->sock, o->reactor, addr.type, BSOCKET_TYPE_DGRAM) < 0) {
  286. BLog(BLOG_ERROR, "BSocket_Init failed");
  287. goto fail1;
  288. }
  289. // connect the socket
  290. // Windows needs this or receive will fail
  291. if (BSocket_Connect(&o->sock, &addr) < 0) {
  292. BLog(BLOG_ERROR, "BSocket_Connect failed");
  293. goto fail2;
  294. }
  295. // init receiving
  296. init_receiving(o);
  297. // init sending
  298. BIPAddr local_addr;
  299. BIPAddr_InitInvalid(&local_addr);
  300. init_sending(o, addr, local_addr);
  301. // set mode
  302. o->mode = DATAGRAMPEERIO_MODE_CONNECT;
  303. return 1;
  304. fail2:
  305. BSocket_Free(&o->sock);
  306. fail1:
  307. return 0;
  308. }
  309. int DatagramPeerIO_Bind (DatagramPeerIO *o, BAddr addr)
  310. {
  311. ASSERT(BAddr_IsRecognized(&addr) && !BAddr_IsInvalid(&addr))
  312. DebugObject_Access(&o->d_obj);
  313. // reset mode
  314. reset_mode(o);
  315. // init socket
  316. if (BSocket_Init(&o->sock, o->reactor, addr.type, BSOCKET_TYPE_DGRAM) < 0) {
  317. BLog(BLOG_ERROR, "BSocket_Init failed");
  318. goto fail1;
  319. }
  320. // bind socket
  321. if (BSocket_Bind(&o->sock, &addr) < 0) {
  322. BLog(BLOG_INFO, "BSocket_Bind failed");
  323. goto fail2;
  324. }
  325. // init receiving
  326. init_receiving(o);
  327. // set recv notifier handler
  328. PacketPassNotifier_SetHandler(&o->recv_notifier, (PacketPassNotifier_handler_notify)recv_decoder_notifier_handler, o);
  329. // set mode
  330. o->mode = DATAGRAMPEERIO_MODE_BIND;
  331. // set sending not up
  332. o->bind_sending_up = 0;
  333. return 1;
  334. fail2:
  335. BSocket_Free(&o->sock);
  336. fail1:
  337. return 0;
  338. }
  339. void DatagramPeerIO_SetEncryptionKey (DatagramPeerIO *o, uint8_t *encryption_key)
  340. {
  341. ASSERT(o->sp_params.encryption_mode != SPPROTO_ENCRYPTION_MODE_NONE)
  342. DebugObject_Access(&o->d_obj);
  343. // set sending key
  344. SPProtoEncoder_SetEncryptionKey(&o->send_encoder, encryption_key);
  345. // set receiving key
  346. SPProtoDecoder_SetEncryptionKey(&o->recv_decoder, encryption_key);
  347. }
  348. void DatagramPeerIO_RemoveEncryptionKey (DatagramPeerIO *o)
  349. {
  350. ASSERT(o->sp_params.encryption_mode != SPPROTO_ENCRYPTION_MODE_NONE)
  351. DebugObject_Access(&o->d_obj);
  352. // remove sending key
  353. SPProtoEncoder_RemoveEncryptionKey(&o->send_encoder);
  354. // remove receiving key
  355. SPProtoDecoder_RemoveEncryptionKey(&o->recv_decoder);
  356. }
  357. void DatagramPeerIO_SetOTPSendSeed (DatagramPeerIO *o, uint16_t seed_id, uint8_t *key, uint8_t *iv)
  358. {
  359. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  360. DebugObject_Access(&o->d_obj);
  361. // set sending seed
  362. SPProtoEncoder_SetOTPSeed(&o->send_encoder, seed_id, key, iv);
  363. }
  364. void DatagramPeerIO_RemoveOTPSendSeed (DatagramPeerIO *o)
  365. {
  366. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  367. DebugObject_Access(&o->d_obj);
  368. // remove sending seed
  369. SPProtoEncoder_RemoveOTPSeed(&o->send_encoder);
  370. }
  371. void DatagramPeerIO_AddOTPRecvSeed (DatagramPeerIO *o, uint16_t seed_id, uint8_t *key, uint8_t *iv)
  372. {
  373. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  374. DebugObject_Access(&o->d_obj);
  375. // add receiving seed
  376. SPProtoDecoder_AddOTPSeed(&o->recv_decoder, seed_id, key, iv);
  377. }
  378. void DatagramPeerIO_RemoveOTPRecvSeeds (DatagramPeerIO *o)
  379. {
  380. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  381. DebugObject_Access(&o->d_obj);
  382. // remove receiving seeds
  383. SPProtoDecoder_RemoveOTPSeeds(&o->recv_decoder);
  384. }
  385. void DatagramPeerIO_SetOTPWarningHandler (DatagramPeerIO *o, DatagramPeerIO_handler_otp_warning handler, void *user, int num_used)
  386. {
  387. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  388. ASSERT(!handler || num_used > 0)
  389. DebugObject_Access(&o->d_obj);
  390. o->handler_otp_warning = handler;
  391. o->handler_otp_warning_user = user;
  392. o->handler_otp_warning_num_used = num_used;
  393. }