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 #ifdef USE_WOLFSSH
28*6236dae4SAndroid Build Coastguard Worker
29*6236dae4SAndroid Build Coastguard Worker #include <limits.h>
30*6236dae4SAndroid Build Coastguard Worker
31*6236dae4SAndroid Build Coastguard Worker #include "urldata.h"
32*6236dae4SAndroid Build Coastguard Worker #include "cfilters.h"
33*6236dae4SAndroid Build Coastguard Worker #include "connect.h"
34*6236dae4SAndroid Build Coastguard Worker #include "sendf.h"
35*6236dae4SAndroid Build Coastguard Worker #include "progress.h"
36*6236dae4SAndroid Build Coastguard Worker #include "curl_path.h"
37*6236dae4SAndroid Build Coastguard Worker #include "strtoofft.h"
38*6236dae4SAndroid Build Coastguard Worker #include "transfer.h"
39*6236dae4SAndroid Build Coastguard Worker #include "speedcheck.h"
40*6236dae4SAndroid Build Coastguard Worker #include "select.h"
41*6236dae4SAndroid Build Coastguard Worker #include "multiif.h"
42*6236dae4SAndroid Build Coastguard Worker #include "warnless.h"
43*6236dae4SAndroid Build Coastguard Worker #include "strdup.h"
44*6236dae4SAndroid Build Coastguard Worker
45*6236dae4SAndroid Build Coastguard Worker /* The last 3 #include files should be in this order */
46*6236dae4SAndroid Build Coastguard Worker #include "curl_printf.h"
47*6236dae4SAndroid Build Coastguard Worker #include "curl_memory.h"
48*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h"
49*6236dae4SAndroid Build Coastguard Worker
50*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_connect(struct Curl_easy *data, bool *done);
51*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_multi_statemach(struct Curl_easy *data, bool *done);
52*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_do(struct Curl_easy *data, bool *done);
53*6236dae4SAndroid Build Coastguard Worker #if 0
54*6236dae4SAndroid Build Coastguard Worker static CURLcode wscp_done(struct Curl_easy *data,
55*6236dae4SAndroid Build Coastguard Worker CURLcode, bool premature);
56*6236dae4SAndroid Build Coastguard Worker static CURLcode wscp_doing(struct Curl_easy *data,
57*6236dae4SAndroid Build Coastguard Worker bool *dophase_done);
58*6236dae4SAndroid Build Coastguard Worker static CURLcode wscp_disconnect(struct Curl_easy *data,
59*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn,
60*6236dae4SAndroid Build Coastguard Worker bool dead_connection);
61*6236dae4SAndroid Build Coastguard Worker #endif
62*6236dae4SAndroid Build Coastguard Worker static CURLcode wsftp_done(struct Curl_easy *data,
63*6236dae4SAndroid Build Coastguard Worker CURLcode, bool premature);
64*6236dae4SAndroid Build Coastguard Worker static CURLcode wsftp_doing(struct Curl_easy *data,
65*6236dae4SAndroid Build Coastguard Worker bool *dophase_done);
66*6236dae4SAndroid Build Coastguard Worker static CURLcode wsftp_disconnect(struct Curl_easy *data,
67*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn,
68*6236dae4SAndroid Build Coastguard Worker bool dead);
69*6236dae4SAndroid Build Coastguard Worker static int wssh_getsock(struct Curl_easy *data,
70*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn,
71*6236dae4SAndroid Build Coastguard Worker curl_socket_t *sock);
72*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_setup_connection(struct Curl_easy *data,
73*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn);
74*6236dae4SAndroid Build Coastguard Worker
75*6236dae4SAndroid Build Coastguard Worker #if 0
76*6236dae4SAndroid Build Coastguard Worker /*
77*6236dae4SAndroid Build Coastguard Worker * SCP protocol handler.
78*6236dae4SAndroid Build Coastguard Worker */
79*6236dae4SAndroid Build Coastguard Worker
80*6236dae4SAndroid Build Coastguard Worker const struct Curl_handler Curl_handler_scp = {
81*6236dae4SAndroid Build Coastguard Worker "SCP", /* scheme */
82*6236dae4SAndroid Build Coastguard Worker wssh_setup_connection, /* setup_connection */
83*6236dae4SAndroid Build Coastguard Worker wssh_do, /* do_it */
84*6236dae4SAndroid Build Coastguard Worker wscp_done, /* done */
85*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* do_more */
86*6236dae4SAndroid Build Coastguard Worker wssh_connect, /* connect_it */
87*6236dae4SAndroid Build Coastguard Worker wssh_multi_statemach, /* connecting */
88*6236dae4SAndroid Build Coastguard Worker wscp_doing, /* doing */
89*6236dae4SAndroid Build Coastguard Worker wssh_getsock, /* proto_getsock */
90*6236dae4SAndroid Build Coastguard Worker wssh_getsock, /* doing_getsock */
91*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* domore_getsock */
92*6236dae4SAndroid Build Coastguard Worker wssh_getsock, /* perform_getsock */
93*6236dae4SAndroid Build Coastguard Worker wscp_disconnect, /* disconnect */
94*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* write_resp */
95*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* write_resp_hd */
96*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* connection_check */
97*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* attach connection */
98*6236dae4SAndroid Build Coastguard Worker PORT_SSH, /* defport */
99*6236dae4SAndroid Build Coastguard Worker CURLPROTO_SCP, /* protocol */
100*6236dae4SAndroid Build Coastguard Worker PROTOPT_DIRLOCK | PROTOPT_CLOSEACTION
101*6236dae4SAndroid Build Coastguard Worker | PROTOPT_NOURLQUERY /* flags */
102*6236dae4SAndroid Build Coastguard Worker };
103*6236dae4SAndroid Build Coastguard Worker
104*6236dae4SAndroid Build Coastguard Worker #endif
105*6236dae4SAndroid Build Coastguard Worker
106*6236dae4SAndroid Build Coastguard Worker /*
107*6236dae4SAndroid Build Coastguard Worker * SFTP protocol handler.
108*6236dae4SAndroid Build Coastguard Worker */
109*6236dae4SAndroid Build Coastguard Worker
110*6236dae4SAndroid Build Coastguard Worker const struct Curl_handler Curl_handler_sftp = {
111*6236dae4SAndroid Build Coastguard Worker "SFTP", /* scheme */
112*6236dae4SAndroid Build Coastguard Worker wssh_setup_connection, /* setup_connection */
113*6236dae4SAndroid Build Coastguard Worker wssh_do, /* do_it */
114*6236dae4SAndroid Build Coastguard Worker wsftp_done, /* done */
115*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* do_more */
116*6236dae4SAndroid Build Coastguard Worker wssh_connect, /* connect_it */
117*6236dae4SAndroid Build Coastguard Worker wssh_multi_statemach, /* connecting */
118*6236dae4SAndroid Build Coastguard Worker wsftp_doing, /* doing */
119*6236dae4SAndroid Build Coastguard Worker wssh_getsock, /* proto_getsock */
120*6236dae4SAndroid Build Coastguard Worker wssh_getsock, /* doing_getsock */
121*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* domore_getsock */
122*6236dae4SAndroid Build Coastguard Worker wssh_getsock, /* perform_getsock */
123*6236dae4SAndroid Build Coastguard Worker wsftp_disconnect, /* disconnect */
124*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* write_resp */
125*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* write_resp_hd */
126*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* connection_check */
127*6236dae4SAndroid Build Coastguard Worker ZERO_NULL, /* attach connection */
128*6236dae4SAndroid Build Coastguard Worker PORT_SSH, /* defport */
129*6236dae4SAndroid Build Coastguard Worker CURLPROTO_SFTP, /* protocol */
130*6236dae4SAndroid Build Coastguard Worker CURLPROTO_SFTP, /* family */
131*6236dae4SAndroid Build Coastguard Worker PROTOPT_DIRLOCK | PROTOPT_CLOSEACTION
132*6236dae4SAndroid Build Coastguard Worker | PROTOPT_NOURLQUERY /* flags */
133*6236dae4SAndroid Build Coastguard Worker };
134*6236dae4SAndroid Build Coastguard Worker
135*6236dae4SAndroid Build Coastguard Worker /*
136*6236dae4SAndroid Build Coastguard Worker * SSH State machine related code
137*6236dae4SAndroid Build Coastguard Worker */
138*6236dae4SAndroid Build Coastguard Worker /* This is the ONLY way to change SSH state! */
state(struct Curl_easy * data,sshstate nowstate)139*6236dae4SAndroid Build Coastguard Worker static void state(struct Curl_easy *data, sshstate nowstate)
140*6236dae4SAndroid Build Coastguard Worker {
141*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
142*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc = &conn->proto.sshc;
143*6236dae4SAndroid Build Coastguard Worker #if defined(DEBUGBUILD) && !defined(CURL_DISABLE_VERBOSE_STRINGS)
144*6236dae4SAndroid Build Coastguard Worker /* for debug purposes */
145*6236dae4SAndroid Build Coastguard Worker static const char * const names[] = {
146*6236dae4SAndroid Build Coastguard Worker "SSH_STOP",
147*6236dae4SAndroid Build Coastguard Worker "SSH_INIT",
148*6236dae4SAndroid Build Coastguard Worker "SSH_S_STARTUP",
149*6236dae4SAndroid Build Coastguard Worker "SSH_HOSTKEY",
150*6236dae4SAndroid Build Coastguard Worker "SSH_AUTHLIST",
151*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_PKEY_INIT",
152*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_PKEY",
153*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_PASS_INIT",
154*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_PASS",
155*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_AGENT_INIT",
156*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_AGENT_LIST",
157*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_AGENT",
158*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_HOST_INIT",
159*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_HOST",
160*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_KEY_INIT",
161*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_KEY",
162*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_GSSAPI",
163*6236dae4SAndroid Build Coastguard Worker "SSH_AUTH_DONE",
164*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_INIT",
165*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_REALPATH",
166*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_INIT",
167*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_POSTQUOTE_INIT",
168*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE",
169*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_NEXT_QUOTE",
170*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_STAT",
171*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_SETSTAT",
172*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_SYMLINK",
173*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_MKDIR",
174*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_RENAME",
175*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_RMDIR",
176*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_UNLINK",
177*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_QUOTE_STATVFS",
178*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_GETINFO",
179*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_FILETIME",
180*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_TRANS_INIT",
181*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_UPLOAD_INIT",
182*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_CREATE_DIRS_INIT",
183*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_CREATE_DIRS",
184*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_CREATE_DIRS_MKDIR",
185*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_READDIR_INIT",
186*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_READDIR",
187*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_READDIR_LINK",
188*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_READDIR_BOTTOM",
189*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_READDIR_DONE",
190*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_DOWNLOAD_INIT",
191*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_DOWNLOAD_STAT",
192*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_CLOSE",
193*6236dae4SAndroid Build Coastguard Worker "SSH_SFTP_SHUTDOWN",
194*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_TRANS_INIT",
195*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_UPLOAD_INIT",
196*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_DOWNLOAD_INIT",
197*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_DOWNLOAD",
198*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_DONE",
199*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_SEND_EOF",
200*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_WAIT_EOF",
201*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_WAIT_CLOSE",
202*6236dae4SAndroid Build Coastguard Worker "SSH_SCP_CHANNEL_FREE",
203*6236dae4SAndroid Build Coastguard Worker "SSH_SESSION_DISCONNECT",
204*6236dae4SAndroid Build Coastguard Worker "SSH_SESSION_FREE",
205*6236dae4SAndroid Build Coastguard Worker "QUIT"
206*6236dae4SAndroid Build Coastguard Worker };
207*6236dae4SAndroid Build Coastguard Worker
208*6236dae4SAndroid Build Coastguard Worker /* a precaution to make sure the lists are in sync */
209*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(sizeof(names)/sizeof(names[0]) == SSH_LAST);
210*6236dae4SAndroid Build Coastguard Worker
211*6236dae4SAndroid Build Coastguard Worker if(sshc->state != nowstate) {
212*6236dae4SAndroid Build Coastguard Worker infof(data, "wolfssh %p state change from %s to %s",
213*6236dae4SAndroid Build Coastguard Worker (void *)sshc, names[sshc->state], names[nowstate]);
214*6236dae4SAndroid Build Coastguard Worker }
215*6236dae4SAndroid Build Coastguard Worker #endif
216*6236dae4SAndroid Build Coastguard Worker
217*6236dae4SAndroid Build Coastguard Worker sshc->state = nowstate;
218*6236dae4SAndroid Build Coastguard Worker }
219*6236dae4SAndroid Build Coastguard Worker
wscp_send(struct Curl_easy * data,int sockindex,const void * mem,size_t len,bool eos,CURLcode * err)220*6236dae4SAndroid Build Coastguard Worker static ssize_t wscp_send(struct Curl_easy *data, int sockindex,
221*6236dae4SAndroid Build Coastguard Worker const void *mem, size_t len, bool eos,
222*6236dae4SAndroid Build Coastguard Worker CURLcode *err)
223*6236dae4SAndroid Build Coastguard Worker {
224*6236dae4SAndroid Build Coastguard Worker ssize_t nwrite = 0;
225*6236dae4SAndroid Build Coastguard Worker (void)data;
226*6236dae4SAndroid Build Coastguard Worker (void)sockindex; /* we only support SCP on the fixed known primary socket */
227*6236dae4SAndroid Build Coastguard Worker (void)mem;
228*6236dae4SAndroid Build Coastguard Worker (void)len;
229*6236dae4SAndroid Build Coastguard Worker (void)eos;
230*6236dae4SAndroid Build Coastguard Worker (void)err;
231*6236dae4SAndroid Build Coastguard Worker
232*6236dae4SAndroid Build Coastguard Worker return nwrite;
233*6236dae4SAndroid Build Coastguard Worker }
234*6236dae4SAndroid Build Coastguard Worker
wscp_recv(struct Curl_easy * data,int sockindex,char * mem,size_t len,CURLcode * err)235*6236dae4SAndroid Build Coastguard Worker static ssize_t wscp_recv(struct Curl_easy *data, int sockindex,
236*6236dae4SAndroid Build Coastguard Worker char *mem, size_t len, CURLcode *err)
237*6236dae4SAndroid Build Coastguard Worker {
238*6236dae4SAndroid Build Coastguard Worker ssize_t nread = 0;
239*6236dae4SAndroid Build Coastguard Worker (void)data;
240*6236dae4SAndroid Build Coastguard Worker (void)sockindex; /* we only support SCP on the fixed known primary socket */
241*6236dae4SAndroid Build Coastguard Worker (void)mem;
242*6236dae4SAndroid Build Coastguard Worker (void)len;
243*6236dae4SAndroid Build Coastguard Worker (void)err;
244*6236dae4SAndroid Build Coastguard Worker
245*6236dae4SAndroid Build Coastguard Worker return nread;
246*6236dae4SAndroid Build Coastguard Worker }
247*6236dae4SAndroid Build Coastguard Worker
248*6236dae4SAndroid Build Coastguard Worker /* return number of sent bytes */
wsftp_send(struct Curl_easy * data,int sockindex,const void * mem,size_t len,bool eos,CURLcode * err)249*6236dae4SAndroid Build Coastguard Worker static ssize_t wsftp_send(struct Curl_easy *data, int sockindex,
250*6236dae4SAndroid Build Coastguard Worker const void *mem, size_t len, bool eos, CURLcode *err)
251*6236dae4SAndroid Build Coastguard Worker {
252*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
253*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc = &conn->proto.sshc;
254*6236dae4SAndroid Build Coastguard Worker word32 offset[2];
255*6236dae4SAndroid Build Coastguard Worker int rc;
256*6236dae4SAndroid Build Coastguard Worker (void)sockindex;
257*6236dae4SAndroid Build Coastguard Worker (void)eos;
258*6236dae4SAndroid Build Coastguard Worker
259*6236dae4SAndroid Build Coastguard Worker offset[0] = (word32)sshc->offset & 0xFFFFFFFF;
260*6236dae4SAndroid Build Coastguard Worker offset[1] = (word32)(sshc->offset >> 32) & 0xFFFFFFFF;
261*6236dae4SAndroid Build Coastguard Worker
262*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_SendWritePacket(sshc->ssh_session, sshc->handle,
263*6236dae4SAndroid Build Coastguard Worker sshc->handleSz,
264*6236dae4SAndroid Build Coastguard Worker &offset[0],
265*6236dae4SAndroid Build Coastguard Worker (byte *)mem, (word32)len);
266*6236dae4SAndroid Build Coastguard Worker
267*6236dae4SAndroid Build Coastguard Worker if(rc == WS_FATAL_ERROR)
268*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
269*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
270*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
271*6236dae4SAndroid Build Coastguard Worker *err = CURLE_AGAIN;
272*6236dae4SAndroid Build Coastguard Worker return -1;
273*6236dae4SAndroid Build Coastguard Worker }
274*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
275*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
276*6236dae4SAndroid Build Coastguard Worker *err = CURLE_AGAIN;
277*6236dae4SAndroid Build Coastguard Worker return -1;
278*6236dae4SAndroid Build Coastguard Worker }
279*6236dae4SAndroid Build Coastguard Worker if(rc < 0) {
280*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfSSH_SFTP_SendWritePacket returned %d", rc);
281*6236dae4SAndroid Build Coastguard Worker return -1;
282*6236dae4SAndroid Build Coastguard Worker }
283*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(rc == (int)len);
284*6236dae4SAndroid Build Coastguard Worker infof(data, "sent %zu bytes SFTP from offset %" FMT_OFF_T,
285*6236dae4SAndroid Build Coastguard Worker len, sshc->offset);
286*6236dae4SAndroid Build Coastguard Worker sshc->offset += len;
287*6236dae4SAndroid Build Coastguard Worker return (ssize_t)rc;
288*6236dae4SAndroid Build Coastguard Worker }
289*6236dae4SAndroid Build Coastguard Worker
290*6236dae4SAndroid Build Coastguard Worker /*
291*6236dae4SAndroid Build Coastguard Worker * Return number of received (decrypted) bytes
292*6236dae4SAndroid Build Coastguard Worker * or <0 on error
293*6236dae4SAndroid Build Coastguard Worker */
wsftp_recv(struct Curl_easy * data,int sockindex,char * mem,size_t len,CURLcode * err)294*6236dae4SAndroid Build Coastguard Worker static ssize_t wsftp_recv(struct Curl_easy *data, int sockindex,
295*6236dae4SAndroid Build Coastguard Worker char *mem, size_t len, CURLcode *err)
296*6236dae4SAndroid Build Coastguard Worker {
297*6236dae4SAndroid Build Coastguard Worker int rc;
298*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
299*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc = &conn->proto.sshc;
300*6236dae4SAndroid Build Coastguard Worker word32 offset[2];
301*6236dae4SAndroid Build Coastguard Worker (void)sockindex;
302*6236dae4SAndroid Build Coastguard Worker
303*6236dae4SAndroid Build Coastguard Worker offset[0] = (word32)sshc->offset & 0xFFFFFFFF;
304*6236dae4SAndroid Build Coastguard Worker offset[1] = (word32)(sshc->offset >> 32) & 0xFFFFFFFF;
305*6236dae4SAndroid Build Coastguard Worker
306*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_SendReadPacket(sshc->ssh_session, sshc->handle,
307*6236dae4SAndroid Build Coastguard Worker sshc->handleSz,
308*6236dae4SAndroid Build Coastguard Worker &offset[0],
309*6236dae4SAndroid Build Coastguard Worker (byte *)mem, (word32)len);
310*6236dae4SAndroid Build Coastguard Worker if(rc == WS_FATAL_ERROR)
311*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
312*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
313*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
314*6236dae4SAndroid Build Coastguard Worker *err = CURLE_AGAIN;
315*6236dae4SAndroid Build Coastguard Worker return -1;
316*6236dae4SAndroid Build Coastguard Worker }
317*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
318*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
319*6236dae4SAndroid Build Coastguard Worker *err = CURLE_AGAIN;
320*6236dae4SAndroid Build Coastguard Worker return -1;
321*6236dae4SAndroid Build Coastguard Worker }
322*6236dae4SAndroid Build Coastguard Worker
323*6236dae4SAndroid Build Coastguard Worker DEBUGASSERT(rc <= (int)len);
324*6236dae4SAndroid Build Coastguard Worker
325*6236dae4SAndroid Build Coastguard Worker if(rc < 0) {
326*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfSSH_SFTP_SendReadPacket returned %d", rc);
327*6236dae4SAndroid Build Coastguard Worker return -1;
328*6236dae4SAndroid Build Coastguard Worker }
329*6236dae4SAndroid Build Coastguard Worker sshc->offset += len;
330*6236dae4SAndroid Build Coastguard Worker
331*6236dae4SAndroid Build Coastguard Worker return (ssize_t)rc;
332*6236dae4SAndroid Build Coastguard Worker }
333*6236dae4SAndroid Build Coastguard Worker
334*6236dae4SAndroid Build Coastguard Worker /*
335*6236dae4SAndroid Build Coastguard Worker * SSH setup and connection
336*6236dae4SAndroid Build Coastguard Worker */
wssh_setup_connection(struct Curl_easy * data,struct connectdata * conn)337*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_setup_connection(struct Curl_easy *data,
338*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn)
339*6236dae4SAndroid Build Coastguard Worker {
340*6236dae4SAndroid Build Coastguard Worker struct SSHPROTO *ssh;
341*6236dae4SAndroid Build Coastguard Worker (void)conn;
342*6236dae4SAndroid Build Coastguard Worker
343*6236dae4SAndroid Build Coastguard Worker data->req.p.ssh = ssh = calloc(1, sizeof(struct SSHPROTO));
344*6236dae4SAndroid Build Coastguard Worker if(!ssh)
345*6236dae4SAndroid Build Coastguard Worker return CURLE_OUT_OF_MEMORY;
346*6236dae4SAndroid Build Coastguard Worker
347*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
348*6236dae4SAndroid Build Coastguard Worker }
349*6236dae4SAndroid Build Coastguard Worker
userauth(byte authtype,WS_UserAuthData * authdata,void * ctx)350*6236dae4SAndroid Build Coastguard Worker static int userauth(byte authtype,
351*6236dae4SAndroid Build Coastguard Worker WS_UserAuthData* authdata,
352*6236dae4SAndroid Build Coastguard Worker void *ctx)
353*6236dae4SAndroid Build Coastguard Worker {
354*6236dae4SAndroid Build Coastguard Worker struct Curl_easy *data = ctx;
355*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "wolfssh callback: type %s",
356*6236dae4SAndroid Build Coastguard Worker authtype == WOLFSSH_USERAUTH_PASSWORD ? "PASSWORD" :
357*6236dae4SAndroid Build Coastguard Worker "PUBLICCKEY"));
358*6236dae4SAndroid Build Coastguard Worker if(authtype == WOLFSSH_USERAUTH_PASSWORD) {
359*6236dae4SAndroid Build Coastguard Worker authdata->sf.password.password = (byte *)data->conn->passwd;
360*6236dae4SAndroid Build Coastguard Worker authdata->sf.password.passwordSz = (word32) strlen(data->conn->passwd);
361*6236dae4SAndroid Build Coastguard Worker }
362*6236dae4SAndroid Build Coastguard Worker
363*6236dae4SAndroid Build Coastguard Worker return 0;
364*6236dae4SAndroid Build Coastguard Worker }
365*6236dae4SAndroid Build Coastguard Worker
wssh_connect(struct Curl_easy * data,bool * done)366*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_connect(struct Curl_easy *data, bool *done)
367*6236dae4SAndroid Build Coastguard Worker {
368*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
369*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc;
370*6236dae4SAndroid Build Coastguard Worker curl_socket_t sock = conn->sock[FIRSTSOCKET];
371*6236dae4SAndroid Build Coastguard Worker int rc;
372*6236dae4SAndroid Build Coastguard Worker
373*6236dae4SAndroid Build Coastguard Worker /* initialize per-handle data if not already */
374*6236dae4SAndroid Build Coastguard Worker if(!data->req.p.ssh)
375*6236dae4SAndroid Build Coastguard Worker wssh_setup_connection(data, conn);
376*6236dae4SAndroid Build Coastguard Worker
377*6236dae4SAndroid Build Coastguard Worker /* We default to persistent connections. We set this already in this connect
378*6236dae4SAndroid Build Coastguard Worker function to make the reuse checks properly be able to check this bit. */
379*6236dae4SAndroid Build Coastguard Worker connkeep(conn, "SSH default");
380*6236dae4SAndroid Build Coastguard Worker
381*6236dae4SAndroid Build Coastguard Worker if(conn->handler->protocol & CURLPROTO_SCP) {
382*6236dae4SAndroid Build Coastguard Worker conn->recv[FIRSTSOCKET] = wscp_recv;
383*6236dae4SAndroid Build Coastguard Worker conn->send[FIRSTSOCKET] = wscp_send;
384*6236dae4SAndroid Build Coastguard Worker }
385*6236dae4SAndroid Build Coastguard Worker else {
386*6236dae4SAndroid Build Coastguard Worker conn->recv[FIRSTSOCKET] = wsftp_recv;
387*6236dae4SAndroid Build Coastguard Worker conn->send[FIRSTSOCKET] = wsftp_send;
388*6236dae4SAndroid Build Coastguard Worker }
389*6236dae4SAndroid Build Coastguard Worker sshc = &conn->proto.sshc;
390*6236dae4SAndroid Build Coastguard Worker sshc->ctx = wolfSSH_CTX_new(WOLFSSH_ENDPOINT_CLIENT, NULL);
391*6236dae4SAndroid Build Coastguard Worker if(!sshc->ctx) {
392*6236dae4SAndroid Build Coastguard Worker failf(data, "No wolfSSH context");
393*6236dae4SAndroid Build Coastguard Worker goto error;
394*6236dae4SAndroid Build Coastguard Worker }
395*6236dae4SAndroid Build Coastguard Worker
396*6236dae4SAndroid Build Coastguard Worker sshc->ssh_session = wolfSSH_new(sshc->ctx);
397*6236dae4SAndroid Build Coastguard Worker if(!sshc->ssh_session) {
398*6236dae4SAndroid Build Coastguard Worker failf(data, "No wolfSSH session");
399*6236dae4SAndroid Build Coastguard Worker goto error;
400*6236dae4SAndroid Build Coastguard Worker }
401*6236dae4SAndroid Build Coastguard Worker
402*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SetUsername(sshc->ssh_session, conn->user);
403*6236dae4SAndroid Build Coastguard Worker if(rc != WS_SUCCESS) {
404*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfSSH failed to set username");
405*6236dae4SAndroid Build Coastguard Worker goto error;
406*6236dae4SAndroid Build Coastguard Worker }
407*6236dae4SAndroid Build Coastguard Worker
408*6236dae4SAndroid Build Coastguard Worker /* set callback for authentication */
409*6236dae4SAndroid Build Coastguard Worker wolfSSH_SetUserAuth(sshc->ctx, userauth);
410*6236dae4SAndroid Build Coastguard Worker wolfSSH_SetUserAuthCtx(sshc->ssh_session, data);
411*6236dae4SAndroid Build Coastguard Worker
412*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_set_fd(sshc->ssh_session, (int)sock);
413*6236dae4SAndroid Build Coastguard Worker if(rc) {
414*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfSSH failed to set socket");
415*6236dae4SAndroid Build Coastguard Worker goto error;
416*6236dae4SAndroid Build Coastguard Worker }
417*6236dae4SAndroid Build Coastguard Worker
418*6236dae4SAndroid Build Coastguard Worker #if 0
419*6236dae4SAndroid Build Coastguard Worker wolfSSH_Debugging_ON();
420*6236dae4SAndroid Build Coastguard Worker #endif
421*6236dae4SAndroid Build Coastguard Worker
422*6236dae4SAndroid Build Coastguard Worker *done = TRUE;
423*6236dae4SAndroid Build Coastguard Worker if(conn->handler->protocol & CURLPROTO_SCP)
424*6236dae4SAndroid Build Coastguard Worker state(data, SSH_INIT);
425*6236dae4SAndroid Build Coastguard Worker else
426*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_INIT);
427*6236dae4SAndroid Build Coastguard Worker
428*6236dae4SAndroid Build Coastguard Worker return wssh_multi_statemach(data, done);
429*6236dae4SAndroid Build Coastguard Worker error:
430*6236dae4SAndroid Build Coastguard Worker wolfSSH_free(sshc->ssh_session);
431*6236dae4SAndroid Build Coastguard Worker wolfSSH_CTX_free(sshc->ctx);
432*6236dae4SAndroid Build Coastguard Worker return CURLE_FAILED_INIT;
433*6236dae4SAndroid Build Coastguard Worker }
434*6236dae4SAndroid Build Coastguard Worker
435*6236dae4SAndroid Build Coastguard Worker /*
436*6236dae4SAndroid Build Coastguard Worker * wssh_statemach_act() runs the SSH state machine as far as it can without
437*6236dae4SAndroid Build Coastguard Worker * blocking and without reaching the end. The data the pointer 'block' points
438*6236dae4SAndroid Build Coastguard Worker * to will be set to TRUE if the wolfssh function returns EAGAIN meaning it
439*6236dae4SAndroid Build Coastguard Worker * wants to be called again when the socket is ready
440*6236dae4SAndroid Build Coastguard Worker */
441*6236dae4SAndroid Build Coastguard Worker
wssh_statemach_act(struct Curl_easy * data,bool * block)442*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_statemach_act(struct Curl_easy *data, bool *block)
443*6236dae4SAndroid Build Coastguard Worker {
444*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
445*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
446*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc = &conn->proto.sshc;
447*6236dae4SAndroid Build Coastguard Worker struct SSHPROTO *sftp_scp = data->req.p.ssh;
448*6236dae4SAndroid Build Coastguard Worker WS_SFTPNAME *name;
449*6236dae4SAndroid Build Coastguard Worker int rc = 0;
450*6236dae4SAndroid Build Coastguard Worker *block = FALSE; /* we are not blocking by default */
451*6236dae4SAndroid Build Coastguard Worker
452*6236dae4SAndroid Build Coastguard Worker do {
453*6236dae4SAndroid Build Coastguard Worker switch(sshc->state) {
454*6236dae4SAndroid Build Coastguard Worker case SSH_INIT:
455*6236dae4SAndroid Build Coastguard Worker state(data, SSH_S_STARTUP);
456*6236dae4SAndroid Build Coastguard Worker break;
457*6236dae4SAndroid Build Coastguard Worker
458*6236dae4SAndroid Build Coastguard Worker case SSH_S_STARTUP:
459*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_connect(sshc->ssh_session);
460*6236dae4SAndroid Build Coastguard Worker if(rc != WS_SUCCESS)
461*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
462*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
463*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
464*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
465*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
466*6236dae4SAndroid Build Coastguard Worker }
467*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
468*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
469*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
470*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
471*6236dae4SAndroid Build Coastguard Worker }
472*6236dae4SAndroid Build Coastguard Worker else if(rc != WS_SUCCESS) {
473*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
474*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
475*6236dae4SAndroid Build Coastguard Worker }
476*6236dae4SAndroid Build Coastguard Worker infof(data, "wolfssh connected");
477*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
478*6236dae4SAndroid Build Coastguard Worker break;
479*6236dae4SAndroid Build Coastguard Worker case SSH_STOP:
480*6236dae4SAndroid Build Coastguard Worker break;
481*6236dae4SAndroid Build Coastguard Worker
482*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_INIT:
483*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_connect(sshc->ssh_session);
484*6236dae4SAndroid Build Coastguard Worker if(rc != WS_SUCCESS)
485*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
486*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
487*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
488*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
489*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
490*6236dae4SAndroid Build Coastguard Worker }
491*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
492*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
493*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
494*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
495*6236dae4SAndroid Build Coastguard Worker }
496*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_SUCCESS) {
497*6236dae4SAndroid Build Coastguard Worker infof(data, "wolfssh SFTP connected");
498*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_REALPATH);
499*6236dae4SAndroid Build Coastguard Worker }
500*6236dae4SAndroid Build Coastguard Worker else {
501*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfssh SFTP connect error %d", rc);
502*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
503*6236dae4SAndroid Build Coastguard Worker }
504*6236dae4SAndroid Build Coastguard Worker break;
505*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_REALPATH:
506*6236dae4SAndroid Build Coastguard Worker name = wolfSSH_SFTP_RealPath(sshc->ssh_session, (char *)".");
507*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
508*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
509*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
510*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
511*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
512*6236dae4SAndroid Build Coastguard Worker }
513*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
514*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
515*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
516*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
517*6236dae4SAndroid Build Coastguard Worker }
518*6236dae4SAndroid Build Coastguard Worker else if(name && (rc == WS_SUCCESS)) {
519*6236dae4SAndroid Build Coastguard Worker sshc->homedir = Curl_memdup0(name->fName, name->fSz);
520*6236dae4SAndroid Build Coastguard Worker if(!sshc->homedir)
521*6236dae4SAndroid Build Coastguard Worker sshc->actualcode = CURLE_OUT_OF_MEMORY;
522*6236dae4SAndroid Build Coastguard Worker wolfSSH_SFTPNAME_list_free(name);
523*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
524*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
525*6236dae4SAndroid Build Coastguard Worker }
526*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfssh SFTP realpath %d", rc);
527*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
528*6236dae4SAndroid Build Coastguard Worker
529*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_QUOTE_INIT:
530*6236dae4SAndroid Build Coastguard Worker result = Curl_getworkingpath(data, sshc->homedir, &sftp_scp->path);
531*6236dae4SAndroid Build Coastguard Worker if(result) {
532*6236dae4SAndroid Build Coastguard Worker sshc->actualcode = result;
533*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
534*6236dae4SAndroid Build Coastguard Worker break;
535*6236dae4SAndroid Build Coastguard Worker }
536*6236dae4SAndroid Build Coastguard Worker
537*6236dae4SAndroid Build Coastguard Worker if(data->set.quote) {
538*6236dae4SAndroid Build Coastguard Worker infof(data, "Sending quote commands");
539*6236dae4SAndroid Build Coastguard Worker sshc->quote_item = data->set.quote;
540*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_QUOTE);
541*6236dae4SAndroid Build Coastguard Worker }
542*6236dae4SAndroid Build Coastguard Worker else {
543*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_GETINFO);
544*6236dae4SAndroid Build Coastguard Worker }
545*6236dae4SAndroid Build Coastguard Worker break;
546*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_GETINFO:
547*6236dae4SAndroid Build Coastguard Worker if(data->set.get_filetime) {
548*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_FILETIME);
549*6236dae4SAndroid Build Coastguard Worker }
550*6236dae4SAndroid Build Coastguard Worker else {
551*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_TRANS_INIT);
552*6236dae4SAndroid Build Coastguard Worker }
553*6236dae4SAndroid Build Coastguard Worker break;
554*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_TRANS_INIT:
555*6236dae4SAndroid Build Coastguard Worker if(data->state.upload)
556*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_UPLOAD_INIT);
557*6236dae4SAndroid Build Coastguard Worker else {
558*6236dae4SAndroid Build Coastguard Worker if(sftp_scp->path[strlen(sftp_scp->path)-1] == '/')
559*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_READDIR_INIT);
560*6236dae4SAndroid Build Coastguard Worker else
561*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_DOWNLOAD_INIT);
562*6236dae4SAndroid Build Coastguard Worker }
563*6236dae4SAndroid Build Coastguard Worker break;
564*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_UPLOAD_INIT: {
565*6236dae4SAndroid Build Coastguard Worker word32 flags;
566*6236dae4SAndroid Build Coastguard Worker WS_SFTP_FILEATRB createattrs;
567*6236dae4SAndroid Build Coastguard Worker if(data->state.resume_from) {
568*6236dae4SAndroid Build Coastguard Worker WS_SFTP_FILEATRB attrs;
569*6236dae4SAndroid Build Coastguard Worker if(data->state.resume_from < 0) {
570*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_STAT(sshc->ssh_session, sftp_scp->path,
571*6236dae4SAndroid Build Coastguard Worker &attrs);
572*6236dae4SAndroid Build Coastguard Worker if(rc != WS_SUCCESS)
573*6236dae4SAndroid Build Coastguard Worker break;
574*6236dae4SAndroid Build Coastguard Worker
575*6236dae4SAndroid Build Coastguard Worker if(rc) {
576*6236dae4SAndroid Build Coastguard Worker data->state.resume_from = 0;
577*6236dae4SAndroid Build Coastguard Worker }
578*6236dae4SAndroid Build Coastguard Worker else {
579*6236dae4SAndroid Build Coastguard Worker curl_off_t size = ((curl_off_t)attrs.sz[1] << 32) | attrs.sz[0];
580*6236dae4SAndroid Build Coastguard Worker if(size < 0) {
581*6236dae4SAndroid Build Coastguard Worker failf(data, "Bad file size (%" FMT_OFF_T ")", size);
582*6236dae4SAndroid Build Coastguard Worker return CURLE_BAD_DOWNLOAD_RESUME;
583*6236dae4SAndroid Build Coastguard Worker }
584*6236dae4SAndroid Build Coastguard Worker data->state.resume_from = size;
585*6236dae4SAndroid Build Coastguard Worker }
586*6236dae4SAndroid Build Coastguard Worker }
587*6236dae4SAndroid Build Coastguard Worker }
588*6236dae4SAndroid Build Coastguard Worker
589*6236dae4SAndroid Build Coastguard Worker if(data->set.remote_append)
590*6236dae4SAndroid Build Coastguard Worker /* Try to open for append, but create if nonexisting */
591*6236dae4SAndroid Build Coastguard Worker flags = WOLFSSH_FXF_WRITE|WOLFSSH_FXF_CREAT|WOLFSSH_FXF_APPEND;
592*6236dae4SAndroid Build Coastguard Worker else if(data->state.resume_from > 0)
593*6236dae4SAndroid Build Coastguard Worker /* If we have restart position then open for append */
594*6236dae4SAndroid Build Coastguard Worker flags = WOLFSSH_FXF_WRITE|WOLFSSH_FXF_APPEND;
595*6236dae4SAndroid Build Coastguard Worker else
596*6236dae4SAndroid Build Coastguard Worker /* Clear file before writing (normal behavior) */
597*6236dae4SAndroid Build Coastguard Worker flags = WOLFSSH_FXF_WRITE|WOLFSSH_FXF_CREAT|WOLFSSH_FXF_TRUNC;
598*6236dae4SAndroid Build Coastguard Worker
599*6236dae4SAndroid Build Coastguard Worker memset(&createattrs, 0, sizeof(createattrs));
600*6236dae4SAndroid Build Coastguard Worker createattrs.per = (word32)data->set.new_file_perms;
601*6236dae4SAndroid Build Coastguard Worker sshc->handleSz = sizeof(sshc->handle);
602*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_Open(sshc->ssh_session, sftp_scp->path,
603*6236dae4SAndroid Build Coastguard Worker flags, &createattrs,
604*6236dae4SAndroid Build Coastguard Worker sshc->handle, &sshc->handleSz);
605*6236dae4SAndroid Build Coastguard Worker if(rc == WS_FATAL_ERROR)
606*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
607*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
608*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
609*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
610*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
611*6236dae4SAndroid Build Coastguard Worker }
612*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
613*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
614*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
615*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
616*6236dae4SAndroid Build Coastguard Worker }
617*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_SUCCESS) {
618*6236dae4SAndroid Build Coastguard Worker infof(data, "wolfssh SFTP open succeeded");
619*6236dae4SAndroid Build Coastguard Worker }
620*6236dae4SAndroid Build Coastguard Worker else {
621*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfssh SFTP upload open failed: %d", rc);
622*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
623*6236dae4SAndroid Build Coastguard Worker }
624*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_DOWNLOAD_STAT);
625*6236dae4SAndroid Build Coastguard Worker
626*6236dae4SAndroid Build Coastguard Worker /* If we have a restart point then we need to seek to the correct
627*6236dae4SAndroid Build Coastguard Worker position. */
628*6236dae4SAndroid Build Coastguard Worker if(data->state.resume_from > 0) {
629*6236dae4SAndroid Build Coastguard Worker /* Let's read off the proper amount of bytes from the input. */
630*6236dae4SAndroid Build Coastguard Worker int seekerr = CURL_SEEKFUNC_OK;
631*6236dae4SAndroid Build Coastguard Worker if(data->set.seek_func) {
632*6236dae4SAndroid Build Coastguard Worker Curl_set_in_callback(data, TRUE);
633*6236dae4SAndroid Build Coastguard Worker seekerr = data->set.seek_func(data->set.seek_client,
634*6236dae4SAndroid Build Coastguard Worker data->state.resume_from, SEEK_SET);
635*6236dae4SAndroid Build Coastguard Worker Curl_set_in_callback(data, FALSE);
636*6236dae4SAndroid Build Coastguard Worker }
637*6236dae4SAndroid Build Coastguard Worker
638*6236dae4SAndroid Build Coastguard Worker if(seekerr != CURL_SEEKFUNC_OK) {
639*6236dae4SAndroid Build Coastguard Worker curl_off_t passed = 0;
640*6236dae4SAndroid Build Coastguard Worker
641*6236dae4SAndroid Build Coastguard Worker if(seekerr != CURL_SEEKFUNC_CANTSEEK) {
642*6236dae4SAndroid Build Coastguard Worker failf(data, "Could not seek stream");
643*6236dae4SAndroid Build Coastguard Worker return CURLE_FTP_COULDNT_USE_REST;
644*6236dae4SAndroid Build Coastguard Worker }
645*6236dae4SAndroid Build Coastguard Worker /* seekerr == CURL_SEEKFUNC_CANTSEEK (cannot seek to offset) */
646*6236dae4SAndroid Build Coastguard Worker do {
647*6236dae4SAndroid Build Coastguard Worker char scratch[4*1024];
648*6236dae4SAndroid Build Coastguard Worker size_t readthisamountnow =
649*6236dae4SAndroid Build Coastguard Worker (data->state.resume_from - passed >
650*6236dae4SAndroid Build Coastguard Worker (curl_off_t)sizeof(scratch)) ?
651*6236dae4SAndroid Build Coastguard Worker sizeof(scratch) : curlx_sotouz(data->state.resume_from - passed);
652*6236dae4SAndroid Build Coastguard Worker
653*6236dae4SAndroid Build Coastguard Worker size_t actuallyread;
654*6236dae4SAndroid Build Coastguard Worker Curl_set_in_callback(data, TRUE);
655*6236dae4SAndroid Build Coastguard Worker actuallyread = data->state.fread_func(scratch, 1,
656*6236dae4SAndroid Build Coastguard Worker readthisamountnow,
657*6236dae4SAndroid Build Coastguard Worker data->state.in);
658*6236dae4SAndroid Build Coastguard Worker Curl_set_in_callback(data, FALSE);
659*6236dae4SAndroid Build Coastguard Worker
660*6236dae4SAndroid Build Coastguard Worker passed += actuallyread;
661*6236dae4SAndroid Build Coastguard Worker if((actuallyread == 0) || (actuallyread > readthisamountnow)) {
662*6236dae4SAndroid Build Coastguard Worker /* this checks for greater-than only to make sure that the
663*6236dae4SAndroid Build Coastguard Worker CURL_READFUNC_ABORT return code still aborts */
664*6236dae4SAndroid Build Coastguard Worker failf(data, "Failed to read data");
665*6236dae4SAndroid Build Coastguard Worker return CURLE_FTP_COULDNT_USE_REST;
666*6236dae4SAndroid Build Coastguard Worker }
667*6236dae4SAndroid Build Coastguard Worker } while(passed < data->state.resume_from);
668*6236dae4SAndroid Build Coastguard Worker }
669*6236dae4SAndroid Build Coastguard Worker
670*6236dae4SAndroid Build Coastguard Worker /* now, decrease the size of the read */
671*6236dae4SAndroid Build Coastguard Worker if(data->state.infilesize > 0) {
672*6236dae4SAndroid Build Coastguard Worker data->state.infilesize -= data->state.resume_from;
673*6236dae4SAndroid Build Coastguard Worker data->req.size = data->state.infilesize;
674*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetUploadSize(data, data->state.infilesize);
675*6236dae4SAndroid Build Coastguard Worker }
676*6236dae4SAndroid Build Coastguard Worker
677*6236dae4SAndroid Build Coastguard Worker sshc->offset += data->state.resume_from;
678*6236dae4SAndroid Build Coastguard Worker }
679*6236dae4SAndroid Build Coastguard Worker if(data->state.infilesize > 0) {
680*6236dae4SAndroid Build Coastguard Worker data->req.size = data->state.infilesize;
681*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetUploadSize(data, data->state.infilesize);
682*6236dae4SAndroid Build Coastguard Worker }
683*6236dae4SAndroid Build Coastguard Worker /* upload data */
684*6236dae4SAndroid Build Coastguard Worker Curl_xfer_setup1(data, CURL_XFER_SEND, -1, FALSE);
685*6236dae4SAndroid Build Coastguard Worker
686*6236dae4SAndroid Build Coastguard Worker /* not set by Curl_xfer_setup to preserve keepon bits */
687*6236dae4SAndroid Build Coastguard Worker conn->sockfd = conn->writesockfd;
688*6236dae4SAndroid Build Coastguard Worker
689*6236dae4SAndroid Build Coastguard Worker if(result) {
690*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_CLOSE);
691*6236dae4SAndroid Build Coastguard Worker sshc->actualcode = result;
692*6236dae4SAndroid Build Coastguard Worker }
693*6236dae4SAndroid Build Coastguard Worker else {
694*6236dae4SAndroid Build Coastguard Worker /* store this original bitmask setup to use later on if we cannot
695*6236dae4SAndroid Build Coastguard Worker figure out a "real" bitmask */
696*6236dae4SAndroid Build Coastguard Worker sshc->orig_waitfor = data->req.keepon;
697*6236dae4SAndroid Build Coastguard Worker
698*6236dae4SAndroid Build Coastguard Worker /* we want to use the _sending_ function even when the socket turns
699*6236dae4SAndroid Build Coastguard Worker out readable as the underlying libssh2 sftp send function will deal
700*6236dae4SAndroid Build Coastguard Worker with both accordingly */
701*6236dae4SAndroid Build Coastguard Worker data->state.select_bits = CURL_CSELECT_OUT;
702*6236dae4SAndroid Build Coastguard Worker
703*6236dae4SAndroid Build Coastguard Worker /* since we do not really wait for anything at this point, we want the
704*6236dae4SAndroid Build Coastguard Worker state machine to move on as soon as possible so we set a very short
705*6236dae4SAndroid Build Coastguard Worker timeout here */
706*6236dae4SAndroid Build Coastguard Worker Curl_expire(data, 0, EXPIRE_RUN_NOW);
707*6236dae4SAndroid Build Coastguard Worker
708*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
709*6236dae4SAndroid Build Coastguard Worker }
710*6236dae4SAndroid Build Coastguard Worker break;
711*6236dae4SAndroid Build Coastguard Worker }
712*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_DOWNLOAD_INIT:
713*6236dae4SAndroid Build Coastguard Worker sshc->handleSz = sizeof(sshc->handle);
714*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_Open(sshc->ssh_session, sftp_scp->path,
715*6236dae4SAndroid Build Coastguard Worker WOLFSSH_FXF_READ, NULL,
716*6236dae4SAndroid Build Coastguard Worker sshc->handle, &sshc->handleSz);
717*6236dae4SAndroid Build Coastguard Worker if(rc == WS_FATAL_ERROR)
718*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
719*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
720*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
721*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
722*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
723*6236dae4SAndroid Build Coastguard Worker }
724*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
725*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
726*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
727*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
728*6236dae4SAndroid Build Coastguard Worker }
729*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_SUCCESS) {
730*6236dae4SAndroid Build Coastguard Worker infof(data, "wolfssh SFTP open succeeded");
731*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_DOWNLOAD_STAT);
732*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
733*6236dae4SAndroid Build Coastguard Worker }
734*6236dae4SAndroid Build Coastguard Worker
735*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfssh SFTP open failed: %d", rc);
736*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
737*6236dae4SAndroid Build Coastguard Worker
738*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_DOWNLOAD_STAT: {
739*6236dae4SAndroid Build Coastguard Worker WS_SFTP_FILEATRB attrs;
740*6236dae4SAndroid Build Coastguard Worker curl_off_t size;
741*6236dae4SAndroid Build Coastguard Worker
742*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_STAT(sshc->ssh_session, sftp_scp->path, &attrs);
743*6236dae4SAndroid Build Coastguard Worker if(rc == WS_FATAL_ERROR)
744*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
745*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
746*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
747*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
748*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
749*6236dae4SAndroid Build Coastguard Worker }
750*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
751*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
752*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
753*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
754*6236dae4SAndroid Build Coastguard Worker }
755*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_SUCCESS) {
756*6236dae4SAndroid Build Coastguard Worker infof(data, "wolfssh STAT succeeded");
757*6236dae4SAndroid Build Coastguard Worker }
758*6236dae4SAndroid Build Coastguard Worker else {
759*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfssh SFTP open failed: %d", rc);
760*6236dae4SAndroid Build Coastguard Worker data->req.size = -1;
761*6236dae4SAndroid Build Coastguard Worker data->req.maxdownload = -1;
762*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetDownloadSize(data, -1);
763*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
764*6236dae4SAndroid Build Coastguard Worker }
765*6236dae4SAndroid Build Coastguard Worker
766*6236dae4SAndroid Build Coastguard Worker size = ((curl_off_t)attrs.sz[1] << 32) | attrs.sz[0];
767*6236dae4SAndroid Build Coastguard Worker
768*6236dae4SAndroid Build Coastguard Worker data->req.size = size;
769*6236dae4SAndroid Build Coastguard Worker data->req.maxdownload = size;
770*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetDownloadSize(data, size);
771*6236dae4SAndroid Build Coastguard Worker
772*6236dae4SAndroid Build Coastguard Worker infof(data, "SFTP download %" FMT_OFF_T " bytes", size);
773*6236dae4SAndroid Build Coastguard Worker
774*6236dae4SAndroid Build Coastguard Worker /* We cannot seek with wolfSSH so resuming and range requests are not
775*6236dae4SAndroid Build Coastguard Worker possible */
776*6236dae4SAndroid Build Coastguard Worker if(data->state.use_range || data->state.resume_from) {
777*6236dae4SAndroid Build Coastguard Worker infof(data, "wolfSSH cannot do range/seek on SFTP");
778*6236dae4SAndroid Build Coastguard Worker return CURLE_BAD_DOWNLOAD_RESUME;
779*6236dae4SAndroid Build Coastguard Worker }
780*6236dae4SAndroid Build Coastguard Worker
781*6236dae4SAndroid Build Coastguard Worker /* Setup the actual download */
782*6236dae4SAndroid Build Coastguard Worker if(data->req.size == 0) {
783*6236dae4SAndroid Build Coastguard Worker /* no data to transfer */
784*6236dae4SAndroid Build Coastguard Worker Curl_xfer_setup_nop(data);
785*6236dae4SAndroid Build Coastguard Worker infof(data, "File already completely downloaded");
786*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
787*6236dae4SAndroid Build Coastguard Worker break;
788*6236dae4SAndroid Build Coastguard Worker }
789*6236dae4SAndroid Build Coastguard Worker Curl_xfer_setup1(data, CURL_XFER_RECV, data->req.size, FALSE);
790*6236dae4SAndroid Build Coastguard Worker
791*6236dae4SAndroid Build Coastguard Worker /* not set by Curl_xfer_setup to preserve keepon bits */
792*6236dae4SAndroid Build Coastguard Worker conn->writesockfd = conn->sockfd;
793*6236dae4SAndroid Build Coastguard Worker
794*6236dae4SAndroid Build Coastguard Worker /* we want to use the _receiving_ function even when the socket turns
795*6236dae4SAndroid Build Coastguard Worker out writableable as the underlying libssh2 recv function will deal
796*6236dae4SAndroid Build Coastguard Worker with both accordingly */
797*6236dae4SAndroid Build Coastguard Worker data->state.select_bits = CURL_CSELECT_IN;
798*6236dae4SAndroid Build Coastguard Worker
799*6236dae4SAndroid Build Coastguard Worker if(result) {
800*6236dae4SAndroid Build Coastguard Worker /* this should never occur; the close state should be entered
801*6236dae4SAndroid Build Coastguard Worker at the time the error occurs */
802*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_CLOSE);
803*6236dae4SAndroid Build Coastguard Worker sshc->actualcode = result;
804*6236dae4SAndroid Build Coastguard Worker }
805*6236dae4SAndroid Build Coastguard Worker else {
806*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
807*6236dae4SAndroid Build Coastguard Worker }
808*6236dae4SAndroid Build Coastguard Worker break;
809*6236dae4SAndroid Build Coastguard Worker }
810*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_CLOSE:
811*6236dae4SAndroid Build Coastguard Worker if(sshc->handleSz)
812*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_SFTP_Close(sshc->ssh_session, sshc->handle,
813*6236dae4SAndroid Build Coastguard Worker sshc->handleSz);
814*6236dae4SAndroid Build Coastguard Worker else
815*6236dae4SAndroid Build Coastguard Worker rc = WS_SUCCESS; /* directory listing */
816*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
817*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
818*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
819*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
820*6236dae4SAndroid Build Coastguard Worker }
821*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
822*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
823*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
824*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
825*6236dae4SAndroid Build Coastguard Worker }
826*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_SUCCESS) {
827*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
828*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
829*6236dae4SAndroid Build Coastguard Worker }
830*6236dae4SAndroid Build Coastguard Worker
831*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfssh SFTP CLOSE failed: %d", rc);
832*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
833*6236dae4SAndroid Build Coastguard Worker
834*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_READDIR_INIT:
835*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetDownloadSize(data, -1);
836*6236dae4SAndroid Build Coastguard Worker if(data->req.no_body) {
837*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
838*6236dae4SAndroid Build Coastguard Worker break;
839*6236dae4SAndroid Build Coastguard Worker }
840*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_READDIR);
841*6236dae4SAndroid Build Coastguard Worker break;
842*6236dae4SAndroid Build Coastguard Worker
843*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_READDIR:
844*6236dae4SAndroid Build Coastguard Worker name = wolfSSH_SFTP_LS(sshc->ssh_session, sftp_scp->path);
845*6236dae4SAndroid Build Coastguard Worker if(!name)
846*6236dae4SAndroid Build Coastguard Worker rc = wolfSSH_get_error(sshc->ssh_session);
847*6236dae4SAndroid Build Coastguard Worker else
848*6236dae4SAndroid Build Coastguard Worker rc = WS_SUCCESS;
849*6236dae4SAndroid Build Coastguard Worker
850*6236dae4SAndroid Build Coastguard Worker if(rc == WS_WANT_READ) {
851*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
852*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_RECV;
853*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
854*6236dae4SAndroid Build Coastguard Worker }
855*6236dae4SAndroid Build Coastguard Worker else if(rc == WS_WANT_WRITE) {
856*6236dae4SAndroid Build Coastguard Worker *block = TRUE;
857*6236dae4SAndroid Build Coastguard Worker conn->waitfor = KEEP_SEND;
858*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
859*6236dae4SAndroid Build Coastguard Worker }
860*6236dae4SAndroid Build Coastguard Worker else if(name && (rc == WS_SUCCESS)) {
861*6236dae4SAndroid Build Coastguard Worker WS_SFTPNAME *origname = name;
862*6236dae4SAndroid Build Coastguard Worker result = CURLE_OK;
863*6236dae4SAndroid Build Coastguard Worker while(name) {
864*6236dae4SAndroid Build Coastguard Worker char *line = aprintf("%s\n",
865*6236dae4SAndroid Build Coastguard Worker data->set.list_only ?
866*6236dae4SAndroid Build Coastguard Worker name->fName : name->lName);
867*6236dae4SAndroid Build Coastguard Worker if(!line) {
868*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_CLOSE);
869*6236dae4SAndroid Build Coastguard Worker sshc->actualcode = CURLE_OUT_OF_MEMORY;
870*6236dae4SAndroid Build Coastguard Worker break;
871*6236dae4SAndroid Build Coastguard Worker }
872*6236dae4SAndroid Build Coastguard Worker result = Curl_client_write(data, CLIENTWRITE_BODY,
873*6236dae4SAndroid Build Coastguard Worker line, strlen(line));
874*6236dae4SAndroid Build Coastguard Worker free(line);
875*6236dae4SAndroid Build Coastguard Worker if(result) {
876*6236dae4SAndroid Build Coastguard Worker sshc->actualcode = result;
877*6236dae4SAndroid Build Coastguard Worker break;
878*6236dae4SAndroid Build Coastguard Worker }
879*6236dae4SAndroid Build Coastguard Worker name = name->next;
880*6236dae4SAndroid Build Coastguard Worker }
881*6236dae4SAndroid Build Coastguard Worker wolfSSH_SFTPNAME_list_free(origname);
882*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
883*6236dae4SAndroid Build Coastguard Worker return result;
884*6236dae4SAndroid Build Coastguard Worker }
885*6236dae4SAndroid Build Coastguard Worker failf(data, "wolfssh SFTP ls failed: %d", rc);
886*6236dae4SAndroid Build Coastguard Worker return CURLE_SSH;
887*6236dae4SAndroid Build Coastguard Worker
888*6236dae4SAndroid Build Coastguard Worker case SSH_SFTP_SHUTDOWN:
889*6236dae4SAndroid Build Coastguard Worker Curl_safefree(sshc->homedir);
890*6236dae4SAndroid Build Coastguard Worker wolfSSH_free(sshc->ssh_session);
891*6236dae4SAndroid Build Coastguard Worker wolfSSH_CTX_free(sshc->ctx);
892*6236dae4SAndroid Build Coastguard Worker state(data, SSH_STOP);
893*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
894*6236dae4SAndroid Build Coastguard Worker default:
895*6236dae4SAndroid Build Coastguard Worker break;
896*6236dae4SAndroid Build Coastguard Worker }
897*6236dae4SAndroid Build Coastguard Worker } while(!rc && (sshc->state != SSH_STOP));
898*6236dae4SAndroid Build Coastguard Worker return result;
899*6236dae4SAndroid Build Coastguard Worker }
900*6236dae4SAndroid Build Coastguard Worker
901*6236dae4SAndroid Build Coastguard Worker /* called repeatedly until done from multi.c */
wssh_multi_statemach(struct Curl_easy * data,bool * done)902*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_multi_statemach(struct Curl_easy *data, bool *done)
903*6236dae4SAndroid Build Coastguard Worker {
904*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
905*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc = &conn->proto.sshc;
906*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
907*6236dae4SAndroid Build Coastguard Worker bool block; /* we store the status and use that to provide a ssh_getsock()
908*6236dae4SAndroid Build Coastguard Worker implementation */
909*6236dae4SAndroid Build Coastguard Worker do {
910*6236dae4SAndroid Build Coastguard Worker result = wssh_statemach_act(data, &block);
911*6236dae4SAndroid Build Coastguard Worker *done = (sshc->state == SSH_STOP);
912*6236dae4SAndroid Build Coastguard Worker /* if there is no error, it is not done and it did not EWOULDBLOCK, then
913*6236dae4SAndroid Build Coastguard Worker try again */
914*6236dae4SAndroid Build Coastguard Worker if(*done) {
915*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "wssh_statemach_act says DONE"));
916*6236dae4SAndroid Build Coastguard Worker }
917*6236dae4SAndroid Build Coastguard Worker } while(!result && !*done && !block);
918*6236dae4SAndroid Build Coastguard Worker
919*6236dae4SAndroid Build Coastguard Worker return result;
920*6236dae4SAndroid Build Coastguard Worker }
921*6236dae4SAndroid Build Coastguard Worker
922*6236dae4SAndroid Build Coastguard Worker static
wscp_perform(struct Curl_easy * data,bool * connected,bool * dophase_done)923*6236dae4SAndroid Build Coastguard Worker CURLcode wscp_perform(struct Curl_easy *data,
924*6236dae4SAndroid Build Coastguard Worker bool *connected,
925*6236dae4SAndroid Build Coastguard Worker bool *dophase_done)
926*6236dae4SAndroid Build Coastguard Worker {
927*6236dae4SAndroid Build Coastguard Worker (void)data;
928*6236dae4SAndroid Build Coastguard Worker (void)connected;
929*6236dae4SAndroid Build Coastguard Worker (void)dophase_done;
930*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
931*6236dae4SAndroid Build Coastguard Worker }
932*6236dae4SAndroid Build Coastguard Worker
933*6236dae4SAndroid Build Coastguard Worker static
wsftp_perform(struct Curl_easy * data,bool * connected,bool * dophase_done)934*6236dae4SAndroid Build Coastguard Worker CURLcode wsftp_perform(struct Curl_easy *data,
935*6236dae4SAndroid Build Coastguard Worker bool *connected,
936*6236dae4SAndroid Build Coastguard Worker bool *dophase_done)
937*6236dae4SAndroid Build Coastguard Worker {
938*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
939*6236dae4SAndroid Build Coastguard Worker
940*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "DO phase starts"));
941*6236dae4SAndroid Build Coastguard Worker
942*6236dae4SAndroid Build Coastguard Worker *dophase_done = FALSE; /* not done yet */
943*6236dae4SAndroid Build Coastguard Worker
944*6236dae4SAndroid Build Coastguard Worker /* start the first command in the DO phase */
945*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_QUOTE_INIT);
946*6236dae4SAndroid Build Coastguard Worker
947*6236dae4SAndroid Build Coastguard Worker /* run the state-machine */
948*6236dae4SAndroid Build Coastguard Worker result = wssh_multi_statemach(data, dophase_done);
949*6236dae4SAndroid Build Coastguard Worker
950*6236dae4SAndroid Build Coastguard Worker *connected = Curl_conn_is_connected(data->conn, FIRSTSOCKET);
951*6236dae4SAndroid Build Coastguard Worker
952*6236dae4SAndroid Build Coastguard Worker if(*dophase_done) {
953*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "DO phase is complete"));
954*6236dae4SAndroid Build Coastguard Worker }
955*6236dae4SAndroid Build Coastguard Worker
956*6236dae4SAndroid Build Coastguard Worker return result;
957*6236dae4SAndroid Build Coastguard Worker }
958*6236dae4SAndroid Build Coastguard Worker
959*6236dae4SAndroid Build Coastguard Worker /*
960*6236dae4SAndroid Build Coastguard Worker * The DO function is generic for both protocols.
961*6236dae4SAndroid Build Coastguard Worker */
wssh_do(struct Curl_easy * data,bool * done)962*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_do(struct Curl_easy *data, bool *done)
963*6236dae4SAndroid Build Coastguard Worker {
964*6236dae4SAndroid Build Coastguard Worker CURLcode result;
965*6236dae4SAndroid Build Coastguard Worker bool connected = FALSE;
966*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
967*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc = &conn->proto.sshc;
968*6236dae4SAndroid Build Coastguard Worker
969*6236dae4SAndroid Build Coastguard Worker *done = FALSE; /* default to false */
970*6236dae4SAndroid Build Coastguard Worker data->req.size = -1; /* make sure this is unknown at this point */
971*6236dae4SAndroid Build Coastguard Worker sshc->actualcode = CURLE_OK; /* reset error code */
972*6236dae4SAndroid Build Coastguard Worker sshc->secondCreateDirs = 0; /* reset the create dir attempt state
973*6236dae4SAndroid Build Coastguard Worker variable */
974*6236dae4SAndroid Build Coastguard Worker
975*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetUploadCounter(data, 0);
976*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetDownloadCounter(data, 0);
977*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetUploadSize(data, -1);
978*6236dae4SAndroid Build Coastguard Worker Curl_pgrsSetDownloadSize(data, -1);
979*6236dae4SAndroid Build Coastguard Worker
980*6236dae4SAndroid Build Coastguard Worker if(conn->handler->protocol & CURLPROTO_SCP)
981*6236dae4SAndroid Build Coastguard Worker result = wscp_perform(data, &connected, done);
982*6236dae4SAndroid Build Coastguard Worker else
983*6236dae4SAndroid Build Coastguard Worker result = wsftp_perform(data, &connected, done);
984*6236dae4SAndroid Build Coastguard Worker
985*6236dae4SAndroid Build Coastguard Worker return result;
986*6236dae4SAndroid Build Coastguard Worker }
987*6236dae4SAndroid Build Coastguard Worker
wssh_block_statemach(struct Curl_easy * data,bool disconnect)988*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_block_statemach(struct Curl_easy *data,
989*6236dae4SAndroid Build Coastguard Worker bool disconnect)
990*6236dae4SAndroid Build Coastguard Worker {
991*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn = data->conn;
992*6236dae4SAndroid Build Coastguard Worker struct ssh_conn *sshc = &conn->proto.sshc;
993*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
994*6236dae4SAndroid Build Coastguard Worker
995*6236dae4SAndroid Build Coastguard Worker while((sshc->state != SSH_STOP) && !result) {
996*6236dae4SAndroid Build Coastguard Worker bool block;
997*6236dae4SAndroid Build Coastguard Worker timediff_t left = 1000;
998*6236dae4SAndroid Build Coastguard Worker struct curltime now = Curl_now();
999*6236dae4SAndroid Build Coastguard Worker
1000*6236dae4SAndroid Build Coastguard Worker result = wssh_statemach_act(data, &block);
1001*6236dae4SAndroid Build Coastguard Worker if(result)
1002*6236dae4SAndroid Build Coastguard Worker break;
1003*6236dae4SAndroid Build Coastguard Worker
1004*6236dae4SAndroid Build Coastguard Worker if(!disconnect) {
1005*6236dae4SAndroid Build Coastguard Worker if(Curl_pgrsUpdate(data))
1006*6236dae4SAndroid Build Coastguard Worker return CURLE_ABORTED_BY_CALLBACK;
1007*6236dae4SAndroid Build Coastguard Worker
1008*6236dae4SAndroid Build Coastguard Worker result = Curl_speedcheck(data, now);
1009*6236dae4SAndroid Build Coastguard Worker if(result)
1010*6236dae4SAndroid Build Coastguard Worker break;
1011*6236dae4SAndroid Build Coastguard Worker
1012*6236dae4SAndroid Build Coastguard Worker left = Curl_timeleft(data, NULL, FALSE);
1013*6236dae4SAndroid Build Coastguard Worker if(left < 0) {
1014*6236dae4SAndroid Build Coastguard Worker failf(data, "Operation timed out");
1015*6236dae4SAndroid Build Coastguard Worker return CURLE_OPERATION_TIMEDOUT;
1016*6236dae4SAndroid Build Coastguard Worker }
1017*6236dae4SAndroid Build Coastguard Worker }
1018*6236dae4SAndroid Build Coastguard Worker
1019*6236dae4SAndroid Build Coastguard Worker if(!result) {
1020*6236dae4SAndroid Build Coastguard Worker int dir = conn->waitfor;
1021*6236dae4SAndroid Build Coastguard Worker curl_socket_t sock = conn->sock[FIRSTSOCKET];
1022*6236dae4SAndroid Build Coastguard Worker curl_socket_t fd_read = CURL_SOCKET_BAD;
1023*6236dae4SAndroid Build Coastguard Worker curl_socket_t fd_write = CURL_SOCKET_BAD;
1024*6236dae4SAndroid Build Coastguard Worker if(dir == KEEP_RECV)
1025*6236dae4SAndroid Build Coastguard Worker fd_read = sock;
1026*6236dae4SAndroid Build Coastguard Worker else if(dir == KEEP_SEND)
1027*6236dae4SAndroid Build Coastguard Worker fd_write = sock;
1028*6236dae4SAndroid Build Coastguard Worker
1029*6236dae4SAndroid Build Coastguard Worker /* wait for the socket to become ready */
1030*6236dae4SAndroid Build Coastguard Worker (void)Curl_socket_check(fd_read, CURL_SOCKET_BAD, fd_write,
1031*6236dae4SAndroid Build Coastguard Worker left > 1000 ? 1000 : left); /* ignore result */
1032*6236dae4SAndroid Build Coastguard Worker }
1033*6236dae4SAndroid Build Coastguard Worker }
1034*6236dae4SAndroid Build Coastguard Worker
1035*6236dae4SAndroid Build Coastguard Worker return result;
1036*6236dae4SAndroid Build Coastguard Worker }
1037*6236dae4SAndroid Build Coastguard Worker
1038*6236dae4SAndroid Build Coastguard Worker /* generic done function for both SCP and SFTP called from their specific
1039*6236dae4SAndroid Build Coastguard Worker done functions */
wssh_done(struct Curl_easy * data,CURLcode status)1040*6236dae4SAndroid Build Coastguard Worker static CURLcode wssh_done(struct Curl_easy *data, CURLcode status)
1041*6236dae4SAndroid Build Coastguard Worker {
1042*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
1043*6236dae4SAndroid Build Coastguard Worker struct SSHPROTO *sftp_scp = data->req.p.ssh;
1044*6236dae4SAndroid Build Coastguard Worker
1045*6236dae4SAndroid Build Coastguard Worker if(!status) {
1046*6236dae4SAndroid Build Coastguard Worker /* run the state-machine */
1047*6236dae4SAndroid Build Coastguard Worker result = wssh_block_statemach(data, FALSE);
1048*6236dae4SAndroid Build Coastguard Worker }
1049*6236dae4SAndroid Build Coastguard Worker else
1050*6236dae4SAndroid Build Coastguard Worker result = status;
1051*6236dae4SAndroid Build Coastguard Worker
1052*6236dae4SAndroid Build Coastguard Worker if(sftp_scp)
1053*6236dae4SAndroid Build Coastguard Worker Curl_safefree(sftp_scp->path);
1054*6236dae4SAndroid Build Coastguard Worker if(Curl_pgrsDone(data))
1055*6236dae4SAndroid Build Coastguard Worker return CURLE_ABORTED_BY_CALLBACK;
1056*6236dae4SAndroid Build Coastguard Worker
1057*6236dae4SAndroid Build Coastguard Worker data->req.keepon = 0; /* clear all bits */
1058*6236dae4SAndroid Build Coastguard Worker return result;
1059*6236dae4SAndroid Build Coastguard Worker }
1060*6236dae4SAndroid Build Coastguard Worker
1061*6236dae4SAndroid Build Coastguard Worker #if 0
1062*6236dae4SAndroid Build Coastguard Worker static CURLcode wscp_done(struct Curl_easy *data,
1063*6236dae4SAndroid Build Coastguard Worker CURLcode code, bool premature)
1064*6236dae4SAndroid Build Coastguard Worker {
1065*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
1066*6236dae4SAndroid Build Coastguard Worker (void)conn;
1067*6236dae4SAndroid Build Coastguard Worker (void)code;
1068*6236dae4SAndroid Build Coastguard Worker (void)premature;
1069*6236dae4SAndroid Build Coastguard Worker
1070*6236dae4SAndroid Build Coastguard Worker return result;
1071*6236dae4SAndroid Build Coastguard Worker }
1072*6236dae4SAndroid Build Coastguard Worker
1073*6236dae4SAndroid Build Coastguard Worker static CURLcode wscp_doing(struct Curl_easy *data,
1074*6236dae4SAndroid Build Coastguard Worker bool *dophase_done)
1075*6236dae4SAndroid Build Coastguard Worker {
1076*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
1077*6236dae4SAndroid Build Coastguard Worker (void)conn;
1078*6236dae4SAndroid Build Coastguard Worker (void)dophase_done;
1079*6236dae4SAndroid Build Coastguard Worker
1080*6236dae4SAndroid Build Coastguard Worker return result;
1081*6236dae4SAndroid Build Coastguard Worker }
1082*6236dae4SAndroid Build Coastguard Worker
1083*6236dae4SAndroid Build Coastguard Worker static CURLcode wscp_disconnect(struct Curl_easy *data,
1084*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn, bool dead_connection)
1085*6236dae4SAndroid Build Coastguard Worker {
1086*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
1087*6236dae4SAndroid Build Coastguard Worker (void)data;
1088*6236dae4SAndroid Build Coastguard Worker (void)conn;
1089*6236dae4SAndroid Build Coastguard Worker (void)dead_connection;
1090*6236dae4SAndroid Build Coastguard Worker
1091*6236dae4SAndroid Build Coastguard Worker return result;
1092*6236dae4SAndroid Build Coastguard Worker }
1093*6236dae4SAndroid Build Coastguard Worker #endif
1094*6236dae4SAndroid Build Coastguard Worker
wsftp_done(struct Curl_easy * data,CURLcode code,bool premature)1095*6236dae4SAndroid Build Coastguard Worker static CURLcode wsftp_done(struct Curl_easy *data,
1096*6236dae4SAndroid Build Coastguard Worker CURLcode code, bool premature)
1097*6236dae4SAndroid Build Coastguard Worker {
1098*6236dae4SAndroid Build Coastguard Worker (void)premature;
1099*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_CLOSE);
1100*6236dae4SAndroid Build Coastguard Worker
1101*6236dae4SAndroid Build Coastguard Worker return wssh_done(data, code);
1102*6236dae4SAndroid Build Coastguard Worker }
1103*6236dae4SAndroid Build Coastguard Worker
wsftp_doing(struct Curl_easy * data,bool * dophase_done)1104*6236dae4SAndroid Build Coastguard Worker static CURLcode wsftp_doing(struct Curl_easy *data,
1105*6236dae4SAndroid Build Coastguard Worker bool *dophase_done)
1106*6236dae4SAndroid Build Coastguard Worker {
1107*6236dae4SAndroid Build Coastguard Worker CURLcode result = wssh_multi_statemach(data, dophase_done);
1108*6236dae4SAndroid Build Coastguard Worker
1109*6236dae4SAndroid Build Coastguard Worker if(*dophase_done) {
1110*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "DO phase is complete"));
1111*6236dae4SAndroid Build Coastguard Worker }
1112*6236dae4SAndroid Build Coastguard Worker return result;
1113*6236dae4SAndroid Build Coastguard Worker }
1114*6236dae4SAndroid Build Coastguard Worker
wsftp_disconnect(struct Curl_easy * data,struct connectdata * conn,bool dead)1115*6236dae4SAndroid Build Coastguard Worker static CURLcode wsftp_disconnect(struct Curl_easy *data,
1116*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn,
1117*6236dae4SAndroid Build Coastguard Worker bool dead)
1118*6236dae4SAndroid Build Coastguard Worker {
1119*6236dae4SAndroid Build Coastguard Worker CURLcode result = CURLE_OK;
1120*6236dae4SAndroid Build Coastguard Worker (void)dead;
1121*6236dae4SAndroid Build Coastguard Worker
1122*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "SSH DISCONNECT starts now"));
1123*6236dae4SAndroid Build Coastguard Worker
1124*6236dae4SAndroid Build Coastguard Worker if(conn->proto.sshc.ssh_session) {
1125*6236dae4SAndroid Build Coastguard Worker /* only if there is a session still around to use! */
1126*6236dae4SAndroid Build Coastguard Worker state(data, SSH_SFTP_SHUTDOWN);
1127*6236dae4SAndroid Build Coastguard Worker result = wssh_block_statemach(data, TRUE);
1128*6236dae4SAndroid Build Coastguard Worker }
1129*6236dae4SAndroid Build Coastguard Worker
1130*6236dae4SAndroid Build Coastguard Worker DEBUGF(infof(data, "SSH DISCONNECT is done"));
1131*6236dae4SAndroid Build Coastguard Worker return result;
1132*6236dae4SAndroid Build Coastguard Worker }
1133*6236dae4SAndroid Build Coastguard Worker
wssh_getsock(struct Curl_easy * data,struct connectdata * conn,curl_socket_t * sock)1134*6236dae4SAndroid Build Coastguard Worker static int wssh_getsock(struct Curl_easy *data,
1135*6236dae4SAndroid Build Coastguard Worker struct connectdata *conn,
1136*6236dae4SAndroid Build Coastguard Worker curl_socket_t *sock)
1137*6236dae4SAndroid Build Coastguard Worker {
1138*6236dae4SAndroid Build Coastguard Worker int bitmap = GETSOCK_BLANK;
1139*6236dae4SAndroid Build Coastguard Worker int dir = conn->waitfor;
1140*6236dae4SAndroid Build Coastguard Worker (void)data;
1141*6236dae4SAndroid Build Coastguard Worker sock[0] = conn->sock[FIRSTSOCKET];
1142*6236dae4SAndroid Build Coastguard Worker
1143*6236dae4SAndroid Build Coastguard Worker if(dir == KEEP_RECV)
1144*6236dae4SAndroid Build Coastguard Worker bitmap |= GETSOCK_READSOCK(FIRSTSOCKET);
1145*6236dae4SAndroid Build Coastguard Worker else if(dir == KEEP_SEND)
1146*6236dae4SAndroid Build Coastguard Worker bitmap |= GETSOCK_WRITESOCK(FIRSTSOCKET);
1147*6236dae4SAndroid Build Coastguard Worker
1148*6236dae4SAndroid Build Coastguard Worker return bitmap;
1149*6236dae4SAndroid Build Coastguard Worker }
1150*6236dae4SAndroid Build Coastguard Worker
Curl_ssh_version(char * buffer,size_t buflen)1151*6236dae4SAndroid Build Coastguard Worker void Curl_ssh_version(char *buffer, size_t buflen)
1152*6236dae4SAndroid Build Coastguard Worker {
1153*6236dae4SAndroid Build Coastguard Worker (void)msnprintf(buffer, buflen, "wolfssh/%s", LIBWOLFSSH_VERSION_STRING);
1154*6236dae4SAndroid Build Coastguard Worker }
1155*6236dae4SAndroid Build Coastguard Worker
Curl_ssh_init(void)1156*6236dae4SAndroid Build Coastguard Worker CURLcode Curl_ssh_init(void)
1157*6236dae4SAndroid Build Coastguard Worker {
1158*6236dae4SAndroid Build Coastguard Worker if(WS_SUCCESS != wolfSSH_Init()) {
1159*6236dae4SAndroid Build Coastguard Worker DEBUGF(fprintf(stderr, "Error: wolfSSH_Init failed\n"));
1160*6236dae4SAndroid Build Coastguard Worker return CURLE_FAILED_INIT;
1161*6236dae4SAndroid Build Coastguard Worker }
1162*6236dae4SAndroid Build Coastguard Worker
1163*6236dae4SAndroid Build Coastguard Worker return CURLE_OK;
1164*6236dae4SAndroid Build Coastguard Worker }
Curl_ssh_cleanup(void)1165*6236dae4SAndroid Build Coastguard Worker void Curl_ssh_cleanup(void)
1166*6236dae4SAndroid Build Coastguard Worker {
1167*6236dae4SAndroid Build Coastguard Worker (void)wolfSSH_Cleanup();
1168*6236dae4SAndroid Build Coastguard Worker }
1169*6236dae4SAndroid Build Coastguard Worker
1170*6236dae4SAndroid Build Coastguard Worker #endif /* USE_WOLFSSH */
1171