* PERFORMANCE OF THIS SOFTWARE.
*/
-/* $Id: win32os.c,v 1.5 2007/06/19 23:47:19 tbox Exp $ */
-
#include <windows.h>
+#ifndef TESTVERSION
#include <isc/win32os.h>
+#else
+#include <stdio.h>
+#endif
-static BOOL bInit = FALSE;
-static OSVERSIONINFOEX osVer;
-
-static void
-initialize_action(void) {
- BOOL bSuccess;
+int
+isc_win32os_versioncheck(unsigned int major, unsigned int minor,
+ unsigned int spmajor, unsigned int spminor)
+{
+ OSVERSIONINFOEX osVer;
+ DWORD typeMask;
+ ULONGLONG conditionMask;
- if (bInit)
- return;
- /*
- * NOTE: VC++ 6.0 gets this function declaration wrong
- * so we compensate by casting the argument
- * NOTE: GetVersionEx() deprecated in Windows 8.1
- */
+ memset(&osVer, 0, sizeof(OSVERSIONINFOEX));
osVer.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
- bSuccess = GetVersionEx((OSVERSIONINFO *) &osVer);
-
- /*
- * Versions of NT before NT4.0 SP6 did not return the
- * extra info that the EX structure provides and returns
- * a failure so we need to retry with the old structure.
- */
- if(!bSuccess) {
- osVer.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
- (void)GetVersionEx((OSVERSIONINFO *) &osVer);
- }
- bInit = TRUE;
-}
+ typeMask = 0;
+ conditionMask = 0;
-unsigned int
-isc_win32os_majorversion(void) {
- initialize_action();
- return ((unsigned int)osVer.dwMajorVersion);
-}
-
-unsigned int
-isc_win32os_minorversion(void) {
- initialize_action();
- return ((unsigned int)osVer.dwMinorVersion);
-}
-
-unsigned int
-isc_win32os_servicepackmajor(void) {
- initialize_action();
- return ((unsigned int)osVer.wServicePackMajor);
-}
+ /* Optimistic: likely greater */
+ osVer.dwMajorVersion = major;
+ typeMask |= VER_MAJORVERSION;
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_MAJORVERSION,
+ VER_GREATER);
+ osVer.dwMinorVersion = minor;
+ typeMask |= VER_MINORVERSION;
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_MINORVERSION,
+ VER_GREATER);
+ osVer.wServicePackMajor = spmajor;
+ typeMask |= VER_SERVICEPACKMAJOR;
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_SERVICEPACKMAJOR,
+ VER_GREATER);
+ osVer.wServicePackMinor = spminor;
+ typeMask |= VER_SERVICEPACKMINOR;
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_SERVICEPACKMINOR,
+ VER_GREATER);
+ if (VerifyVersionInfo(&osVer, typeMask, conditionMask))
+ return (1);
-unsigned int
-isc_win32os_servicepackminor(void) {
- initialize_action();
- return ((unsigned int)osVer.wServicePackMinor);
+ /* Failed: retry with equal */
+ conditionMask = 0;
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_MAJORVERSION,
+ VER_EQUAL);
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_MINORVERSION,
+ VER_EQUAL);
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_SERVICEPACKMAJOR,
+ VER_EQUAL);
+ conditionMask = VerSetConditionMask(conditionMask,
+ VER_SERVICEPACKMINOR,
+ VER_EQUAL);
+ if (VerifyVersionInfo(&osVer, typeMask, conditionMask))
+ return (0);
+ else
+ return (-1);
}
+#ifdef TESTVERSION
int
-isc_win32os_versioncheck(unsigned int major, unsigned int minor,
- unsigned int spmajor, unsigned int spminor) {
+main(int argc, char **argv) {
+ unsigned int major = 0;
+ unsigned int minor = 0;
+ unsigned int spmajor = 0;
+ unsigned int spminor = 0;
+ int ret;
- initialize_action();
+ if (argc > 1) {
+ --argc;
+ ++argv;
+ major = (unsigned int) atoi(argv[0]);
+ }
+ if (argc > 1) {
+ --argc;
+ ++argv;
+ minor = (unsigned int) atoi(argv[0]);
+ }
+ if (argc > 1) {
+ --argc;
+ ++argv;
+ spmajor = (unsigned int) atoi(argv[0]);
+ }
+ if (argc > 1) {
+ --argc;
+ ++argv;
+ spminor = (unsigned int) atoi(argv[0]);
+ }
- if (major < isc_win32os_majorversion())
- return (1);
- if (major > isc_win32os_majorversion())
- return (-1);
- if (minor < isc_win32os_minorversion())
- return (1);
- if (minor > isc_win32os_minorversion())
- return (-1);
- if (spmajor < isc_win32os_servicepackmajor())
- return (1);
- if (spmajor > isc_win32os_servicepackmajor())
- return (-1);
- if (spminor < isc_win32os_servicepackminor())
- return (1);
- if (spminor > isc_win32os_servicepackminor())
- return (-1);
+ ret = isc_win32os_versioncheck(major, minor, spmajor, spminor);
- /* Exact */
- return (0);
+ printf("%s major %u minor %u SP major %u SP minor %u\n",
+ ret > 0 ? "greater" : (ret == 0 ? "equal" : "less"),
+ major, minor, spmajor, spminor);
+ return (ret);
}
+#endif