如何防止非托管dll调用访问冲突?

时间:2009-10-01 12:54:24

标签: c# .net windows unmanaged

我们继承了一种传统系统,用于读取和从电子枪中读取数据。该系统最初是在带有.Net 1.1(VS2003?)的XP上构建的。在VS2008上使用.net 3.5重新编译它时,我们在调用dll时遇到访问冲突(未触及dll)。原始程序(使用基本相同的代码)在我们的生产机器上正常运行。

崩溃:

[System.AccessViolationException]
Attempted to read or write protected memory. This is often an indication that other memory is corrupt.

代码:

[DllImport("tinydb.dll",CallingConvention=CallingConvention.Cdecl)]
static extern Int16 db_fillnew(Int16 recno,IntPtr buffer,Int16 dbn);
     :
     :
    IntPtr buffer = Marshal.AllocHGlobal(1024); 
     :
     :
foreach (Round round in roundList)
{
    RoundRec roundRec=new RoundRec();
    roundRec.Book=Int16.Parse(round.Reference);
    roundRec.NLegend=round.Name;
    Marshal.StructureToPtr(roundRec,buffer,true);
    status = db_fillnew(ROUND_REC,buffer,0); // <=CRASHES HERE

它总是第二次在循环中崩溃。

这是记录结构:

    [StructLayout(LayoutKind.Sequential, Pack = 1)]
struct RoundRec
{
    public Int16 Book;
    [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 9)]
    public string NLegend; public string LastReadRefNo;
    [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 9)]
    public string LastNoAccessRefNo;
}

1 个答案:

答案 0 :(得分:2)

尝试为每条记录分配和释放内存:

foreach (Round round in roundList)
{
  RoundRec roundRec = new RoundRec();
  roundRec.Book=Int16.Parse(round.Reference);
  roundRec.NLegend = round.Name;

  IntPtr buffer = Marshal.AllocHGlobal(Marshal.SizeOf(roundRec));

  Marshal.StructureToPtr(roundRec, buffer, true);
  status = db_fillnew(ROUND_REC, buffer, 0);

  Marshal.FreeHGlobal(buffer);
}