xref: /aosp_15_r20/external/cronet/net/websockets/websocket_frame.h (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2012 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #ifndef NET_WEBSOCKETS_WEBSOCKET_FRAME_H_
6*6777b538SAndroid Build Coastguard Worker #define NET_WEBSOCKETS_WEBSOCKET_FRAME_H_
7*6777b538SAndroid Build Coastguard Worker 
8*6777b538SAndroid Build Coastguard Worker #include <stddef.h>
9*6777b538SAndroid Build Coastguard Worker #include <stdint.h>
10*6777b538SAndroid Build Coastguard Worker 
11*6777b538SAndroid Build Coastguard Worker #include <memory>
12*6777b538SAndroid Build Coastguard Worker #include <vector>
13*6777b538SAndroid Build Coastguard Worker 
14*6777b538SAndroid Build Coastguard Worker #include "base/containers/span.h"
15*6777b538SAndroid Build Coastguard Worker #include "base/memory/scoped_refptr.h"
16*6777b538SAndroid Build Coastguard Worker #include "net/base/net_export.h"
17*6777b538SAndroid Build Coastguard Worker 
18*6777b538SAndroid Build Coastguard Worker namespace net {
19*6777b538SAndroid Build Coastguard Worker 
20*6777b538SAndroid Build Coastguard Worker // Represents a WebSocket frame header.
21*6777b538SAndroid Build Coastguard Worker //
22*6777b538SAndroid Build Coastguard Worker // Members of this class correspond to each element in WebSocket frame header
23*6777b538SAndroid Build Coastguard Worker // (see http://tools.ietf.org/html/rfc6455#section-5.2).
24*6777b538SAndroid Build Coastguard Worker struct NET_EXPORT WebSocketFrameHeader {
25*6777b538SAndroid Build Coastguard Worker   typedef int OpCode;
26*6777b538SAndroid Build Coastguard Worker 
27*6777b538SAndroid Build Coastguard Worker   // Originally these constants were static const int, but to make it possible
28*6777b538SAndroid Build Coastguard Worker   // to use them in a switch statement they were changed to an enum.
29*6777b538SAndroid Build Coastguard Worker   enum OpCodeEnum {
30*6777b538SAndroid Build Coastguard Worker     kOpCodeContinuation = 0x0,
31*6777b538SAndroid Build Coastguard Worker     kOpCodeText = 0x1,
32*6777b538SAndroid Build Coastguard Worker     kOpCodeBinary = 0x2,
33*6777b538SAndroid Build Coastguard Worker     kOpCodeDataUnused = 0x3,
34*6777b538SAndroid Build Coastguard Worker     kOpCodeClose = 0x8,
35*6777b538SAndroid Build Coastguard Worker     kOpCodePing = 0x9,
36*6777b538SAndroid Build Coastguard Worker     kOpCodePong = 0xA,
37*6777b538SAndroid Build Coastguard Worker     kOpCodeControlUnused = 0xB,
38*6777b538SAndroid Build Coastguard Worker   };
39*6777b538SAndroid Build Coastguard Worker 
40*6777b538SAndroid Build Coastguard Worker   // Return true if |opcode| is one of the data opcodes known to this
41*6777b538SAndroid Build Coastguard Worker   // implementation.
IsKnownDataOpCodeWebSocketFrameHeader42*6777b538SAndroid Build Coastguard Worker   static bool IsKnownDataOpCode(OpCode opcode) {
43*6777b538SAndroid Build Coastguard Worker     return opcode == kOpCodeContinuation || opcode == kOpCodeText ||
44*6777b538SAndroid Build Coastguard Worker            opcode == kOpCodeBinary;
45*6777b538SAndroid Build Coastguard Worker   }
46*6777b538SAndroid Build Coastguard Worker 
47*6777b538SAndroid Build Coastguard Worker   // Return true if |opcode| is one of the control opcodes known to this
48*6777b538SAndroid Build Coastguard Worker   // implementation.
IsKnownControlOpCodeWebSocketFrameHeader49*6777b538SAndroid Build Coastguard Worker   static bool IsKnownControlOpCode(OpCode opcode) {
50*6777b538SAndroid Build Coastguard Worker     return opcode == kOpCodeClose || opcode == kOpCodePing ||
51*6777b538SAndroid Build Coastguard Worker            opcode == kOpCodePong;
52*6777b538SAndroid Build Coastguard Worker   }
53*6777b538SAndroid Build Coastguard Worker 
54*6777b538SAndroid Build Coastguard Worker   // These values must be compile-time constants.
55*6777b538SAndroid Build Coastguard Worker   static constexpr size_t kBaseHeaderSize = 2;
56*6777b538SAndroid Build Coastguard Worker   static constexpr size_t kMaximumExtendedLengthSize = 8;
57*6777b538SAndroid Build Coastguard Worker   static constexpr size_t kMaskingKeyLength = 4;
58*6777b538SAndroid Build Coastguard Worker 
59*6777b538SAndroid Build Coastguard Worker   // Contains four-byte data representing "masking key" of WebSocket frames.
60*6777b538SAndroid Build Coastguard Worker   struct WebSocketMaskingKey {
61*6777b538SAndroid Build Coastguard Worker     uint8_t key[WebSocketFrameHeader::kMaskingKeyLength];
62*6777b538SAndroid Build Coastguard Worker   };
63*6777b538SAndroid Build Coastguard Worker 
64*6777b538SAndroid Build Coastguard Worker   // Constructor to avoid a lot of repetitive initialisation.
WebSocketFrameHeaderWebSocketFrameHeader65*6777b538SAndroid Build Coastguard Worker   explicit WebSocketFrameHeader(OpCode opCode) : opcode(opCode) {}
66*6777b538SAndroid Build Coastguard Worker 
67*6777b538SAndroid Build Coastguard Worker   WebSocketFrameHeader(const WebSocketFrameHeader&) = delete;
68*6777b538SAndroid Build Coastguard Worker   WebSocketFrameHeader& operator=(const WebSocketFrameHeader&) = delete;
69*6777b538SAndroid Build Coastguard Worker 
70*6777b538SAndroid Build Coastguard Worker   // Create a clone of this object on the heap.
71*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<WebSocketFrameHeader> Clone() const;
72*6777b538SAndroid Build Coastguard Worker 
73*6777b538SAndroid Build Coastguard Worker   // Overwrite this object with the fields from |source|.
74*6777b538SAndroid Build Coastguard Worker   void CopyFrom(const WebSocketFrameHeader& source);
75*6777b538SAndroid Build Coastguard Worker 
76*6777b538SAndroid Build Coastguard Worker   // Members below correspond to each item in WebSocket frame header.
77*6777b538SAndroid Build Coastguard Worker   // See <http://tools.ietf.org/html/rfc6455#section-5.2> for details.
78*6777b538SAndroid Build Coastguard Worker   bool final = false;
79*6777b538SAndroid Build Coastguard Worker   bool reserved1 = false;
80*6777b538SAndroid Build Coastguard Worker   bool reserved2 = false;
81*6777b538SAndroid Build Coastguard Worker   bool reserved3 = false;
82*6777b538SAndroid Build Coastguard Worker   OpCode opcode;
83*6777b538SAndroid Build Coastguard Worker   bool masked = false;
84*6777b538SAndroid Build Coastguard Worker   WebSocketMaskingKey masking_key = {};
85*6777b538SAndroid Build Coastguard Worker   uint64_t payload_length = 0;
86*6777b538SAndroid Build Coastguard Worker };
87*6777b538SAndroid Build Coastguard Worker 
88*6777b538SAndroid Build Coastguard Worker // Contains an entire WebSocket frame including payload. This is used by APIs
89*6777b538SAndroid Build Coastguard Worker // that are not concerned about retaining the original frame boundaries (because
90*6777b538SAndroid Build Coastguard Worker // frames may need to be split in order for the data to fit in memory).
91*6777b538SAndroid Build Coastguard Worker struct NET_EXPORT_PRIVATE WebSocketFrame {
92*6777b538SAndroid Build Coastguard Worker   // A frame must always have an opcode, so this parameter is compulsory.
93*6777b538SAndroid Build Coastguard Worker   explicit WebSocketFrame(WebSocketFrameHeader::OpCode opcode);
94*6777b538SAndroid Build Coastguard Worker 
95*6777b538SAndroid Build Coastguard Worker   WebSocketFrame(const WebSocketFrame&) = delete;
96*6777b538SAndroid Build Coastguard Worker   WebSocketFrame& operator=(const WebSocketFrame&) = delete;
97*6777b538SAndroid Build Coastguard Worker 
98*6777b538SAndroid Build Coastguard Worker   ~WebSocketFrame();
99*6777b538SAndroid Build Coastguard Worker 
100*6777b538SAndroid Build Coastguard Worker   // |header| is always present.
101*6777b538SAndroid Build Coastguard Worker   WebSocketFrameHeader header;
102*6777b538SAndroid Build Coastguard Worker 
103*6777b538SAndroid Build Coastguard Worker   // |payload| is always unmasked even if the frame is masked. The size of
104*6777b538SAndroid Build Coastguard Worker   // |payload| is given by |header.payload_length|.
105*6777b538SAndroid Build Coastguard Worker   // The lifetime of |payload| is not defined by WebSocketFrameChunk. It is the
106*6777b538SAndroid Build Coastguard Worker   // responsibility of the creator to ensure it remains valid for the lifetime
107*6777b538SAndroid Build Coastguard Worker   // of this object. This should be documented in the code that creates this
108*6777b538SAndroid Build Coastguard Worker   // object.
109*6777b538SAndroid Build Coastguard Worker   // TODO(crbug.com/1001915): Find more better way to clarify the life cycle.
110*6777b538SAndroid Build Coastguard Worker   const char* payload = nullptr;
111*6777b538SAndroid Build Coastguard Worker };
112*6777b538SAndroid Build Coastguard Worker 
113*6777b538SAndroid Build Coastguard Worker // Structure describing one chunk of a WebSocket frame.
114*6777b538SAndroid Build Coastguard Worker //
115*6777b538SAndroid Build Coastguard Worker // The payload of a WebSocket frame may be divided into multiple chunks.
116*6777b538SAndroid Build Coastguard Worker // You need to look at |final_chunk| member variable to detect the end of a
117*6777b538SAndroid Build Coastguard Worker // series of chunk objects of a WebSocket frame.
118*6777b538SAndroid Build Coastguard Worker //
119*6777b538SAndroid Build Coastguard Worker // Frame dissection is necessary to handle frames that are too large to store in
120*6777b538SAndroid Build Coastguard Worker // the browser memory without losing information about the frame boundaries. In
121*6777b538SAndroid Build Coastguard Worker // practice, most code does not need to worry about the original frame
122*6777b538SAndroid Build Coastguard Worker // boundaries and can use the WebSocketFrame type declared above.
123*6777b538SAndroid Build Coastguard Worker //
124*6777b538SAndroid Build Coastguard Worker // Users of this struct should treat WebSocket frames as a data stream; it's
125*6777b538SAndroid Build Coastguard Worker // important to keep the frame data flowing, especially in the browser process.
126*6777b538SAndroid Build Coastguard Worker // Users should not let the data stuck somewhere in the pipeline.
127*6777b538SAndroid Build Coastguard Worker //
128*6777b538SAndroid Build Coastguard Worker // This struct is used for reading WebSocket frame data (created by
129*6777b538SAndroid Build Coastguard Worker // WebSocketFrameParser). To construct WebSocket frames, use functions below.
130*6777b538SAndroid Build Coastguard Worker struct NET_EXPORT WebSocketFrameChunk {
131*6777b538SAndroid Build Coastguard Worker   WebSocketFrameChunk();
132*6777b538SAndroid Build Coastguard Worker 
133*6777b538SAndroid Build Coastguard Worker   WebSocketFrameChunk(const WebSocketFrameChunk&) = delete;
134*6777b538SAndroid Build Coastguard Worker   WebSocketFrameChunk& operator=(const WebSocketFrameChunk&) = delete;
135*6777b538SAndroid Build Coastguard Worker 
136*6777b538SAndroid Build Coastguard Worker   ~WebSocketFrameChunk();
137*6777b538SAndroid Build Coastguard Worker 
138*6777b538SAndroid Build Coastguard Worker   // Non-null |header| is provided only if this chunk is the first part of
139*6777b538SAndroid Build Coastguard Worker   // a series of chunks.
140*6777b538SAndroid Build Coastguard Worker   std::unique_ptr<WebSocketFrameHeader> header;
141*6777b538SAndroid Build Coastguard Worker 
142*6777b538SAndroid Build Coastguard Worker   // Indicates this part is the last chunk of a frame.
143*6777b538SAndroid Build Coastguard Worker   bool final_chunk = false;
144*6777b538SAndroid Build Coastguard Worker 
145*6777b538SAndroid Build Coastguard Worker   // |payload| is always unmasked even if the frame is masked. |payload| might
146*6777b538SAndroid Build Coastguard Worker   // be empty in the first chunk.
147*6777b538SAndroid Build Coastguard Worker   // The lifetime of |payload| is not defined by WebSocketFrameChunk. It is the
148*6777b538SAndroid Build Coastguard Worker   // responsibility of the creator to ensure it remains valid for the lifetime
149*6777b538SAndroid Build Coastguard Worker   // of this object. This should be documented in the code that creates this
150*6777b538SAndroid Build Coastguard Worker   // object.
151*6777b538SAndroid Build Coastguard Worker   // TODO(crbug.com/1001915): Find more better way to clarify the life cycle.
152*6777b538SAndroid Build Coastguard Worker   base::span<const char> payload;
153*6777b538SAndroid Build Coastguard Worker };
154*6777b538SAndroid Build Coastguard Worker 
155*6777b538SAndroid Build Coastguard Worker using WebSocketMaskingKey = WebSocketFrameHeader::WebSocketMaskingKey;
156*6777b538SAndroid Build Coastguard Worker 
157*6777b538SAndroid Build Coastguard Worker // Returns the size of WebSocket frame header. The size of WebSocket frame
158*6777b538SAndroid Build Coastguard Worker // header varies from 2 bytes to 14 bytes depending on the payload length
159*6777b538SAndroid Build Coastguard Worker // and maskedness.
160*6777b538SAndroid Build Coastguard Worker NET_EXPORT size_t
161*6777b538SAndroid Build Coastguard Worker GetWebSocketFrameHeaderSize(const WebSocketFrameHeader& header);
162*6777b538SAndroid Build Coastguard Worker 
163*6777b538SAndroid Build Coastguard Worker // Writes wire format of a WebSocket frame header into |output|, and returns
164*6777b538SAndroid Build Coastguard Worker // the number of bytes written.
165*6777b538SAndroid Build Coastguard Worker //
166*6777b538SAndroid Build Coastguard Worker // WebSocket frame format is defined at:
167*6777b538SAndroid Build Coastguard Worker // <http://tools.ietf.org/html/rfc6455#section-5.2>. This function writes
168*6777b538SAndroid Build Coastguard Worker // everything but payload data in a WebSocket frame to |buffer|.
169*6777b538SAndroid Build Coastguard Worker //
170*6777b538SAndroid Build Coastguard Worker // If |header->masked| is true, |masking_key| must point to a valid
171*6777b538SAndroid Build Coastguard Worker // WebSocketMaskingKey object containing the masking key for that frame
172*6777b538SAndroid Build Coastguard Worker // (possibly generated by GenerateWebSocketMaskingKey() function below).
173*6777b538SAndroid Build Coastguard Worker // Otherwise, |masking_key| must be NULL.
174*6777b538SAndroid Build Coastguard Worker //
175*6777b538SAndroid Build Coastguard Worker // |buffer| should have enough size to contain the frame header.
176*6777b538SAndroid Build Coastguard Worker // GetWebSocketFrameHeaderSize() can be used to know the size of header
177*6777b538SAndroid Build Coastguard Worker // beforehand. If the size of |buffer| is insufficient, this function returns
178*6777b538SAndroid Build Coastguard Worker // ERR_INVALID_ARGUMENT and does not write any data to |buffer|.
179*6777b538SAndroid Build Coastguard Worker NET_EXPORT int WriteWebSocketFrameHeader(const WebSocketFrameHeader& header,
180*6777b538SAndroid Build Coastguard Worker                                          const WebSocketMaskingKey* masking_key,
181*6777b538SAndroid Build Coastguard Worker                                          char* buffer,
182*6777b538SAndroid Build Coastguard Worker                                          int buffer_size);
183*6777b538SAndroid Build Coastguard Worker 
184*6777b538SAndroid Build Coastguard Worker // Generates a masking key suitable for use in a new WebSocket frame.
185*6777b538SAndroid Build Coastguard Worker NET_EXPORT WebSocketMaskingKey GenerateWebSocketMaskingKey();
186*6777b538SAndroid Build Coastguard Worker 
187*6777b538SAndroid Build Coastguard Worker // Masks WebSocket frame payload.
188*6777b538SAndroid Build Coastguard Worker //
189*6777b538SAndroid Build Coastguard Worker // A client must mask every WebSocket frame by XOR'ing the frame payload
190*6777b538SAndroid Build Coastguard Worker // with four-byte random data (masking key). This function applies the masking
191*6777b538SAndroid Build Coastguard Worker // to the given payload data.
192*6777b538SAndroid Build Coastguard Worker //
193*6777b538SAndroid Build Coastguard Worker // This function masks |data| with |masking_key|, assuming |data| is partial
194*6777b538SAndroid Build Coastguard Worker // data starting from |frame_offset| bytes from the beginning of the payload
195*6777b538SAndroid Build Coastguard Worker // data.
196*6777b538SAndroid Build Coastguard Worker //
197*6777b538SAndroid Build Coastguard Worker // Since frame masking is a reversible operation, this function can also be
198*6777b538SAndroid Build Coastguard Worker // used for unmasking a WebSocket frame.
199*6777b538SAndroid Build Coastguard Worker NET_EXPORT void MaskWebSocketFramePayload(
200*6777b538SAndroid Build Coastguard Worker     const WebSocketMaskingKey& masking_key,
201*6777b538SAndroid Build Coastguard Worker     uint64_t frame_offset,
202*6777b538SAndroid Build Coastguard Worker     char* data,
203*6777b538SAndroid Build Coastguard Worker     int data_size);
204*6777b538SAndroid Build Coastguard Worker 
205*6777b538SAndroid Build Coastguard Worker }  // namespace net
206*6777b538SAndroid Build Coastguard Worker 
207*6777b538SAndroid Build Coastguard Worker #endif  // NET_WEBSOCKETS_WEBSOCKET_FRAME_H_
208