xref: /aosp_15_r20/external/libwebsockets/lib/roles/http/compression/stream.c (revision 1c60b9aca93fdbc9b5f19b2d2194c91294b22281)
1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker  * libwebsockets - small server side websockets and web server implementation
3*1c60b9acSAndroid Build Coastguard Worker  *
4*1c60b9acSAndroid Build Coastguard Worker  * Copyright (C) 2010 - 2019 Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker  *
6*1c60b9acSAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a copy
7*1c60b9acSAndroid Build Coastguard Worker  * of this software and associated documentation files (the "Software"), to
8*1c60b9acSAndroid Build Coastguard Worker  * deal in the Software without restriction, including without limitation the
9*1c60b9acSAndroid Build Coastguard Worker  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10*1c60b9acSAndroid Build Coastguard Worker  * sell copies of the Software, and to permit persons to whom the Software is
11*1c60b9acSAndroid Build Coastguard Worker  * furnished to do so, subject to the following conditions:
12*1c60b9acSAndroid Build Coastguard Worker  *
13*1c60b9acSAndroid Build Coastguard Worker  * The above copyright notice and this permission notice shall be included in
14*1c60b9acSAndroid Build Coastguard Worker  * all copies or substantial portions of the Software.
15*1c60b9acSAndroid Build Coastguard Worker  *
16*1c60b9acSAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*1c60b9acSAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*1c60b9acSAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*1c60b9acSAndroid Build Coastguard Worker  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*1c60b9acSAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*1c60b9acSAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*1c60b9acSAndroid Build Coastguard Worker  * IN THE SOFTWARE.
23*1c60b9acSAndroid Build Coastguard Worker  */
24*1c60b9acSAndroid Build Coastguard Worker 
25*1c60b9acSAndroid Build Coastguard Worker #include "private-lib-core.h"
26*1c60b9acSAndroid Build Coastguard Worker 
27*1c60b9acSAndroid Build Coastguard Worker /* compression methods listed in order of preference */
28*1c60b9acSAndroid Build Coastguard Worker 
29*1c60b9acSAndroid Build Coastguard Worker struct lws_compression_support *lcs_available[] = {
30*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_HTTP_BROTLI)
31*1c60b9acSAndroid Build Coastguard Worker 	&lcs_brotli,
32*1c60b9acSAndroid Build Coastguard Worker #endif
33*1c60b9acSAndroid Build Coastguard Worker 	&lcs_deflate,
34*1c60b9acSAndroid Build Coastguard Worker };
35*1c60b9acSAndroid Build Coastguard Worker 
36*1c60b9acSAndroid Build Coastguard Worker /* compute acceptable compression encodings while we still have an ah */
37*1c60b9acSAndroid Build Coastguard Worker 
38*1c60b9acSAndroid Build Coastguard Worker int
lws_http_compression_validate(struct lws * wsi)39*1c60b9acSAndroid Build Coastguard Worker lws_http_compression_validate(struct lws *wsi)
40*1c60b9acSAndroid Build Coastguard Worker {
41*1c60b9acSAndroid Build Coastguard Worker 	const char *a;
42*1c60b9acSAndroid Build Coastguard Worker 	size_t n;
43*1c60b9acSAndroid Build Coastguard Worker 
44*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.comp_accept_mask = 0;
45*1c60b9acSAndroid Build Coastguard Worker 
46*1c60b9acSAndroid Build Coastguard Worker 	if (!wsi->http.ah || !lwsi_role_server(wsi))
47*1c60b9acSAndroid Build Coastguard Worker 		return 0;
48*1c60b9acSAndroid Build Coastguard Worker 
49*1c60b9acSAndroid Build Coastguard Worker 	a = lws_hdr_simple_ptr(wsi, WSI_TOKEN_HTTP_ACCEPT_ENCODING);
50*1c60b9acSAndroid Build Coastguard Worker 	if (!a)
51*1c60b9acSAndroid Build Coastguard Worker 		return 0;
52*1c60b9acSAndroid Build Coastguard Worker 
53*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < LWS_ARRAY_SIZE(lcs_available); n++)
54*1c60b9acSAndroid Build Coastguard Worker 		if (strstr(a, lcs_available[n]->encoding_name))
55*1c60b9acSAndroid Build Coastguard Worker 			wsi->http.comp_accept_mask = (uint8_t)(wsi->http.comp_accept_mask | (1 << n));
56*1c60b9acSAndroid Build Coastguard Worker 
57*1c60b9acSAndroid Build Coastguard Worker 	return 0;
58*1c60b9acSAndroid Build Coastguard Worker }
59*1c60b9acSAndroid Build Coastguard Worker 
60*1c60b9acSAndroid Build Coastguard Worker int
lws_http_compression_apply(struct lws * wsi,const char * name,unsigned char ** p,unsigned char * end,char decomp)61*1c60b9acSAndroid Build Coastguard Worker lws_http_compression_apply(struct lws *wsi, const char *name,
62*1c60b9acSAndroid Build Coastguard Worker 			   unsigned char **p, unsigned char *end, char decomp)
63*1c60b9acSAndroid Build Coastguard Worker {
64*1c60b9acSAndroid Build Coastguard Worker 	size_t n;
65*1c60b9acSAndroid Build Coastguard Worker 
66*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < LWS_ARRAY_SIZE(lcs_available); n++) {
67*1c60b9acSAndroid Build Coastguard Worker 		/* if name is non-NULL, choose only that compression method */
68*1c60b9acSAndroid Build Coastguard Worker 		if (name && !strcmp(lcs_available[n]->encoding_name, name))
69*1c60b9acSAndroid Build Coastguard Worker 			continue;
70*1c60b9acSAndroid Build Coastguard Worker 		/*
71*1c60b9acSAndroid Build Coastguard Worker 		 * If we're the server, confirm that the client told us he could
72*1c60b9acSAndroid Build Coastguard Worker 		 * handle this kind of compression transform...
73*1c60b9acSAndroid Build Coastguard Worker 		 */
74*1c60b9acSAndroid Build Coastguard Worker 		if (!decomp && !(wsi->http.comp_accept_mask & (1 << n)))
75*1c60b9acSAndroid Build Coastguard Worker 			continue;
76*1c60b9acSAndroid Build Coastguard Worker 
77*1c60b9acSAndroid Build Coastguard Worker 		/* let's go with this one then... */
78*1c60b9acSAndroid Build Coastguard Worker 		break;
79*1c60b9acSAndroid Build Coastguard Worker 	}
80*1c60b9acSAndroid Build Coastguard Worker 
81*1c60b9acSAndroid Build Coastguard Worker 	if (n == LWS_ARRAY_SIZE(lcs_available))
82*1c60b9acSAndroid Build Coastguard Worker 		return 1;
83*1c60b9acSAndroid Build Coastguard Worker 
84*1c60b9acSAndroid Build Coastguard Worker 	lcs_available[n]->init_compression(&wsi->http.comp_ctx, decomp);
85*1c60b9acSAndroid Build Coastguard Worker 	if (!wsi->http.comp_ctx.u.generic_ctx_ptr) {
86*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: init_compression %d failed\n", __func__, (int)n);
87*1c60b9acSAndroid Build Coastguard Worker 		return 1;
88*1c60b9acSAndroid Build Coastguard Worker 	}
89*1c60b9acSAndroid Build Coastguard Worker 
90*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.lcs = lcs_available[n];
91*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.comp_ctx.may_have_more = 0;
92*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.comp_ctx.final_on_input_side = 0;
93*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.comp_ctx.chunking = 0;
94*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.comp_ctx.is_decompression = !!decomp;
95*1c60b9acSAndroid Build Coastguard Worker 
96*1c60b9acSAndroid Build Coastguard Worker 	if (lws_add_http_header_by_token(wsi, WSI_TOKEN_HTTP_CONTENT_ENCODING,
97*1c60b9acSAndroid Build Coastguard Worker 			(unsigned char *)lcs_available[n]->encoding_name,
98*1c60b9acSAndroid Build Coastguard Worker 			(int)strlen(lcs_available[n]->encoding_name), p, end))
99*1c60b9acSAndroid Build Coastguard Worker 		return -1;
100*1c60b9acSAndroid Build Coastguard Worker 
101*1c60b9acSAndroid Build Coastguard Worker 	lwsl_info("%s: %s: applied %s content-encoding\n", __func__,
102*1c60b9acSAndroid Build Coastguard Worker 		    lws_wsi_tag(wsi), lcs_available[n]->encoding_name);
103*1c60b9acSAndroid Build Coastguard Worker 
104*1c60b9acSAndroid Build Coastguard Worker 	return 0;
105*1c60b9acSAndroid Build Coastguard Worker }
106*1c60b9acSAndroid Build Coastguard Worker 
107*1c60b9acSAndroid Build Coastguard Worker void
lws_http_compression_destroy(struct lws * wsi)108*1c60b9acSAndroid Build Coastguard Worker lws_http_compression_destroy(struct lws *wsi)
109*1c60b9acSAndroid Build Coastguard Worker {
110*1c60b9acSAndroid Build Coastguard Worker 	if (!wsi->http.lcs || !wsi->http.comp_ctx.u.generic_ctx_ptr)
111*1c60b9acSAndroid Build Coastguard Worker 		return;
112*1c60b9acSAndroid Build Coastguard Worker 
113*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.lcs->destroy(&wsi->http.comp_ctx);
114*1c60b9acSAndroid Build Coastguard Worker 
115*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.lcs = NULL;
116*1c60b9acSAndroid Build Coastguard Worker }
117*1c60b9acSAndroid Build Coastguard Worker 
118*1c60b9acSAndroid Build Coastguard Worker /*
119*1c60b9acSAndroid Build Coastguard Worker  * This manages the compression transform independent of h1 or h2.
120*1c60b9acSAndroid Build Coastguard Worker  *
121*1c60b9acSAndroid Build Coastguard Worker  * wsi->buflist_comp stashes pre-transform input that was not yet compressed
122*1c60b9acSAndroid Build Coastguard Worker  */
123*1c60b9acSAndroid Build Coastguard Worker 
124*1c60b9acSAndroid Build Coastguard Worker int
lws_http_compression_transform(struct lws * wsi,unsigned char * buf,size_t len,enum lws_write_protocol * wp,unsigned char ** outbuf,size_t * olen_oused)125*1c60b9acSAndroid Build Coastguard Worker lws_http_compression_transform(struct lws *wsi, unsigned char *buf,
126*1c60b9acSAndroid Build Coastguard Worker 			       size_t len, enum lws_write_protocol *wp,
127*1c60b9acSAndroid Build Coastguard Worker 			       unsigned char **outbuf, size_t *olen_oused)
128*1c60b9acSAndroid Build Coastguard Worker {
129*1c60b9acSAndroid Build Coastguard Worker 	size_t ilen_iused = len;
130*1c60b9acSAndroid Build Coastguard Worker 	int n, use = 0, wp1f = (*wp) & 0x1f;
131*1c60b9acSAndroid Build Coastguard Worker 	lws_comp_ctx_t *ctx = &wsi->http.comp_ctx;
132*1c60b9acSAndroid Build Coastguard Worker 
133*1c60b9acSAndroid Build Coastguard Worker 	ctx->may_have_more = 0;
134*1c60b9acSAndroid Build Coastguard Worker 
135*1c60b9acSAndroid Build Coastguard Worker 	if (!wsi->http.lcs ||
136*1c60b9acSAndroid Build Coastguard Worker 	    (wp1f != LWS_WRITE_HTTP && wp1f != LWS_WRITE_HTTP_FINAL)) {
137*1c60b9acSAndroid Build Coastguard Worker 		*outbuf = buf;
138*1c60b9acSAndroid Build Coastguard Worker 		*olen_oused = len;
139*1c60b9acSAndroid Build Coastguard Worker 
140*1c60b9acSAndroid Build Coastguard Worker 		return 0;
141*1c60b9acSAndroid Build Coastguard Worker 	}
142*1c60b9acSAndroid Build Coastguard Worker 
143*1c60b9acSAndroid Build Coastguard Worker 	if (wp1f == LWS_WRITE_HTTP_FINAL) {
144*1c60b9acSAndroid Build Coastguard Worker 		/*
145*1c60b9acSAndroid Build Coastguard Worker 		 * ...we may get a large buffer that represents the final input
146*1c60b9acSAndroid Build Coastguard Worker 		 * buffer, but it may form multiple frames after being
147*1c60b9acSAndroid Build Coastguard Worker 		 * tranformed by compression; only the last of those is actually
148*1c60b9acSAndroid Build Coastguard Worker 		 * the final frame on the output stream.
149*1c60b9acSAndroid Build Coastguard Worker 		 *
150*1c60b9acSAndroid Build Coastguard Worker 		 * Note that we have received the FINAL input, and downgrade it
151*1c60b9acSAndroid Build Coastguard Worker 		 * to a non-final for now.
152*1c60b9acSAndroid Build Coastguard Worker 		 */
153*1c60b9acSAndroid Build Coastguard Worker 		ctx->final_on_input_side = 1;
154*1c60b9acSAndroid Build Coastguard Worker 		*wp = (unsigned int)(LWS_WRITE_HTTP | ((*wp) & ~0x1fu));
155*1c60b9acSAndroid Build Coastguard Worker 	}
156*1c60b9acSAndroid Build Coastguard Worker 
157*1c60b9acSAndroid Build Coastguard Worker 	if (ctx->buflist_comp) {
158*1c60b9acSAndroid Build Coastguard Worker 		/*
159*1c60b9acSAndroid Build Coastguard Worker 		 * we can't send this new stuff when we have old stuff
160*1c60b9acSAndroid Build Coastguard Worker 		 * buffered and not compressed yet.  Add it to the tail
161*1c60b9acSAndroid Build Coastguard Worker 		 * and switch to trying to process the head.
162*1c60b9acSAndroid Build Coastguard Worker 		 */
163*1c60b9acSAndroid Build Coastguard Worker 		if (buf && len) {
164*1c60b9acSAndroid Build Coastguard Worker 			if (lws_buflist_append_segment(
165*1c60b9acSAndroid Build Coastguard Worker 					&ctx->buflist_comp, buf, len) < 0)
166*1c60b9acSAndroid Build Coastguard Worker 				return -1;
167*1c60b9acSAndroid Build Coastguard Worker 			lwsl_debug("%s: %s: adding %d to comp buflist\n",
168*1c60b9acSAndroid Build Coastguard Worker 				   __func__, lws_wsi_tag(wsi), (int)len);
169*1c60b9acSAndroid Build Coastguard Worker 		}
170*1c60b9acSAndroid Build Coastguard Worker 
171*1c60b9acSAndroid Build Coastguard Worker 		len = lws_buflist_next_segment_len(&ctx->buflist_comp, &buf);
172*1c60b9acSAndroid Build Coastguard Worker 		ilen_iused = len;
173*1c60b9acSAndroid Build Coastguard Worker 		use = 1;
174*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: %s: trying comp buflist %d\n", __func__,
175*1c60b9acSAndroid Build Coastguard Worker 				lws_wsi_tag(wsi), (int)len);
176*1c60b9acSAndroid Build Coastguard Worker 	}
177*1c60b9acSAndroid Build Coastguard Worker 
178*1c60b9acSAndroid Build Coastguard Worker 	if (!buf && ilen_iused)
179*1c60b9acSAndroid Build Coastguard Worker 		return 0;
180*1c60b9acSAndroid Build Coastguard Worker 
181*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: %s: pre-process: ilen_iused %d, olen_oused %d\n",
182*1c60b9acSAndroid Build Coastguard Worker 		   __func__, lws_wsi_tag(wsi), (int)ilen_iused, (int)*olen_oused);
183*1c60b9acSAndroid Build Coastguard Worker 
184*1c60b9acSAndroid Build Coastguard Worker 	n = wsi->http.lcs->process(ctx, buf, &ilen_iused, *outbuf, olen_oused);
185*1c60b9acSAndroid Build Coastguard Worker 
186*1c60b9acSAndroid Build Coastguard Worker 	if (n && n != 1) {
187*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: problem with compression\n", __func__);
188*1c60b9acSAndroid Build Coastguard Worker 
189*1c60b9acSAndroid Build Coastguard Worker 		return -1;
190*1c60b9acSAndroid Build Coastguard Worker 	}
191*1c60b9acSAndroid Build Coastguard Worker 
192*1c60b9acSAndroid Build Coastguard Worker 	if (!ctx->may_have_more && ctx->final_on_input_side)
193*1c60b9acSAndroid Build Coastguard Worker 
194*1c60b9acSAndroid Build Coastguard Worker 		*wp = (unsigned int)(LWS_WRITE_HTTP_FINAL | ((*wp) & ~0x1fu));
195*1c60b9acSAndroid Build Coastguard Worker 
196*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: %s: more %d, ilen_iused %d\n", __func__, lws_wsi_tag(wsi),
197*1c60b9acSAndroid Build Coastguard Worker 		   ctx->may_have_more, (int)ilen_iused);
198*1c60b9acSAndroid Build Coastguard Worker 
199*1c60b9acSAndroid Build Coastguard Worker 	if (use && ilen_iused) {
200*1c60b9acSAndroid Build Coastguard Worker 		/*
201*1c60b9acSAndroid Build Coastguard Worker 		 * we were flushing stuff from the buflist head... account for
202*1c60b9acSAndroid Build Coastguard Worker 		 * however much actually got processed by the compression
203*1c60b9acSAndroid Build Coastguard Worker 		 * transform
204*1c60b9acSAndroid Build Coastguard Worker 		 */
205*1c60b9acSAndroid Build Coastguard Worker 		lws_buflist_use_segment(&ctx->buflist_comp, ilen_iused);
206*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: %s: marking %d of comp buflist as used "
207*1c60b9acSAndroid Build Coastguard Worker 			   "(ctx->buflist_comp %p)\n", __func__,
208*1c60b9acSAndroid Build Coastguard Worker 			   lws_wsi_tag(wsi), (int)len, ctx->buflist_comp);
209*1c60b9acSAndroid Build Coastguard Worker 	}
210*1c60b9acSAndroid Build Coastguard Worker 
211*1c60b9acSAndroid Build Coastguard Worker 	if (!use && ilen_iused != len) {
212*1c60b9acSAndroid Build Coastguard Worker 		 /*
213*1c60b9acSAndroid Build Coastguard Worker 		  * ...we were sending stuff from the caller directly and not
214*1c60b9acSAndroid Build Coastguard Worker 		  * all of it got processed... stash on the buflist tail
215*1c60b9acSAndroid Build Coastguard Worker 		  */
216*1c60b9acSAndroid Build Coastguard Worker 		if (lws_buflist_append_segment(&ctx->buflist_comp,
217*1c60b9acSAndroid Build Coastguard Worker 					   buf + ilen_iused, len - ilen_iused) < 0)
218*1c60b9acSAndroid Build Coastguard Worker 			return -1;
219*1c60b9acSAndroid Build Coastguard Worker 
220*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: buffering %d unused comp input\n", __func__,
221*1c60b9acSAndroid Build Coastguard Worker 			   (int)(len - ilen_iused));
222*1c60b9acSAndroid Build Coastguard Worker 	}
223*1c60b9acSAndroid Build Coastguard Worker 	if (ctx->buflist_comp || ctx->may_have_more)
224*1c60b9acSAndroid Build Coastguard Worker 		lws_callback_on_writable(wsi);
225*1c60b9acSAndroid Build Coastguard Worker 
226*1c60b9acSAndroid Build Coastguard Worker 	return 0;
227*1c60b9acSAndroid Build Coastguard Worker }
228