MATLAB调用DLL函数时的堆栈限制是多少?

3
我正在尝试弄清楚MATLAB在调用DLL函数时的堆栈大小限制。是否有一种方法可以配置这个限制?
我使用loadlibrarycalllib函数来调用C语言实现的函数(在动态链接库中)。
我创建了一个测试来确定堆栈限制。
我使用MATLAB 2016a(64位)和Visual Studio 2010来构建DLL。
以下是我的MATLAB源代码:
loadlibrary('MyDll','MyDll.h')

size_in_bytes = 1000000;

res = calllib('MyDll', 'Test', size_in_bytes);

if (res == -1)
    disp(['Stack Overflow... (size = ', num2str(size_in_bytes), ')']);
else
    disp(['Successful stack allocation... (size = ', num2str(size_in_bytes), ')']);
end

unloadlibrary MyDll

以下是我的C源代码:

MyDll.h

// MyDll.h : DLL interface.

#ifndef MY_DLL_H
#define MY_DLL_H

#ifdef MY_DLL_EXPORTS
    #define MY_DLL_API   __declspec(dllexport)
#else
    #define MY_DLL_API   __declspec(dllimport)
#endif

extern MY_DLL_API int Test(int size);

#endif

MyDll.c

// MyDll.c

#include "MyDll.h"

#include <windows.h>
#include <stdio.h>
#include <string.h>
#include <malloc.h>


//Allocate <size> bytes in stack using _alloca(size).
//Return 0 if OK.
//Return (-1) in case of stack overflow.
int Test(int size)
{
    //Not allocated on the stack...
    static wchar_t errorMsg[100];
    static wchar_t okMsg[100];

    int errcode = 0;
    void *pData = NULL;

    //Prepare messages from advance.
    swprintf_s(errorMsg, 100, L"Stack Overflow (size = %d)", size);
    swprintf_s(okMsg, 100, L"Successful stack allocation (size = %d)", size);

    __try 
    {
        pData = _alloca(size);
    }
    // If an exception occurred with the _alloca function
    __except (GetExceptionCode() == STATUS_STACK_OVERFLOW)
    {
        MessageBox(NULL, errorMsg, TEXT("Error"), MB_OK | MB_ICONERROR);

        // If the stack overflows, use this function to restore.
        errcode = _resetstkoflw();
        if (errcode)
        {
            MessageBox(NULL, TEXT("Could not reset the stack!"), TEXT("Error"), MB_OK | MB_ICONERROR);
            _exit(1);
        }

        pData = NULL;
    };

    if (pData != NULL)
    {
        //Fill allocated buffer with zeros
        memset(pData, 0, size);

        MessageBox(NULL, okMsg, TEXT("OK"), MB_OK);

        return 0;
    }

    return -1;
}
__try__except块取自Microsoft示例:https://msdn.microsoft.com/en-us/library/wb1s57t5.aspx DLL编译器标志:/Zi /nologo /W4 /WX- /Od /D "WIN32" /D "_DEBUG" /D "_CONSOLE" /D "_USRDLL" /D "MY_DLL_EXPORTS" /D "_WINDLL" /D "_UNICODE" /D "UNICODE" /Gm /EHsc /RTC1 /MTd /GS /fp:precise /Zc:wchar_t /Zc:forScope /Fp"x64\Debug\MyDll.pch" /Fa"x64\Debug\" /Fo"x64\Debug\" /Fd"x64\Debug\vc100.pdb" /Gd /errorReport:queue DLL链接器标志:/OUT:"x64\Debug\MyDll.dll" /INCREMENTAL:NO /NOLOGO /DLL "kernel32.lib" "user32.lib" "gdi32.lib" "winspool.lib" "comdlg32.lib" "advapi32.lib" "shell32.lib" "ole32.lib" "oleaut32.lib" "uuid.lib" "odbc32.lib" "odbccp32.lib" /MANIFEST /ManifestFile:"x64\Debug\MyDll.dll.intermediate.manifest" /ALLOWISOLATION /MANIFESTUAC:"level='asInvoker' uiAccess='false'" /DEBUG /PDB:"c:\Tmp\MyDll\x64\Debug\MyDll.pdb" /SUBSYSTEM:CONSOLE /PGD:"c:\Tmp\MyDll\x64\Debug\MyDll.pgd" /TLBID:1 /DYNAMICBASE /NXCOMPAT /MACHINE:X64 /ERRORREPORT:QUEUE 我使用不同的size_in_bytes值执行了MATLAB代码:size_in_bytes = 1000000:通过!size_in_bytes = 10000000:通过!size_in_bytes = 50000000:通过!size_in_bytes = 60000000:通过!size_in_bytes = 70000000:栈溢出!
看起来我的系统限制大约为64MByte(但我不知道这个数字是否适用于所有系统)。

我试图使用editbin工具修改Matlab.exe的栈大小。
我尝试了以下命令(例如):
editbin /STACK:250000000 "c:\Program Files\MATLAB\R2016a\bin\matlab.exe"

这个选项按字节设置栈的大小,并使用十进制或C语言表示法作为参数。/STACK选项仅适用于可执行文件。

似乎没有任何影响...


堆栈大小在执行时设置,因此您需要检查操作系统分配的堆栈大小。顺便说一句,在 C 语言中使用 __try 让我吓了一跳。 - Stargateur
那么我在Windows 10中如何设置大小? - Rotem
1个回答

1
似乎在Windows上,堆栈的大小在编译时设置。因此,您可以使用选项/F或二进制文件EDITBIN
例如,您可以编辑以下文件:
EDITBIN /STACK:134217728 "C:\Program Files\MATLAB\R2016a\bin\win64\MATLAB.exe"

这将设置堆栈大小为128 MB(128 x 1024 x 1024字节=134217728字节)。
注意:请注意编辑C:\Program Files\MATLAB\R2016a\bin\matlab.exe无效。

1
你是指使用“editbin”更改“Matlab.exe”吗?就像这样:“editbin /stack:250000000“ c:\Program Files\MATLAB\R2016a\bin\matlab.exe”。看起来没有任何影响... - Rotem
1
你的答案似乎是错误的。你能否请删除它? - Rotem
@Wollmich 8_000_000 B / 1024 / 1024 => 7.62 MB,我不明白你说的128 MB是什么意思。 - Stargateur
@Stargateur 抱歉我的错。DUMPBIN /header 返回的堆栈大小是十六进制的。所以我认为 EDITBIN 需要以十六进制指定。 - Wollmich

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接