Windows Driver Model >> Assembly >> Synchronization
Spin Lock
參考資訊:
1. Source Code
在多核心CPU環境中,可能有些程式區段,每次只能讓一顆CPU進入執行,這時候可以使用Spin Lock機制,Spin Lock一次只能有一個Lock,因此,當有一顆CPU取得Lock後,其餘CPU便無法取得Lock,在無法取得Lock狀況下,這些CPU會不斷嘗試取得Lock,因此,Spin Lock對於CPU資源的開銷是很大的,在寫Spin Lock程式時,必須注意不能Lock太久,否則將導致系統效能下降,同時,在取得Lock後,將會提昇到DISPATCH_LEVEL執行,使用步驟如下:
1. KeInitializeSpinLock()
2. KeAcquireSpinLock()
3. KeReleaseSpinLock()
main.asm
.386p .model flat, stdcall option casemap:none include c:\masm32\include\w2k\hal.inc include c:\masm32\include\w2k\ntstatus.inc include c:\masm32\include\w2k\ntddk.inc include c:\masm32\include\w2k\ntoskrnl.inc include c:\masm32\include\w2k\ntddkbd.inc include c:\masm32\Macros\Strings.mac includelib c:\masm32\lib\wxp\i386\hal.lib includelib c:\masm32\lib\wxp\i386\ntoskrnl.lib public DriverEntry MAX_THREAD equ 3 .data myLock KSPIN_LOCK 0 pNextDev PDEVICE_OBJECT 0 .const DEV_NAME word "\","D","e","v","i","c","e","\","M","y","D","r","i","v","e","r",0 SYM_NAME word "\","D","o","s","D","e","v","i","c","e","s","\","M","y","D","r","i","v","e","r",0 MSG_LOCKING byte "Thread%d, Locking",0 MSG_LOCKED byte "Thread%d, Locked",0 MSG_UNLOCKING byte "Thread%d, Unlocking",0 MSG_UNLOCKED byte "Thread%d, Unlocked",0 .code RunMe proc t:DWORD local oldirql:KIRQL local c0:DWORD local c1:DWORD invoke DbgPrint, offset MSG_LOCKING, t invoke KeAcquireSpinLock, offset myLock, addr oldirql invoke DbgPrint, offset MSG_LOCKED, t mov c0, 10000 d0: mov c1, 10000 d1: dec c1 jnz d1 dec c0 jnz d0 invoke DbgPrint, offset MSG_UNLOCKING, t invoke KeReleaseSpinLock, offset myLock, oldirql invoke DbgPrint, offset MSG_UNLOCKED, t ret RunMe endp MyThread proc pParam:DWORD local stTime:LARGE_INTEGER or stTime.HighPart, -1 mov stTime.LowPart, -10000000 invoke KeDelayExecutionThread, KernelMode, FALSE, addr stTime invoke RunMe, pParam invoke PsTerminateSystemThread, STATUS_SUCCESS ret MyThread endp IrpPnp proc pDevObj:PDEVICE_OBJECT, pIrp:PIRP local pdx:PTR OurDeviceExtension local szSymName:UNICODE_STRING mov eax, pDevObj push (DEVICE_OBJECT PTR [eax]).DeviceExtension pop pdx IoGetCurrentIrpStackLocation pIrp movzx eax, (IO_STACK_LOCATION PTR [eax]).MinorFunction .if eax == IRP_MN_START_DEVICE mov eax, pIrp mov (_IRP PTR [eax]).IoStatus.Status, STATUS_SUCCESS .elseif eax == IRP_MN_REMOVE_DEVICE invoke RtlInitUnicodeString, addr szSymName, offset SYM_NAME invoke IoDeleteSymbolicLink, addr szSymName mov eax, pIrp mov (_IRP PTR [eax]).IoStatus.Status, STATUS_SUCCESS mov eax, pdx invoke IoDetachDevice, pNextDev invoke IoDeleteDevice, pDevObj .endif IoSkipCurrentIrpStackLocation pIrp mov eax, pdx invoke IoCallDriver, pNextDev, pIrp ret IrpPnp endp AddDevice proc pOurDriver:PDRIVER_OBJECT, pPhyDevice:PDEVICE_OBJECT local pOurDevice:PDEVICE_OBJECT local suDevName:UNICODE_STRING local szSymName:UNICODE_STRING invoke RtlInitUnicodeString, addr suDevName, offset DEV_NAME invoke RtlInitUnicodeString, addr szSymName, offset SYM_NAME invoke IoCreateDevice, pOurDriver, 0, addr suDevName, FILE_DEVICE_UNKNOWN, 0, FALSE, addr pOurDevice .if eax == STATUS_SUCCESS invoke IoAttachDeviceToDeviceStack, pOurDevice, pPhyDevice .if eax != NULL push eax pop pNextDev mov eax, pOurDevice or (DEVICE_OBJECT PTR [eax]).Flags, DO_BUFFERED_IO and (DEVICE_OBJECT PTR [eax]).Flags, not DO_DEVICE_INITIALIZING invoke IoCreateSymbolicLink, addr szSymName, addr suDevName .endif .endif ret AddDevice endp Unload proc pOurDriver:PDRIVER_OBJECT ret Unload endp DriverEntry proc pOurDriver:PDRIVER_OBJECT, pOurRegistry:PUNICODE_STRING local hThread:DWORD local cnt:DWORD mov eax, pOurDriver mov (DRIVER_OBJECT PTR [eax]).MajorFunction[IRP_MJ_PNP * (sizeof PVOID)], offset IrpPnp mov (DRIVER_OBJECT PTR [eax]).DriverUnload, offset Unload mov eax, (DRIVER_OBJECT PTR [eax]).DriverExtension mov (DRIVER_EXTENSION PTR [eax]).AddDevice, AddDevice invoke KeInitializeSpinLock, offset myLock mov cnt, 0 th: invoke PsCreateSystemThread, addr hThread, THREAD_ALL_ACCESS, NULL, -1, NULL, offset MyThread, cnt .if eax == STATUS_SUCCESS invoke ZwClose, hThread .endif inc cnt cmp cnt, MAX_THREAD jnz th mov eax, STATUS_SUCCESS ret DriverEntry endp end DriverEntry .end
DriverEntry()產生3個Thread,每個Thread延遲1秒,接著呼叫RunMe(),在RunMe()裡面會進行Spin Lock的動作,確保每次只能有一個CPU進入執行
使用四顆CPU測試
每次只會有一顆CPU進入執行