Noções básicas sobre VirtualBox e CPUID
Você precisa definir os VBoxInternal/CPUM/HostCPUID
extradados da máquina virtual. Isso fará o VirtualBox relatar resultados personalizados para a instrução CPUID ao convidado. Dependendo do valor do registro EAX, esta instrução retorna informações sobre o processador - itens como fornecedor, tipo, família, etapa, marca, tamanho do cache, recursos (MMX, SSE, SSE2, PAE, HTT), etc. Quanto mais resultados você mutilar, maiores as chances de enganar o hóspede.
Você pode usar o vboxmanage setextradata
comando para configurar a máquina virtual. Por exemplo,
vboxmanage setextradata WinXP VBoxInternal/CPUM/HostCPUID/80000003/ebx 0x50202952
fará com que CPUID retorne 50202952₍₁₆₎ no registro EBX, quando chamado com EAX definido como 80000003₍₁₆₎. (A partir de agora, os números hexadecimais serão gravados como 0xNN ou NNh.)
Definindo a cadeia de fornecedores da CPU
Se EAX for 0 (ou 80000000h na AMD), o CPUID retornará o fornecedor como uma sequência ASCII nos registros EBX, EDX, ECX (observe o pedido). Para uma CPU AMD, eles se parecem com isso:
| Register | Value | Description |
|----------|------------|--------------------------------|
| EBX | 6874_7541h | The ASCII characters "h t u A" |
| ECX | 444D_4163h | The ASCII characters "D M A c" |
| EDX | 6974_6E65h | The ASCII characters "i t n e" |
(Extraído da especificação CPUID da AMD , subseção "CPUID Fn0000_0000_E")
Se você concatenar EBX, EDX e ECX, você conseguirá AuthenticAMD
.
Se você possui o Bash e os utilitários tradicionais do Unix, pode configurar facilmente o fornecedor com os seguintes comandos:
vm='WinXP' # UUID works as well
# The vendor string needs to have 12 characters!
vendor='AuthenticAMD'
if [ ${#vendor} -ne 12 ]; then
exit 1
fi
ascii2hex() { echo -n 0x; od -A n --endian little -t x4 | sed 's/ //g'; }
registers=(ebx edx ecx)
for (( i=0; i<${#vendor}; i+=4 )); do
register=${registers[$(($i/4))]}
value=`echo -n "${vendor:$i:4}" | ascii2hex`
# set value to an empty string to reset the CPUID, i.e.
# value=""
for eax in 00000000 80000000; do
key=VBoxInternal/CPUM/HostCPUID/${eax}/${register}
vboxmanage setextradata "$vm" $key $value
done
done
Definindo a cadeia de caracteres da marca da CPU
Se EAX for 80000002h, 80000003h, 80000004h, CPUID retornará 16 caracteres ASCII da string da marca nos registros EAX, EBX, ECX, EDX, totalizando 3 * 16 = 48 caracteres; a cadeia é finalizada com um caractere nulo . Observe que esse recurso foi introduzido nos processadores Pentium 4. É assim que a string da marca pode parecer em um processador Pentium 4:
| EAX Input Value | Return Values | ASCII Equivalent |
|-----------------|-----------------|------------------|
| 80000002h | EAX = 20202020h | " " |
| | EBX = 20202020h | " " |
| | ECX = 20202020h | " " |
| | EDX = 6E492020h | "nI " |
|-----------------|-----------------|------------------|
| 80000003h | EAX = 286C6574h | "(let" |
| | EBX = 50202952h | "P )R" |
| | ECX = 69746E65h | "itne" |
| | EDX = 52286D75h | "R(mu" |
|-----------------|-----------------|------------------|
| 80000004h | EAX = 20342029h | " 4 )" |
| | EBX = 20555043h | " UPC" |
| | ECX = 30303531h | "0051" |
| | EDX = 007A484Dh | "☠zHM" |
|-----------------|-----------------|------------------|
(Extraído da Referência de programação das extensões do conjunto de instruções da arquitetura Intel , subseção 2.9, "Instrução CPUID", tabela 2-30. ☠ é o caractere nulo (valor numérico 0).)
Se você juntar os resultados, obterá Intel(R) Pentium(R) 4 CPU 1500MHz☠
.
Se você possui o Bash e os utilitários tradicionais do Unix, pode definir facilmente a marca com os seguintes comandos:
vm='WinXP' # UUID works as well
# The brand string needs to have 47 characters!
# The null terminator is added automatically
brand=' Intel(R) Pentium(R) 4 CPU 1500MHz'
if [ ${#brand} -ne 47 ]; then
exit 1
fi
ascii2hex() { echo -n 0x; od -A n --endian little -t x4 | sed 's/ //g'; }
eax_values=(80000002 80000003 80000004)
registers=(edx ecx ebx eax)
for (( i=0; i<${#brand}; i+=4 )); do
eax=${eax_values[$((${i} / 4 / 4))]}
register=${registers[$((${i} / 4 % 4 ))]}
key=VBoxInternal/CPUM/HostCPUID/${eax}/${register}
value=`echo -n "${brand:$i:4}" | ascii2hex`
# set value to an empty string to reset the CPUID, i.e.
# value=""
vboxmanage setextradata "$vm" $key $value
done
Se você tiver um prompt de comando do Windows, poderá definir a marca como Intel(R) Core(TM)2 CPU 6600 @ 2.40 GHz
1 executando:
set vm=your-vm-name-or-uuid
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000002/eax 0x65746e49
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000002/ebx 0x2952286c
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000002/ecx 0x726f4320
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000002/edx 0x4d542865
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000003/eax 0x43203229
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000003/ebx 0x20205550
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000003/ecx 0x20202020
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000003/edx 0x20202020
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000004/eax 0x30303636
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000004/ebx 0x20402020
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000004/ecx 0x30342e32
vboxmanage setextradata %vm% VBoxInternal/CPUM/HostCPUID/80000004/edx 0x007a4847
1 Os HostCPUID
valores foram retirados do relatório de bug do VirtualBox # 7865 .