调试实战——PInvoke导致栈破坏

简介: .net 程序也能崩溃?本文记录了 PInvoke 导致的栈破坏的排查过程

缘起

最近项目中遇到一个诡异的问题,程序在升级到.net4.6.1后,执行某个功能时会崩溃,提示访问只读内存区。大概规律如下:

  1. debug版不崩溃,release版稳定崩溃。
  2. 只有x64位的程序崩溃,32位anycpu编译出来的程序运行不会崩溃。
  3. 出问题的代码范围很小(按钮点击事件代码不多)。

根据以上信息,各位小伙伴有什么思路吗?

排查

由于release版可以稳定重现,而且范围不大,故通过二分法(注释掉一半代码,看看是否崩溃,如果崩溃,接着注释掉一半代码,如果不崩溃说明崩溃跟注释掉的那段代码有关...)很快定位到了导致问题的代码。

最后发现并不是由于升级.net版本导致的,而是程序本身的问题:
代码中通过P/Invoke调用了原生API GlobalMemoryStatus。在定义MemoryStatus结构体的时候强制按4字节定义了每一个字段。而在x64MemoryStatus结构体中的成员有些不是4字节大小,而是8字节大小!这样,传递给GlobalMemoryStatusMemoryStatus参数(32字节)比GlobalMemoryStatus预期的(56字节)小,导致GlobalMemoryStatus写了不该写的内存!:bomb::bomb::bomb:

重现

我把会出问题的代码独立出来了,完整的测试代码如下(请编译x64版本):

using System;
using System.Runtime.InteropServices;
namespace ConsoleApplication1
{
   
   
    class Program
    {
   
   
          [StructLayout(LayoutKind.Sequential)]
        public struct MemoryStatus
        {
   
   
            [MarshalAs(UnmanagedType.U4)]
            public uint dwLength;
            [MarshalAs(UnmanagedType.U4)]
            public uint dwMemoryLoad;
            [MarshalAs(UnmanagedType.U4)]
            public uint dwTotalPhys;
            [MarshalAs(UnmanagedType.U4)]
            public uint dwAvailPhys;
            [MarshalAs(UnmanagedType.U4)]
            public uint dwTotalPageFile;
            [MarshalAs(UnmanagedType.U4)]
            public uint dwAvailPageFile;
            [MarshalAs(UnmanagedType.U4)]
            public uint dwTotalVirtual;
            [MarshalAs(UnmanagedType.U4)]
            public uint dwAvailVirtual;
        }

        [DllImport("kernel32.dll")]
        public static extern void GlobalMemoryStatus(ref MemoryStatus memoryStatus);
        class CMyClass
        {
   
   
            public int n1 = 0;
        }
        struct CMyStruct
        {
   
   
            public CMyClass data;
        }
        static void Main(string[] args)
        {
   
   
            CMyStruct myObj = new CMyStruct(); myObj.data = new CMyClass();
            MemoryStatus memoryStatus = new MemoryStatus();
            // this line will corrupt the stack if we run in x64.
            // because memoryStatus is defined on the stack.
            GlobalMemoryStatus(ref memoryStatus);
            // myObj.data is corrupted
            System.Console.WriteLine("{0}", myObj.data);
        }
    }
}

修复

只需要定义MemoryStatus的时候,注意字段的大小即可。正确的MemoryStatus定义如下:

public struct MemoryStatus
{
   
   
    [MarshalAs(UnmanagedType.U4)]
    public uint dwLength;
    [MarshalAs(UnmanagedType.U4)]
    public uint dwMemoryLoad;
    // 以下字段 4 bytes on 32-bit Windows, 8 bytes on 64-bit Windows.
    [MarshalAs(UnmanagedType.SysUInt)]
    public IntPtr dwTotalPhys;
    [MarshalAs(UnmanagedType.SysUInt)]
    public IntPtr dwAvailPhys;
    [MarshalAs(UnmanagedType.SysUInt)]
    public IntPtr dwTotalPageFile;
    [MarshalAs(UnmanagedType.SysUInt)]
    public IntPtr dwAvailPageFile;
    [MarshalAs(UnmanagedType.SysUInt)]
    public IntPtr dwTotalVirtual;
    [MarshalAs(UnmanagedType.SysUInt)]
    public IntPtr dwAvailVirtual;
}

思考

  • 为什么debug版不崩溃?release版崩溃?

    我在测试机器上调查的原因是debug版本运行的时候,关键内存恰巧没被破坏(太“幸运”或者太不幸了),而在release版本中暴露了问题。可能在其它机器上debug版本也会崩溃或者发生其它诡异的问题。

说明:测试代码与项目中的实际代码不一样,有可能现象不一样,但问题的本质是一样的。

  • 为什么运行Any CPU编译出来的程序不崩溃?

    Platform targetAny CPU的时候,在工程属性,Build下的Prefer 32-bit的选项默认是勾选的,编译的程序会作为32位进程运行,所以不会崩溃。如果取消勾选,则编译出来的程序会作为64位应用程序运行,会崩溃。
    build选项

    关于Platform target的作用,具体参考《CLR via C#》,下图是从《CLR via C#》中文版第 4 版上截取的.
    Effects of /platform switch 摘自《CLR via C#》

总结

.net程序中,令人头疼的堆破坏问题很难出现了,这极大的提高了程序的稳定性。如果出现堆破坏,很有可能跟P/Invoke或者unsafe代码相关,可以重点排查相关代码。

{% note info %}
启用托管调试助手(Managed Debugging Assistants, 下文简称MDAs) 有时候会对调试问题有极大的帮助,虽然我这次调试没有借助MDAs,但我第一个想到的就是MDAs
{% endnote %}

关于MDAs的介绍请点击这里

参考资料

相关文章
|
1月前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
187 9
|
1月前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
32 1
|
2月前
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
74 2
|
24天前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
44 5
|
1月前
|
存储 算法 Java
数据结构的栈
栈作为一种简单而高效的数据结构,在计算机科学和软件开发中有着广泛的应用。通过合理地使用栈,可以有效地解决许多与数据存储和操作相关的问题。
|
1月前
|
存储 JavaScript 前端开发
执行上下文和执行栈
执行上下文是JavaScript运行代码时的环境,每个执行上下文都有自己的变量对象、作用域链和this值。执行栈用于管理函数调用,每当调用一个函数,就会在栈中添加一个新的执行上下文。
|
1月前
|
存储
系统调用处理程序在内核栈中保存了哪些上下文信息?
【10月更文挑战第29天】系统调用处理程序在内核栈中保存的这些上下文信息对于保证系统调用的正确执行和用户程序的正常恢复至关重要。通过准确地保存和恢复这些信息,操作系统能够实现用户模式和内核模式之间的无缝切换,为用户程序提供稳定、可靠的系统服务。
51 4
|
1月前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
1月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
48 0
|
2月前
数据结构(栈与列队)
数据结构(栈与列队)
23 1