You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
lnav/src/curl_looper.cc

259 lines
8.4 KiB
C++

/**
* Copyright (c) 2015, Timothy Stack
*
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* * Neither the name of Timothy Stack nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ''AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* @file curl_looper.cc
*/
#include "config.h"
#ifdef HAVE_LIBCURL
#include <curl/multi.h>
#include "curl_looper.hh"
using namespace std;
struct curl_request_eq {
curl_request_eq(const std::string &name) : cre_name(name) {
};
bool operator()(const curl_request *cr) const {
return this->cre_name == cr->get_name();
};
bool operator()(const pair<mstime_t, curl_request *> &pair) const {
return this->cre_name == pair.second->get_name();
};
const std::string &cre_name;
};
int curl_request::debug_cb(CURL *handle,
curl_infotype type,
char *data,
size_t size,
void *userp) {
curl_request *cr = (curl_request *) userp;
if (type == CURLINFO_TEXT) {
while (size > 0 && isspace(data[size - 1])) {
size -= 1;
}
log_debug("%s:%.*s", cr->get_name().c_str(), size, data);
}
return 0;
}
void *curl_looper::trampoline(void *arg)
{
curl_looper *cl = (curl_looper *) arg;
return cl->run();
}
void *curl_looper::run()
{
log_info("curl looper thread started");
while (this->cl_looping) {
this->loop_body();
}
log_info("curl looper thread exiting");
return NULL;
}
void curl_looper::loop_body()
{
mstime_t current_time = getmstime();
int timeout = this->compute_timeout(current_time);
if (this->cl_handle_to_request.empty()) {
mutex_guard mg(this->cl_mutex);
if (this->cl_new_requests.empty() && this->cl_close_requests.empty()) {
mstime_t deadline = current_time + timeout;
struct timespec ts;
ts.tv_sec = deadline / 1000ULL;
ts.tv_nsec = (deadline % 1000ULL) * 1000 * 1000;
log_trace("no requests in progress, waiting %d ms for new ones",
timeout);
pthread_cond_timedwait(&this->cl_cond, &this->cl_mutex, &ts);
}
}
this->perform_io();
this->check_for_finished_requests();
this->check_for_new_requests();
this->requeue_requests(current_time + 5);
}
void curl_looper::perform_io()
{
if (this->cl_handle_to_request.empty()) {
return;
}
mstime_t current_time = getmstime();
int timeout = this->compute_timeout(current_time);
int running_handles;
curl_multi_wait(this->cl_curl_multi,
NULL,
0,
timeout,
NULL);
curl_multi_perform(this->cl_curl_multi, &running_handles);
}
void curl_looper::requeue_requests(mstime_t up_to_time)
{
while (!this->cl_poll_queue.empty() &&
this->cl_poll_queue.front().first <= up_to_time) {
curl_request *cr = this->cl_poll_queue.front().second;
log_debug("%s:polling request is ready again -- %p",
cr->get_name().c_str(), cr);
this->cl_handle_to_request[cr->get_handle()] = cr;
curl_multi_add_handle(this->cl_curl_multi, cr->get_handle());
this->cl_poll_queue.erase(this->cl_poll_queue.begin());
}
}
void curl_looper::check_for_new_requests() {
mutex_guard mg(this->cl_mutex);
while (!this->cl_new_requests.empty()) {
curl_request *cr = this->cl_new_requests.back();
log_info("%s:new curl request %p",
cr->get_name().c_str(),
cr);
this->cl_handle_to_request[cr->get_handle()] = cr;
curl_multi_add_handle(this->cl_curl_multi, cr->get_handle());
this->cl_new_requests.pop_back();
}
while (!this->cl_close_requests.empty()) {
const std::string &name = this->cl_close_requests.back();
vector<curl_request *>::iterator all_iter = find_if(
this->cl_all_requests.begin(),
this->cl_all_requests.end(),
curl_request_eq(name));
log_info("attempting to close request -- %s", name.c_str());
if (all_iter != this->cl_all_requests.end()) {
map<CURL *, curl_request *>::iterator act_iter;
vector<pair<mstime_t, curl_request *> >::iterator poll_iter;
curl_request *cr = *all_iter;
log_info("%s:closing request -- %p",
cr->get_name().c_str(), cr);
(*all_iter)->close();
act_iter = this->cl_handle_to_request.find(cr);
if (act_iter != this->cl_handle_to_request.end()) {
this->cl_handle_to_request.erase(act_iter);
curl_multi_remove_handle(this->cl_curl_multi,
cr->get_handle());
delete cr;
}
poll_iter = find_if(this->cl_poll_queue.begin(),
this->cl_poll_queue.end(),
curl_request_eq(name));
if (poll_iter != this->cl_poll_queue.end()) {
this->cl_poll_queue.erase(poll_iter);
}
this->cl_all_requests.erase(all_iter);
}
else {
log_error("Unable to find request with the name -- %s",
name.c_str());
}
this->cl_close_requests.pop_back();
pthread_cond_broadcast(&this->cl_cond);
}
}
void curl_looper::check_for_finished_requests()
{
CURLMsg *msg;
int msgs_left;
while ((msg = curl_multi_info_read(this->cl_curl_multi, &msgs_left)) != NULL) {
if (msg->msg != CURLMSG_DONE) {
continue;
}
CURL *easy = msg->easy_handle;
map<CURL *, curl_request *>::iterator iter = this->cl_handle_to_request.find(easy);
curl_multi_remove_handle(this->cl_curl_multi, easy);
if (iter != this->cl_handle_to_request.end()) {
curl_request *cr = iter->second;
long delay_ms;
this->cl_handle_to_request.erase(iter);
delay_ms = cr->complete(msg->data.result);
if (delay_ms < 0) {
vector<curl_request *>::iterator all_iter;
log_info("%s:curl_request %p finished, deleting...",
cr->get_name().c_str(), cr);
{
mutex_guard mg(this->cl_mutex);
all_iter = find(this->cl_all_requests.begin(),
this->cl_all_requests.end(),
cr);
if (all_iter != this->cl_all_requests.end()) {
this->cl_all_requests.erase(all_iter);
}
pthread_cond_broadcast(&cl_cond);
}
delete cr;
}
else {
log_debug("%s:curl_request %p is polling, requeueing in %d",
cr->get_name().c_str(),
cr,
delay_ms);
this->cl_poll_queue.push_back(
make_pair(getmstime() + delay_ms, cr));
sort(this->cl_poll_queue.begin(), this->cl_poll_queue.end());
}
}
}
}
#endif