打印

奇怪的Heap corrupt问题,_CrtIsValidHeapPointer失败

[复制链接]
8557|45
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
guoyt|  楼主 | 2010-7-30 20:50 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
一个由VC6迁移到VC2008SP1的工程。
使用了Cypress的USB库:CyUSB.lib(静态链接,无dll文件)
在主线程中我new了一个字符串,然后作为参数,传递给线程,线程中使用该字符串,然后delete[],结果引发异常。
沙发
zhuww| | 2010-7-30 21:01 | 只看该作者
你的设置?

使用特权

评论回复
板凳
zhuww| | 2010-7-30 21:01 | 只看该作者
详细点,可以的话上个代码吧

使用特权

评论回复
地板
guoyt|  楼主 | 2010-7-30 21:08 | 只看该作者
工程设置:Debug multithread DLL(/MDd)

使用特权

评论回复
5
guoyt|  楼主 | 2010-7-30 21:08 | 只看该作者
改用GlobalAlloc/GlobalFree依然无济于事。

使用特权

评论回复
6
guoyt|  楼主 | 2010-7-30 21:09 | 只看该作者
我认为CyUSB.lib可能静态链接到了CRT库,而我的版本是使用动态链接。

使用特权

评论回复
7
zhuww| | 2010-7-30 21:25 | 只看该作者
改下工程设置

使用特权

评论回复
8
guoyt|  楼主 | 2010-7-30 21:27 | 只看该作者
设置为“静态使用MFC、使用CRT库的多线程非DLL版本”,依然会报错。

使用特权

评论回复
9
guoyt|  楼主 | 2010-7-30 21:27 | 只看该作者
在VC6中一切正常。

使用特权

评论回复
10
guoyt|  楼主 | 2010-7-30 21:31 | 只看该作者
代码:
C/C++ code
//CEmuDevice继承自CCyUSBDevice,该类由CyUSB.lib导出
LRESULT CEmuDevice::SendCommand(COMMAND nCommand, WPARAM wParam/* =0 */, LPARAM lParam/* =0 */)
{
    ASSERT(IsOpen());

    WaitWorkerThreadExit();

    PWORKITEM pWorkItem = new WORKITEM;
    pWorkItem->wParam = wParam;
    pWorkItem->lParam = lParam;
    pWorkItem->pDevice = this;
    switch(nCommand)
    {
        case CONFIG_EMULATOR:
        {
            LPCTSTR lpszConfigFile = (LPCTSTR)wParam;
             LPTSTR lpszPathName = new TCHAR[_tcslen(lpszConfigFile)+1];
             ZeroMemory(lpszPathName, (_tcslen(lpszConfigFile)+1)*sizeof(TCHAR));
             _tcscpy_s(lpszPathName, (_tcslen(lpszConfigFile)+1)*sizeof(TCHAR), lpszConfigFile);

            pWorkItem->wParam = (WPARAM)lpszPathName;
            pWorkItem->uCmd = CONFIG_EMULATOR;

            break;
        }

        ……
    }
   
    if(pWorkItem != NULL)
    {
        if(m_hWorkerThread == NULL)
        {
            m_hWorkerThread = ::CreateThread(NULL, 0, CommandDispatchWorker, (LPVOID)pWorkItem, 0, &m_dwThreadID);//参数传入
            if(m_hWorkerThread == NULL)
            {
                TRACE0("Create command dispatch worker thread failed.\n");
                return 0;
            }
        }
    }

//线程函数
DWORD WINAPI CEmuDevice::CommandDispatchWorker(LPVOID lpParameter)
{
    ASSERT(lpParameter != NULL);

    WORKITEM* pWorkItem = (WORKITEM*)lpParameter;
    CEmuDevice* pDevice = pWorkItem->pDevice;
    WPARAM wParam = pWorkItem->wParam;
    LPARAM lParam = pWorkItem->lParam;

    ASSERT(pDevice != NULL);
    ASSERT(pDevice->m_hNeedStop != NULL);

    if(!pDevice->IsOpen())
    {
        return 1;
    }

    switch(pWorkItem->uCmd)
    {
        /* Board level commands */        
        case CONFIG_EMULATOR:
        {
            LPTSTR lpszConfigFile = (LPTSTR)wParam;
            pDevice->ConfigEmulator(lpszConfigFile);
            delete[] lpszConfigFile; //<------------异常

            break;
        }
        
        ……

    }
    delete[] pWorkItem;
    return 0;
}

使用特权

评论回复
11
zhuww| | 2010-7-30 21:37 | 只看该作者
是否存在SendCommand中没有运行case CONFIG_EMULATOR:里面的语句就到后面创建函数的代码啦的情况

使用特权

评论回复
12
langgq| | 2010-7-30 21:38 | 只看该作者
估计与Unicode字符集有关,你把项目改成多字节字符集验证一下。

使用特权

评论回复
13
lium| | 2010-7-30 21:39 | 只看该作者
另外再调试一下,开一个内存窗口,监视从lpszPathName-0x20开始的内存,看看执行过程中是否发生变化。

使用特权

评论回复
14
hanwe| | 2010-7-30 21:39 | 只看该作者
建议好好看一下CCyUSBDevice的说明文档。

使用特权

评论回复
15
guoyt|  楼主 | 2010-7-30 21:40 | 只看该作者
这个……

使用特权

评论回复
16
hanwe| | 2010-7-30 21:41 | 只看该作者
你要delete的指针是在SendCommand中由CCyUSBDevice作为参数传给你的,我说的没错吧。

使用特权

评论回复
17
hanwe| | 2010-7-30 21:41 | 只看该作者
那么你知道这个指针在CCyUSBDevice中是如何创建的?

使用特权

评论回复
18
yufe| | 2010-7-30 21:43 | 只看该作者
至少你怎么知道要用delete []而不是delete?

使用特权

评论回复
19
yufe| | 2010-7-30 21:43 | 只看该作者
甚至可能你根本不用自己释放,CCyUSBDevice会自己释放。

使用特权

评论回复
20
wangpe| | 2010-7-30 21:49 | 只看该作者
一个很重要很容易出你这种问题的做法是在dll中分配内存在exe中释放,这是非常不推荐的

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

1018

主题

9067

帖子

3

粉丝