Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Vulkan: Destroy Vulkan instance on exit #10989

Open
wants to merge 2 commits into
base: master
Choose a base branch
from

Conversation

0cc4m
Copy link
Collaborator

@0cc4m 0cc4m commented Dec 26, 2024

This seems to fix the Nvidia driver segfault on exit (#10528). If someone can reproduce it reliably, please test if this resolves it.

It's a little more hacky than I had hoped, but I couldn't think of a better way to check when the instance can be destroyed.

Basically the backend is now counting backend contexts and backend (device) buffer allocations and when a backend or buffer is destroyed checks whether any are left. If not, it destroys the instance. This seems to happen early enough that it avoids the issue with the Nvidia driver. Usually it should happen on a backend unload or model unload, depending on which happens last.

Let me know if you see any issues, I had to touch a lot of code to remove references to the devices that would prevent them and thus the instance from being destroyed.

@github-actions github-actions bot added Vulkan Issues specific to the Vulkan backend ggml changes relating to the ggml tensor library for machine learning labels Dec 26, 2024
@jeffbolznv
Copy link
Collaborator

Have you tried this on Windows? What I was seeing when I looked into this recently is that the static destructors in ggml-vulkan are run after the Vulkan driver is unloaded, so things crash if you try to call into the driver at all then. I was specifically seeing it crash when vkDestroyDevice is called.

@0cc4m
Copy link
Collaborator Author

0cc4m commented Dec 26, 2024

Have you tried this on Windows? What I was seeing when I looked into this recently is that the static destructors in ggml-vulkan are run after the Vulkan driver is unloaded, so things crash if you try to call into the driver at all then. I was specifically seeing it crash when vkDestroyDevice is called.

No, I don't have a Windows setup. But this does not rely on the static destructors on exit, it should destroy the devices and instance once the controlling program unloaded the backend and freed the device buffers (unloaded the model) At least for the examples I tested this seems to fix the segfault on Linux.

@jeffbolznv
Copy link
Collaborator

Cool! I fetched your change and I'm not seeing the crash - the backend is indeed freed before the process is terminated.

@jeffbolznv
Copy link
Collaborator

I've had this running overnight on a Linux system and no crashes so far. But the previous repro rate was maybe once a day, so not definitive. I can keep running it for a couple more days.

@LostRuins
Copy link
Collaborator

LostRuins commented Dec 28, 2024

Unfortunately, this did not solve the BSOD issue for me. Loading the models was fine, but upon unload I got a VIDEO_MEMORY_MANAGEMENT_INTERNAL BSOD again. Again, it only happens whenever I offload enough layers that available VRAM is nearly depleted (> 90% vram utilization) otherwise it will not BSOD.

(sorry for poor quality)

image

and here's BlueScreenView's info for this minidump

image

@jeffbolznv
Copy link
Collaborator

IMO the VIDEO_MEMORY_MANAGEMENT_INTERNAL error is likely unrelated to the crash in driver threads.

@LostRuins
Copy link
Collaborator

Do you think it's something that can be fixed on the llama.cpp side? Or is it a bug in the graphics driver? I'm running it in user mode without admin permissions, so it shouldn't be able to trigger a BSOD under normal circumstances correct?

@jeffbolznv
Copy link
Collaborator

Do you think it's something that can be fixed on the llama.cpp side? Or is it a bug in the graphics driver? I'm running it in user mode without admin permissions, so it shouldn't be able to trigger a BSOD under normal circumstances correct?

I'd guess it's one of: OS bug, kernel driver bug, or hardware failure (e.g. memory corruption).

@jeffbolznv
Copy link
Collaborator

My borrowed Linux system has been running for 3 days now with no failures.

@0cc4m
Copy link
Collaborator Author

0cc4m commented Dec 29, 2024

I also haven't seen any segfaults with it and I get them quite often. I'll rebase it, then it's probably good to merge, unless you see some issue with the code.

@0cc4m 0cc4m force-pushed the 0cc4m/vulkan-instance-cleanup branch from 01ab364 to d9b0958 Compare December 29, 2024 20:41

~vk_buffer_struct() {
if (size == 0) {
if (size == 0 || device_ref.expired()) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If we get here with an expired pointer, does that indicate that we failed to free the VkBuffer/VkDeviceMemory? Is there any way to avoid that?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This case happens in the vk_device_struct deconstructor, there is still the sync_staging buffer around. But at that point the weak pointer is invalid, so I added a ggml_vk_destroy_buffer overload that uses the vk::Device directly and added this check. There might be a better way to handle that.


vk_instance_buffers--;

if (vk_instance_contexts == 0 && vk_instance_buffers == 0) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If these counters really need to be atomic, then this check should use the return value from the decrement rather than separately reading the value, to avoid the situation where two threads decrement and then both read zero. But with two counters, it gets more complex and maybe a mutex would be better?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

They don't. In the beginning I had only one counter, so it made more sense. I replaced them with a mutex now.

There's still the open issue about making the Vulkan backend multithread-capable, which needs some refactoring. I'm trying not to make the situation worse when I add new features.

@ggerganov
Copy link
Owner

Hm, ggml-ci still seems to segfault using this PR. Here is one of the tests that fails on exit:

ggml@ggml-6-x86-vulkan-t4:~/work/llama.cpp/build-ci-debug$ ./bin/test-arg-parser 
test-arg-parser: make sure there is no duplicated arguments in any examples

ggml_vulkan: Found 1 Vulkan devices:
ggml_vulkan: 0 = Tesla T4 (NVIDIA) | uma: 0 | fp16: 1 | warp size: 32 | matrix cores: KHR_coopmat
register_backend: registered backend Vulkan (1 devices)
register_device: registered device Vulkan0 (Tesla T4)
register_backend: registered backend CPU (1 devices)
register_device: registered device CPU (AMD EPYC 7V12 64-Core Processor)
test-arg-parser: test invalid usage

error while handling argument "-m": expected value for argument

usage:
-m,    --model FNAME                    model path (default: `models/$filename` with filename from `--hf-file`
                                        or `--model-url` if set, otherwise models/7B/ggml-model-f16.gguf)
                                        (env: LLAMA_ARG_MODEL)


to show complete usage, run with -h
error while handling argument "-ngl": stoi

usage:
-ngl,  --gpu-layers, --n-gpu-layers N   number of layers to store in VRAM
                                        (env: LLAMA_ARG_N_GPU_LAYERS)


to show complete usage, run with -h
error while handling argument "-sm": invalid value

usage:
-sm,   --split-mode {none,layer,row}    how to split the model across multiple GPUs, one of:
                                        - none: use one GPU only
                                        - layer (default): split layers and KV across GPUs
                                        - row: split rows across GPUs
                                        (env: LLAMA_ARG_SPLIT_MODE)


to show complete usage, run with -h
error: invalid argument: --draft
test-arg-parser: test valid usage

test-arg-parser: test environment variables (valid + invalid usages)

error while handling environment variable "LLAMA_ARG_THREADS": stoi


test-arg-parser: test environment variables being overwritten

warn: LLAMA_ARG_MODEL environment variable is set, but will be overwritten by command line argument -m
test-arg-parser: all tests OK

Segmentation fault (core dumped)
...

@0cc4m
Copy link
Collaborator Author

0cc4m commented Jan 2, 2025

I haven't merged this yet because I can reproduce the crash reported in #10528 (comment) , but couldn't figure out why it occurs, yet.

@ggerganov
Copy link
Owner

For the time-being, I will turn off the ggml-org / ggml-6-x86-vulkan-t4 node because the runs always fail due to this segfault. We can bring it back up when the issue is resolved.

@0cc4m
Copy link
Collaborator Author

0cc4m commented Jan 22, 2025

For the time-being, I will turn off the ggml-org / ggml-6-x86-vulkan-t4 node because the runs always fail due to this segfault. We can bring it back up when the issue is resolved.

Fair enough, this is a really annoying bug. It only happens on Nvidia, so a temporary workaround may be using an AMD node (if this is available) or run the tests on CPU using llvmpipe.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
ggml changes relating to the ggml tensor library for machine learning Vulkan Issues specific to the Vulkan backend
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants