2020-01-06 15:11:57 +01:00
|
|
|
|
/*
|
2024-03-05 18:37:42 +01:00
|
|
|
|
* Copyright (c) 2024 ETH Zürich, IT Services
|
2020-01-06 15:11:57 +01:00
|
|
|
|
*
|
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
*/
|
|
|
|
|
|
2020-09-29 14:37:54 +02:00
|
|
|
|
using System.Linq;
|
2020-01-06 15:11:57 +01:00
|
|
|
|
using SafeExamBrowser.Logging.Contracts;
|
|
|
|
|
using SafeExamBrowser.SystemComponents.Contracts;
|
2023-07-17 16:40:33 +02:00
|
|
|
|
using SafeExamBrowser.SystemComponents.Contracts.Registry;
|
2020-01-06 15:11:57 +01:00
|
|
|
|
|
|
|
|
|
namespace SafeExamBrowser.SystemComponents
|
|
|
|
|
{
|
|
|
|
|
public class VirtualMachineDetector : IVirtualMachineDetector
|
|
|
|
|
{
|
2023-07-31 10:52:40 +02:00
|
|
|
|
private const string MANIPULATED = "000000000000";
|
2023-05-02 14:56:15 +02:00
|
|
|
|
private const string QEMU_MAC_PREFIX = "525400";
|
|
|
|
|
private const string VIRTUALBOX_MAC_PREFIX = "080027";
|
|
|
|
|
|
|
|
|
|
private static readonly string[] DeviceBlacklist =
|
2023-02-23 16:40:26 +01:00
|
|
|
|
{
|
2023-03-07 23:41:56 +01:00
|
|
|
|
// Hyper-V
|
|
|
|
|
"PROD_VIRTUAL", "HYPER_V",
|
|
|
|
|
// QEMU
|
|
|
|
|
"qemu", "ven_1af4", "ven_1b36", "subsys_11001af4",
|
|
|
|
|
// VirtualBox
|
2023-11-21 15:53:58 +01:00
|
|
|
|
"VEN_VBOX", "vid_80ee",
|
2023-03-07 23:41:56 +01:00
|
|
|
|
// VMware
|
|
|
|
|
"PROD_VMWARE", "VEN_VMWARE", "VMWARE_IDE"
|
2023-02-23 16:40:26 +01:00
|
|
|
|
};
|
2023-05-02 14:56:15 +02:00
|
|
|
|
|
|
|
|
|
private static readonly string[] DeviceWhitelist =
|
|
|
|
|
{
|
2023-05-30 15:28:21 +02:00
|
|
|
|
// Microsoft Virtual Disk Device
|
|
|
|
|
"PROD_VIRTUAL_DISK",
|
|
|
|
|
// Microsoft Virtual DVD Device
|
|
|
|
|
"PROD_VIRTUAL_DVD"
|
2023-05-02 14:56:15 +02:00
|
|
|
|
};
|
2022-07-29 13:49:26 +02:00
|
|
|
|
|
|
|
|
|
private readonly ILogger logger;
|
2023-07-17 16:40:33 +02:00
|
|
|
|
private readonly IRegistry registry;
|
2022-07-29 13:49:26 +02:00
|
|
|
|
private readonly ISystemInfo systemInfo;
|
2020-05-06 18:44:08 +02:00
|
|
|
|
|
2023-07-17 16:40:33 +02:00
|
|
|
|
public VirtualMachineDetector(ILogger logger, IRegistry registry, ISystemInfo systemInfo)
|
2020-01-06 15:11:57 +01:00
|
|
|
|
{
|
|
|
|
|
this.logger = logger;
|
2023-07-31 10:06:09 +02:00
|
|
|
|
this.registry = registry;
|
2020-01-06 15:11:57 +01:00
|
|
|
|
this.systemInfo = systemInfo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool IsVirtualMachine()
|
|
|
|
|
{
|
2023-07-31 10:06:09 +02:00
|
|
|
|
var isVirtualMachine = false;
|
2020-05-07 13:21:57 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
isVirtualMachine |= HasVirtualDevice();
|
|
|
|
|
isVirtualMachine |= HasVirtualMacAddress();
|
2023-07-22 14:19:42 +02:00
|
|
|
|
isVirtualMachine |= IsVirtualCpu();
|
2023-07-31 10:06:09 +02:00
|
|
|
|
isVirtualMachine |= IsVirtualRegistry();
|
2023-07-31 10:52:40 +02:00
|
|
|
|
isVirtualMachine |= IsVirtualSystem(systemInfo.BiosInfo, systemInfo.Manufacturer, systemInfo.Model);
|
|
|
|
|
|
|
|
|
|
logger.Debug($"Computer '{systemInfo.Name}' appears {(isVirtualMachine ? "" : "not ")}to be a virtual machine.");
|
2023-04-14 19:56:22 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
return isVirtualMachine;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool HasVirtualDevice()
|
|
|
|
|
{
|
|
|
|
|
var hasVirtualDevice = false;
|
2023-04-14 19:56:22 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
foreach (var device in systemInfo.PlugAndPlayDeviceIds)
|
2023-04-14 19:56:22 +02:00
|
|
|
|
{
|
2023-07-31 10:52:40 +02:00
|
|
|
|
hasVirtualDevice |= DeviceBlacklist.Any(d => device.ToLower().Contains(d.ToLower())) && DeviceWhitelist.All(d => !device.ToLower().Contains(d.ToLower()));
|
2023-04-14 19:56:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
return hasVirtualDevice;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool HasVirtualMacAddress()
|
|
|
|
|
{
|
|
|
|
|
var hasVirtualMacAddress = false;
|
|
|
|
|
var macAddress = systemInfo.MacAddress;
|
|
|
|
|
|
|
|
|
|
if (macAddress != null && macAddress.Length > 2)
|
2023-04-14 19:56:22 +02:00
|
|
|
|
{
|
2023-07-31 10:52:40 +02:00
|
|
|
|
hasVirtualMacAddress |= macAddress.StartsWith(MANIPULATED);
|
|
|
|
|
hasVirtualMacAddress |= macAddress.StartsWith(QEMU_MAC_PREFIX);
|
|
|
|
|
hasVirtualMacAddress |= macAddress.StartsWith(VIRTUALBOX_MAC_PREFIX);
|
2023-04-14 19:56:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
return hasVirtualMacAddress;
|
2023-04-14 19:56:22 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
private bool IsVirtualCpu()
|
2020-01-06 15:11:57 +01:00
|
|
|
|
{
|
2023-07-31 10:52:40 +02:00
|
|
|
|
var isVirtualCpu = false;
|
2020-05-07 13:21:57 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
isVirtualCpu |= systemInfo.CpuName.ToLower().Contains(" kvm ");
|
2020-05-07 13:21:57 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
return isVirtualCpu;
|
2023-04-01 16:25:56 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool IsVirtualRegistry()
|
|
|
|
|
{
|
2023-07-31 10:52:40 +02:00
|
|
|
|
var isVirtualRegistry = false;
|
2023-04-01 16:25:56 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
isVirtualRegistry |= HasLocalVirtualMachineDeviceCache();
|
2023-07-17 17:33:21 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
return isVirtualRegistry;
|
2023-07-17 17:33:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
private bool IsVirtualSystem(string biosInfo, string manufacturer, string model)
|
2023-07-17 17:33:21 +02:00
|
|
|
|
{
|
2023-07-31 10:52:40 +02:00
|
|
|
|
var isVirtualSystem = false;
|
2023-07-31 10:06:09 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
biosInfo = biosInfo.ToLower();
|
|
|
|
|
manufacturer = manufacturer.ToLower();
|
|
|
|
|
model = model.ToLower();
|
2023-07-31 10:06:09 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
isVirtualSystem |= biosInfo.Contains("hyper-v");
|
|
|
|
|
isVirtualSystem |= biosInfo.Contains("virtualbox");
|
|
|
|
|
isVirtualSystem |= biosInfo.Contains("vmware");
|
|
|
|
|
isVirtualSystem |= biosInfo.Contains("ovmf");
|
|
|
|
|
isVirtualSystem |= biosInfo.Contains("edk ii unknown");
|
|
|
|
|
isVirtualSystem |= manufacturer.Contains("microsoft corporation") && !model.Contains("surface");
|
|
|
|
|
isVirtualSystem |= manufacturer.Contains("parallels software");
|
|
|
|
|
isVirtualSystem |= manufacturer.Contains("qemu");
|
|
|
|
|
isVirtualSystem |= manufacturer.Contains("vmware");
|
|
|
|
|
isVirtualSystem |= model.Contains("virtualbox");
|
|
|
|
|
isVirtualSystem |= model.Contains("Q35 +");
|
|
|
|
|
|
|
|
|
|
return isVirtualSystem;
|
|
|
|
|
}
|
2020-05-07 13:21:57 +02:00
|
|
|
|
|
2023-07-18 15:11:44 +02:00
|
|
|
|
private bool HasLocalVirtualMachineDeviceCache()
|
2023-07-17 17:33:21 +02:00
|
|
|
|
{
|
|
|
|
|
var deviceName = System.Environment.GetEnvironmentVariable("COMPUTERNAME");
|
2023-07-31 10:52:40 +02:00
|
|
|
|
var hasDeviceCache = false;
|
2023-09-01 12:28:03 +02:00
|
|
|
|
var hasDeviceCacheKeys = registry.TryGetSubKeys(RegistryValue.UserHive.DeviceCache_Key, out var deviceCacheKeys);
|
2023-04-14 21:18:08 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
if (deviceName != default && hasDeviceCacheKeys)
|
2023-04-01 19:09:01 +02:00
|
|
|
|
{
|
2023-07-18 14:32:54 +02:00
|
|
|
|
foreach (var cacheId in deviceCacheKeys)
|
2023-04-01 19:09:01 +02:00
|
|
|
|
{
|
2023-09-01 12:28:03 +02:00
|
|
|
|
var cacheIdKey = $@"{RegistryValue.UserHive.DeviceCache_Key}\{cacheId}";
|
2023-07-18 14:32:54 +02:00
|
|
|
|
var didReadKeys = true;
|
2023-04-01 19:09:01 +02:00
|
|
|
|
|
2023-07-18 14:32:54 +02:00
|
|
|
|
didReadKeys &= registry.TryRead(cacheIdKey, "DeviceName", out var cacheDeviceName);
|
2023-07-31 10:06:09 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
if (didReadKeys && deviceName.ToLower() == ((string) cacheDeviceName).ToLower())
|
2023-07-18 15:02:02 +02:00
|
|
|
|
{
|
2023-07-31 10:52:40 +02:00
|
|
|
|
didReadKeys &= registry.TryRead(cacheIdKey, "DeviceMake", out var cacheDeviceManufacturer);
|
|
|
|
|
didReadKeys &= registry.TryRead(cacheIdKey, "DeviceModel", out var cacheDeviceModel);
|
2023-07-31 10:06:09 +02:00
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
if (didReadKeys)
|
|
|
|
|
{
|
|
|
|
|
hasDeviceCache |= IsVirtualSystem("", (string) cacheDeviceManufacturer, (string) cacheDeviceModel);
|
|
|
|
|
}
|
2023-07-18 15:02:02 +02:00
|
|
|
|
}
|
2023-04-01 19:09:01 +02:00
|
|
|
|
}
|
2023-04-01 16:25:56 +02:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-31 10:52:40 +02:00
|
|
|
|
return hasDeviceCache;
|
2020-01-06 15:11:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|