Rubrik: Verschiedenes / Sonstiges | VB-Versionen: VB2005, VB2008 | 14.07.08 |
Assembler Code ausführen Maschinencode mit VB Bordmitteln ausführen | ||
Autor: Arne Elster | Bewertung: | Views: 16.571 |
actorics.de/rm_code | System: Win2k, WinXP, Win7, Win8, Win10, Win11 | Beispielprojekt auf CD |
Ab und an ist es nützlich, auf die unterste Programmebene zurückzugreifen, den Assembler. Zum Beispiel dann, wenn Algorithmen besonders performant laufen sollen oder näher an der Hardware programmiert werden muss.
Folgendes Beispiel zeigt, wie bereits assemblierter Code, der den Herstellernamen der CPU ausliest, mithilfe der Marshal Klasse ausgeführt werden kann:
Imports System.Runtime.InteropServices Module Module1 Private Declare Function VirtualAlloc Lib "kernel32" ( _ ByVal addr As IntPtr, _ ByVal size As Integer, _ ByVal alloctype As Integer, _ ByVal protect As Integer) As IntPtr Private Declare Function VirtualFree Lib "kernel32" ( _ ByVal addr As IntPtr, _ ByVal size As Integer, _ ByVal freetype As Integer) As Integer Private Const MEM_COMMIT As Integer = &H1000 Private Const MEM_DECOMMIT As Integer = &H4000 Private Const PAGE_EXEC_RW As Integer = &H40 ' Über dieses Delegate wird der Maschinencode aufgerufen. ' Das Attribut "Out" ist wichtig, damit VB weiß, dass ' ASMGetCPUName Daten in str schreiben wird Delegate Sub ASMGetCPUName(<Out()> ByVal str() As Char)
Sub Main() ' ; Dieser ASM Code soll ausgeführt werden: ' ' push ebx ' xor eax, eax ' cpuid ' mov eax, [esp+8] ' mov [eax+0], ebx ' mov [eax+4], edx ' mov [eax+8], ecx ' pop ebx ' ret 4 ' obiges Assembler Listing in Maschinencode übersetzt (mit Nasm) ' und in 4 Byte Blöcke (Integer) aufgeteilt Dim code() As Integer = {264253779, 608471970, -1994880760, _ 1216939088, 79846152, 0} ' Buffer für CPU Hersteller Name Dim name() As Char = New String(" "c, 12).ToCharArray() ' ausführbaren Speicher beim System beantragen Dim mem As IntPtr = VirtualAlloc(Nothing, code.Length, _ MEM_COMMIT, PAGE_EXEC_RW) If mem.ToInt32 = 0 Then MsgBox("Fehler bei Aufruf von VirtualAlloc!", MsgBoxStyle.Exclamation) Else ' Delegate für Unmanaged Code erzeugen Dim fnc As ASMGetCPUName = CType( _ Marshal.GetDelegateForFunctionPointer(mem, _ GetType(ASMGetCPUName)), ASMGetCPUName) ' Code in Speicher kopieren Marshal.Copy(code, 0, mem, code.Length) ' Maschinencode ausführen fnc(name) ' Speicher wieder freigeben VirtualFree(mem, code.Length, MEM_DECOMMIT) MsgBox("CPU Name: " & name) End If End Sub
End Module