c++dllmingwmingw-w64msys2

How to debug DLL issues in MinGW?


When using MinGW, it's easy to run into a DLL hell, i.e. your program not running because it either doesn't find the required DLLs or finds incompatible versions of them.

This can manifest itself as:

How do I fix this?


(This is intended as a canonical duplicate on this topic.)


Solution

  • What do the errors mean?

    Your program requires several .dll ("shared libraries") to run. Those are either parts of the standard library (distributed with your compiler), or third-party libraries you're using.

    Where does my program look for the DLLs?

    You can refer to the manual, but a short summary is: (the search happens in this order)

    How to fix?

    There are several options, from simplest to harderst, pick one.

    1. Static linking

      Add -static to your linker flags. This makes your program not depend on the DLLs in the first place, embedding their contents into the executable.

      Why is it not good?

      • Your users can't easily update the libraries to newer versions if they need to.

      • Some libraries are licensed in a way that hinders this. (E.g. with LGPL license, linking statically requires to open-source your software.)

    2. Run from MSYS2 terminal (if you're using MSYS2)

      The MSYS2 terminal automatically adjusts its own PATH to have the compiler installation as the first thing in there (which includes the standard library DLLs you likely have problems with).

      There are several MSYS2 shortcuts in the start menu, and you MUST pick the correct one (usually UCRT64 or MINGW64, but read What are MSYS2 environments? How do I pick one? to know for sure).

      While this fixes the problem for you, you can't expect your users to install MSYS2, so you should look into other solutions in the long term.

      If this didn't help, you likely didn't use the correct MSYS2 shortcut. You can also consult "What not to do" below, the part about junk in C:\Windows.

    3. Fix your PATH

      Modify your PATH to include the directory with the DLLs. Usually it will be the compiler installation directory (since that's where its standard library is), e.g. C:\path\to\MinGW\bin (substitute your path). If you're using MSYS2, this will be C:\msys64\ucrt64\bin or C:\msys64\mingw64\bin (read What are MSYS2 environments? How do I pick one? to know which directory to choose).

      (This only makes sense for the standard library DLLs. For third-party libraries, always copy them next to the executable as explained in (4) below.)

      This directory must be the first thing in the PATH. Otherwise you risk picking up incompatible versions of the same DLLs from preceding directories.

      There are two PATH settings, system-wide and user-specific. The former has priority, so use the system-wide PATH.

      While this fixes the problem for you, you can't expect your users to install MinGW, so you should look into other solutions in the long term.

      How do I set PATH?

      • In the settings, type "env" into the search box, click Edit the system environment variables, click Environment Variables... at the bottom, then in the second list titled System variables double-click Path, type your path (e.g. C:\path\to\MinGW\bin) into an empty cell at the bottom, click Move Up repeatedly to make it the first entry, then click OK in all the open dialogues.

      • Don't forget to restart the console/IDE after changing PATH!

      If this doesn't work and you're sure you did everything correctly, consult "What not to do" below, the part about junk in C:\Windows.

    4. Copy the required DLLs next to the executable

      Copying DLLs is the best long-term solution. You need to do this anyway to distribute your app to others (this or link statically, see above). The next section explains how to do this.

    Which DLLs should I copy next to the executable?

    The required DLLs could be mentioned in the error message, but:

    There are tools that will tell you the list of DLLs your app depends on, e.g. ntldd. It is dated, but gets the job done.

    Note: To do this, you need to have PATH set up correctly (see above; or be running in the MSYS2 terminal as explained above), so you should be already able to run your app. And you must not use -static.

    MSYS2 users can install ntldd using pacman -S ...-ntldd (e.g. mingw-w64-ucrt-x86_64-ntldd or mingw-w64-x86_64-ntldd, consult What are MSYS2 environments? How do I pick one?). If you don't use MSYS2, download from the GitHub link above.

    Run ntldd -R my_program.exe. You will see a long list. Ignore anything that's not in your compiler directory (and isn't a third-party library you installed separately). In MSYS2 terminal you can run ntldd -R my_program.exe | grep 'C:\\msys64' to filter automatically.

    I ended up with libstdc++-6.dll, libgcc_s_seh-1.dll, libwinpthread-1.dll, but your list can be different depending on the type of MinGW you use.

    Copy those DLLs next to your executable, and you should be done. Writing a script to automate this is left as an exercise to the reader.

    What not to do?