LoadLibraryEx配合LOAD_WITH_ALTERED_SEARCH_PATH不生效

dahaiI0 2020-04-15 03:37:50
有A.exe和BC2个dll,其中A.exe加载B.dll,B.dll又加载C.dll,现在要把BC这2个dll放到另一个目录下,搜索了下发现LoadLibraryEx可以实现此功能,但实际并没有生效,比如我把BC放到F盘根目录下

HMODULE hDll = ::LoadLibraryEx(_T("F:\\B.dll"), NULL, LOAD_WITH_ALTERED_SEARCH_PATH);

然后就发现C这个DLL加载不起来了,有大佬指点下?
...全文
1268 13 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
13 条回复
切换为时间正序
请发表友善的回复…
发表回复
dahaiI0 2020-04-16
  • 打赏
  • 举报
回复
引用 12 楼 zgl7903 的回复:
GetCurrentDirectory 记录当前路径 SetCurrentDirectory 设置目标路径 加载DLL SetCurrentDirectory 恢复路径 或者 Dll main入口函数处记录 HMODULE 入口地址, GetModuleFileName 可以获取模块文件路径名,就可以构造出绝对路径了
GetCurrentDirectory 这个不可行的,dll6大搜索路径,第一个就是exe所在路径,如果exe目录下也有这个dll,会优先加载exe目录下的dll。 loadlibraryex这个发现我用法用错了,我以为DLL2的加载也会从DLL1目录开始搜索,实际是loadlibraryex只对静态链接生效,举个例子如果DLL1静态链接DLL3,用loadlibraryex会加载DLL1目录下的DLL3,而如果是loadlibrary则会加载exe目录下的DLL3。至于动态链接的,感觉必须指定全路径(因为exe搜索路径优先,什么环境变量,SetCurrentDirectory ,setdlldirectory都没用,只有在exe下没有这个DLL时上述方法才有效果)。 晚点结贴,留个备注供后续人参考
zgl7903 2020-04-16
  • 打赏
  • 举报
回复
GetCurrentDirectory 记录当前路径 SetCurrentDirectory 设置目标路径 加载DLL SetCurrentDirectory 恢复路径 或者 Dll main入口函数处记录 HMODULE 入口地址, GetModuleFileName 可以获取模块文件路径名,就可以构造出绝对路径了
dahaiI0 2020-04-16
  • 打赏
  • 举报
回复
引用 10 楼 zgl7903 的回复:
没有静态链接啊, 生成的是两个DLL和一个显式加载调用的exe 你指的动态链接是说 LoadLibrary 显式加载?
嗯,我指的静态链接是DLLA对DLLB的加载方式,如果是静态的,好像默认是当前目录,如果是显示动态(loadlibrary),会优先加载exe所在目录,但我的目的是想让他优先加载DLLA所在的目录下的DLLB
zgl7903 2020-04-16
  • 打赏
  • 举报
回复
没有静态链接啊, 生成的是两个DLL和一个显式加载调用的exe 你指的动态链接是说 LoadLibrary 显式加载?
dahaiI0 2020-04-15
  • 打赏
  • 举报
回复
引用 8 楼 zgl7903 的回复:
ERROR_CHILD_NOT_COMPLETE 129 The %1 application cannot be run in Win32 mode. 我测试的可以的,

// DllB.cpp : 定义 DLL 应用程序的导出函数。
//
#include "stdafx.h"
#include <stdio.h>

extern "C" __declspec( dllexport ) void __cdecl funB()
{
  printf("%s was called\n", __FUNCDNAME__);
}


// DllA.cpp : 定义 DLL 应用程序的导出函数。

#include "stdafx.h"
#include <stdio.h>

#pragma comment(lib, "../debug/DllB.lib")
extern "C" void __cdecl funB();
extern "C" __declspec( dllexport ) void __cdecl funA()
{
  printf("%s was called\n", __FUNCDNAME__);
  funB();
}


// Win32Dll.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <windows.h>
#include <crtdbg.h>

int _tmain(int argc, _TCHAR* argv[])
{
  HMODULE hDll = LoadLibrary(_T("d:\\temp\\DllA.dll")); //_T("DllA.dll")
  if(hDll != NULL)
  {
    typedef void (__cdecl *FUNA)();
    FUNA pf = (FUNA)GetProcAddress(hDll, "funA");
    if(pf)
    {
      pf();
    }

    FreeLibrary(hDll);
  }
  else
  {
    printf("LoadLibrary failed %u\n", GetLastError());
    _ASSERT(0);
  }

	return 0;
}

静态链接好像是可以,动态链接不行
zgl7903 2020-04-15
  • 打赏
  • 举报
回复
ERROR_CHILD_NOT_COMPLETE
129
The %1 application cannot be run in Win32 mode.

我测试的可以的,


// DllB.cpp : 定义 DLL 应用程序的导出函数。
//
#include "stdafx.h"
#include <stdio.h>

extern "C" __declspec( dllexport ) void __cdecl funB()
{
printf("%s was called\n", __FUNCDNAME__);
}




// DllA.cpp : 定义 DLL 应用程序的导出函数。

#include "stdafx.h"
#include <stdio.h>

#pragma comment(lib, "../debug/DllB.lib")
extern "C" void __cdecl funB();
extern "C" __declspec( dllexport ) void __cdecl funA()
{
printf("%s was called\n", __FUNCDNAME__);
funB();
}




// Win32Dll.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <windows.h>
#include <crtdbg.h>

int _tmain(int argc, _TCHAR* argv[])
{
HMODULE hDll = LoadLibrary(_T("d:\\temp\\DllA.dll")); //_T("DllA.dll")
if(hDll != NULL)
{
typedef void (__cdecl *FUNA)();
FUNA pf = (FUNA)GetProcAddress(hDll, "funA");
if(pf)
{
pf();
}

FreeLibrary(hDll);
}
else
{
printf("LoadLibrary failed %u\n", GetLastError());
_ASSERT(0);
}

return 0;
}



dahaiI0 2020-04-15
  • 打赏
  • 举报
回复
引用 5 楼 zgl7903 的回复:
GetLastError() 看看错误信息
126.
引用 6 楼 schlafenhamster 的回复:
如果设定了LOAD_WITH_ALTERED_SEARCH_PATH标志,那么LoadLibraryEx函数就按照下面的顺序来搜索文件: 1)pszDLLPathName参数中设定的目录。 2)进程的当前目录。 3)Windows的系统目录。 4)Windows目录。 5)PATH环境变量中列出的目录。
这个博客看过了,但没生效,我还特意看了LOAD_WITH_ALTERED_SEARCH_PATH这个标志也不需要安装KBXXX
schlafenhamster 2020-04-15
  • 打赏
  • 举报
回复
如果设定了LOAD_WITH_ALTERED_SEARCH_PATH标志,那么LoadLibraryEx函数就按照下面的顺序来搜索文件:
1)pszDLLPathName参数中设定的目录。
2)进程的当前目录。
3)Windows的系统目录。
4)Windows目录。
5)PATH环境变量中列出的目录。
zgl7903 2020-04-15
  • 打赏
  • 举报
回复
GetLastError() 看看错误信息
dahaiI0 2020-04-15
  • 打赏
  • 举报
回复
引用 2 楼 zgl7903 的回复:
B 和 C 是否还有其它的依赖项? Depends查看下 加载前,试试 SetCurrentDirectory 修改当前路径
SetCurrentDirectory 有2个问题,首先是有副作用会修改目录,其次我想优先加载其他目录的dll用SetCurrentDirectory 仍然是无法实现的
dahaiI0 2020-04-15
  • 打赏
  • 举报
回复
引用 2 楼 zgl7903 的回复:
B 和 C 是否还有其它的依赖项? Depends查看下 加载前,试试 SetCurrentDirectory 修改当前路径
没有了,就写了个简单的demo,放在同一个目录下(就3个文件)能加载起来。
zgl7903 2020-04-15
  • 打赏
  • 举报
回复
B 和 C 是否还有其它的依赖项? Depends查看下 加载前,试试 SetCurrentDirectory 修改当前路径
dahaiI0 2020-04-15
  • 打赏
  • 举报
回复
其他方案就不考虑了,什么每次使用绝对路径,setdlldirectory(xp如果没有sp1补丁包就不支持),修改环境变量之类的。

15,473

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC 进程/线程/DLL
社区管理员
  • 进程/线程/DLL社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧