DatagramPeerIO.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500
  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 sending base
  47. // init disassembler
  48. FragmentProtoDisassembler_Init(&o->send_disassembler, o->reactor, o->payload_mtu, o->spproto_payload_mtu, -1, latency);
  49. // init encoder
  50. if (!SPProtoEncoder_Init(&o->send_encoder, o->sp_params, FragmentProtoDisassembler_GetOutput(&o->send_disassembler), BReactor_PendingGroup(o->reactor))) {
  51. BLog(BLOG_ERROR, "SPProtoEncoder_Init failed");
  52. goto fail1;
  53. }
  54. // init notifier
  55. PacketRecvNotifier_Init(&o->send_notifier, SPProtoEncoder_GetOutput(&o->send_encoder), BReactor_PendingGroup(o->reactor));
  56. if (SPPROTO_HAVE_OTP(o->sp_params)) {
  57. PacketRecvNotifier_SetHandler(&o->send_notifier, (PacketRecvNotifier_handler_notify)send_encoder_notifier_handler, o);
  58. }
  59. // init connector
  60. PacketPassConnector_Init(&o->send_connector, o->effective_socket_mtu, BReactor_PendingGroup(o->reactor));
  61. // init buffer
  62. if (!SinglePacketBuffer_Init(&o->send_buffer, PacketRecvNotifier_GetOutput(&o->send_notifier), PacketPassConnector_GetInput(&o->send_connector), BReactor_PendingGroup(o->reactor))) {
  63. BLog(BLOG_ERROR, "SinglePacketBuffer_Init failed");
  64. goto fail2;
  65. }
  66. // init receiving
  67. // init assembler
  68. 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))) {
  69. goto fail3;
  70. }
  71. // init notifier
  72. PacketPassNotifier_Init(&o->recv_notifier, FragmentProtoAssembler_GetInput(&o->recv_assembler), BReactor_PendingGroup(o->reactor));
  73. // init decoder
  74. if (!SPProtoDecoder_Init(&o->recv_decoder, PacketPassNotifier_GetInput(&o->recv_notifier), o->sp_params, 2, BReactor_PendingGroup(o->reactor))) {
  75. goto fail4;
  76. }
  77. // init connector
  78. PacketRecvConnector_Init(&o->recv_connector, o->effective_socket_mtu, BReactor_PendingGroup(o->reactor));
  79. // init buffer
  80. if (!SinglePacketBuffer_Init(&o->recv_buffer, PacketRecvConnector_GetOutput(&o->recv_connector), SPProtoDecoder_GetInput(&o->recv_decoder), BReactor_PendingGroup(o->reactor))) {
  81. goto fail5;
  82. }
  83. return 1;
  84. fail5:
  85. PacketRecvConnector_Free(&o->recv_connector);
  86. SPProtoDecoder_Free(&o->recv_decoder);
  87. fail4:
  88. PacketPassNotifier_Free(&o->recv_notifier);
  89. FragmentProtoAssembler_Free(&o->recv_assembler);
  90. fail3:
  91. SinglePacketBuffer_Free(&o->send_buffer);
  92. fail2:
  93. PacketPassConnector_Free(&o->send_connector);
  94. PacketRecvNotifier_Free(&o->send_notifier);
  95. SPProtoEncoder_Free(&o->send_encoder);
  96. fail1:
  97. FragmentProtoDisassembler_Free(&o->send_disassembler);
  98. return 0;
  99. }
  100. void free_persistent_io (DatagramPeerIO *o)
  101. {
  102. // free receiving
  103. SinglePacketBuffer_Free(&o->recv_buffer);
  104. PacketRecvConnector_Free(&o->recv_connector);
  105. SPProtoDecoder_Free(&o->recv_decoder);
  106. PacketPassNotifier_Free(&o->recv_notifier);
  107. FragmentProtoAssembler_Free(&o->recv_assembler);
  108. // free sending base
  109. SinglePacketBuffer_Free(&o->send_buffer);
  110. PacketPassConnector_Free(&o->send_connector);
  111. PacketRecvNotifier_Free(&o->send_notifier);
  112. SPProtoEncoder_Free(&o->send_encoder);
  113. FragmentProtoDisassembler_Free(&o->send_disassembler);
  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. int error = *((int *)data);
  147. switch (component) {
  148. case DATAGRAMPEERIO_COMPONENT_SINK:
  149. switch (error) {
  150. case DATAGRAMSOCKETSINK_ERROR_BSOCKET:
  151. BLog(BLOG_NOTICE, "sink BSocket error %d", BSocket_GetError(&o->sock));
  152. break;
  153. case DATAGRAMSOCKETSINK_ERROR_WRONGSIZE:
  154. BLog(BLOG_NOTICE, "sink wrong size error");
  155. break;
  156. default:
  157. ASSERT(0);
  158. }
  159. break;
  160. case DATAGRAMPEERIO_COMPONENT_SOURCE:
  161. switch (error) {
  162. case DATAGRAMSOCKETSOURCE_ERROR_BSOCKET:
  163. BLog(BLOG_NOTICE, "source BSocket error %d", BSocket_GetError(&o->sock));
  164. break;
  165. default:
  166. ASSERT(0);
  167. }
  168. break;
  169. default:
  170. ASSERT(0);
  171. }
  172. }
  173. void reset_mode (DatagramPeerIO *o)
  174. {
  175. switch (o->mode) {
  176. case DATAGRAMPEERIO_MODE_NONE:
  177. break;
  178. case DATAGRAMPEERIO_MODE_CONNECT:
  179. // kill mode dead var
  180. DEAD_KILL(o->mode_dead);
  181. // set default mode
  182. o->mode = DATAGRAMPEERIO_MODE_NONE;
  183. // free receiving
  184. free_receiving(o);
  185. // free sending
  186. free_sending(o);
  187. // free socket
  188. BSocket_Free(&o->sock);
  189. break;
  190. case DATAGRAMPEERIO_MODE_BIND:
  191. // kill mode dead var
  192. DEAD_KILL(o->mode_dead);
  193. // set default mode
  194. o->mode = DATAGRAMPEERIO_MODE_NONE;
  195. // remove recv notifier handler
  196. PacketPassNotifier_SetHandler(&o->recv_notifier, NULL, NULL);
  197. // free receiving
  198. free_receiving(o);
  199. // free sending
  200. if (o->bind_sending_up) {
  201. free_sending(o);
  202. }
  203. // free socket
  204. BSocket_Free(&o->sock);
  205. break;
  206. default:
  207. ASSERT(0);
  208. }
  209. }
  210. void recv_decoder_notifier_handler (DatagramPeerIO *o, uint8_t *data, int data_len)
  211. {
  212. ASSERT(o->mode == DATAGRAMPEERIO_MODE_BIND)
  213. // obtain addresses from last received packet
  214. BAddr addr;
  215. BIPAddr local_addr;
  216. DatagramSocketSource_GetLastAddresses(&o->recv_source, &addr, &local_addr);
  217. if (!o->bind_sending_up) {
  218. // init sending
  219. init_sending(o, addr, local_addr);
  220. // set sending up
  221. o->bind_sending_up = 1;
  222. } else {
  223. // update addresses
  224. DatagramSocketSink_SetAddresses(&o->send_sink, addr, local_addr);
  225. }
  226. }
  227. void send_encoder_notifier_handler (DatagramPeerIO *o, uint8_t *data, int data_len)
  228. {
  229. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  230. if (o->handler_otp_warning && SPProtoEncoder_GetOTPPosition(&o->send_encoder) == o->handler_otp_warning_num_used) {
  231. o->handler_otp_warning(o->handler_otp_warning_user);
  232. return;
  233. }
  234. }
  235. 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)
  236. {
  237. ASSERT(payload_mtu >= 0)
  238. ASSERT(payload_mtu <= UINT16_MAX)
  239. ASSERT(socket_mtu >= 0)
  240. ASSERT(spproto_validate_security_params(sp_params))
  241. ASSERT(spproto_payload_mtu_for_carrier_mtu(sp_params, socket_mtu) > sizeof(struct fragmentproto_chunk_header))
  242. ASSERT(PacketPassInterface_GetMTU(recv_userif) >= payload_mtu)
  243. // set parameters
  244. o->reactor = reactor;
  245. o->payload_mtu = payload_mtu;
  246. o->sp_params = sp_params;
  247. // init dead var
  248. DEAD_INIT(o->dead);
  249. // calculate SPProto payload MTU
  250. o->spproto_payload_mtu = spproto_payload_mtu_for_carrier_mtu(o->sp_params, socket_mtu);
  251. // calculate effective socket MTU
  252. o->effective_socket_mtu = spproto_carrier_mtu_for_payload_mtu(o->sp_params, o->spproto_payload_mtu);
  253. // set no OTP warning handler
  254. if (SPPROTO_HAVE_OTP(o->sp_params)) {
  255. o->handler_otp_warning = NULL;
  256. }
  257. // set mode none
  258. o->mode = DATAGRAMPEERIO_MODE_NONE;
  259. // init persistent I/O objects
  260. if (!init_persistent_io(o, latency, recv_userif)) {
  261. goto fail1;
  262. }
  263. // init debug object
  264. DebugObject_Init(&o->d_obj);
  265. return 1;
  266. fail1:
  267. return 0;
  268. }
  269. void DatagramPeerIO_Free (DatagramPeerIO *o)
  270. {
  271. // free debug object
  272. DebugObject_Free(&o->d_obj);
  273. // reset mode
  274. reset_mode(o);
  275. // free persistent I/O objects
  276. free_persistent_io(o);
  277. // free dead var
  278. DEAD_KILL(o->dead);
  279. }
  280. PacketPassInterface * DatagramPeerIO_GetSendInput (DatagramPeerIO *o)
  281. {
  282. return FragmentProtoDisassembler_GetInput(&o->send_disassembler);
  283. }
  284. void DatagramPeerIO_Disconnect (DatagramPeerIO *o)
  285. {
  286. // reset mode
  287. reset_mode(o);
  288. }
  289. int DatagramPeerIO_Connect (DatagramPeerIO *o, BAddr addr)
  290. {
  291. ASSERT(BAddr_IsRecognized(&addr) && !BAddr_IsInvalid(&addr))
  292. // reset mode
  293. reset_mode(o);
  294. // init socket
  295. if (BSocket_Init(&o->sock, o->reactor, addr.type, BSOCKET_TYPE_DGRAM) < 0) {
  296. BLog(BLOG_ERROR, "BSocket_Init failed");
  297. goto fail1;
  298. }
  299. // connect the socket
  300. // Windows needs this or receive will fail
  301. if (BSocket_Connect(&o->sock, &addr) < 0) {
  302. BLog(BLOG_ERROR, "BSocket_Connect failed");
  303. goto fail2;
  304. }
  305. // init sending
  306. BIPAddr local_addr;
  307. BIPAddr_InitInvalid(&local_addr);
  308. init_sending(o, addr, local_addr);
  309. // init receiving
  310. init_receiving(o);
  311. // set mode
  312. o->mode = DATAGRAMPEERIO_MODE_CONNECT;
  313. // init mode dead var
  314. DEAD_INIT(o->mode_dead);
  315. return 1;
  316. fail2:
  317. BSocket_Free(&o->sock);
  318. fail1:
  319. return 0;
  320. }
  321. int DatagramPeerIO_Bind (DatagramPeerIO *o, BAddr addr)
  322. {
  323. ASSERT(BAddr_IsRecognized(&addr) && !BAddr_IsInvalid(&addr))
  324. // reset mode
  325. reset_mode(o);
  326. // init socket
  327. if (BSocket_Init(&o->sock, o->reactor, addr.type, BSOCKET_TYPE_DGRAM) < 0) {
  328. BLog(BLOG_ERROR, "BSocket_Init failed");
  329. goto fail1;
  330. }
  331. // bind socket
  332. if (BSocket_Bind(&o->sock, &addr) < 0) {
  333. BLog(BLOG_INFO, "BSocket_Bind failed");
  334. goto fail2;
  335. }
  336. // init receiving
  337. init_receiving(o);
  338. // set recv notifier handler
  339. PacketPassNotifier_SetHandler(&o->recv_notifier, (PacketPassNotifier_handler_notify)recv_decoder_notifier_handler, o);
  340. // set mode
  341. o->mode = DATAGRAMPEERIO_MODE_BIND;
  342. // init mode dead var
  343. DEAD_INIT(o->mode_dead);
  344. // set sending not up
  345. o->bind_sending_up = 0;
  346. return 1;
  347. fail2:
  348. BSocket_Free(&o->sock);
  349. fail1:
  350. return 0;
  351. }
  352. void DatagramPeerIO_Flush (DatagramPeerIO *o)
  353. {
  354. BLog(BLOG_ERROR, "Flushing not implemented");
  355. }
  356. void DatagramPeerIO_SetEncryptionKey (DatagramPeerIO *o, uint8_t *encryption_key)
  357. {
  358. ASSERT(o->sp_params.encryption_mode != SPPROTO_ENCRYPTION_MODE_NONE)
  359. // set sending key
  360. SPProtoEncoder_SetEncryptionKey(&o->send_encoder, encryption_key);
  361. // set receiving key
  362. SPProtoDecoder_SetEncryptionKey(&o->recv_decoder, encryption_key);
  363. }
  364. void DatagramPeerIO_RemoveEncryptionKey (DatagramPeerIO *o)
  365. {
  366. ASSERT(o->sp_params.encryption_mode != SPPROTO_ENCRYPTION_MODE_NONE)
  367. // remove sending key
  368. SPProtoEncoder_RemoveEncryptionKey(&o->send_encoder);
  369. // remove receiving key
  370. SPProtoDecoder_RemoveEncryptionKey(&o->recv_decoder);
  371. }
  372. void DatagramPeerIO_SetOTPSendSeed (DatagramPeerIO *o, uint16_t seed_id, uint8_t *key, uint8_t *iv)
  373. {
  374. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  375. // set sending seed
  376. SPProtoEncoder_SetOTPSeed(&o->send_encoder, seed_id, key, iv);
  377. }
  378. void DatagramPeerIO_RemoveOTPSendSeed (DatagramPeerIO *o)
  379. {
  380. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  381. // remove sending seed
  382. SPProtoEncoder_RemoveOTPSeed(&o->send_encoder);
  383. }
  384. void DatagramPeerIO_AddOTPRecvSeed (DatagramPeerIO *o, uint16_t seed_id, uint8_t *key, uint8_t *iv)
  385. {
  386. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  387. // add receiving seed
  388. SPProtoDecoder_AddOTPSeed(&o->recv_decoder, seed_id, key, iv);
  389. }
  390. void DatagramPeerIO_RemoveOTPRecvSeeds (DatagramPeerIO *o)
  391. {
  392. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  393. // remove receiving seeds
  394. SPProtoDecoder_RemoveOTPSeeds(&o->recv_decoder);
  395. }
  396. void DatagramPeerIO_SetOTPWarningHandler (DatagramPeerIO *o, DatagramPeerIO_handler_otp_warning handler, void *user, int num_used)
  397. {
  398. ASSERT(SPPROTO_HAVE_OTP(o->sp_params))
  399. ASSERT(!handler || num_used > 0)
  400. o->handler_otp_warning = handler;
  401. o->handler_otp_warning_user = user;
  402. o->handler_otp_warning_num_used = num_used;
  403. }