// Copyright 2016 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #include #include #include #include #include "Common/Analytics.h" #include "Common/CommonTypes.h" #include "Common/StringUtil.h" #include "Common/Thread.h" namespace Common { namespace { // Format version number, used as the first byte of every report sent. // Increment for any change to the wire format. constexpr u8 WIRE_FORMAT_VERSION = 0; // Identifiers for the value types supported by the analytics reporting wire // format. enum class TypeId : u8 { STRING = 0, BOOL = 1, UINT = 2, SINT = 3, FLOAT = 4, }; void AppendBool(std::string* out, bool v) { out->push_back(v ? '\xFF' : '\x00'); } void AppendVarInt(std::string* out, u64 v) { do { u8 current_byte = v & 0x7F; v >>= 7; current_byte |= (!!v) << 7; out->push_back(current_byte); } while (v); } void AppendBytes(std::string* out, const u8* bytes, u32 length, bool encode_length = true) { if (encode_length) { AppendVarInt(out, length); } out->append(reinterpret_cast(bytes), length); } void AppendType(std::string* out, TypeId type) { out->push_back(static_cast(type)); } // Dummy write function for curl. size_t DummyCurlWriteFunction(char* ptr, size_t size, size_t nmemb, void* userdata) { return size * nmemb; } } // namespace AnalyticsReportBuilder::AnalyticsReportBuilder() { m_report.push_back(WIRE_FORMAT_VERSION); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, const std::string& v) { AppendType(report, TypeId::STRING); AppendBytes(report, reinterpret_cast(v.data()), static_cast(v.size())); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, const char* v) { AppendSerializedValue(report, std::string(v)); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, bool v) { AppendType(report, TypeId::BOOL); AppendBool(report, v); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, u64 v) { AppendType(report, TypeId::UINT); AppendVarInt(report, v); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, s64 v) { AppendType(report, TypeId::SINT); AppendBool(report, v >= 0); AppendVarInt(report, static_cast(std::abs(v))); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, u32 v) { AppendSerializedValue(report, static_cast(v)); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, s32 v) { AppendSerializedValue(report, static_cast(v)); } void AnalyticsReportBuilder::AppendSerializedValue(std::string* report, float v) { AppendType(report, TypeId::FLOAT); AppendBytes(report, reinterpret_cast(&v), sizeof(v), false); } AnalyticsReporter::AnalyticsReporter() { m_reporter_thread = std::thread(&AnalyticsReporter::ThreadProc, this); } AnalyticsReporter::~AnalyticsReporter() { // Set the exit request flag and wait for the thread to honor it. m_reporter_stop_request.Set(); m_reporter_event.Set(); m_reporter_thread.join(); } void AnalyticsReporter::Send(AnalyticsReportBuilder&& report) { #if defined(USE_ANALYTICS) && USE_ANALYTICS // Put a bound on the size of the queue to avoid uncontrolled memory growth. constexpr u32 QUEUE_SIZE_LIMIT = 25; if (m_reports_queue.Size() < QUEUE_SIZE_LIMIT) { m_reports_queue.Push(report.Consume()); m_reporter_event.Set(); } #endif } void AnalyticsReporter::ThreadProc() { Common::SetCurrentThreadName("Analytics"); while (true) { m_reporter_event.Wait(); if (m_reporter_stop_request.IsSet()) { return; } while (!m_reports_queue.Empty()) { std::shared_ptr backend(m_backend); if (backend) { std::string report; m_reports_queue.Pop(report); backend->Send(std::move(report)); } else { break; } // Recheck after each report sent. if (m_reporter_stop_request.IsSet()) { return; } } } } void StdoutAnalyticsBackend::Send(std::string report) { printf("Analytics report sent:\n%s", HexDump(reinterpret_cast(report.data()), report.size()).c_str()); } HttpAnalyticsBackend::HttpAnalyticsBackend(const std::string& endpoint) { CURL* curl = curl_easy_init(); if (curl) { // libcurl may not have been built with async DNS support, so we disable // signal handlers to avoid a possible and likely crash if a resolve times out. curl_easy_setopt(curl, CURLOPT_NOSIGNAL, true); curl_easy_setopt(curl, CURLOPT_URL, endpoint.c_str()); curl_easy_setopt(curl, CURLOPT_POST, true); curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &DummyCurlWriteFunction); curl_easy_setopt(curl, CURLOPT_TIMEOUT_MS, 3000); #ifdef _WIN32 // ALPN support is enabled by default but requires Windows >= 8.1. curl_easy_setopt(curl, CURLOPT_SSL_ENABLE_ALPN, false); #endif m_curl = curl; } } HttpAnalyticsBackend::~HttpAnalyticsBackend() { if (m_curl) { curl_easy_cleanup(m_curl); } } void HttpAnalyticsBackend::Send(std::string report) { if (!m_curl) return; curl_easy_setopt(m_curl, CURLOPT_POSTFIELDS, report.c_str()); curl_easy_setopt(m_curl, CURLOPT_POSTFIELDSIZE, report.size()); curl_easy_perform(m_curl); } } // namespace Common