xref: /aosp_15_r20/external/curl/lib/http_proxy.c (revision 6236dae45794135f37c4eb022389c904c8b0090d)
1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker  *                                  _   _ ____  _
3*6236dae4SAndroid Build Coastguard Worker  *  Project                     ___| | | |  _ \| |
4*6236dae4SAndroid Build Coastguard Worker  *                             / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker  *                            | (__| |_| |  _ <| |___
6*6236dae4SAndroid Build Coastguard Worker  *                             \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker  *
8*6236dae4SAndroid Build Coastguard Worker  * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker  *
10*6236dae4SAndroid Build Coastguard Worker  * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker  * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker  * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker  *
14*6236dae4SAndroid Build Coastguard Worker  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker  * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker  * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker  *
18*6236dae4SAndroid Build Coastguard Worker  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker  * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker  *
21*6236dae4SAndroid Build Coastguard Worker  * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker  *
23*6236dae4SAndroid Build Coastguard Worker  ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker 
25*6236dae4SAndroid Build Coastguard Worker #include "curl_setup.h"
26*6236dae4SAndroid Build Coastguard Worker 
27*6236dae4SAndroid Build Coastguard Worker #include "http_proxy.h"
28*6236dae4SAndroid Build Coastguard Worker 
29*6236dae4SAndroid Build Coastguard Worker #if !defined(CURL_DISABLE_HTTP) && !defined(CURL_DISABLE_PROXY)
30*6236dae4SAndroid Build Coastguard Worker 
31*6236dae4SAndroid Build Coastguard Worker #include <curl/curl.h>
32*6236dae4SAndroid Build Coastguard Worker #ifdef USE_HYPER
33*6236dae4SAndroid Build Coastguard Worker #include <hyper.h>
34*6236dae4SAndroid Build Coastguard Worker #endif
35*6236dae4SAndroid Build Coastguard Worker #include "sendf.h"
36*6236dae4SAndroid Build Coastguard Worker #include "http.h"
37*6236dae4SAndroid Build Coastguard Worker #include "url.h"
38*6236dae4SAndroid Build Coastguard Worker #include "select.h"
39*6236dae4SAndroid Build Coastguard Worker #include "progress.h"
40*6236dae4SAndroid Build Coastguard Worker #include "cfilters.h"
41*6236dae4SAndroid Build Coastguard Worker #include "cf-h1-proxy.h"
42*6236dae4SAndroid Build Coastguard Worker #include "cf-h2-proxy.h"
43*6236dae4SAndroid Build Coastguard Worker #include "connect.h"
44*6236dae4SAndroid Build Coastguard Worker #include "curlx.h"
45*6236dae4SAndroid Build Coastguard Worker #include "vtls/vtls.h"
46*6236dae4SAndroid Build Coastguard Worker #include "transfer.h"
47*6236dae4SAndroid Build Coastguard Worker #include "multiif.h"
48*6236dae4SAndroid Build Coastguard Worker 
49*6236dae4SAndroid Build Coastguard Worker /* The last 3 #include files should be in this order */
50*6236dae4SAndroid Build Coastguard Worker #include "curl_printf.h"
51*6236dae4SAndroid Build Coastguard Worker #include "curl_memory.h"
52*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h"
53*6236dae4SAndroid Build Coastguard Worker 
54*6236dae4SAndroid Build Coastguard Worker 
Curl_http_proxy_get_destination(struct Curl_cfilter * cf,const char ** phostname,int * pport,bool * pipv6_ip)55*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_http_proxy_get_destination(struct Curl_cfilter *cf,
56*6236dae4SAndroid Build Coastguard Worker                                          const char **phostname,
57*6236dae4SAndroid Build Coastguard Worker                                          int *pport, bool *pipv6_ip)
58*6236dae4SAndroid Build Coastguard Worker {
59*6236dae4SAndroid Build Coastguard Worker   DEBUGASSERT(cf);
60*6236dae4SAndroid Build Coastguard Worker   DEBUGASSERT(cf->conn);
61*6236dae4SAndroid Build Coastguard Worker 
62*6236dae4SAndroid Build Coastguard Worker   if(cf->conn->bits.conn_to_host)
63*6236dae4SAndroid Build Coastguard Worker     *phostname = cf->conn->conn_to_host.name;
64*6236dae4SAndroid Build Coastguard Worker   else if(cf->sockindex == SECONDARYSOCKET)
65*6236dae4SAndroid Build Coastguard Worker     *phostname = cf->conn->secondaryhostname;
66*6236dae4SAndroid Build Coastguard Worker   else
67*6236dae4SAndroid Build Coastguard Worker     *phostname = cf->conn->host.name;
68*6236dae4SAndroid Build Coastguard Worker 
69*6236dae4SAndroid Build Coastguard Worker   if(cf->sockindex == SECONDARYSOCKET)
70*6236dae4SAndroid Build Coastguard Worker     *pport = cf->conn->secondary_port;
71*6236dae4SAndroid Build Coastguard Worker   else if(cf->conn->bits.conn_to_port)
72*6236dae4SAndroid Build Coastguard Worker     *pport = cf->conn->conn_to_port;
73*6236dae4SAndroid Build Coastguard Worker   else
74*6236dae4SAndroid Build Coastguard Worker     *pport = cf->conn->remote_port;
75*6236dae4SAndroid Build Coastguard Worker 
76*6236dae4SAndroid Build Coastguard Worker   if(*phostname != cf->conn->host.name)
77*6236dae4SAndroid Build Coastguard Worker     *pipv6_ip = (strchr(*phostname, ':') != NULL);
78*6236dae4SAndroid Build Coastguard Worker   else
79*6236dae4SAndroid Build Coastguard Worker     *pipv6_ip = cf->conn->bits.ipv6_ip;
80*6236dae4SAndroid Build Coastguard Worker 
81*6236dae4SAndroid Build Coastguard Worker   return CURLE_OK;
82*6236dae4SAndroid Build Coastguard Worker }
83*6236dae4SAndroid Build Coastguard Worker 
Curl_http_proxy_create_CONNECT(struct httpreq ** preq,struct Curl_cfilter * cf,struct Curl_easy * data,int http_version_major)84*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_http_proxy_create_CONNECT(struct httpreq **preq,
85*6236dae4SAndroid Build Coastguard Worker                                         struct Curl_cfilter *cf,
86*6236dae4SAndroid Build Coastguard Worker                                         struct Curl_easy *data,
87*6236dae4SAndroid Build Coastguard Worker                                         int http_version_major)
88*6236dae4SAndroid Build Coastguard Worker {
89*6236dae4SAndroid Build Coastguard Worker   const char *hostname = NULL;
90*6236dae4SAndroid Build Coastguard Worker   char *authority = NULL;
91*6236dae4SAndroid Build Coastguard Worker   int port;
92*6236dae4SAndroid Build Coastguard Worker   bool ipv6_ip;
93*6236dae4SAndroid Build Coastguard Worker   CURLcode result;
94*6236dae4SAndroid Build Coastguard Worker   struct httpreq *req = NULL;
95*6236dae4SAndroid Build Coastguard Worker 
96*6236dae4SAndroid Build Coastguard Worker   result = Curl_http_proxy_get_destination(cf, &hostname, &port, &ipv6_ip);
97*6236dae4SAndroid Build Coastguard Worker   if(result)
98*6236dae4SAndroid Build Coastguard Worker     goto out;
99*6236dae4SAndroid Build Coastguard Worker 
100*6236dae4SAndroid Build Coastguard Worker   authority = aprintf("%s%s%s:%d", ipv6_ip ? "[" : "", hostname,
101*6236dae4SAndroid Build Coastguard Worker                       ipv6_ip ?"]" : "", port);
102*6236dae4SAndroid Build Coastguard Worker   if(!authority) {
103*6236dae4SAndroid Build Coastguard Worker     result = CURLE_OUT_OF_MEMORY;
104*6236dae4SAndroid Build Coastguard Worker     goto out;
105*6236dae4SAndroid Build Coastguard Worker   }
106*6236dae4SAndroid Build Coastguard Worker 
107*6236dae4SAndroid Build Coastguard Worker   result = Curl_http_req_make(&req, "CONNECT", sizeof("CONNECT")-1,
108*6236dae4SAndroid Build Coastguard Worker                               NULL, 0, authority, strlen(authority),
109*6236dae4SAndroid Build Coastguard Worker                               NULL, 0);
110*6236dae4SAndroid Build Coastguard Worker   if(result)
111*6236dae4SAndroid Build Coastguard Worker     goto out;
112*6236dae4SAndroid Build Coastguard Worker 
113*6236dae4SAndroid Build Coastguard Worker   /* Setup the proxy-authorization header, if any */
114*6236dae4SAndroid Build Coastguard Worker   result = Curl_http_output_auth(data, cf->conn, req->method, HTTPREQ_GET,
115*6236dae4SAndroid Build Coastguard Worker                                  req->authority, TRUE);
116*6236dae4SAndroid Build Coastguard Worker   if(result)
117*6236dae4SAndroid Build Coastguard Worker     goto out;
118*6236dae4SAndroid Build Coastguard Worker 
119*6236dae4SAndroid Build Coastguard Worker   /* If user is not overriding Host: header, we add for HTTP/1.x */
120*6236dae4SAndroid Build Coastguard Worker   if(http_version_major == 1 &&
121*6236dae4SAndroid Build Coastguard Worker      !Curl_checkProxyheaders(data, cf->conn, STRCONST("Host"))) {
122*6236dae4SAndroid Build Coastguard Worker     result = Curl_dynhds_cadd(&req->headers, "Host", authority);
123*6236dae4SAndroid Build Coastguard Worker     if(result)
124*6236dae4SAndroid Build Coastguard Worker       goto out;
125*6236dae4SAndroid Build Coastguard Worker   }
126*6236dae4SAndroid Build Coastguard Worker 
127*6236dae4SAndroid Build Coastguard Worker   if(data->state.aptr.proxyuserpwd) {
128*6236dae4SAndroid Build Coastguard Worker     result = Curl_dynhds_h1_cadd_line(&req->headers,
129*6236dae4SAndroid Build Coastguard Worker                                       data->state.aptr.proxyuserpwd);
130*6236dae4SAndroid Build Coastguard Worker     if(result)
131*6236dae4SAndroid Build Coastguard Worker       goto out;
132*6236dae4SAndroid Build Coastguard Worker   }
133*6236dae4SAndroid Build Coastguard Worker 
134*6236dae4SAndroid Build Coastguard Worker   if(!Curl_checkProxyheaders(data, cf->conn, STRCONST("User-Agent")) &&
135*6236dae4SAndroid Build Coastguard Worker      data->set.str[STRING_USERAGENT] && *data->set.str[STRING_USERAGENT]) {
136*6236dae4SAndroid Build Coastguard Worker     result = Curl_dynhds_cadd(&req->headers, "User-Agent",
137*6236dae4SAndroid Build Coastguard Worker                               data->set.str[STRING_USERAGENT]);
138*6236dae4SAndroid Build Coastguard Worker     if(result)
139*6236dae4SAndroid Build Coastguard Worker       goto out;
140*6236dae4SAndroid Build Coastguard Worker   }
141*6236dae4SAndroid Build Coastguard Worker 
142*6236dae4SAndroid Build Coastguard Worker   if(http_version_major == 1 &&
143*6236dae4SAndroid Build Coastguard Worker     !Curl_checkProxyheaders(data, cf->conn, STRCONST("Proxy-Connection"))) {
144*6236dae4SAndroid Build Coastguard Worker     result = Curl_dynhds_cadd(&req->headers, "Proxy-Connection", "Keep-Alive");
145*6236dae4SAndroid Build Coastguard Worker     if(result)
146*6236dae4SAndroid Build Coastguard Worker       goto out;
147*6236dae4SAndroid Build Coastguard Worker   }
148*6236dae4SAndroid Build Coastguard Worker 
149*6236dae4SAndroid Build Coastguard Worker   result = Curl_dynhds_add_custom(data, TRUE, &req->headers);
150*6236dae4SAndroid Build Coastguard Worker 
151*6236dae4SAndroid Build Coastguard Worker out:
152*6236dae4SAndroid Build Coastguard Worker   if(result && req) {
153*6236dae4SAndroid Build Coastguard Worker     Curl_http_req_free(req);
154*6236dae4SAndroid Build Coastguard Worker     req = NULL;
155*6236dae4SAndroid Build Coastguard Worker   }
156*6236dae4SAndroid Build Coastguard Worker   free(authority);
157*6236dae4SAndroid Build Coastguard Worker   *preq = req;
158*6236dae4SAndroid Build Coastguard Worker   return result;
159*6236dae4SAndroid Build Coastguard Worker }
160*6236dae4SAndroid Build Coastguard Worker 
161*6236dae4SAndroid Build Coastguard Worker 
162*6236dae4SAndroid Build Coastguard Worker struct cf_proxy_ctx {
163*6236dae4SAndroid Build Coastguard Worker   /* the protocol specific sub-filter we install during connect */
164*6236dae4SAndroid Build Coastguard Worker   struct Curl_cfilter *cf_protocol;
165*6236dae4SAndroid Build Coastguard Worker };
166*6236dae4SAndroid Build Coastguard Worker 
http_proxy_cf_connect(struct Curl_cfilter * cf,struct Curl_easy * data,bool blocking,bool * done)167*6236dae4SAndroid Build Coastguard Worker static CURLcode http_proxy_cf_connect(struct Curl_cfilter *cf,
168*6236dae4SAndroid Build Coastguard Worker                                       struct Curl_easy *data,
169*6236dae4SAndroid Build Coastguard Worker                                       bool blocking, bool *done)
170*6236dae4SAndroid Build Coastguard Worker {
171*6236dae4SAndroid Build Coastguard Worker   struct cf_proxy_ctx *ctx = cf->ctx;
172*6236dae4SAndroid Build Coastguard Worker   CURLcode result;
173*6236dae4SAndroid Build Coastguard Worker 
174*6236dae4SAndroid Build Coastguard Worker   if(cf->connected) {
175*6236dae4SAndroid Build Coastguard Worker     *done = TRUE;
176*6236dae4SAndroid Build Coastguard Worker     return CURLE_OK;
177*6236dae4SAndroid Build Coastguard Worker   }
178*6236dae4SAndroid Build Coastguard Worker 
179*6236dae4SAndroid Build Coastguard Worker   CURL_TRC_CF(data, cf, "connect");
180*6236dae4SAndroid Build Coastguard Worker connect_sub:
181*6236dae4SAndroid Build Coastguard Worker   result = cf->next->cft->do_connect(cf->next, data, blocking, done);
182*6236dae4SAndroid Build Coastguard Worker   if(result || !*done)
183*6236dae4SAndroid Build Coastguard Worker     return result;
184*6236dae4SAndroid Build Coastguard Worker 
185*6236dae4SAndroid Build Coastguard Worker   *done = FALSE;
186*6236dae4SAndroid Build Coastguard Worker   if(!ctx->cf_protocol) {
187*6236dae4SAndroid Build Coastguard Worker     struct Curl_cfilter *cf_protocol = NULL;
188*6236dae4SAndroid Build Coastguard Worker     int alpn = Curl_conn_cf_is_ssl(cf->next) ?
189*6236dae4SAndroid Build Coastguard Worker       cf->conn->proxy_alpn : CURL_HTTP_VERSION_1_1;
190*6236dae4SAndroid Build Coastguard Worker 
191*6236dae4SAndroid Build Coastguard Worker     /* First time call after the subchain connected */
192*6236dae4SAndroid Build Coastguard Worker     switch(alpn) {
193*6236dae4SAndroid Build Coastguard Worker     case CURL_HTTP_VERSION_NONE:
194*6236dae4SAndroid Build Coastguard Worker     case CURL_HTTP_VERSION_1_0:
195*6236dae4SAndroid Build Coastguard Worker     case CURL_HTTP_VERSION_1_1:
196*6236dae4SAndroid Build Coastguard Worker       CURL_TRC_CF(data, cf, "installing subfilter for HTTP/1.1");
197*6236dae4SAndroid Build Coastguard Worker       infof(data, "CONNECT tunnel: HTTP/1.%d negotiated",
198*6236dae4SAndroid Build Coastguard Worker             (alpn == CURL_HTTP_VERSION_1_0) ? 0 : 1);
199*6236dae4SAndroid Build Coastguard Worker       result = Curl_cf_h1_proxy_insert_after(cf, data);
200*6236dae4SAndroid Build Coastguard Worker       if(result)
201*6236dae4SAndroid Build Coastguard Worker         goto out;
202*6236dae4SAndroid Build Coastguard Worker       cf_protocol = cf->next;
203*6236dae4SAndroid Build Coastguard Worker       break;
204*6236dae4SAndroid Build Coastguard Worker #ifdef USE_NGHTTP2
205*6236dae4SAndroid Build Coastguard Worker     case CURL_HTTP_VERSION_2:
206*6236dae4SAndroid Build Coastguard Worker       CURL_TRC_CF(data, cf, "installing subfilter for HTTP/2");
207*6236dae4SAndroid Build Coastguard Worker       infof(data, "CONNECT tunnel: HTTP/2 negotiated");
208*6236dae4SAndroid Build Coastguard Worker       result = Curl_cf_h2_proxy_insert_after(cf, data);
209*6236dae4SAndroid Build Coastguard Worker       if(result)
210*6236dae4SAndroid Build Coastguard Worker         goto out;
211*6236dae4SAndroid Build Coastguard Worker       cf_protocol = cf->next;
212*6236dae4SAndroid Build Coastguard Worker       break;
213*6236dae4SAndroid Build Coastguard Worker #endif
214*6236dae4SAndroid Build Coastguard Worker     default:
215*6236dae4SAndroid Build Coastguard Worker       infof(data, "CONNECT tunnel: unsupported ALPN(%d) negotiated", alpn);
216*6236dae4SAndroid Build Coastguard Worker       result = CURLE_COULDNT_CONNECT;
217*6236dae4SAndroid Build Coastguard Worker       goto out;
218*6236dae4SAndroid Build Coastguard Worker     }
219*6236dae4SAndroid Build Coastguard Worker 
220*6236dae4SAndroid Build Coastguard Worker     ctx->cf_protocol = cf_protocol;
221*6236dae4SAndroid Build Coastguard Worker     /* after we installed the filter "below" us, we call connect
222*6236dae4SAndroid Build Coastguard Worker      * on out sub-chain again.
223*6236dae4SAndroid Build Coastguard Worker      */
224*6236dae4SAndroid Build Coastguard Worker     goto connect_sub;
225*6236dae4SAndroid Build Coastguard Worker   }
226*6236dae4SAndroid Build Coastguard Worker   else {
227*6236dae4SAndroid Build Coastguard Worker     /* subchain connected and we had already installed the protocol filter.
228*6236dae4SAndroid Build Coastguard Worker      * This means the protocol tunnel is established, we are done.
229*6236dae4SAndroid Build Coastguard Worker      */
230*6236dae4SAndroid Build Coastguard Worker     DEBUGASSERT(ctx->cf_protocol);
231*6236dae4SAndroid Build Coastguard Worker     result = CURLE_OK;
232*6236dae4SAndroid Build Coastguard Worker   }
233*6236dae4SAndroid Build Coastguard Worker 
234*6236dae4SAndroid Build Coastguard Worker out:
235*6236dae4SAndroid Build Coastguard Worker   if(!result) {
236*6236dae4SAndroid Build Coastguard Worker     cf->connected = TRUE;
237*6236dae4SAndroid Build Coastguard Worker     *done = TRUE;
238*6236dae4SAndroid Build Coastguard Worker   }
239*6236dae4SAndroid Build Coastguard Worker   return result;
240*6236dae4SAndroid Build Coastguard Worker }
241*6236dae4SAndroid Build Coastguard Worker 
Curl_cf_http_proxy_get_host(struct Curl_cfilter * cf,struct Curl_easy * data,const char ** phost,const char ** pdisplay_host,int * pport)242*6236dae4SAndroid Build Coastguard Worker void Curl_cf_http_proxy_get_host(struct Curl_cfilter *cf,
243*6236dae4SAndroid Build Coastguard Worker                                  struct Curl_easy *data,
244*6236dae4SAndroid Build Coastguard Worker                                  const char **phost,
245*6236dae4SAndroid Build Coastguard Worker                                  const char **pdisplay_host,
246*6236dae4SAndroid Build Coastguard Worker                                  int *pport)
247*6236dae4SAndroid Build Coastguard Worker {
248*6236dae4SAndroid Build Coastguard Worker   (void)data;
249*6236dae4SAndroid Build Coastguard Worker   if(!cf->connected) {
250*6236dae4SAndroid Build Coastguard Worker     *phost = cf->conn->http_proxy.host.name;
251*6236dae4SAndroid Build Coastguard Worker     *pdisplay_host = cf->conn->http_proxy.host.dispname;
252*6236dae4SAndroid Build Coastguard Worker     *pport = (int)cf->conn->http_proxy.port;
253*6236dae4SAndroid Build Coastguard Worker   }
254*6236dae4SAndroid Build Coastguard Worker   else {
255*6236dae4SAndroid Build Coastguard Worker     cf->next->cft->get_host(cf->next, data, phost, pdisplay_host, pport);
256*6236dae4SAndroid Build Coastguard Worker   }
257*6236dae4SAndroid Build Coastguard Worker }
258*6236dae4SAndroid Build Coastguard Worker 
http_proxy_cf_destroy(struct Curl_cfilter * cf,struct Curl_easy * data)259*6236dae4SAndroid Build Coastguard Worker static void http_proxy_cf_destroy(struct Curl_cfilter *cf,
260*6236dae4SAndroid Build Coastguard Worker                                   struct Curl_easy *data)
261*6236dae4SAndroid Build Coastguard Worker {
262*6236dae4SAndroid Build Coastguard Worker   struct cf_proxy_ctx *ctx = cf->ctx;
263*6236dae4SAndroid Build Coastguard Worker 
264*6236dae4SAndroid Build Coastguard Worker   (void)data;
265*6236dae4SAndroid Build Coastguard Worker   CURL_TRC_CF(data, cf, "destroy");
266*6236dae4SAndroid Build Coastguard Worker   free(ctx);
267*6236dae4SAndroid Build Coastguard Worker }
268*6236dae4SAndroid Build Coastguard Worker 
http_proxy_cf_close(struct Curl_cfilter * cf,struct Curl_easy * data)269*6236dae4SAndroid Build Coastguard Worker static void http_proxy_cf_close(struct Curl_cfilter *cf,
270*6236dae4SAndroid Build Coastguard Worker                                 struct Curl_easy *data)
271*6236dae4SAndroid Build Coastguard Worker {
272*6236dae4SAndroid Build Coastguard Worker   struct cf_proxy_ctx *ctx = cf->ctx;
273*6236dae4SAndroid Build Coastguard Worker 
274*6236dae4SAndroid Build Coastguard Worker   CURL_TRC_CF(data, cf, "close");
275*6236dae4SAndroid Build Coastguard Worker   cf->connected = FALSE;
276*6236dae4SAndroid Build Coastguard Worker   if(ctx->cf_protocol) {
277*6236dae4SAndroid Build Coastguard Worker     struct Curl_cfilter *f;
278*6236dae4SAndroid Build Coastguard Worker     /* if someone already removed it, we assume he also
279*6236dae4SAndroid Build Coastguard Worker      * took care of destroying it. */
280*6236dae4SAndroid Build Coastguard Worker     for(f = cf->next; f; f = f->next) {
281*6236dae4SAndroid Build Coastguard Worker       if(f == ctx->cf_protocol) {
282*6236dae4SAndroid Build Coastguard Worker         /* still in our sub-chain */
283*6236dae4SAndroid Build Coastguard Worker         Curl_conn_cf_discard_sub(cf, ctx->cf_protocol, data, FALSE);
284*6236dae4SAndroid Build Coastguard Worker         break;
285*6236dae4SAndroid Build Coastguard Worker       }
286*6236dae4SAndroid Build Coastguard Worker     }
287*6236dae4SAndroid Build Coastguard Worker     ctx->cf_protocol = NULL;
288*6236dae4SAndroid Build Coastguard Worker   }
289*6236dae4SAndroid Build Coastguard Worker   if(cf->next)
290*6236dae4SAndroid Build Coastguard Worker     cf->next->cft->do_close(cf->next, data);
291*6236dae4SAndroid Build Coastguard Worker }
292*6236dae4SAndroid Build Coastguard Worker 
293*6236dae4SAndroid Build Coastguard Worker 
294*6236dae4SAndroid Build Coastguard Worker struct Curl_cftype Curl_cft_http_proxy = {
295*6236dae4SAndroid Build Coastguard Worker   "HTTP-PROXY",
296*6236dae4SAndroid Build Coastguard Worker   CF_TYPE_IP_CONNECT|CF_TYPE_PROXY,
297*6236dae4SAndroid Build Coastguard Worker   0,
298*6236dae4SAndroid Build Coastguard Worker   http_proxy_cf_destroy,
299*6236dae4SAndroid Build Coastguard Worker   http_proxy_cf_connect,
300*6236dae4SAndroid Build Coastguard Worker   http_proxy_cf_close,
301*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_shutdown,
302*6236dae4SAndroid Build Coastguard Worker   Curl_cf_http_proxy_get_host,
303*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_adjust_pollset,
304*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_data_pending,
305*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_send,
306*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_recv,
307*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_cntrl,
308*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_conn_is_alive,
309*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_conn_keep_alive,
310*6236dae4SAndroid Build Coastguard Worker   Curl_cf_def_query,
311*6236dae4SAndroid Build Coastguard Worker };
312*6236dae4SAndroid Build Coastguard Worker 
Curl_cf_http_proxy_insert_after(struct Curl_cfilter * cf_at,struct Curl_easy * data)313*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_cf_http_proxy_insert_after(struct Curl_cfilter *cf_at,
314*6236dae4SAndroid Build Coastguard Worker                                          struct Curl_easy *data)
315*6236dae4SAndroid Build Coastguard Worker {
316*6236dae4SAndroid Build Coastguard Worker   struct Curl_cfilter *cf;
317*6236dae4SAndroid Build Coastguard Worker   struct cf_proxy_ctx *ctx = NULL;
318*6236dae4SAndroid Build Coastguard Worker   CURLcode result;
319*6236dae4SAndroid Build Coastguard Worker 
320*6236dae4SAndroid Build Coastguard Worker   (void)data;
321*6236dae4SAndroid Build Coastguard Worker   ctx = calloc(1, sizeof(*ctx));
322*6236dae4SAndroid Build Coastguard Worker   if(!ctx) {
323*6236dae4SAndroid Build Coastguard Worker     result = CURLE_OUT_OF_MEMORY;
324*6236dae4SAndroid Build Coastguard Worker     goto out;
325*6236dae4SAndroid Build Coastguard Worker   }
326*6236dae4SAndroid Build Coastguard Worker   result = Curl_cf_create(&cf, &Curl_cft_http_proxy, ctx);
327*6236dae4SAndroid Build Coastguard Worker   if(result)
328*6236dae4SAndroid Build Coastguard Worker     goto out;
329*6236dae4SAndroid Build Coastguard Worker   ctx = NULL;
330*6236dae4SAndroid Build Coastguard Worker   Curl_conn_cf_insert_after(cf_at, cf);
331*6236dae4SAndroid Build Coastguard Worker 
332*6236dae4SAndroid Build Coastguard Worker out:
333*6236dae4SAndroid Build Coastguard Worker   free(ctx);
334*6236dae4SAndroid Build Coastguard Worker   return result;
335*6236dae4SAndroid Build Coastguard Worker }
336*6236dae4SAndroid Build Coastguard Worker 
337*6236dae4SAndroid Build Coastguard Worker #endif /* ! CURL_DISABLE_HTTP && !CURL_DISABLE_PROXY */
338