• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * nghttp2 - HTTP/2 C Library
3  *
4  * Copyright (c) 2021 Tatsuhiro Tsujikawa
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining
7  * a copy of this software and associated documentation files (the
8  * "Software"), to deal in the Software without restriction, including
9  * without limitation the rights to use, copy, modify, merge, publish,
10  * distribute, sublicense, and/or sell copies of the Software, and to
11  * permit persons to whom the Software is furnished to do so, subject to
12  * the following conditions:
13  *
14  * The above copyright notice and this permission notice shall be
15  * included in all copies or substantial portions of the Software.
16  *
17  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
21  * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22  * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23  * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24  */
25 #ifndef SHRPX_QUIC_CONNECTION_HANDLER_H
26 #define SHRPX_QUIC_CONNECTION_HANDLER_H
27 
28 #include "shrpx.h"
29 
30 #include <memory>
31 #include <unordered_map>
32 #include <string>
33 #include <vector>
34 
35 #include <ngtcp2/ngtcp2.h>
36 
37 #include <ev.h>
38 
39 #include "shrpx_quic.h"
40 #include "network.h"
41 
42 using namespace nghttp2;
43 
44 namespace shrpx {
45 
46 struct UpstreamAddr;
47 class ClientHandler;
48 class Worker;
49 
50 // CloseWait handles packets received in close-wait (draining or
51 // closing period).
52 struct CloseWait {
53   CloseWait(Worker *worker, std::vector<ngtcp2_cid> scids,
54             std::vector<uint8_t> pkt, ev_tstamp period);
55   ~CloseWait();
56 
57   int handle_packet(const UpstreamAddr *faddr, const Address &remote_addr,
58                     const Address &local_addr, const ngtcp2_pkt_info &pi,
59                     const uint8_t *data, size_t datalen);
60 
61   Worker *worker;
62   // Source Connection IDs of the connection.
63   std::vector<ngtcp2_cid> scids;
64   // QUIC packet which is sent in response to the incoming packet.  It
65   // might be empty.
66   std::vector<uint8_t> pkt;
67   // Close-wait (draining or closing period) timer.
68   ev_timer timer;
69   // The number of bytes received during close-wait period.
70   size_t bytes_recv;
71   // The number of bytes sent during close-wait period.
72   size_t bytes_sent;
73   // The number of packets received during close-wait period.
74   size_t num_pkts_recv;
75   // If the number of packets received reaches this number, send a
76   // QUIC packet.
77   size_t next_pkts_recv;
78 };
79 
80 class QUICConnectionHandler {
81 public:
82   QUICConnectionHandler(Worker *worker);
83   ~QUICConnectionHandler();
84   int handle_packet(const UpstreamAddr *faddr, const Address &remote_addr,
85                     const Address &local_addr, const ngtcp2_pkt_info &pi,
86                     const uint8_t *data, size_t datalen);
87   // Send Retry packet.  |ini_dcid| is the destination Connection ID
88   // which appeared in Client Initial packet and its length is
89   // |dcidlen|.  |ini_scid| is the source Connection ID which appeared
90   // in Client Initial packet and its length is |scidlen|.
91   int send_retry(const UpstreamAddr *faddr, uint32_t version,
92                  const uint8_t *ini_dcid, size_t ini_dcidlen,
93                  const uint8_t *ini_scid, size_t ini_scidlen,
94                  const Address &remote_addr, const Address &local_addr,
95                  size_t max_pktlen);
96   // Send Version Negotiation packet.  |ini_dcid| is the destination
97   // Connection ID which appeared in Client Initial packet and its
98   // length is |dcidlen|.  |ini_scid| is the source Connection ID
99   // which appeared in Client Initial packet and its length is
100   // |scidlen|.
101   int send_version_negotiation(const UpstreamAddr *faddr, uint32_t version,
102                                const uint8_t *ini_dcid, size_t ini_dcidlen,
103                                const uint8_t *ini_scid, size_t ini_scidlen,
104                                const Address &remote_addr,
105                                const Address &local_addr);
106   int send_stateless_reset(const UpstreamAddr *faddr, const uint8_t *dcid,
107                            size_t dcidlen, const Address &remote_addr,
108                            const Address &local_addr);
109   // Send Initial CONNECTION_CLOSE.  |ini_dcid| is the destination
110   // Connection ID which appeared in Client Initial packet.
111   // |ini_scid| is the source Connection ID which appeared in Client
112   // Initial packet.
113   int send_connection_close(const UpstreamAddr *faddr, uint32_t version,
114                             const ngtcp2_cid &ini_dcid,
115                             const ngtcp2_cid &ini_scid,
116                             const Address &remote_addr,
117                             const Address &local_addr, uint64_t error_code,
118                             size_t max_pktlen);
119   ClientHandler *
120   handle_new_connection(const UpstreamAddr *faddr, const Address &remote_addr,
121                         const Address &local_addr, const ngtcp2_pkt_hd &hd,
122                         const ngtcp2_cid *odcid, const uint8_t *token,
123                         size_t tokenlen, ngtcp2_token_type token_type);
124   void add_connection_id(const ngtcp2_cid &cid, ClientHandler *handler);
125   void remove_connection_id(const ngtcp2_cid &cid);
126 
127   void add_close_wait(CloseWait *cw);
128   void remove_close_wait(const CloseWait *cw);
129 
130   void on_stateless_reset_bucket_regen();
131 
132 private:
133   Worker *worker_;
134   std::unordered_map<ngtcp2_cid, ClientHandler *> connections_;
135   std::unordered_map<ngtcp2_cid, CloseWait *> close_waits_;
136   ev_timer stateless_reset_bucket_regen_timer_;
137   size_t stateless_reset_bucket_;
138 };
139 
140 } // namespace shrpx
141 
142 #endif // SHRPX_QUIC_CONNECTION_HANDLER_H
143