草庐IT

c++ - 如何在混合语言应用程序中创建堆?

coder 2024-06-05 原文

我们有一个用 Visual Basic 6.0 编写的前端,它调用几个用混合 C/C++ 编写的后端 DLL。问题是每个 DLL 似乎都有自己的堆,但其中一个堆不够大。当我们分配了足够的内存时,堆就会与程序堆栈发生冲突。 每个 DLL 都完全用 C 编写,除了基本的 DLL 包装器是用 C++ 编写的。每个 DLL 都有一些入口点。每个入口点立即调用一个 C 例程。我们想增加 DLL 中堆的大小,但一直无法弄清楚该怎么做。我搜索了指导并找到了这些 MSDN 文章:

http://msdn.microsoft.com/en-us/library/hh405351(v=VS.85).aspx

这些文章很有趣,但提供的信息相互矛盾。在我们的问题中,似乎每个 DLL 都有自己的堆。这与“Heaps: Pleasures and Pains”一文相符,该文章称 C 运行时 (C RT) 库在启动时创建了自己的堆。 “Managing Heap Memory”一文说 C RT 库是从默认进程堆中分配的。 “Win32 中的内存管理选项”一文说行为取决于所使用的 C RT 库的版本。

我们通过从私有(private)堆中分配内存暂时解决了这个问题。但是,为了改进这个非常大的复杂程序的结构,我们希望从带有薄 C++ 包装器的 C 切换到带有类的真正 C++。我们非常确定 new 和 free 运算符不会从我们的私有(private)堆中分配内存,我们想知道如何控制 C++ 用于在每个 DLL 中分配对象的堆的大小。该应用程序需要在从 2000 到 7 的所有版本的桌面 Windows-NT 中运行。

问题

Can anyone point us to definitive and correct documentation that explains how to control the size of the heap C++ uses to allocate objects?

有几个人断言,由于堆分配覆盖堆栈而导致的堆栈损坏是不可能的。这是我们观察到的。 VB 前端使用四个动态加载的 DLL。每个 DLL 都独立于其他 DLL,并提供少数由前端调用的方法。所有 DLL 都通过写入磁盘文件的数据结构进行通信。这些数据结构都是静态结构化的。它们不包含指针,仅包含值类型和固定大小的值类型数组。问题 DLL 由传递文件名的单个调用调用。它旨在分配完成其处理所需的大约 20MB 的数据结构。它进行大量计算,将结果写入磁盘,释放 20MB 的数据结构,并返回和错误代码。然后前端卸载 DLL。在调试所讨论的问题时,我们在数据结构分配代码的开头设置了一个断点,观察从 calloc 调用返回的内存值,并将它们与当前堆栈指针进行比较。我们看着分配的 block 接近堆栈。分配完成后,堆栈开始增长,直到与堆重叠。最终,计算写入了堆并破坏了堆栈。当堆栈展开时,它试图返回到一个无效地址并因段错误而崩溃。

我们的每个 DLL 都静态链接到 CRT,因此每个 DLL 都有自己的 CRT 堆和堆管理器。微软在 http://msdn.microsoft.com/en-us/library/ms235460(v=vs.80).aspx 中表示:

Each copy of the CRT library has a separate and distinct state. As such, CRT objects such as file handles, environment variables, and locales are only valid for the copy of the CRT where these objects are allocated or set. When a DLL and its users use different copies of the CRT library, you cannot pass these CRT objects across the DLL boundary and expect them to be picked up correctly on the other side.
Also, because each copy of the CRT library has its own heap manager, allocating memory in one CRT library and passing the pointer across a DLL boundary to be freed by a different copy of the CRT library is a potential cause for heap corruption.

我们不在 DLL 之间传递指针。我们遇到的不是堆损坏,而是堆栈损坏。

最佳答案

好的,问题是:

Can anyone point us to definitive and correct documentation that explains how to control the size of the heap C++ uses to allocate objects?

我要回答我自己的问题。我从阅读中得到了答案 Raymond Chen的博客 The Old New Thing , 具体来说 There's also a large object heap for unmanaged code, but it's inside the regular heap .雷蒙德在那篇文章中推荐Advanced Windows Debugging作者:马里奥·赫沃特 (Mario Hewardt) 和丹尼尔·普拉瓦特 (Daniel Pravat)。这本书有关于堆栈和堆损坏的非常具体的信息,这是我想知道的。此外,它还提供了有关如何调试这些问题的各种信息。

关于c++ - 如何在混合语言应用程序中创建堆?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7577976/

有关c++ - 如何在混合语言应用程序中创建堆?的更多相关文章

  1. ruby - 如何在 Ruby 中顺序创建 PI - 2

    出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits

  2. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  3. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  4. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  5. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  6. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  7. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  8. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  9. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  10. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

随机推荐