| 
									
										
										
										
											2017-05-02 00:09:15 -04:00
										 |  |  | // Copyright 2017 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-09 14:42:49 -04:00
										 |  |  | #include <array>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-09 14:47:24 -04:00
										 |  |  | #include <mbedtls/ctr_drbg.h>
 | 
					
						
							|  |  |  | #include <mbedtls/entropy.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-12 22:19:34 -04:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-14 18:48:28 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  | #include "common/file_util.h"
 | 
					
						
							| 
									
										
										
										
											2018-10-09 14:47:24 -04:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-14 18:48:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  | #include "core/file_sys/control_metadata.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/patch_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-31 12:21:34 -04:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-12 22:19:34 -04:00
										 |  |  | #include "core/settings.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-02 00:09:15 -04:00
										 |  |  | #include "core/telemetry_session.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-09 17:52:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  | #ifdef ENABLE_WEB_SERVICE
 | 
					
						
							|  |  |  | #include "web_service/telemetry_json.h"
 | 
					
						
							|  |  |  | #include "web_service/verify_login.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-02 00:09:15 -04:00
										 |  |  | namespace Core { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  | static u64 GenerateTelemetryId() { | 
					
						
							|  |  |  |     u64 telemetry_id{}; | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     mbedtls_entropy_context entropy; | 
					
						
							|  |  |  |     mbedtls_entropy_init(&entropy); | 
					
						
							|  |  |  |     mbedtls_ctr_drbg_context ctr_drbg; | 
					
						
							| 
									
										
										
										
											2018-10-09 14:42:49 -04:00
										 |  |  |     constexpr std::array<char, 18> personalization{{"yuzu Telemetry ID"}}; | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     mbedtls_ctr_drbg_init(&ctr_drbg); | 
					
						
							| 
									
										
										
										
											2018-09-17 17:16:01 +02:00
										 |  |  |     ASSERT(mbedtls_ctr_drbg_seed(&ctr_drbg, mbedtls_entropy_func, &entropy, | 
					
						
							| 
									
										
										
										
											2018-10-09 14:42:49 -04:00
										 |  |  |                                  reinterpret_cast<const unsigned char*>(personalization.data()), | 
					
						
							| 
									
										
										
										
											2018-09-17 20:58:24 +02:00
										 |  |  |                                  personalization.size()) == 0); | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  |     ASSERT(mbedtls_ctr_drbg_random(&ctr_drbg, reinterpret_cast<unsigned char*>(&telemetry_id), | 
					
						
							|  |  |  |                                    sizeof(u64)) == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     mbedtls_ctr_drbg_free(&ctr_drbg); | 
					
						
							|  |  |  |     mbedtls_entropy_free(&entropy); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  |     return telemetry_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-23 00:08:07 -04:00
										 |  |  | u64 GetTelemetryId() { | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  |     u64 telemetry_id{}; | 
					
						
							| 
									
										
										
										
											2018-08-20 20:06:22 -04:00
										 |  |  |     const std::string filename{FileUtil::GetUserPath(FileUtil::UserPath::ConfigDir) + | 
					
						
							|  |  |  |                                "telemetry_id"}; | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  |     bool generate_new_id = !FileUtil::Exists(filename); | 
					
						
							|  |  |  |     if (!generate_new_id) { | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  |         FileUtil::IOFile file(filename, "rb"); | 
					
						
							|  |  |  |         if (!file.IsOpen()) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |             LOG_ERROR(Core, "failed to open telemetry_id: {}", filename); | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  |             return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         file.ReadBytes(&telemetry_id, sizeof(u64)); | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  |         if (telemetry_id == 0) { | 
					
						
							|  |  |  |             LOG_ERROR(Frontend, "telemetry_id is 0. Generating a new one.", telemetry_id); | 
					
						
							|  |  |  |             generate_new_id = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (generate_new_id) { | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  |         FileUtil::IOFile file(filename, "wb"); | 
					
						
							|  |  |  |         if (!file.IsOpen()) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |             LOG_ERROR(Core, "failed to open telemetry_id: {}", filename); | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  |             return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         telemetry_id = GenerateTelemetryId(); | 
					
						
							|  |  |  |         file.WriteBytes(&telemetry_id, sizeof(u64)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return telemetry_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-23 00:08:07 -04:00
										 |  |  | u64 RegenerateTelemetryId() { | 
					
						
							|  |  |  |     const u64 new_telemetry_id{GenerateTelemetryId()}; | 
					
						
							| 
									
										
										
										
											2018-08-20 20:06:22 -04:00
										 |  |  |     const std::string filename{FileUtil::GetUserPath(FileUtil::UserPath::ConfigDir) + | 
					
						
							|  |  |  |                                "telemetry_id"}; | 
					
						
							| 
									
										
										
										
											2017-08-23 00:08:07 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     FileUtil::IOFile file(filename, "wb"); | 
					
						
							|  |  |  |     if (!file.IsOpen()) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_ERROR(Core, "failed to open telemetry_id: {}", filename); | 
					
						
							| 
									
										
										
										
											2017-08-23 00:08:07 -04:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     file.WriteBytes(&new_telemetry_id, sizeof(u64)); | 
					
						
							|  |  |  |     return new_telemetry_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-17 17:16:01 +02:00
										 |  |  | bool VerifyLogin(const std::string& username, const std::string& token) { | 
					
						
							| 
									
										
										
										
											2017-09-19 03:18:26 +02:00
										 |  |  | #ifdef ENABLE_WEB_SERVICE
 | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  |     return WebService::VerifyLogin(Settings::values.web_api_url, username, token); | 
					
						
							| 
									
										
										
										
											2017-09-19 03:18:26 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2017-09-19 03:18:26 +02:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  | TelemetrySession::TelemetrySession() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TelemetrySession::~TelemetrySession() { | 
					
						
							|  |  |  |     // Log one-time session end information
 | 
					
						
							|  |  |  |     const s64 shutdown_time{std::chrono::duration_cast<std::chrono::milliseconds>( | 
					
						
							|  |  |  |                                 std::chrono::system_clock::now().time_since_epoch()) | 
					
						
							|  |  |  |                                 .count()}; | 
					
						
							|  |  |  |     AddField(Telemetry::FieldType::Session, "Shutdown_Time", shutdown_time); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef ENABLE_WEB_SERVICE
 | 
					
						
							|  |  |  |     auto backend = std::make_unique<WebService::TelemetryJson>( | 
					
						
							|  |  |  |         Settings::values.web_api_url, Settings::values.yuzu_username, Settings::values.yuzu_token); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     auto backend = std::make_unique<Telemetry::NullVisitor>(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-28 21:32:48 -04:00
										 |  |  |     // Complete the session, submitting to the web service backend if necessary
 | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |     field_collection.Accept(*backend); | 
					
						
							|  |  |  |     if (Settings::values.enable_telemetry) { | 
					
						
							|  |  |  |         backend->Complete(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void TelemetrySession::AddInitialInfo(Loader::AppLoader& app_loader) { | 
					
						
							| 
									
										
										
										
											2017-08-22 20:58:19 -04:00
										 |  |  |     // Log one-time top-level information
 | 
					
						
							|  |  |  |     AddField(Telemetry::FieldType::None, "TelemetryId", GetTelemetryId()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-02 00:10:04 -04:00
										 |  |  |     // Log one-time session start information
 | 
					
						
							| 
									
										
										
										
											2017-07-12 22:19:34 -04:00
										 |  |  |     const s64 init_time{std::chrono::duration_cast<std::chrono::milliseconds>( | 
					
						
							|  |  |  |                             std::chrono::system_clock::now().time_since_epoch()) | 
					
						
							|  |  |  |                             .count()}; | 
					
						
							|  |  |  |     AddField(Telemetry::FieldType::Session, "Init_Time", init_time); | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     u64 program_id{}; | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |     const Loader::ResultStatus res{app_loader.ReadProgramId(program_id)}; | 
					
						
							| 
									
										
										
										
											2017-08-01 19:55:48 -04:00
										 |  |  |     if (res == Loader::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2018-09-17 17:16:01 +02:00
										 |  |  |         const std::string formatted_program_id{fmt::format("{:016X}", program_id)}; | 
					
						
							| 
									
										
										
										
											2018-09-16 20:05:51 +02:00
										 |  |  |         AddField(Telemetry::FieldType::Session, "ProgramId", formatted_program_id); | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         std::string name; | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |         app_loader.ReadTitle(name); | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (name.empty()) { | 
					
						
							|  |  |  |             auto [nacp, icon_file] = FileSys::PatchManager(program_id).GetControlMetadata(); | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |             if (nacp != nullptr) { | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  |                 name = nacp->GetApplicationName(); | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |         if (!name.empty()) { | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  |             AddField(Telemetry::FieldType::Session, "ProgramName", name); | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-08-01 19:55:48 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-05-02 00:10:04 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  |     AddField(Telemetry::FieldType::Session, "ProgramFormat", | 
					
						
							| 
									
										
										
										
											2019-05-28 21:12:23 -04:00
										 |  |  |              static_cast<u8>(app_loader.GetFileType())); | 
					
						
							| 
									
										
										
										
											2018-09-03 19:00:51 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-12 22:19:34 -04:00
										 |  |  |     // Log application information
 | 
					
						
							| 
									
										
										
										
											2018-08-14 18:48:28 -04:00
										 |  |  |     Telemetry::AppendBuildInfo(field_collection); | 
					
						
							| 
									
										
										
										
											2017-07-12 22:19:34 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-14 18:48:28 -04:00
										 |  |  |     // Log user system information
 | 
					
						
							|  |  |  |     Telemetry::AppendCPUInfo(field_collection); | 
					
						
							|  |  |  |     Telemetry::AppendOSInfo(field_collection); | 
					
						
							| 
									
										
										
										
											2017-07-12 22:19:34 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Log user configuration information
 | 
					
						
							| 
									
										
										
										
											2019-12-02 16:49:46 -03:00
										 |  |  |     constexpr auto field_type = Telemetry::FieldType::UserConfig; | 
					
						
							|  |  |  |     AddField(field_type, "Audio_SinkId", Settings::values.sink_id); | 
					
						
							|  |  |  |     AddField(field_type, "Audio_EnableAudioStretching", Settings::values.enable_audio_stretching); | 
					
						
							|  |  |  |     AddField(field_type, "Core_UseMultiCore", Settings::values.use_multi_core); | 
					
						
							|  |  |  |     AddField(field_type, "Renderer_ResolutionFactor", Settings::values.resolution_factor); | 
					
						
							|  |  |  |     AddField(field_type, "Renderer_UseFrameLimit", Settings::values.use_frame_limit); | 
					
						
							|  |  |  |     AddField(field_type, "Renderer_FrameLimit", Settings::values.frame_limit); | 
					
						
							|  |  |  |     AddField(field_type, "Renderer_UseDiskShaderCache", Settings::values.use_disk_shader_cache); | 
					
						
							|  |  |  |     AddField(field_type, "Renderer_UseAccurateGpuEmulation", | 
					
						
							| 
									
										
										
										
											2018-10-16 17:02:29 -04:00
										 |  |  |              Settings::values.use_accurate_gpu_emulation); | 
					
						
							| 
									
										
										
										
											2019-12-02 16:49:46 -03:00
										 |  |  |     AddField(field_type, "Renderer_UseAsynchronousGpuEmulation", | 
					
						
							| 
									
										
										
										
											2019-01-07 21:46:33 -05:00
										 |  |  |              Settings::values.use_asynchronous_gpu_emulation); | 
					
						
							| 
									
										
										
										
											2019-12-02 16:49:46 -03:00
										 |  |  |     AddField(field_type, "System_UseDockedMode", Settings::values.use_docked_mode); | 
					
						
							| 
									
										
										
										
											2017-05-02 00:09:15 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-26 16:21:45 +02:00
										 |  |  | bool TelemetrySession::SubmitTestcase() { | 
					
						
							|  |  |  | #ifdef ENABLE_WEB_SERVICE
 | 
					
						
							| 
									
										
										
										
											2019-04-19 22:02:49 +08:00
										 |  |  |     auto backend = std::make_unique<WebService::TelemetryJson>( | 
					
						
							|  |  |  |         Settings::values.web_api_url, Settings::values.yuzu_username, Settings::values.yuzu_token); | 
					
						
							| 
									
										
										
										
											2018-10-26 16:21:45 +02:00
										 |  |  |     field_collection.Accept(*backend); | 
					
						
							|  |  |  |     return backend->SubmitTestcase(); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-02 00:09:15 -04:00
										 |  |  | } // namespace Core
 |