2 * Copyright (C) 2005-2013 Team XBMC
5 * This Program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2, or (at your option)
10 * This Program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with XBMC; see the file COPYING. If not, see
17 * <http://www.gnu.org/licenses/>.
23 #include "threads/SystemClock.h"
25 #include "SystemInfo.h"
29 #include <sys/utsname.h>
31 #include "GUIInfoManager.h"
32 #include "filesystem/CurlFile.h"
33 #include "network/Network.h"
34 #include "Application.h"
35 #include "windowing/WindowingFactory.h"
36 #include "guilib/LocalizeStrings.h"
38 #include "utils/TimeUtils.h"
39 #include "utils/log.h"
43 #if defined(TARGET_DARWIN)
44 #include "osx/DarwinUtils.h"
45 #include "osx/CocoaInterface.h"
47 #include "powermanagement/PowerManager.h"
48 #include "utils/StringUtils.h"
49 #include "utils/XMLUtils.h"
50 #if defined(TARGET_ANDROID)
51 #include "android/jni/Build.h"
54 /* Target identification */
55 #if defined(TARGET_DARWIN)
56 #include <Availability.h>
57 #elif defined(TARGET_ANDROID)
58 #include <android/api-level.h>
59 #elif defined(TARGET_FREEBSD)
60 #include <sys/param.h>
61 #elif defined(TARGET_LINUX)
62 #include <linux/version.h>
67 CSysInfoJob::CSysInfoJob()
71 bool CSysInfoJob::DoWork()
73 m_info.systemUptime = GetSystemUpTime(false);
74 m_info.systemTotalUptime = GetSystemUpTime(true);
75 m_info.internetState = GetInternetState();
76 m_info.videoEncoder = GetVideoEncoder();
77 m_info.cpuFrequency = GetCPUFreqInfo();
78 m_info.kernelVersion = CSysInfo::GetKernelVersion();
79 m_info.macAddress = GetMACAddress();
80 m_info.batteryLevel = GetBatteryLevel();
84 const CSysData &CSysInfoJob::GetData() const
89 CStdString CSysInfoJob::GetCPUFreqInfo()
91 double CPUFreq = GetCPUFrequency();
92 return StringUtils::Format("%4.2fMHz", CPUFreq);;
95 CSysData::INTERNET_STATE CSysInfoJob::GetInternetState()
97 // Internet connection state!
98 XFILE::CCurlFile http;
99 if (http.IsInternet())
100 return CSysData::CONNECTED;
101 if (http.IsInternet(false))
102 return CSysData::NO_DNS;
103 return CSysData::DISCONNECTED;
106 CStdString CSysInfoJob::GetMACAddress()
108 #if defined(HAS_LINUX_NETWORK) || defined(HAS_WIN32_NETWORK)
109 CNetworkInterface* iface = g_application.getNetwork().GetFirstConnectedInterface();
111 return iface->GetMacAddress();
116 CStdString CSysInfoJob::GetVideoEncoder()
118 return "GPU: " + g_Windowing.GetRenderRenderer();
121 CStdString CSysInfoJob::GetBatteryLevel()
123 return StringUtils::Format("%d%%", g_powerManager.BatteryLevel());
126 double CSysInfoJob::GetCPUFrequency()
128 #if defined (TARGET_POSIX) || defined(TARGET_WINDOWS)
129 return double (g_cpuInfo.getCPUFrequency());
135 bool CSysInfoJob::SystemUpTime(int iInputMinutes, int &iMinutes, int &iHours, int &iDays)
137 iMinutes=0;iHours=0;iDays=0;
138 iMinutes = iInputMinutes;
139 if (iMinutes >= 60) // Hour's
141 iHours = iMinutes / 60;
142 iMinutes = iMinutes - (iHours *60);
144 if (iHours >= 24) // Days
147 iHours = iHours - (iDays * 24);
152 CStdString CSysInfoJob::GetSystemUpTime(bool bTotalUptime)
154 CStdString strSystemUptime;
155 int iInputMinutes, iMinutes,iHours,iDays;
160 iInputMinutes = g_sysinfo.GetTotalUptime() + ((int)(XbmcThreads::SystemClockMillis() / 60000));
165 iInputMinutes = (int)(XbmcThreads::SystemClockMillis() / 60000);
168 SystemUpTime(iInputMinutes,iMinutes, iHours, iDays);
171 strSystemUptime = StringUtils::Format("%i %s, %i %s, %i %s",
172 iDays, g_localizeStrings.Get(12393).c_str(),
173 iHours, g_localizeStrings.Get(12392).c_str(),
174 iMinutes, g_localizeStrings.Get(12391).c_str());
176 else if (iDays == 0 && iHours >= 1 )
178 strSystemUptime = StringUtils::Format("%i %s, %i %s",
179 iHours, g_localizeStrings.Get(12392).c_str(),
180 iMinutes, g_localizeStrings.Get(12391).c_str());
182 else if (iDays == 0 && iHours == 0 && iMinutes >= 0)
184 strSystemUptime = StringUtils::Format("%i %s",
185 iMinutes, g_localizeStrings.Get(12391).c_str());
187 return strSystemUptime;
190 CStdString CSysInfo::TranslateInfo(int info) const
194 case SYSTEM_VIDEO_ENCODER_INFO:
195 return m_info.videoEncoder;
196 case NETWORK_MAC_ADDRESS:
197 return m_info.macAddress;
198 case SYSTEM_KERNEL_VERSION:
199 return m_info.kernelVersion;
200 case SYSTEM_CPUFREQUENCY:
201 return m_info.cpuFrequency;
203 return m_info.systemUptime;
204 case SYSTEM_TOTALUPTIME:
205 return m_info.systemTotalUptime;
206 case SYSTEM_INTERNET_STATE:
207 if (m_info.internetState == CSysData::CONNECTED)
208 return g_localizeStrings.Get(13296);
209 else if (m_info.internetState == CSysData::NO_DNS)
210 return g_localizeStrings.Get(13274);
212 return g_localizeStrings.Get(13297);
213 case SYSTEM_BATTERY_LEVEL:
214 return m_info.batteryLevel;
220 void CSysInfo::Reset()
225 CSysInfo::CSysInfo(void) : CInfoLoader(15 * 1000)
227 memset(MD5_Sign, 0, sizeof(MD5_Sign));
228 m_iSystemTimeTotalUp = 0;
231 CSysInfo::~CSysInfo()
235 bool CSysInfo::Load(const TiXmlNode *settings)
237 if (settings == NULL)
240 const TiXmlElement *pElement = settings->FirstChildElement("general");
242 XMLUtils::GetInt(pElement, "systemtotaluptime", m_iSystemTimeTotalUp, 0, INT_MAX);
247 bool CSysInfo::Save(TiXmlNode *settings) const
249 if (settings == NULL)
252 TiXmlNode *generalNode = settings->FirstChild("general");
253 if (generalNode == NULL)
255 TiXmlElement generalNodeNew("general");
256 generalNode = settings->InsertEndChild(generalNodeNew);
257 if (generalNode == NULL)
260 XMLUtils::SetInt(generalNode, "systemtotaluptime", m_iSystemTimeTotalUp);
265 bool CSysInfo::GetDiskSpace(const CStdString& drive,int& iTotal, int& iTotalFree, int& iTotalUsed, int& iPercentFree, int& iPercentUsed)
268 ULARGE_INTEGER ULTotal= { { 0 } };
269 ULARGE_INTEGER ULTotalFree= { { 0 } };
271 if( !drive.empty() && !drive.Equals("*") )
273 #ifdef TARGET_WINDOWS
274 UINT uidriveType = GetDriveType(( drive + ":\\" ));
275 if(uidriveType != DRIVE_UNKNOWN && uidriveType != DRIVE_NO_ROOT_DIR)
277 bRet= ( 0 != GetDiskFreeSpaceEx( ( drive + ":\\" ), NULL, &ULTotal, &ULTotalFree) );
281 ULARGE_INTEGER ULTotalTmp= { { 0 } };
282 ULARGE_INTEGER ULTotalFreeTmp= { { 0 } };
283 #ifdef TARGET_WINDOWS
284 char* pcBuffer= NULL;
285 DWORD dwStrLength= GetLogicalDriveStrings( 0, pcBuffer );
286 if( dwStrLength != 0 )
289 pcBuffer= new char [dwStrLength];
290 GetLogicalDriveStrings( dwStrLength, pcBuffer );
293 if( DRIVE_FIXED == GetDriveType( pcBuffer + iPos ) &&
294 GetDiskFreeSpaceEx( ( pcBuffer + iPos ), NULL, &ULTotal, &ULTotalFree ) )
296 ULTotalTmp.QuadPart+= ULTotal.QuadPart;
297 ULTotalFreeTmp.QuadPart+= ULTotalFree.QuadPart;
299 iPos += (strlen( pcBuffer + iPos) + 1 );
300 }while( strlen( pcBuffer + iPos ) > 0 );
303 #else // for linux and osx
304 static const char *drv_letter[] = { "C:\\", "E:\\", "F:\\", "G:\\", "X:\\", "Y:\\", "Z:\\", NULL };
305 for( int i = 0; drv_letter[i]; i++)
307 if( GetDiskFreeSpaceEx( drv_letter[i], NULL, &ULTotal, &ULTotalFree ) )
309 ULTotalTmp.QuadPart+= ULTotal.QuadPart;
310 ULTotalFreeTmp.QuadPart+= ULTotalFree.QuadPart;
314 if( ULTotalTmp.QuadPart || ULTotalFreeTmp.QuadPart )
316 ULTotal.QuadPart= ULTotalTmp.QuadPart;
317 ULTotalFree.QuadPart= ULTotalFreeTmp.QuadPart;
324 iTotal = (int)( ULTotal.QuadPart / MB );
325 iTotalFree = (int)( ULTotalFree.QuadPart / MB );
326 iTotalUsed = iTotal - iTotalFree;
327 if( ULTotal.QuadPart > 0 )
329 iPercentUsed = (int)( 100.0f * ( ULTotal.QuadPart - ULTotalFree.QuadPart ) / ULTotal.QuadPart + 0.5f );
335 iPercentFree = 100 - iPercentUsed;
341 CStdString CSysInfo::GetCPUModel()
343 return "CPU: " + g_cpuInfo.getCPUModel();
346 CStdString CSysInfo::GetCPUBogoMips()
348 return "BogoMips: " + g_cpuInfo.getCPUBogoMips();
351 CStdString CSysInfo::GetCPUHardware()
353 return "Hardware: " + g_cpuInfo.getCPUHardware();
356 CStdString CSysInfo::GetCPURevision()
358 return "Revision: " + g_cpuInfo.getCPURevision();
361 CStdString CSysInfo::GetCPUSerial()
363 return "Serial: " + g_cpuInfo.getCPUSerial();
366 CStdString CSysInfo::GetManufacturer()
368 CStdString manufacturer = "";
369 #if defined(TARGET_ANDROID)
370 manufacturer = CJNIBuild::MANUFACTURER;
375 CStdString CSysInfo::GetModel()
377 CStdString model = "";
378 #if defined(TARGET_ANDROID)
379 model = CJNIBuild::MODEL;
384 CStdString CSysInfo::GetProduct()
386 CStdString product = "";
387 #if defined(TARGET_ANDROID)
388 product = CJNIBuild::PRODUCT;
393 bool CSysInfo::IsAeroDisabled()
395 #ifdef TARGET_WINDOWS
396 BOOL aeroEnabled = FALSE;
397 HRESULT res = DwmIsCompositionEnabled(&aeroEnabled);
404 CSysInfo::WindowsVersion CSysInfo::m_WinVer = WindowsVersionUnknown;
406 bool CSysInfo::IsWindowsVersion(WindowsVersion ver)
408 if (ver == WindowsVersionUnknown)
410 return GetWindowsVersion() == ver;
413 bool CSysInfo::IsWindowsVersionAtLeast(WindowsVersion ver)
415 if (ver == WindowsVersionUnknown)
417 return GetWindowsVersion() >= ver;
420 CSysInfo::WindowsVersion CSysInfo::GetWindowsVersion()
422 #ifdef TARGET_WINDOWS
423 if (m_WinVer == WindowsVersionUnknown)
425 OSVERSIONINFOEX osvi;
426 ZeroMemory(&osvi, sizeof(OSVERSIONINFOEX));
427 osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
428 if (GetVersionEx((OSVERSIONINFO *)&osvi))
430 if (osvi.dwMajorVersion == 6 && osvi.dwMinorVersion == 0)
431 m_WinVer = WindowsVersionVista;
432 else if (osvi.dwMajorVersion == 6 && osvi.dwMinorVersion == 1)
433 m_WinVer = WindowsVersionWin7;
434 else if (osvi.dwMajorVersion == 6 && osvi.dwMinorVersion == 2)
435 m_WinVer = WindowsVersionWin8;
436 else if (osvi.dwMajorVersion == 6 && osvi.dwMinorVersion == 3)
437 m_WinVer = WindowsVersionWin8_1;
438 /* Insert checks for new Windows versions here */
439 else if ( (osvi.dwMajorVersion == 6 && osvi.dwMinorVersion > 3) || osvi.dwMajorVersion > 6)
440 m_WinVer = WindowsVersionFuture;
443 #endif // TARGET_WINDOWS
447 int CSysInfo::GetKernelBitness(void)
449 #ifdef TARGET_WINDOWS
452 if (si.wProcessorArchitecture == PROCESSOR_ARCHITECTURE_AMD64)
455 BOOL (WINAPI *ptrIsWow64) (HANDLE, PBOOL);
456 HMODULE hKernel32 = GetModuleHandleA("kernel32");
457 if (hKernel32 == NULL)
458 return 0; // Can't detect OS
459 ptrIsWow64 = (BOOL (WINAPI *) (HANDLE, PBOOL)) GetProcAddress(hKernel32, "IsWow64Process");
460 BOOL wow64proc = FALSE;
461 if (ptrIsWow64 == NULL || ptrIsWow64(GetCurrentProcess(), &wow64proc) == FALSE)
462 return 0; // Can't detect OS
463 return (wow64proc == FALSE) ? 32 : 64;
464 #elif defined(TARGET_POSIX)
468 std::string machine(un.machine);
469 if (machine == "x86_64" || machine == "amd64" || machine == "arm64" || machine == "aarch64" || machine == "ppc64" || machine == "ia64")
473 return 0; // can't detect
479 int CSysInfo::GetXbmcBitness(void)
481 #if defined (__aarch64__) || defined(__arm64__) || defined(__amd64__) || defined(__amd64) || defined(__x86_64__) || defined(__x86_64) || defined(_M_X64) || defined(_M_AMD64) || defined(__ppc64__)
483 #elif defined(__thumb__) || defined(_M_ARMT) || defined(__arm__) || defined(_M_ARM) || defined(__mips__) || defined(mips) || defined(__mips) || defined(i386) || \
484 defined(__i386) || defined(__i386__) || defined(__i486__) || defined(__i586__) || defined(__i686__) || defined(_M_IX86) || defined(_X86_) || defined(__powerpc) || \
485 defined(__powerpc__) || defined(__ppc__) || defined(_M_PPC)
492 CStdString CSysInfo::GetKernelVersion()
494 #if defined(TARGET_DARWIN)
495 return g_sysinfo.GetUnameVersion();
496 #elif defined (TARGET_POSIX)
500 return StringUtils::Format("%s %s %s %s", un.sysname, un.release, un.version, un.machine);;
505 OSVERSIONINFOEX osvi;
506 ZeroMemory(&osvi, sizeof(OSVERSIONINFOEX));
507 osvi.dwOSVersionInfoSize = sizeof(osvi);
509 std::string strKernel = "Windows";
510 if (GetVersionEx((OSVERSIONINFO *)&osvi))
512 switch (GetWindowsVersion())
514 case WindowsVersionVista:
515 if (osvi.wProductType == VER_NT_WORKSTATION)
516 strKernel.append(" Vista");
518 strKernel.append(" Server 2008");
520 case WindowsVersionWin7:
521 if (osvi.wProductType == VER_NT_WORKSTATION)
522 strKernel.append(" 7");
524 strKernel.append(" Server 2008 R2");
526 case WindowsVersionWin8:
527 if (osvi.wProductType == VER_NT_WORKSTATION)
528 strKernel.append(" 8");
530 strKernel.append(" Server 2012");
532 case WindowsVersionWin8_1:
533 if (osvi.wProductType == VER_NT_WORKSTATION)
534 strKernel.append(" 8.1");
536 strKernel.append(" Server 2012 R2");
538 case WindowsVersionFuture:
539 strKernel.append(" Unknown Future Version");
542 strKernel.append(" Unknown version");
546 // Append Service Pack version if any
547 if (osvi.wServicePackMajor > 0)
549 strKernel.append(StringUtils::Format(" SP%d", osvi.wServicePackMajor));
550 if (osvi.wServicePackMinor > 0)
552 strKernel.append(StringUtils::Format(".%d", osvi.wServicePackMinor));
556 strKernel.append(StringUtils::Format(" %d-bit", GetKernelBitness()));
558 strKernel.append(StringUtils::Format(", build %d", osvi.dwBuildNumber));
562 strKernel.append(" unknown");
563 strKernel.append(StringUtils::Format(" %d-bit", GetKernelBitness()));
570 bool CSysInfo::HasInternet()
572 if (m_info.internetState != CSysData::UNKNOWN)
573 return m_info.internetState == CSysData::CONNECTED;
574 return (m_info.internetState = CSysInfoJob::GetInternetState()) == CSysData::CONNECTED;
577 CStdString CSysInfo::GetHddSpaceInfo(int drive, bool shortText)
580 return GetHddSpaceInfo( percent, drive, shortText);
583 CStdString CSysInfo::GetHddSpaceInfo(int& percent, int drive, bool shortText)
585 int total, totalFree, totalUsed, percentFree, percentused;
588 if (g_sysinfo.GetDiskSpace("", total, totalFree, totalUsed, percentFree, percentused))
594 case SYSTEM_FREE_SPACE:
595 percent = percentFree;
597 case SYSTEM_USED_SPACE:
598 percent = percentused;
606 case SYSTEM_FREE_SPACE:
607 strRet = StringUtils::Format("%i MB %s", totalFree, g_localizeStrings.Get(160).c_str());
609 case SYSTEM_USED_SPACE:
610 strRet = StringUtils::Format("%i MB %s", totalUsed, g_localizeStrings.Get(20162).c_str());
612 case SYSTEM_TOTAL_SPACE:
613 strRet = StringUtils::Format("%i MB %s", total, g_localizeStrings.Get(20161).c_str());
615 case SYSTEM_FREE_SPACE_PERCENT:
616 strRet = StringUtils::Format("%i %% %s", percentFree, g_localizeStrings.Get(160).c_str());
618 case SYSTEM_USED_SPACE_PERCENT:
619 strRet = StringUtils::Format("%i %% %s", percentused, g_localizeStrings.Get(20162).c_str());
629 strRet = g_localizeStrings.Get(161);
634 #if defined(TARGET_LINUX)
635 CStdString CSysInfo::GetLinuxDistro()
637 #if defined(TARGET_ANDROID)
640 static const char* release_file[] = { "/etc/debian_version",
642 "/etc/mandrake-release",
643 "/etc/fedora-release",
644 "/etc/redhat-release",
645 "/etc/gentoo-release",
646 "/etc/slackware-version",
648 "/etc/buildroot-release",
650 CStdString result("");
651 char buffer[256] = {'\0'};
653 /* Try reading PRETTY_NAME from /etc/os-release first.
654 * If this fails, fall back to lsb_release or distro-specific release-file. */
656 FILE *os_release = fopen("/etc/os-release", "r");
663 while (fgets(buffer, sizeof(buffer), os_release))
668 if (strcmp(key, "PRETTY_NAME") == 0)
670 char *pretty_name = val;
672 // remove newline and enclosing quotes
673 if (pretty_name[strlen(pretty_name) - 1] == '\n')
674 pretty_name[strlen(pretty_name) - 1] = '\0';
676 if (pretty_name[0] == '\'' || pretty_name[0] == '\"')
679 pretty_name[strlen(pretty_name) - 1] = '\0';
682 // unescape quotes and backslashes
683 char *p = pretty_name;
687 char *next_char = p + 1;
689 if (*this_char == '\\' &&
690 (*next_char == '\'' || *next_char == '\"' || *next_char == '\\'))
694 *this_char = *next_char;
703 result = pretty_name;
714 FILE* pipe = popen("unset PYTHONHOME; unset PYTHONPATH; lsb_release -d 2>/dev/null | cut -f2", "r");
717 if (fread(buffer, sizeof(char), sizeof(buffer), pipe) > 0 && !ferror(pipe))
721 return StringUtils::Trim(result);
725 for (int i = 0; result.empty() && release_file[i]; i++)
727 file = fopen(release_file[i], "r");
730 if (fgets(buffer, sizeof(buffer), file))
734 return StringUtils::Trim(result);
740 CLog::Log(LOGWARNING, "Unable to determine Linux distribution");
746 CStdString CSysInfo::GetUnameVersion()
748 CStdString result = "";
750 #if defined(TARGET_ANDROID)
752 if (uname(&name) == -1)
754 result += name.release;
756 result += name.machine;
757 #elif defined(TARGET_DARWIN_IOS)
758 result = GetDarwinOSReleaseString();
760 result += GetDarwinVersionString();
762 FILE* pipe = popen("uname -rm", "r");
766 if (fgets(buffer, sizeof(buffer), pipe))
769 #if defined(TARGET_DARWIN)
770 StringUtils::Trim(result);
772 result += GetDarwinVersionString();
776 CLog::Log(LOGWARNING, "Unable to determine Uname version");
779 #endif//else !TARGET_ANDROID
781 return StringUtils::Trim(result);
785 #if defined(TARGET_WINDOWS)
786 CStdString CSysInfo::GetUAWindowsVersion()
788 OSVERSIONINFOEX osvi = {};
790 osvi.dwOSVersionInfoSize = sizeof(osvi);
791 CStdString strVersion = "Windows NT";
793 if (GetVersionEx((OSVERSIONINFO *)&osvi))
795 strVersion += StringUtils::Format(" %d.%d", osvi.dwMajorVersion, osvi.dwMinorVersion);
802 if (IsWow64Process(GetCurrentProcess(), &bIsWow))
806 strVersion.append(";WOW64");
807 GetNativeSystemInfo(&si); // different function to read the info under Wow
811 if (si.wProcessorArchitecture==PROCESSOR_ARCHITECTURE_AMD64)
812 strVersion.append(";Win64;x64");
813 else if (si.wProcessorArchitecture==PROCESSOR_ARCHITECTURE_IA64)
814 strVersion.append(";Win64;IA64");
821 CStdString CSysInfo::GetUserAgent()
824 result = "XBMC/" + g_infoManager.GetLabel(SYSTEM_BUILD_VERSION) + " (";
825 #if defined(TARGET_WINDOWS)
826 result += GetUAWindowsVersion();
827 #elif defined(TARGET_DARWIN)
828 #if defined(TARGET_DARWIN_IOS)
831 result += "Mac OS X; ";
833 result += GetUnameVersion();
834 #elif defined(TARGET_FREEBSD)
835 result += "FreeBSD; ";
836 result += GetUnameVersion();
837 #elif defined(TARGET_POSIX)
839 result += GetLinuxDistro();
841 result += GetUnameVersion();
843 result += "; http://xbmc.org)";
848 bool CSysInfo::IsAppleTV2()
850 #if defined(TARGET_DARWIN)
851 return DarwinIsAppleTV2();
857 bool CSysInfo::HasVideoToolBoxDecoder()
861 #if defined(HAVE_VIDEOTOOLBOXDECODER)
862 result = DarwinHasVideoToolboxDecoder();
867 std::string CSysInfo::GetBuildTargetPlatformName(void)
869 #if defined(TARGET_DARWIN_OSX)
871 #elif defined(TARGET_DARWIN_IOS_ATV2)
872 return "Darwin iOS ATV2";
873 #elif defined(TARGET_DARWIN_IOS)
875 #elif defined(TARGET_FREEBSD)
877 #elif defined(TARGET_ANDROID)
879 #elif defined(TARGET_LINUX)
881 #elif defined(TARGET_WINDOWS)
884 return "unknown platform";
888 std::string CSysInfo::GetBuildTargetPlatformVersion(void)
890 /* Expand macro before stringify */
891 #define STR_MACRO(x) #x
892 #define XSTR_MACRO(x) STR_MACRO(x)
894 #if defined(TARGET_DARWIN_OSX)
895 return "version " XSTR_MACRO(__MAC_OS_X_VERSION_MIN_REQUIRED);
896 #elif defined(TARGET_DARWIN_IOS)
897 return "version " XSTR_MACRO(__IPHONE_OS_VERSION_MIN_REQUIRED);
898 #elif defined(TARGET_FREEBSD)
899 return "version " XSTR_MACRO(__FreeBSD_version);
900 #elif defined(TARGET_ANDROID)
901 return "API level " XSTR_MACRO(__ANDROID_API__);
902 #elif defined(TARGET_LINUX)
903 std::string ver = StringUtils::Format("%i.%i.%i", LINUX_VERSION_CODE >> 16, (LINUX_VERSION_CODE >> 8) & 0xff, LINUX_VERSION_CODE & 0xff);
905 #elif defined(TARGET_WINDOWS)
906 return "version " XSTR_MACRO(NTDDI_VERSION);
908 return "(unknown platform)";
912 std::string CSysInfo::GetBuildTargetCpuFamily(void)
914 #if defined(__thumb__) || defined(_M_ARMT)
915 return "ARM (Thumb)";
916 #elif defined(__arm__) || defined(_M_ARM) || defined (__aarch64__)
918 #elif defined(__mips__) || defined(mips) || defined(__mips)
920 #elif defined(__amd64__) || defined(__amd64) || defined(__x86_64__) || defined(__x86_64) || defined(_M_X64) || defined(_M_AMD64) || \
921 defined(i386) || defined(__i386) || defined(__i386__) || defined(__i486__) || defined(__i586__) || defined(__i686__) || defined(_M_IX86) || defined(_X86_)
923 #elif defined(__powerpc) || defined(__powerpc__) || defined(__powerpc64__) || defined(__ppc__) || defined(__ppc64__) || defined(_M_PPC)
926 return "unknown CPU family";
931 CJob *CSysInfo::GetJob() const
933 return new CSysInfoJob();
936 void CSysInfo::OnJobComplete(unsigned int jobID, bool success, CJob *job)
938 m_info = ((CSysInfoJob *)job)->GetData();
939 CInfoLoader::OnJobComplete(jobID, success, job);