BUG: Assert When Calling AfxFreeLibrary from ExitInstance (187684)
The information in this article applies to:
- Microsoft Visual C++ 2.0, when used with:
- the operating system: Microsoft Windows 95
- Microsoft Visual C++ 2.1, when used with:
- the operating system: Microsoft Windows 95
- Microsoft Visual C++ 2.2, when used with:
- the operating system: Microsoft Windows 95
- Microsoft Visual C++ 4.0, when used with:
- the operating system: Microsoft Windows 95
- Microsoft Visual C++ 4.1, when used with:
- the operating system: Microsoft Windows 95
- Microsoft Visual C++ 4.2, when used with:
- the operating system: Microsoft Windows 95
This article was previously published under Q187684 SYMPTOMS
The following assertion in Afxcrit.cpp line 73 can fail when you call the
AfxFreeLibrary function from the ExitInstance function of an MFC DLL:
ASSERT(!_afxResourceLocked[i]);
Similarly, an access violation or an unhandled exception may occur when you
call FreeLibrary from ExitInstance or the DllMain function during
DLL_PROCESS_DETACH.
CAUSE
The Windows 95 loader is not reentrant. Causing a DLL to be freed while in
the process of freeing another DLL can cause a dependent DLL to be freed
unexpectedly.
For example, if a DLL (A) with a dependant DLL (B) and a reference count of
0 calls FreeLibrary from DLL_PROCESS_DETACH on another DLL (C), the Windows
95 unloader rescans the dependent DLL list and prematurely releases the
dependent DLL (B). The DLL (A) that calls FreeLibrary is protected because
it is marked. Upon returning from FreeLibrary, DLL (A) might make further
calls into DLL (B) and cause the assertion or access violation.
RESOLUTION
Don't call AfxFreeLibrary from ExitInstance. Likewise, don't call
FreeLibrary from either ExitInstance or from DLL_PROCESS_DETACH of the
DllMain function.
You can work around this bug by providing initialization and
uninitialization functions that are not called from within DllMain. For
example, if you use an ActiveX DLL, you can call AfxLoadLibrary and
AfxFreeLibrary from the COM object's constructor and OnFinalRelease
function. Calling AfxFreeLibrary from a destructor of a static object does
not work because the static object constructors and destructors are called
from DllMain.
An alternative workaround is to increment the reference count of the DLL
with a 0 reference by calling (Afx)LoadLibrary again.
STATUS
Microsoft has confirmed this to be a bug in Windows 95. We are researching
this bug and will post new information here in the Microsoft Knowledge Base
as it becomes available.
MORE INFORMATION
In MFC, DLLs are dynamically loaded and unloaded using AfxLoadLibrary and
AfxFreeLibrary, respectively. A common mistake when using AfxFreeLibrary
(or FreeLibrary) is to call it from ExitInstance of an MFC DLL. A similar
problem occurs when you call FreeLibrary from DllMain with
DLL_PROCESS_DETACH.
When you call FreeLibrary from DllMain during DLL_PROCESS_DETACH under
Windows 95, the entire dependency tree is reevaluated. If any DLL other
than the one currently calling FreeLibrary has a reference count of 0, it
will be unloaded. In some cases, this can cause problems if there are still
calls to the unloaded DLL.
A common example is when the CRT libraries are prematurely unloaded. To
work around this problem, you can make an additional call to LoadLibrary
for the CRT libraries before you call FreeLibrary.
Modification Type: | Major | Last Reviewed: | 12/10/2003 |
---|
Keywords: | kbBug kbDLL kbpending KB187684 |
---|
|