(C++) Odd Thread Count by kernelwernel

Created the Saturday 17 August 2024. Updated 3 weeks ago.

Code

            /**
 * The thread count of the CPU must be an even number, which allows 
 * us to detect whether the thread count has been modified through 
 * the VM creation stage. If the thread count is not an even number, 
 * this is a sign of thread tampering (with a few exceptions which 
 * are already covered).
 *
 * This code snippet is from the VMAware project at https://github.com/kernelwernel/VMAware
 * detection made by kernelwernel, 2024
 */

#if (defined(_MSC_VER) || defined(_WIN32) || defined(_WIN64) || defined(__MINGW32__))
#define MSVC 1
#define LINUX 0
#elif (defined(__linux__))
#define MSVC 0
#define LINUX 1
#else
#define MSVC 0
#define LINUX 0
#endif

#if (LINUX)
#include <cpuid.h>
#elif (MSVC)
#include <intrin.h>
#endif

#include <cstdint>
#include <thread>
#include <array>
#include <iostream>

/*
 * @brief Check for odd CPU threads, usually a sign of modification through VM setting because 99% of CPUs have even numbers of threads
 */ 
bool odd_cpu_threads() {
    struct stepping_struct {
        std::uint8_t model;
        std::uint8_t family;
        std::uint8_t extmodel;
    };

    struct stepping_struct steps {};

    // cross-platform wrapper function for linux and MSVC cpuid
    auto cpuid = [](
        std::uint32_t& a, 
        std::uint32_t& b, 
        std::uint32_t& c, 
        std::uint32_t& d,
        const std::uint32_t a_leaf
    ) -> void {
#if (MSVC)
        std::int32_t x[4]{};
        __cpuid((std::int32_t*)x, static_cast<std::int32_t>(a_leaf));
        a = static_cast<std::uint32_t>(x[0]);
        b = static_cast<std::uint32_t>(x[1]);
        c = static_cast<std::uint32_t>(x[2]);
        d = static_cast<std::uint32_t>(x[3]);
#elif (LINUX)
        __get_cpuid(a_leaf, &a, &b, &c, &d);
#else
        return;
#endif
    };

    auto is_intel = [&]() -> bool {
        constexpr std::uint32_t intel_ecx = 0x6c65746e;

        std::uint32_t unused, ecx = 0;
        cpuid(unused, unused, ecx, unused, 0);

        return (ecx == intel_ecx);
    };

    auto is_amd = [&]() -> bool {
        constexpr std::uint32_t amd_ecx = 0x69746e65;

        std::uint32_t unused, ecx = 0;
        cpuid(unused, unused, ecx, unused, 0);

        return (ecx == amd_ecx);
    };

    std::uint32_t unused, eax = 0;
    cpuid(eax, unused, unused, unused, 1);

    steps.model    = ((eax >> 4)  & 0b1111);
    steps.family   = ((eax >> 8)  & 0b1111);
    steps.extmodel = ((eax >> 16) & 0b1111);
    
    // check if the CPU is an intel celeron
    auto is_celeron = [&]() -> bool {
        if (!is_intel()) {
            return false;
        }

        constexpr std::uint8_t celeron_family   = 0x6;
        constexpr std::uint8_t celeron_extmodel = 0x2;
        constexpr std::uint8_t celeron_model    = 0xA;

        return (
            steps.family   == celeron_family   &&
            steps.extmodel == celeron_extmodel && 
            steps.model    == celeron_model
        );
    };

    // check if the microarchitecture was made before 2006, which was around the time multi-core processors were implemented
    auto old_microarchitecture = [&]() -> bool {
        constexpr std::array<std::array<std::uint8_t, 3>, 32> old_archs = {{
            // 80486
            {{ 0x4, 0x0, 0x1 }},
            {{ 0x4, 0x0, 0x2 }},
            {{ 0x4, 0x0, 0x3 }},
            {{ 0x4, 0x0, 0x4 }},
            {{ 0x4, 0x0, 0x5 }},
            {{ 0x4, 0x0, 0x7 }},
            {{ 0x4, 0x0, 0x8 }},
            {{ 0x4, 0x0, 0x9 }},

            // P5
            {{ 0x5, 0x0, 0x1 }},
            {{ 0x5, 0x0, 0x2 }},
            {{ 0x5, 0x0, 0x4 }},
            {{ 0x5, 0x0, 0x7 }},
            {{ 0x5, 0x0, 0x8 }},

            // P6
            {{ 0x6, 0x0, 0x1 }},
            {{ 0x6, 0x0, 0x3 }},
            {{ 0x6, 0x0, 0x5 }},
            {{ 0x6, 0x0, 0x6 }},
            {{ 0x6, 0x0, 0x7 }},
            {{ 0x6, 0x0, 0x8 }},
            {{ 0x6, 0x0, 0xA }},
            {{ 0x6, 0x0, 0xB }},

            // Netburst
            {{ 0xF, 0x0, 0x6 }},
            {{ 0xF, 0x0, 0x4 }},
            {{ 0xF, 0x0, 0x3 }},
            {{ 0xF, 0x0, 0x2 }},
            {{ 0xF, 0x0, 0x10 }},

            {{ 0x6, 0x1, 0x5 }}, // Pentium M (Talopai)
            {{ 0x6, 0x1, 0x6 }}, // Core (Client)
            {{ 0x6, 0x0, 0x9 }}, // Pentium M
            {{ 0x6, 0x0, 0xD }}, // Pentium M
            {{ 0x6, 0x0, 0xE }}, // Modified Pentium M
            {{ 0x6, 0x0, 0xF }}  // Core (Client)
        }};

        constexpr std::uint8_t FAMILY   = 0;
        constexpr std::uint8_t EXTMODEL = 1;
        constexpr std::uint8_t MODEL    = 2;

        for (const auto& arch : old_archs) {
            if (
                steps.family   == arch.at(FAMILY)   &&
                steps.extmodel == arch.at(EXTMODEL) &&
                steps.model    == arch.at(MODEL)
            ) {
                return true;
            }
        }

        return false;
    };

    // if neither AMD or Intel, return false
    if (!(is_intel() || is_amd())) {
        return false;
    }

    // Intel Celeron CPUs are relatively modern, but they can contain a single or odd thread count
    if (is_celeron()) {
        return false;
    }

    // CPUs before 2006 had no official multi-core processors (for both AMD and Intel)
    if (old_microarchitecture()) {
        return false;
    }

    // Is the thread count odd?
    return (std::thread::hardware_concurrency() & 1);
}

int main() {
    std::cout << "\nIs the thread count odd? = " << (odd_cpu_threads() ? "Yes, very likely a VM" : "No, could be baremetal") << "\n";
    return 0;
}