/*************************************************************************** * _ _ ____ _ * Project ___| | | | _ \| | * / __| | | | |_) | | * | (__| |_| | _ <| |___ * \___|\___/|_| \_\_____| * * Copyright (C) Daniel Stenberg, , et al. * * This software is licensed as described in the file COPYING, which * you should have received as part of this distribution. The terms * are also available at https://curl.se/docs/copyright.html. * * You may opt to use, copy, modify, merge, publish, distribute and/or sell * copies of the Software, and permit persons to whom the Software is * furnished to do so, under the terms of the COPYING file. * * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY * KIND, either express or implied. * * SPDX-License-Identifier: curl * ***************************************************************************/ #include "tool_setup.h" #ifdef HAVE_SYS_SELECT_H #include #endif #ifdef HAVE_POLL_H #include #elif defined(HAVE_SYS_POLL_H) #include #endif #include "tool_cfgable.h" #include "tool_cb_rea.h" #include "tool_operate.h" #include "tool_msgs.h" #ifndef _WIN32 /* Wait up to a number of milliseconds for socket activity. This function waits on read activity on a file descriptor that is not a socket which makes it not work with select() or poll() on Windows. */ static bool waitfd(int waitms, int fd) { #ifdef HAVE_POLL struct pollfd set; set.fd = fd; set.events = POLLIN; set.revents = 0; if(poll(&set, 1, waitms)) return TRUE; /* timeout */ return FALSE; #else fd_set bits; struct timeval timeout; if(fd >= FD_SETSIZE) return FALSE; /* cannot wait! */ /* wait this long at the most */ timeout.tv_sec = waitms / 1000; timeout.tv_usec = (int)((waitms % 1000) * 1000); FD_ZERO(&bits); FD_SET(fd, &bits); if(!select(fd + 1, &bits, NULL, NULL, &timeout)) return TRUE; /* timeout */ return FALSE; #endif } #endif /* ** callback for CURLOPT_READFUNCTION */ size_t tool_read_cb(char *buffer, size_t sz, size_t nmemb, void *userdata) { ssize_t rc = 0; struct per_transfer *per = userdata; struct OperationConfig *config = per->config; if((per->uploadfilesize != -1) && (per->uploadedsofar == per->uploadfilesize)) { /* done */ return 0; } if(config->timeout_ms) { struct curltime now = curlx_now(); long msdelta = (long)curlx_timediff_ms(now, per->start); if(msdelta > config->timeout_ms) return 0; /* timeout */ #ifndef _WIN32 else { long w = config->timeout_ms - msdelta; if(w > INT_MAX) w = INT_MAX; waitfd((int)w, per->infd); } #endif } #ifdef _WIN32 /* If we are on Windows, and using `-T .`, then per->infd points to a socket connected to stdin via a reader thread, and needs to be read with recv() Make sure we are in non-blocking mode and infd is not regular stdin On Linux per->infd should be stdin (0) and the block below should not execute */ if(per->uploadfile && !strcmp(per->uploadfile, ".") && per->infd > 0) { #ifndef CURL_WINDOWS_UWP rc = sread(per->infd, buffer, curlx_uztosi(sz * nmemb)); if(rc < 0) { if(SOCKERRNO == SOCKEWOULDBLOCK) { errno = 0; config->readbusy = TRUE; return CURL_READFUNC_PAUSE; } rc = 0; } #else warnf("per->infd != 0: FD == %d. " "This behavior is only supported on desktop Windows", per->infd); #endif } else #endif /* _WIN32 */ { rc = read(per->infd, buffer, sz * nmemb); if(rc < 0) { if(errno == EAGAIN) { errno = 0; config->readbusy = TRUE; return CURL_READFUNC_PAUSE; } /* since size_t is unsigned we cannot return negative values fine */ rc = 0; } } if((per->uploadfilesize != -1) && (per->uploadedsofar + rc > per->uploadfilesize)) { /* do not allow uploading more than originally set out to do */ curl_off_t delta = per->uploadedsofar + rc - per->uploadfilesize; warnf("File size larger in the end than when " "started. Dropping at least %" CURL_FORMAT_CURL_OFF_T " bytes", delta); rc = (ssize_t)(per->uploadfilesize - per->uploadedsofar); } config->readbusy = FALSE; /* when select() returned zero here, it timed out */ return (size_t)rc; } /* ** callback for CURLOPT_XFERINFOFUNCTION used to unpause busy reads */ int tool_readbusy_cb(void *clientp, curl_off_t dltotal, curl_off_t dlnow, curl_off_t ultotal, curl_off_t ulnow) { struct per_transfer *per = clientp; struct OperationConfig *config = per->config; static curl_off_t ulprev; (void)dltotal; (void)dlnow; (void)ultotal; (void)ulnow; if(config->readbusy) { if(ulprev == ulnow) { #ifndef _WIN32 waitfd(1, per->infd); #else curlx_wait_ms(1); /* sleep */ #endif } config->readbusy = FALSE; curl_easy_pause(per->curl, CURLPAUSE_CONT); } ulprev = ulnow; return per->noprogress ? 0 : CURL_PROGRESSFUNC_CONTINUE; }