[tor-commits] [tor/master] Merge branch 'bug28096-029-squashed' into bug28096-035-squashed
nickm at torproject.org
nickm at torproject.org
Mon Nov 26 22:25:46 UTC 2018
commit 44ced9b750a65d867dc5dd2cbbcf7e42a46ae90d
Merge: d598d834f 2fbc58cf0
Author: teor <teor at torproject.org>
Date: Thu Nov 15 12:23:29 2018 +1000
Merge branch 'bug28096-029-squashed' into bug28096-035-squashed
Move the get_uname() changes from src/common/compat.c to
src/lib/osinfo/uname.c
changes/bug28096 | 13 +++++++
src/lib/osinfo/uname.c | 95 ++++++++++++++++++++++++++++++++++----------------
2 files changed, 77 insertions(+), 31 deletions(-)
diff --cc src/lib/osinfo/uname.c
index 9d1923695,000000000..7111ae31d
mode 100644,000000..100644
--- a/src/lib/osinfo/uname.c
+++ b/src/lib/osinfo/uname.c
@@@ -1,116 -1,0 +1,149 @@@
+/* Copyright (c) 2003-2004, Roger Dingledine
+ * Copyright (c) 2004-2006, Roger Dingledine, Nick Mathewson.
+ * Copyright (c) 2007-2018, The Tor Project, Inc. */
+/* See LICENSE for licensing information */
+
+/**
+ * \file uname.c
+ * \brief Look up a description of the operating system.
+ **/
+
+#include "orconfig.h"
+#include "lib/osinfo/uname.h"
+
+#include "lib/string/compat_string.h"
+#include "lib/string/printf.h"
+
+#ifdef HAVE_UNAME
+#include <sys/utsname.h>
+#endif
+#ifdef _WIN32
+#include <windows.h>
+#endif
+#include <string.h>
+
+/** Hold the result of our call to <b>uname</b>. */
+static char uname_result[256];
+/** True iff uname_result is set. */
+static int uname_result_is_set = 0;
+
+/** Return a pointer to a description of our platform.
+ */
+MOCK_IMPL(const char *,
+get_uname,(void))
+{
+#ifdef HAVE_UNAME
+ struct utsname u;
+#endif
+ if (!uname_result_is_set) {
+#ifdef HAVE_UNAME
+ if (uname(&u) != -1) {
+ /* (Linux says 0 is success, Solaris says 1 is success) */
+ strlcpy(uname_result, u.sysname, sizeof(uname_result));
+ } else
+#endif /* defined(HAVE_UNAME) */
+ {
+#ifdef _WIN32
+ OSVERSIONINFOEX info;
+ int i;
++ int is_client = 0;
++ int is_server = 0;
+ const char *plat = NULL;
+ static struct {
- unsigned major; unsigned minor; const char *version;
++ unsigned major; unsigned minor;
++ const char *client_version; const char *server_version;
+ } win_version_table[] = {
- { 6, 2, "Windows 8" },
- { 6, 1, "Windows 7" },
- { 6, 0, "Windows Vista" },
- { 5, 2, "Windows Server 2003" },
- { 5, 1, "Windows XP" },
- { 5, 0, "Windows 2000" },
- /* { 4, 0, "Windows NT 4.0" }, */
- { 4, 90, "Windows Me" },
- { 4, 10, "Windows 98" },
- /* { 4, 0, "Windows 95" } */
- { 3, 51, "Windows NT 3.51" },
- { 0, 0, NULL }
++ /* This table must be sorted in descending order.
++ * Sources:
++ * https://en.wikipedia.org/wiki/List_of_Microsoft_Windows_versions
++ * https://docs.microsoft.com/en-us/windows/desktop/api/winnt/
++ * ns-winnt-_osversioninfoexa#remarks
++ */
++ /* Windows Server 2019 is indistinguishable from Windows Server 2016
++ * using GetVersionEx().
++ { 10, 0, NULL, "Windows Server 2019" }, */
++ { 10, 0, "Windows 10", "Windows Server 2016" },
++ { 6, 3, "Windows 8.1", "Windows Server 2012 R2" },
++ { 6, 2, "Windows 8", "Windows Server 2012" },
++ { 6, 1, "Windows 7", "Windows Server 2008 R2" },
++ { 6, 0, "Windows Vista", "Windows Server 2008" },
++ { 5, 2, "Windows XP Professional", "Windows Server 2003" },
++ /* Windows XP did not have a server version, but we need something here */
++ { 5, 1, "Windows XP", "Windows XP Server" },
++ { 5, 0, "Windows 2000 Professional", "Windows 2000 Server" },
++ /* Earlier versions are not supported by GetVersionEx(). */
++ { 0, 0, NULL, NULL }
+ };
+ memset(&info, 0, sizeof(info));
+ info.dwOSVersionInfoSize = sizeof(info);
+ if (! GetVersionEx((LPOSVERSIONINFO)&info)) {
+ strlcpy(uname_result, "Bizarre version of Windows where GetVersionEx"
+ " doesn't work.", sizeof(uname_result));
+ uname_result_is_set = 1;
+ return uname_result;
+ }
- if (info.dwMajorVersion == 4 && info.dwMinorVersion == 0) {
- if (info.dwPlatformId == VER_PLATFORM_WIN32_NT)
- plat = "Windows NT 4.0";
- else
- plat = "Windows 95";
++#ifdef VER_NT_SERVER
++ if (info.wProductType == VER_NT_SERVER ||
++ info.wProductType == VER_NT_DOMAIN_CONTROLLER) {
++ is_server = 1;
+ } else {
- for (i=0; win_version_table[i].major>0; ++i) {
- if (win_version_table[i].major == info.dwMajorVersion &&
- win_version_table[i].minor == info.dwMinorVersion) {
- plat = win_version_table[i].version;
- break;
++ is_client = 1;
++ }
++#endif /* defined(VER_NT_SERVER) */
++ /* Search the version table for a matching version */
++ for (i=0; win_version_table[i].major>0; ++i) {
++ if (win_version_table[i].major == info.dwMajorVersion &&
++ win_version_table[i].minor == info.dwMinorVersion) {
++ if (is_server) {
++ plat = win_version_table[i].server_version;
++ } else {
++ /* Use client versions for clients, and when we don't know if it
++ * is a client or a server. */
++ plat = win_version_table[i].client_version;
+ }
++ break;
+ }
+ }
+ if (plat) {
+ strlcpy(uname_result, plat, sizeof(uname_result));
+ } else {
- if (info.dwMajorVersion > 6 ||
- (info.dwMajorVersion==6 && info.dwMinorVersion>2))
++ if (info.dwMajorVersion > win_version_table[0].major ||
++ (info.dwMajorVersion == win_version_table[0].major &&
++ info.dwMinorVersion > win_version_table[0].minor))
+ tor_snprintf(uname_result, sizeof(uname_result),
+ "Very recent version of Windows [major=%d,minor=%d]",
+ (int)info.dwMajorVersion,(int)info.dwMinorVersion);
+ else
+ tor_snprintf(uname_result, sizeof(uname_result),
+ "Unrecognized version of Windows [major=%d,minor=%d]",
+ (int)info.dwMajorVersion,(int)info.dwMinorVersion);
+ }
- #ifdef VER_NT_SERVER
- if (info.wProductType == VER_NT_SERVER ||
- info.wProductType == VER_NT_DOMAIN_CONTROLLER) {
- strlcat(uname_result, " [server]", sizeof(uname_result));
- }
- #endif /* defined(VER_NT_SERVER) */
++ /* Now append extra information to the name.
++ *
++ * Microsoft's API documentation says that on Windows 8.1 and later,
++ * GetVersionEx returns Windows 8 (6.2) for applications without an
++ * app compatibility manifest (including tor's default build).
++ *
++ * But in our testing, we have seen the actual Windows version on
++ * Windows Server 2012 R2, even without a manifest. */
++ if (info.dwMajorVersion > 6 ||
++ (info.dwMajorVersion == 6 && info.dwMinorVersion >= 2)) {
++ /* When GetVersionEx() returns Windows 8, the actual OS may be any
++ * later version. */
++ strlcat(uname_result, " [or later]", sizeof(uname_result));
++ }
++ /* When we don't know if the OS is a client or server version, we use
++ * the client version, and this qualifier. */
++ if (!is_server && !is_client) {
++ strlcat(uname_result, " [client or server]", sizeof(uname_result));
++ }
+#else /* !(defined(_WIN32)) */
+ /* LCOV_EXCL_START -- can't provoke uname failure */
+ strlcpy(uname_result, "Unknown platform", sizeof(uname_result));
+ /* LCOV_EXCL_STOP */
+#endif /* defined(_WIN32) */
+ }
+ uname_result_is_set = 1;
+ }
+ return uname_result;
+}
More information about the tor-commits
mailing list