mirror of
				https://github.com/DarkflameUniverse/DarkflameServer.git
				synced 2025-10-26 10:02:00 +00:00 
			
		
		
		
	 ef6f2f133e
			
		
	
	ef6f2f133e
	
	
	
		
			
			* cast and code cleanup * cast cleanup * bug fixes and improvements * no getBoolField method exists * fixes * unbroke sg cannon scoring * removing comments * Remove the c-style cast warning I added from CMakeLists now that they're gone (it triggers on 3rd party dependencies and slows down compilation) * (Hopefully) fix MacOS compilation error * partially-implemented feedback * more updates to account for feedback * change bool default --------- Co-authored-by: jadebenn <jonahebenn@yahoo.com>
		
			
				
	
	
		
			275 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			275 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "Metrics.hpp"
 | |
| 
 | |
| #include <chrono>
 | |
| 
 | |
| std::unordered_map<MetricVariable, Metric*> Metrics::m_Metrics = {};
 | |
| std::vector<MetricVariable> Metrics::m_Variables = {
 | |
| 	MetricVariable::GameLoop,
 | |
| 	MetricVariable::PacketHandling,
 | |
| 	MetricVariable::UpdateEntities,
 | |
| 	MetricVariable::UpdateSpawners,
 | |
| 	MetricVariable::Physics,
 | |
| 	MetricVariable::UpdateReplica,
 | |
| 	MetricVariable::Ghosting,
 | |
| 	MetricVariable::CPUTime,
 | |
| 	MetricVariable::Sleep,
 | |
| 	MetricVariable::Frame,
 | |
| };
 | |
| 
 | |
| void Metrics::AddMeasurement(MetricVariable variable, int64_t value) {
 | |
| 	const auto& iter = m_Metrics.find(variable);
 | |
| 
 | |
| 	Metric* metric;
 | |
| 
 | |
| 	if (iter == m_Metrics.end()) {
 | |
| 		metric = new Metric();
 | |
| 
 | |
| 		m_Metrics[variable] = metric;
 | |
| 	} else {
 | |
| 		metric = iter->second;
 | |
| 	}
 | |
| 
 | |
| 	AddMeasurement(metric, value);
 | |
| }
 | |
| 
 | |
| void Metrics::AddMeasurement(Metric* metric, int64_t value) {
 | |
| 	const auto index = metric->measurementIndex;
 | |
| 
 | |
| 	metric->measurements[index] = value;
 | |
| 
 | |
| 	if (metric->max == -1 || value > metric->max) {
 | |
| 		metric->max = value;
 | |
| 	} else if (metric->min == -1 || metric->min > value) {
 | |
| 		metric->min = value;
 | |
| 	}
 | |
| 
 | |
| 	if (metric->measurementSize < MAX_MEASURMENT_POINTS) {
 | |
| 		metric->measurementSize++;
 | |
| 	}
 | |
| 
 | |
| 	metric->measurementIndex = (index + 1) % MAX_MEASURMENT_POINTS;
 | |
| }
 | |
| 
 | |
| const Metric* Metrics::GetMetric(MetricVariable variable) {
 | |
| 	const auto& iter = m_Metrics.find(variable);
 | |
| 
 | |
| 	if (iter == m_Metrics.end()) {
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 
 | |
| 	Metric* metric = iter->second;
 | |
| 
 | |
| 	int64_t average = 0;
 | |
| 
 | |
| 	for (size_t i = 0; i < metric->measurementSize; i++) {
 | |
| 		average += metric->measurements[i];
 | |
| 	}
 | |
| 
 | |
| 	average /= metric->measurementSize;
 | |
| 
 | |
| 	metric->average = average;
 | |
| 
 | |
| 	return metric;
 | |
| }
 | |
| 
 | |
| void Metrics::StartMeasurement(MetricVariable variable) {
 | |
| 	const auto& iter = m_Metrics.find(variable);
 | |
| 
 | |
| 	Metric* metric;
 | |
| 
 | |
| 	if (iter == m_Metrics.end()) {
 | |
| 		metric = new Metric();
 | |
| 
 | |
| 		m_Metrics[variable] = metric;
 | |
| 	} else {
 | |
| 		metric = iter->second;
 | |
| 	}
 | |
| 
 | |
| 	metric->activeMeasurement = std::chrono::high_resolution_clock::now();
 | |
| }
 | |
| 
 | |
| void Metrics::EndMeasurement(MetricVariable variable) {
 | |
| 	const auto end = std::chrono::high_resolution_clock::now();
 | |
| 
 | |
| 	const auto& iter = m_Metrics.find(variable);
 | |
| 
 | |
| 	if (iter == m_Metrics.end()) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	Metric* metric = iter->second;
 | |
| 
 | |
| 	const auto elapsed = end - metric->activeMeasurement;
 | |
| 
 | |
| 	const auto nanoseconds = std::chrono::duration_cast<std::chrono::nanoseconds>(elapsed).count();
 | |
| 
 | |
| 	AddMeasurement(metric, nanoseconds);
 | |
| }
 | |
| 
 | |
| float Metrics::ToMiliseconds(int64_t nanoseconds) {
 | |
| 	return static_cast<float>(nanoseconds) / 1e6;
 | |
| }
 | |
| 
 | |
| std::string Metrics::MetricVariableToString(MetricVariable variable) {
 | |
| 	switch (variable) {
 | |
| 	case MetricVariable::GameLoop:
 | |
| 		return "GameLoop";
 | |
| 	case MetricVariable::PacketHandling:
 | |
| 		return "PacketHandling";
 | |
| 	case MetricVariable::UpdateEntities:
 | |
| 		return "UpdateEntities";
 | |
| 	case MetricVariable::UpdateSpawners:
 | |
| 		return "UpdateSpawners";
 | |
| 	case MetricVariable::Physics:
 | |
| 		return "Physics";
 | |
| 	case MetricVariable::UpdateReplica:
 | |
| 		return "UpdateReplica";
 | |
| 	case MetricVariable::Sleep:
 | |
| 		return "Sleep";
 | |
| 	case MetricVariable::CPUTime:
 | |
| 		return "CPUTime";
 | |
| 	case MetricVariable::Frame:
 | |
| 		return "Frame";
 | |
| 	case MetricVariable::Ghosting:
 | |
| 		return "Ghosting";
 | |
| 
 | |
| 	default:
 | |
| 		return "Invalid";
 | |
| 	}
 | |
| }
 | |
| 
 | |
| const std::vector<MetricVariable>& Metrics::GetAllMetrics() {
 | |
| 	return m_Variables;
 | |
| }
 | |
| 
 | |
| void Metrics::Clear() {
 | |
| 	for (const auto& pair : m_Metrics) {
 | |
| 		delete pair.second;
 | |
| 	}
 | |
| 
 | |
| 	m_Metrics.clear();
 | |
| }
 | |
| 
 | |
| /* RSS Memory utilities
 | |
|  *
 | |
|  * Author:  David Robert Nadeau
 | |
|  * Site:    http://NadeauSoftware.com/
 | |
|  * License: Creative Commons Attribution 3.0 Unported License
 | |
|  *          http://creativecommons.org/licenses/by/3.0/deed.en_US
 | |
|  */
 | |
| 
 | |
| 
 | |
| #if defined(_WIN32)
 | |
| #include <windows.h>
 | |
| #include <psapi.h>
 | |
| 
 | |
| #elif defined(__unix__) || defined(__unix) || defined(unix) || (defined(__APPLE__) && defined(__MACH__))
 | |
| #include <unistd.h>
 | |
| #include <sys/resource.h>
 | |
| 
 | |
| #if defined(__APPLE__) && defined(__MACH__)
 | |
| #include <mach/mach.h>
 | |
| 
 | |
| #elif (defined(_AIX) || defined(__TOS__AIX__)) || (defined(__sun__) || defined(__sun) || defined(sun) && (defined(__SVR4) || defined(__svr4__)))
 | |
| #include <fcntl.h>
 | |
| #include <procfs.h>
 | |
| 
 | |
| #elif defined(__linux__) || defined(__linux) || defined(linux) || defined(__gnu_linux__)
 | |
| #include <stdio.h>
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #else
 | |
| #error "Cannot define getPeakRSS( ) or getCurrentRSS( ) for an unknown OS."
 | |
| #endif
 | |
| 
 | |
|  /**
 | |
|   * Returns the peak (maximum so far) resident set size (physical
 | |
|   * memory use) measured in bytes, or zero if the value cannot be
 | |
|   * determined on this OS.
 | |
|   */
 | |
| size_t Metrics::GetPeakRSS() {
 | |
| #if defined(_WIN32)
 | |
| 	/* Windows -------------------------------------------------- */
 | |
| 	PROCESS_MEMORY_COUNTERS info;
 | |
| 	GetProcessMemoryInfo(GetCurrentProcess(), &info, sizeof(info));
 | |
| 	return static_cast<size_t>(info.PeakWorkingSetSize);
 | |
| 
 | |
| #elif (defined(_AIX) || defined(__TOS__AIX__)) || (defined(__sun__) || defined(__sun) || defined(sun) && (defined(__SVR4) || defined(__svr4__)))
 | |
| 	/* AIX and Solaris ------------------------------------------ */
 | |
| 	struct psinfo psinfo;
 | |
| 	int fd = -1;
 | |
| 	if ((fd = open("/proc/self/psinfo", O_RDONLY)) == -1)
 | |
| 		return static_cast<size_t>(0L);      /* Can't open? */
 | |
| 	if (read(fd, &psinfo, sizeof(psinfo)) != sizeof(psinfo)) {
 | |
| 		close(fd);
 | |
| 		return static_cast<size_t>(0L);      /* Can't read? */
 | |
| 	}
 | |
| 	close(fd);
 | |
| 	return static_cast<size_t>(psinfo.pr_rssize * 1024L);
 | |
| 
 | |
| #elif defined(__unix__) || defined(__unix) || defined(unix) || (defined(__APPLE__) && defined(__MACH__))
 | |
| 	/* BSD, Linux, and OSX -------------------------------------- */
 | |
| 	struct rusage rusage;
 | |
| 	getrusage(RUSAGE_SELF, &rusage);
 | |
| #if defined(__APPLE__) && defined(__MACH__)
 | |
| 	return static_cast<size_t>(rusage.ru_maxrss);
 | |
| #else
 | |
| 	return static_cast<size_t>(rusage.ru_maxrss * 1024L);
 | |
| #endif
 | |
| 
 | |
| #else
 | |
| 	/* Unknown OS ----------------------------------------------- */
 | |
| 	return static_cast<size_t>(0L);          /* Unsupported. */
 | |
| #endif
 | |
| 
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Returns the current resident set size (physical memory use) measured
 | |
|  * in bytes, or zero if the value cannot be determined on this OS.
 | |
|  */
 | |
| size_t Metrics::GetCurrentRSS() {
 | |
| #if defined(_WIN32)
 | |
| 	/* Windows -------------------------------------------------- */
 | |
| 	PROCESS_MEMORY_COUNTERS info;
 | |
| 	GetProcessMemoryInfo(GetCurrentProcess(), &info, sizeof(info));
 | |
| 	return static_cast<size_t>(info.WorkingSetSize);
 | |
| 
 | |
| #elif defined(__APPLE__) && defined(__MACH__)
 | |
| 	/* OSX ------------------------------------------------------ */
 | |
| 	struct mach_task_basic_info info;
 | |
| 	mach_msg_type_number_t infoCount = MACH_TASK_BASIC_INFO_COUNT;
 | |
| 	if (task_info(mach_task_self(), MACH_TASK_BASIC_INFO,
 | |
| 		reinterpret_cast<task_info_t>(&info), &infoCount) != KERN_SUCCESS)
 | |
| 		return static_cast<size_t>(0L);      /* Can't access? */
 | |
| 	return static_cast<size_t>(info.resident_size);
 | |
| 
 | |
| #elif defined(__linux__) || defined(__linux) || defined(linux) || defined(__gnu_linux__)
 | |
| 	/* Linux ---------------------------------------------------- */
 | |
| 	long rss = 0L;
 | |
| 	FILE* fp = NULL;
 | |
| 	if ((fp = fopen("/proc/self/statm", "r")) == NULL)
 | |
| 		return static_cast<size_t>(0L);      /* Can't open? */
 | |
| 	if (fscanf(fp, "%*s%ld", &rss) != 1) {
 | |
| 		fclose(fp);
 | |
| 		return static_cast<size_t>(0L);      /* Can't read? */
 | |
| 	}
 | |
| 	fclose(fp);
 | |
| 	return static_cast<size_t>(rss) * static_cast<size_t>(sysconf(_SC_PAGESIZE));
 | |
| 
 | |
| #else
 | |
| 	/* AIX, BSD, Solaris, and Unknown OS ------------------------ */
 | |
| 	return static_cast<size_t>(0L);          /* Unsupported. */
 | |
| #endif
 | |
| 
 | |
| }
 | |
| 
 | |
| size_t Metrics::GetProcessID() {
 | |
| #if defined(_WIN32)
 | |
| 	return GetCurrentProcessId();
 | |
| #else
 | |
| 	return getpid();
 | |
| #endif
 | |
| }
 |