*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
-#ifdef HAVE_CONFIG_H
-# include "config.h"
-#endif
+#include "config.h"
#include <glib.h>
#endif
#ifdef HAVE_OS_X_FRAMEWORKS
-#include <CoreServices/CoreServices.h>
+#include <CoreFoundation/CoreFoundation.h>
+#include "cfutils.h"
#endif
#ifdef HAVE_LIBCAP
#else /* HAVE_LIBCAP */
g_string_append(str, "without POSIX capabilities");
#endif /* HAVE_LIBCAP */
-#ifdef USE_THREADS
- g_string_append_printf(str, ", with threads support");
+
+ /* LIBNL */
+ g_string_append(str, ", ");
+#if defined(HAVE_LIBNL1)
+ g_string_append(str, "with libnl 1");
+#elif defined(HAVE_LIBNL2)
+ g_string_append(str, "with libnl 2");
+#elif defined(HAVE_LIBNL3)
+ g_string_append(str, "with libnl 3");
#else
- g_string_append_printf(str, ", without threads support");
+ g_string_append(str, "without libnl");
#endif
/* Additional application-dependent information */
#endif
/*
- * Get various library run-time versions, and the OS version, and append
- * them to the specified GString.
+ * Handles the rather elaborate process of getting OS version information
+ * from OS X (we want the OS X version, not the Darwin version, the latter
+ * being easy to get with uname()).
*/
-void
-get_runtime_version_info(GString *str, void (*additional_info)(GString *))
+#ifdef HAVE_OS_X_FRAMEWORKS
+
+/*
+ * Fetch a string, as a UTF-8 C string, from a dictionary, given a key.
+ */
+static char *
+get_string_from_dictionary(CFPropertyListRef dict, CFStringRef key)
+{
+ CFStringRef cfstring;
+
+ cfstring = (CFStringRef)CFDictionaryGetValue((CFDictionaryRef)dict,
+ (const void *)key);
+ if (cfstring == NULL)
+ return NULL;
+ if (CFGetTypeID(cfstring) != CFStringGetTypeID()) {
+ /* It isn't a string. Punt. */
+ return NULL;
+ }
+ return CFString_to_C_string(cfstring);
+}
+
+/*
+ * Get the OS X version information, and append it to the GString.
+ * Return TRUE if we succeed, FALSE if we fail.
+ */
+static gboolean
+get_os_x_version_info(GString *str)
+{
+ static const UInt8 server_version_plist_path[] =
+ "/System/Library/CoreServices/ServerVersion.plist";
+ static const UInt8 system_version_plist_path[] =
+ "/System/Library/CoreServices/SystemVersion.plist";
+ CFURLRef version_plist_file_url;
+ CFReadStreamRef version_plist_stream;
+ CFDictionaryRef version_dict;
+ char *string;
+
+ /*
+ * On OS X, report the OS X version number as the OS, and put
+ * the Darwin information in parentheses.
+ *
+ * Alas, Gestalt() is deprecated in Mountain Lion, so the build
+ * fails if you treat deprecation warnings as fatal. I don't
+ * know of any replacement API, so we fall back on reading
+ * /System/Library/CoreServices/ServerVersion.plist if it
+ * exists, otherwise /System/Library/CoreServices/SystemVersion.plist,
+ * and using ProductUserVisibleVersion. We also get the build
+ * version from ProductBuildVersion and the product name from
+ * ProductName.
+ */
+ version_plist_file_url = CFURLCreateFromFileSystemRepresentation(NULL,
+ server_version_plist_path, sizeof server_version_plist_path - 1,
+ false);
+ if (version_plist_file_url == NULL)
+ return FALSE;
+ version_plist_stream = CFReadStreamCreateWithFile(NULL,
+ version_plist_file_url);
+ CFRelease(version_plist_file_url);
+ if (version_plist_stream == NULL)
+ return FALSE;
+ if (!CFReadStreamOpen(version_plist_stream)) {
+ CFRelease(version_plist_stream);
+
+ /*
+ * Try SystemVersion.plist.
+ */
+ version_plist_file_url = CFURLCreateFromFileSystemRepresentation(NULL,
+ system_version_plist_path, sizeof system_version_plist_path - 1,
+ false);
+ if (version_plist_file_url == NULL)
+ return FALSE;
+ version_plist_stream = CFReadStreamCreateWithFile(NULL,
+ version_plist_file_url);
+ CFRelease(version_plist_file_url);
+ if (version_plist_stream == NULL)
+ return FALSE;
+ if (!CFReadStreamOpen(version_plist_stream)) {
+ CFRelease(version_plist_stream);
+ return FALSE;
+ }
+ }
+#ifdef HAVE_CFPROPERTYLISTCREATEWITHSTREAM
+ version_dict = (CFDictionaryRef)CFPropertyListCreateWithStream(NULL,
+ version_plist_stream, 0, kCFPropertyListImmutable,
+ NULL, NULL);
+#else
+ version_dict = (CFDictionaryRef)CFPropertyListCreateFromStream(NULL,
+ version_plist_stream, 0, kCFPropertyListImmutable,
+ NULL, NULL);
+#endif
+ if (version_dict == NULL) {
+ CFRelease(version_plist_stream);
+ return FALSE;
+ }
+ if (CFGetTypeID(version_dict) != CFDictionaryGetTypeID()) {
+ /* This is *supposed* to be a dictionary. Punt. */
+ CFRelease(version_dict);
+ CFReadStreamClose(version_plist_stream);
+ CFRelease(version_plist_stream);
+ return FALSE;
+ }
+ /* Get the product name string. */
+ string = get_string_from_dictionary(version_dict,
+ CFSTR("ProductName"));
+ if (string == NULL) {
+ CFRelease(version_dict);
+ CFReadStreamClose(version_plist_stream);
+ CFRelease(version_plist_stream);
+ return FALSE;
+ }
+ g_string_append_printf(str, "%s", string);
+ g_free(string);
+
+ /* Get the OS version string. */
+ string = get_string_from_dictionary(version_dict,
+ CFSTR("ProductUserVisibleVersion"));
+ if (string == NULL) {
+ CFRelease(version_dict);
+ CFReadStreamClose(version_plist_stream);
+ CFRelease(version_plist_stream);
+ return FALSE;
+ }
+ g_string_append_printf(str, " %s", string);
+ g_free(string);
+
+ /* Get the build string */
+ string = get_string_from_dictionary(version_dict,
+ CFSTR("ProductBuildVersion"));
+ if (string == NULL) {
+ CFRelease(version_dict);
+ CFReadStreamClose(version_plist_stream);
+ CFRelease(version_plist_stream);
+ return FALSE;
+ }
+ g_string_append_printf(str, ", build %s", string);
+ g_free(string);
+ CFRelease(version_dict);
+ CFReadStreamClose(version_plist_stream);
+ CFRelease(version_plist_stream);
+ return TRUE;
+}
+#endif
+
+/*
+ * Get the OS version, and append it to the GString
+ */
+void get_os_version_info(GString *str)
{
#if defined(_WIN32)
OSVERSIONINFOEX info;
#elif defined(HAVE_SYS_UTSNAME_H)
struct utsname name;
#endif
-#if HAVE_OS_X_FRAMEWORKS
- SInt32 macosx_ver, macosx_major_ver, macosx_minor_ver, macosx_bugfix_ver;
-#endif
-#ifndef _WIN32
- gchar *lang;
-#endif
-
- g_string_append(str, "on ");
#if defined(_WIN32)
/*
case 1:
g_string_append_printf(str, is_nt_workstation ? "Windows 7" : "Windows Server 2008 R2");
break;
+ case 2:
+ g_string_append_printf(str, is_nt_workstation ? "Windows 8" : "Windows Server 2012");
+ break;
default:
g_string_append_printf(str, "Windows NT, unknown version %lu.%lu",
info.dwMajorVersion, info.dwMinorVersion);
#ifdef HAVE_OS_X_FRAMEWORKS
/*
* On Mac OS X, report the Mac OS X version number as
- * the OS, and put the Darwin information in parentheses.
- *
- * XXX - can we get the build name? There's no API to
- * get it; it's currently in
- * /System/Library/CoreServices/SystemVersion.plist
- * but there's no guarantee that it will continue to
- * be there.
+ * the OS version if we can, and put the Darwin information
+ * in parentheses.
*/
- Gestalt(gestaltSystemVersion, &macosx_ver);
-
- /* The following functions are only available in Mac OS 10.4+ */
- if(macosx_ver >= 0x1040) {
- Gestalt(gestaltSystemVersionMajor, &macosx_major_ver);
- Gestalt(gestaltSystemVersionMinor, &macosx_minor_ver);
- Gestalt(gestaltSystemVersionBugFix, &macosx_bugfix_ver);
-
- g_string_append_printf(str, "Mac OS %ld.%ld.%ld",
- (long)macosx_major_ver,
- (long)macosx_minor_ver,
- (long)macosx_bugfix_ver);
+ if (get_os_x_version_info(str)) {
+ /* Success - append the Darwin information. */
+ g_string_append_printf(str, " (%s %s)", name.sysname, name.release);
} else {
- g_string_append_printf(str, "Mac OS X < 10.4 [%lx]",
- (long)macosx_ver);
- /* See Apple's Gestalt Manager Reference for meanings
- * of the macosx_ver values. */
+ /* Failure - just use the Darwin information. */
+ g_string_append_printf(str, "%s %s", name.sysname, name.release);
}
- g_string_append_printf(str, " (%s %s)", name.sysname, name.release);
#else /* HAVE_OS_X_FRAMEWORKS */
/*
* XXX - on Linux, are there any APIs to get the distribution
#else
g_string_append(str, "an unknown OS");
#endif
+}
+
+
+/*
+ * Get the CPU info, and append it to the GString
+ */
+
+#if defined(_MSC_VER)
+static void
+do_cpuid(int *CPUInfo, guint32 selector){
+ __cpuid(CPUInfo, selector);
+}
+#elif defined(__GNUC__)
+#if defined(__x86_64__)
+static inline void
+do_cpuid(guint32 *CPUInfo, int selector)
+{
+ __asm__ __volatile__("cpuid"
+ : "=a" (CPUInfo[0]),
+ "=b" (CPUInfo[1]),
+ "=c" (CPUInfo[2]),
+ "=d" (CPUInfo[3])
+ : "a"(selector));
+}
+#else /* (__i386__) */
+/* would need a test if older proccesors have the cpuid instruction */
+static void
+do_cpuid(guint32 *CPUInfo, int selector _U_){
+ CPUInfo[0] = 0;
+}
+
+#endif /* defined(__x86_64__)*/
+#else /* Other compilers */
+static void
+do_cpuid(guint32 *CPUInfo, int selector _U_){
+ CPUInfo[0] = 0;
+}
+#endif
+
+/*
+ * Get CPU info on platforms where the cpuid instruction can be used skip 32 bit versions for GCC
+ * http://www.intel.com/content/dam/www/public/us/en/documents/application-notes/processor-identification-cpuid-instruction-note.pdf
+ * the get_cpuid() routine will return 0 in CPUInfo[0] if cpuinfo isn't available.
+ */
+
+static void get_cpu_info(GString *str _U_)
+{
+#if defined(_MSC_VER)
+ int CPUInfo[4];
+#else
+ guint32 CPUInfo[4];
+#endif
+ char CPUBrandString[0x40];
+ unsigned nExIds;
+
+ /* http://msdn.microsoft.com/en-us/library/hskdteyh(v=vs.100).aspx */
+
+ /* Calling __cpuid with 0x80000000 as the InfoType argument*/
+ /* gets the number of valid extended IDs.*/
+ do_cpuid(CPUInfo, 0x80000000);
+ nExIds = CPUInfo[0];
+
+ if( nExIds<0x80000005)
+ return;
+ memset(CPUBrandString, 0, sizeof(CPUBrandString));
+
+ /* Interpret CPU brand string.*/
+ do_cpuid(CPUInfo, 0x80000002);
+ memcpy(CPUBrandString, CPUInfo, sizeof(CPUInfo));
+ do_cpuid(CPUInfo, 0x80000003);
+ memcpy(CPUBrandString + 16, CPUInfo, sizeof(CPUInfo));
+ do_cpuid(CPUInfo, 0x80000004);
+ memcpy(CPUBrandString + 32, CPUInfo, sizeof(CPUInfo));
+
+ g_string_append_printf(str, "\n%s", CPUBrandString);
+
+}
+
+static void get_mem_info(GString *str _U_)
+{
+#if defined(_WIN32)
+ MEMORYSTATUSEX statex;
+
+ statex.dwLength = sizeof (statex);
+
+ if(GlobalMemoryStatusEx (&statex))
+ g_string_append_printf(str, ", with ""%" G_GINT64_MODIFIER "d" "MB of physical memory.\n", statex.ullTotalPhys/(1024*1024));
+#endif
+
+}
+
+/*
+ * Get various library run-time versions, and the OS version, and append
+ * them to the specified GString.
+ */
+void
+get_runtime_version_info(GString *str, void (*additional_info)(GString *))
+{
+#ifndef _WIN32
+ gchar *lang;
+#endif
+
+ g_string_append(str, "on ");
+
+ get_os_version_info(str);
#ifndef _WIN32
/* Locale */
g_string_append(str, ".");
+ /* CPU Info */
+ get_cpu_info(str);
+
+ /* Get info about installed memory Windows only */
+ get_mem_info(str);
+
/* Compiler info */
/*
get_copyright_info(void)
{
return
-"Copyright 1998-2011 Gerald Combs <gerald@wireshark.org> and contributors.\n"
+"Copyright 1998-2013 Gerald Combs <gerald@wireshark.org> and contributors.\n"
"This is free software; see the source for copying conditions. There is NO\n"
"warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n";
}