LCOV - code coverage report
Current view: top level - lib/osinfo - uname.c (source / functions) Hit Total Coverage
Test: lcov.info Lines: 7 7 100.0 %
Date: 2021-11-24 03:28:48 Functions: 1 1 100.0 %

          Line data    Source code
       1             : /* Copyright (c) 2003-2004, Roger Dingledine
       2             :  * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
       3             :  * Copyright (c) 2007-2021, The Tor Project, Inc. */
       4             : /* See LICENSE for licensing information */
       5             : 
       6             : /**
       7             :  * \file uname.c
       8             :  * \brief Look up a description of the operating system.
       9             :  **/
      10             : 
      11             : #include "orconfig.h"
      12             : #include "lib/osinfo/uname.h"
      13             : 
      14             : #include "lib/string/compat_string.h"
      15             : #include "lib/string/printf.h"
      16             : 
      17             : #ifdef HAVE_UNAME
      18             : #include <sys/utsname.h>
      19             : #endif
      20             : #ifdef _WIN32
      21             : #include <windows.h>
      22             : #endif
      23             : #include <string.h>
      24             : 
      25             : /** Hold the result of our call to <b>uname</b>. */
      26             : static char uname_result[256];
      27             : /** True iff uname_result is set. */
      28             : static int uname_result_is_set = 0;
      29             : 
      30             : #ifdef _WIN32
      31             : /** Table to map claimed windows versions into human-readable windows
      32             :  * versions. */
      33             : static struct {
      34             :   unsigned major;
      35             :   unsigned minor;
      36             :   const char *client_version;
      37             :   const char *server_version;
      38             : } win_version_table[] = {
      39             :   /* This table must be sorted in descending order.
      40             :    * Sources:
      41             :    *   https://en.wikipedia.org/wiki/List_of_Microsoft_Windows_versions
      42             :    *   https://docs.microsoft.com/en-us/windows/desktop/api/winnt/
      43             :    *     ns-winnt-_osversioninfoexa#remarks
      44             :    */
      45             :    /* Windows Server 2019 is indistinguishable from Windows Server 2016
      46             :     * using GetVersionEx().
      47             :    { 10,  0, NULL,                        "Windows Server 2019" }, */
      48             :    // clang-format off
      49             :    { 10,  0, "Windows 10",                "Windows Server 2016" },
      50             :    {  6,  3, "Windows 8.1",               "Windows Server 2012 R2" },
      51             :    {  6,  2, "Windows 8",                 "Windows Server 2012" },
      52             :    {  6,  1, "Windows 7",                 "Windows Server 2008 R2" },
      53             :    {  6,  0, "Windows Vista",             "Windows Server 2008" },
      54             :    {  5,  2, "Windows XP Professional",   "Windows Server 2003" },
      55             :    /* Windows XP did not have a server version, but we need something here */
      56             :    {  5,  1, "Windows XP",                "Windows XP Server" },
      57             :    {  5,  0, "Windows 2000 Professional", "Windows 2000 Server" },
      58             :    /* Earlier versions are not supported by GetVersionEx(). */
      59             :    {  0,  0, NULL,                        NULL }
      60             :    // clang-format on
      61             : };
      62             : #endif /* defined(_WIN32) */
      63             : 
      64             : /** Return a pointer to a description of our platform.
      65             :  */
      66         443 : MOCK_IMPL(const char *,
      67             : get_uname,(void))
      68             : {
      69             : #ifdef HAVE_UNAME
      70         443 :   struct utsname u;
      71             : #endif
      72         443 :   if (!uname_result_is_set) {
      73             : #ifdef HAVE_UNAME
      74         253 :     if (uname(&u) != -1) {
      75             :       /* (Linux says 0 is success, Solaris says 1 is success) */
      76         253 :       strlcpy(uname_result, u.sysname, sizeof(uname_result));
      77             :     } else
      78             : #endif /* defined(HAVE_UNAME) */
      79             :       {
      80             : #ifdef _WIN32
      81             :         OSVERSIONINFOEX info;
      82             :         int i;
      83             :         int is_client = 0;
      84             :         int is_server = 0;
      85             :         const char *plat = NULL;
      86             :         memset(&info, 0, sizeof(info));
      87             :         info.dwOSVersionInfoSize = sizeof(info);
      88             :         if (! GetVersionEx((LPOSVERSIONINFO)&info)) {
      89             :           strlcpy(uname_result, "Bizarre version of Windows where GetVersionEx"
      90             :                   " doesn't work.", sizeof(uname_result));
      91             :           uname_result_is_set = 1;
      92             :           return uname_result;
      93             :         }
      94             : #ifdef VER_NT_SERVER
      95             :         if (info.wProductType == VER_NT_SERVER ||
      96             :             info.wProductType == VER_NT_DOMAIN_CONTROLLER) {
      97             :           is_server = 1;
      98             :         } else {
      99             :           is_client = 1;
     100             :         }
     101             : #endif /* defined(VER_NT_SERVER) */
     102             :         /* Search the version table for a matching version */
     103             :         for (i=0; win_version_table[i].major>0; ++i) {
     104             :           if (win_version_table[i].major == info.dwMajorVersion &&
     105             :               win_version_table[i].minor == info.dwMinorVersion) {
     106             :             if (is_server) {
     107             :               plat = win_version_table[i].server_version;
     108             :             } else {
     109             :               /* Use client versions for clients, and when we don't know if it
     110             :               * is a client or a server. */
     111             :               plat = win_version_table[i].client_version;
     112             :             }
     113             :             break;
     114             :           }
     115             :         }
     116             :         if (plat) {
     117             :           strlcpy(uname_result, plat, sizeof(uname_result));
     118             :         } else {
     119             :           if (info.dwMajorVersion > win_version_table[0].major ||
     120             :               (info.dwMajorVersion == win_version_table[0].major &&
     121             :                info.dwMinorVersion > win_version_table[0].minor))
     122             :             tor_snprintf(uname_result, sizeof(uname_result),
     123             :                          "Very recent version of Windows [major=%d,minor=%d]",
     124             :                          (int)info.dwMajorVersion,(int)info.dwMinorVersion);
     125             :           else
     126             :             tor_snprintf(uname_result, sizeof(uname_result),
     127             :                          "Unrecognized version of Windows [major=%d,minor=%d]",
     128             :                          (int)info.dwMajorVersion,(int)info.dwMinorVersion);
     129             :         }
     130             :         /* Now append extra information to the name.
     131             :          *
     132             :          * Microsoft's API documentation says that on Windows 8.1 and later,
     133             :          * GetVersionEx returns Windows 8 (6.2) for applications without an
     134             :          * app compatibility manifest (including tor's default build).
     135             :          *
     136             :          * But in our testing, we have seen the actual Windows version on
     137             :          * Windows Server 2012 R2, even without a manifest. */
     138             :         if (info.dwMajorVersion > 6 ||
     139             :             (info.dwMajorVersion == 6 && info.dwMinorVersion >= 2)) {
     140             :           /* When GetVersionEx() returns Windows 8, the actual OS may be any
     141             :            * later version. */
     142             :           strlcat(uname_result, " [or later]", sizeof(uname_result));
     143             :         }
     144             :         /* When we don't know if the OS is a client or server version, we use
     145             :          * the client version, and this qualifier. */
     146             :         if (!is_server && !is_client) {
     147             :           strlcat(uname_result, " [client or server]", sizeof(uname_result));
     148             :         }
     149             : #else /* !defined(_WIN32) */
     150             :         /* LCOV_EXCL_START -- can't provoke uname failure */
     151             :         strlcpy(uname_result, "Unknown platform", sizeof(uname_result));
     152             :         /* LCOV_EXCL_STOP */
     153             : #endif /* defined(_WIN32) */
     154             :       }
     155         253 :     uname_result_is_set = 1;
     156             :   }
     157         443 :   return uname_result;
     158             : }

Generated by: LCOV version 1.14