gpt4 book ai didi

c# - 加载 native DLL 的引用中的 DllImport

转载 作者:行者123 更新时间:2023-11-30 22:36:45 31 4
gpt4 key购买 nike

我使用的是使用 VS2008 构建的第三方 dll,它依赖于 msvcr90.dll 打印到 stdout。我尝试使用 SetStdHandle 重定向此输出,但 dll 似乎错过了在返回之前刷新输出缓冲区的最终调用。

简单使用

[DllImport("msvcr90.dll")]

导致 DllNotFoundException。

为了解决这个问题,我从驻留在我的 WinSxS 文件夹中的 msvcr90.dll PInvoke fflush()__iob_func()。我通过设置 dll 的完整路径来做到这一点(我在第三方 dll 上使用 DependencyWalker 获得了路径)。如果我想将我的程序提供给其他人或者 msvcr90.dll 由 Microsoft 更新,那么当然不是很有用。

谁能给我一个提示,如何告诉 DllImport 使用最新版本的 msvcr90.dll

感谢您的回复!

安德烈亚斯

我的解决方案

在 David 的帮助下,我想出了以下解决方案。 msvcr90.dll 不是通过 WinSxS 定位的,而是从加载的模块列表中获取的。

using System;
using System.Diagnostics;
using System.Linq;
using System.Runtime.InteropServices;

namespace siemens.OptiX
{
static unsafe class FlushPrintBuffer
{
struct CFile
{
char* _ptr;
int _cnt;
char* _base;
int _flag;
public int _file;
int _charbuf;
int _bufsiz;
char* _tmpfname;
}

[DllImport("kernel32", CharSet = CharSet.Ansi, ExactSpelling = true, SetLastError = true)]
static extern IntPtr GetProcAddress(IntPtr hModule, string procName);

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
private delegate CFile* iobFuncDelegate();

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
private delegate int flushDelegate(CFile* stream);

static iobFuncDelegate myIobFunc;
static flushDelegate myFlush;

static FlushPrintBuffer()
{
ProcessModuleCollection allLoadedModules =
Process.GetCurrentProcess().Modules;

IntPtr aDllHandle = allLoadedModules
.Cast<ProcessModule>()
.Where(p => string.Compare(p.ModuleName, "msvcr90.dll", true) == 0)
.First()
.BaseAddress;

IntPtr aIobFuncPtr = GetProcAddress(aDllHandle, "__iob_func");
myIobFunc = Marshal.GetDelegateForFunctionPointer(
aIobFuncPtr, typeof(iobFuncDelegate)) as iobFuncDelegate;

IntPtr aFlushPtr = GetProcAddress(aDllHandle, "fflush");
myFlush = Marshal.GetDelegateForFunctionPointer(
aFlushPtr, typeof(flushDelegate)) as flushDelegate;
}

static public void flushNow()
{
CFile* aFilePtr = myIobFunc();
int aRes = myFlush(&(aFilePtr[1]));
}
}
}

最佳答案

我认为您会发现很难使用 DLLImport 链接到正确的运行时。这是因为您的可执行文件将缺少使 WinSxS 神奇工作所需的 list 。

我会为您需要导入的函数声明一些委托(delegate)。使用 GetProcAddress 获取函数指针。使用 Marshal.GetDelegateForFunctionPointer 将函数指针转换为委托(delegate).

为了让模块句柄传递给 GetProcAddress,您可以使用 GetModuleHandle(@"msvcr90.dll"),因为您知道它已加载到您的进程中.

通过 P/invoke 获取 GetProcAddressGetModuleHandle


调用 GetModuleHandle(@"msvcr90.dll") 似乎不起作用。使用完整路径调用 GetModuleHandle 是可行的,但这并不是真正可行的解决方案。

相反,像这样枚举加载的模块很容易:

IntPtr MyGetModuleHandleByPartialName(string ModuleName)
{
foreach (ProcessModule module in Process.GetCurrentProcess().Modules)
if (module.FileName.ToLower().Contains(ModuleName))
return module.BaseAddress;
return IntPtr.Zero;
}

关于c# - 加载 native DLL 的引用中的 DllImport,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6840065/

31 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com