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.
		
		
		
		
		
			
		
			
				
					
					
						
							151 lines
						
					
					
						
							4.5 KiB
						
					
					
				
			
		
		
	
	
							151 lines
						
					
					
						
							4.5 KiB
						
					
					
				| #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);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 |