2
0
mirror of https://git.missingno.dev/kolibrios-nvme-driver/ synced 2025-01-05 04:15:56 +01:00

feat: implement support for multiple NVMe drives

This commit is contained in:
Abdur-Rahman Mansoor 2024-07-30 13:48:50 -04:00
parent 0d7f6b1c8f
commit 6262634324

View File

@ -62,28 +62,26 @@ proc START c, reason:dword
call detect_nvme call detect_nvme
test eax, eax test eax, eax
jz .err jz .err
mov eax, dword [p_nvme_devices] xor ebx, ebx
test eax, eax mov esi, dword [p_nvme_devices]
jz .err DEBUGF DBG_INFO, "p_nvme_devices: 0x%x\n", [p_nvme_devices]
xor ecx, ecx
.loop: .loop:
mov ebx, dword [p_nvme_devices] push ebx
stdcall device_is_compat, ebx lea esi, [esi + ebx * 4]
stdcall device_is_compat, esi
test eax, eax test eax, eax
jz @f jz @f
stdcall nvme_init, ebx stdcall nvme_init, esi
test eax, eax test eax, eax
jz .err jz @f
;@@:
;inc ecx
;cmp ecx, dword [pcidevs_len]
;jne .loop
stdcall add_nvme_disk, [p_nvme_devices]
test eax, eax
jz .err
@@:
pop ebx
inc ebx
cmp ebx, dword [pcidevs_len]
jne .loop
stdcall add_nvme_disk, dword [p_nvme_devices]
invoke RegService, my_service, service_proc invoke RegService, my_service, service_proc
ret ret
@ -149,7 +147,7 @@ proc add_nvme_disk stdcall, pci:dword
ret ret
@@: @@:
DEBUGF DBG_INFO, "nvme%u: Failed to register disk\n", [esi + pcidev.num] DEBUGF DBG_INFO, "nvme%un%u: Failed to register disk\n", [esi + pcidev.num], [esi + pcidev.nsid]
xor eax, eax xor eax, eax
pop esi pop esi
ret ret
@ -729,64 +727,58 @@ endp
proc detect_nvme proc detect_nvme
invoke GetPCIList invoke GetPCIList
mov edx, eax mov esi, eax
mov ebx, eax
.check_dev: .check_dev:
mov ebx, dword [eax + PCIDEV.class] mov eax, dword [esi + PCIDEV.class]
and ebx, 0x00ffff00 ; retrieve class/subclass code only and eax, 0x00ffff00 ; retrieve class/subclass code only
cmp ebx, 0x00010800 ; Mass Storage Controller - Non-Volatile Memory Controller cmp eax, 0x00010800 ; Mass Storage Controller - Non-Volatile Memory Controller
je .found_dev je .found_dev
.next_dev: .next_dev:
mov eax, dword [eax + PCIDEV.fd] mov esi, dword [esi + PCIDEV.fd]
cmp eax, edx cmp esi, ebx
jne .check_dev jne .check_dev
jmp .exit_success
.found_dev:
push edx eax
;PDEBUGF DBG_INFO, "PCI(%u.%u.%u): Detected NVMe device...\n", byte [eax + PCIDEV.bus], byte [eax + PCIDEV.devfn]
cmp dword [pcidevs_len], TOTAL_PCIDEVS
jne @f
pop eax edx
jmp .exit_success
@@:
inc dword [pcidevs_len]
mov ebx, dword [p_nvme_devices]
test ebx, ebx
jnz @f
invoke KernelAlloc, sizeof.pcidev
test eax, eax
jz .err_no_mem
mov dword [p_nvme_devices], eax
;DEBUGF DBG_INFO, "(NVMe) Allocated pcidev struct at 0x%x\n", [p_nvme_devices]
@@:
mov ecx, dword [pcidevs_len]
dec ecx
pop eax
mov ebx, dword [p_nvme_devices]
movzx edx, byte [eax + PCIDEV.bus]
mov byte [ebx + pcidev.bus], dl
movzx edx, byte [eax + PCIDEV.devfn]
mov byte [ebx + pcidev.devfn], dl
mov dword [ebx + pcidev.num], ecx
pop edx
jmp .next_dev
.err_no_mem:
pop eax edx
xor eax, eax
ret
.exit_success: .exit_success:
xor eax, eax xor eax, eax
inc eax inc eax
ret ret
.found_dev:
cmp dword [pcidevs_len], TOTAL_PCIDEVS
jne @f
DEBUGF DBG_INFO, "Can't add any more NVMe devices...\n"
jmp .exit_success
@@:
inc dword [pcidevs_len]
cmp dword [p_nvme_devices], 0
jnz @f ; was the pointer already allocated?
invoke KernelAlloc, sizeof.pcidev * TOTAL_PCIDEVS
test eax, eax
jz .err_no_mem
mov dword [p_nvme_devices], eax
@@:
mov ecx, dword [pcidevs_len]
dec ecx
mov edi, dword [p_nvme_devices]
lea edi, [edi + ecx * 4]
movzx eax, byte [esi + PCIDEV.bus]
mov byte [edi + pcidev.bus], al
movzx eax, byte [esi + PCIDEV.devfn]
mov byte [edi + pcidev.devfn], al
mov dword [edi + pcidev.num], ecx
jmp .next_dev
.err_no_mem:
xor eax, eax
ret
endp endp
proc device_is_compat stdcall, pci:dword proc device_is_compat stdcall, pci:dword
@ -1416,28 +1408,37 @@ endp
proc irq_handler proc irq_handler
push esi edi push ebx esi edi
mov esi, dword [p_nvme_devices] mov edi, dword [p_nvme_devices]
mov esi, edi
sub esi, sizeof.pcidev
mov ebx, dword [pcidevs_len]
xor ecx, ecx
; check if a NVMe device generated an interrupt .check_who_raised_irq:
add esi, sizeof.pcidev
inc ecx
cmp ecx, ebx
ja .not_our_irq
invoke PciRead16, dword [esi + pcidev.bus], dword [esi + pcidev.devfn], PCI_header00.status invoke PciRead16, dword [esi + pcidev.bus], dword [esi + pcidev.devfn], PCI_header00.status
test al, 1000b ; check interrupt status test al, 1000b ; check interrupt status
jz .not_our_irq jz .check_who_raised_irq
mov esi, edi
mov edi, dword [esi + pcidev.io_addr] mov edi, dword [esi + pcidev.io_addr]
mov dword [edi + NVME_MMIO.INTMS], 0x3 mov dword [edi + NVME_MMIO.INTMS], 0x3
mov eax, dword [spinlock] mov eax, dword [spinlock]
test eax, eax test eax, eax
jz @f ; not locked, so it must be an I/O command jz @f ; not locked, so it must be an I/O command
stdcall consume_cq_entries, [p_nvme_devices], 0 stdcall consume_cq_entries, esi, 0
@@: @@:
stdcall consume_cq_entries, [p_nvme_devices], 1 stdcall consume_cq_entries, esi, 1
; Interrupt handled by driver, return 1 ; Interrupt handled by driver, return 1
mov dword [edi + NVME_MMIO.INTMC], 0x3 mov dword [edi + NVME_MMIO.INTMC], 0x3
pop edi esi pop edi esi ebx
xor eax, eax xor eax, eax
xchg eax, dword [spinlock] ; unlock spinlock xchg eax, dword [spinlock] ; unlock spinlock
mov eax, 1 mov eax, 1
@ -1445,7 +1446,7 @@ proc irq_handler
.not_our_irq: .not_our_irq:
; Interrupt not handled by driver, return 0 ; Interrupt not handled by driver, return 0
pop edi esi pop edi esi ebx
xor eax, eax xor eax, eax
ret ret