You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
152 lines
4.5 KiB
152 lines
4.5 KiB
4 years ago
|
#include "selfdrive/ui/replay/util.h"
|
||
|
|
||
|
#include <cassert>
|
||
|
#include <bzlib.h>
|
||
|
#include <curl/curl.h>
|
||
|
|
||
|
#include "selfdrive/common/timing.h"
|
||
|
#include "selfdrive/common/util.h"
|
||
|
|
||
|
struct CURLGlobalInitializer {
|
||
|
CURLGlobalInitializer() { curl_global_init(CURL_GLOBAL_DEFAULT); }
|
||
|
~CURLGlobalInitializer() { curl_global_cleanup(); }
|
||
|
};
|
||
|
|
||
|
struct MultiPartWriter {
|
||
|
int64_t offset;
|
||
|
int64_t end;
|
||
|
FILE *fp;
|
||
|
};
|
||
|
|
||
|
static size_t write_cb(char *data, size_t n, size_t l, void *userp) {
|
||
|
MultiPartWriter *w = (MultiPartWriter *)userp;
|
||
|
fseek(w->fp, w->offset, SEEK_SET);
|
||
|
fwrite(data, l, n, w->fp);
|
||
|
w->offset += n * l;
|
||
|
return n * l;
|
||
|
}
|
||
|
|
||
|
static size_t dumy_write_cb(char *data, size_t n, size_t l, void *userp) { return n * l; }
|
||
|
|
||
|
int64_t getDownloadContentLength(const std::string &url) {
|
||
|
CURL *curl = curl_easy_init();
|
||
|
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
||
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, dumy_write_cb);
|
||
|
curl_easy_setopt(curl, CURLOPT_HEADER, 1);
|
||
|
curl_easy_setopt(curl, CURLOPT_NOBODY, 1);
|
||
|
CURLcode res = curl_easy_perform(curl);
|
||
|
double content_length = -1;
|
||
|
if (res == CURLE_OK) {
|
||
|
res = curl_easy_getinfo(curl, CURLINFO_CONTENT_LENGTH_DOWNLOAD, &content_length);
|
||
|
}
|
||
|
curl_easy_cleanup(curl);
|
||
|
return res == CURLE_OK ? (int64_t)content_length : -1;
|
||
|
}
|
||
|
|
||
|
bool httpMultiPartDownload(const std::string &url, const std::string &target_file, int parts, std::atomic<bool> *abort) {
|
||
|
static CURLGlobalInitializer curl_initializer;
|
||
|
|
||
|
int64_t content_length = getDownloadContentLength(url);
|
||
|
if (content_length == -1) return false;
|
||
|
|
||
|
std::string tmp_file = target_file + ".tmp";
|
||
|
FILE *fp = fopen(tmp_file.c_str(), "wb");
|
||
|
// create a sparse file
|
||
|
fseek(fp, content_length, SEEK_SET);
|
||
|
|
||
|
CURLM *cm = curl_multi_init();
|
||
|
std::map<CURL *, MultiPartWriter> writers;
|
||
|
const int part_size = content_length / parts;
|
||
|
for (int i = 0; i < parts; ++i) {
|
||
|
CURL *eh = curl_easy_init();
|
||
|
writers[eh] = {
|
||
|
.fp = fp,
|
||
|
.offset = i * part_size,
|
||
|
.end = i == parts - 1 ? content_length - 1 : (i + 1) * part_size - 1,
|
||
|
};
|
||
|
curl_easy_setopt(eh, CURLOPT_WRITEFUNCTION, write_cb);
|
||
|
curl_easy_setopt(eh, CURLOPT_WRITEDATA, (void *)(&writers[eh]));
|
||
|
curl_easy_setopt(eh, CURLOPT_URL, url.c_str());
|
||
|
curl_easy_setopt(eh, CURLOPT_RANGE, util::string_format("%d-%d", writers[eh].offset, writers[eh].end).c_str());
|
||
|
curl_easy_setopt(eh, CURLOPT_HTTPGET, 1);
|
||
|
curl_easy_setopt(eh, CURLOPT_NOSIGNAL, 1);
|
||
|
curl_easy_setopt(eh, CURLOPT_FOLLOWLOCATION, 1);
|
||
|
curl_multi_add_handle(cm, eh);
|
||
|
}
|
||
|
|
||
|
int running = 1, success_cnt = 0;
|
||
|
while (!(abort && abort->load())) {
|
||
|
CURLMcode ret = curl_multi_perform(cm, &running);
|
||
|
|
||
|
if (!running) {
|
||
|
CURLMsg *msg;
|
||
|
int msgs_left = -1;
|
||
|
while ((msg = curl_multi_info_read(cm, &msgs_left))) {
|
||
|
if (msg->msg == CURLMSG_DONE && msg->data.result == CURLE_OK) {
|
||
|
int http_status_code = 0;
|
||
|
curl_easy_getinfo(msg->easy_handle, CURLINFO_RESPONSE_CODE, &http_status_code);
|
||
|
success_cnt += (http_status_code == 206);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (ret == CURLM_OK) {
|
||
|
curl_multi_wait(cm, nullptr, 0, 1000, nullptr);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
fclose(fp);
|
||
|
bool success = success_cnt == parts;
|
||
|
if (success) {
|
||
|
success = ::rename(tmp_file.c_str(), target_file.c_str()) == 0;
|
||
|
}
|
||
|
|
||
|
// cleanup curl
|
||
|
for (auto &[e, w] : writers) {
|
||
|
curl_multi_remove_handle(cm, e);
|
||
|
curl_easy_cleanup(e);
|
||
|
}
|
||
|
curl_multi_cleanup(cm);
|
||
|
return success;
|
||
|
}
|
||
|
|
||
|
bool readBZ2File(const std::string_view file, std::ostream &stream) {
|
||
|
std::unique_ptr<FILE, decltype(&fclose)> f(fopen(file.data(), "r"), &fclose);
|
||
|
if (!f) return false;
|
||
|
|
||
|
int bzerror = BZ_OK;
|
||
|
BZFILE *bz_file = BZ2_bzReadOpen(&bzerror, f.get(), 0, 0, nullptr, 0);
|
||
|
if (!bz_file) return false;
|
||
|
|
||
|
std::array<char, 64 * 1024> buf;
|
||
|
do {
|
||
|
int size = BZ2_bzRead(&bzerror, bz_file, buf.data(), buf.size());
|
||
|
if (bzerror == BZ_OK || bzerror == BZ_STREAM_END) {
|
||
|
stream.write(buf.data(), size);
|
||
|
}
|
||
|
} while (bzerror == BZ_OK);
|
||
|
|
||
|
bool success = (bzerror == BZ_STREAM_END);
|
||
|
BZ2_bzReadClose(&bzerror, bz_file);
|
||
|
return success;
|
||
|
}
|
||
|
|
||
|
void precise_nano_sleep(long sleep_ns) {
|
||
|
const long estimate_ns = 1 * 1e6; // 1ms
|
||
|
struct timespec req = {.tv_nsec = estimate_ns};
|
||
|
uint64_t start_sleep = nanos_since_boot();
|
||
|
while (sleep_ns > estimate_ns) {
|
||
|
nanosleep(&req, nullptr);
|
||
|
uint64_t end_sleep = nanos_since_boot();
|
||
|
sleep_ns -= (end_sleep - start_sleep);
|
||
|
start_sleep = end_sleep;
|
||
|
}
|
||
|
// spin wait
|
||
|
if (sleep_ns > 0) {
|
||
|
while ((nanos_since_boot() - start_sleep) <= sleep_ns) {
|
||
|
usleep(0);
|
||
|
}
|
||
|
}
|
||
|
}
|