IOCP закрывает сокет и удаляет каждый буфер, связанный с IO

Попытка остановить рабочий поток IOCP и закрыть сокет и связанные с ним данные. Это основано на подсчете ссылок, а когда счетчик ссылок достигнет 0, закройте сокет и удалите буферы. Ниже приведен скелет рабочего потока IOCP. Когда я хочу остановить IO, вызовите функцию 'FFT_stop_session'. который просто 'PostQueuedCompletion.."Этот способ, кажется, работает. Но сбой происходит время от времени.. сообщение об ошибке, как показано ниже..

"HEAP: Free Heap block ??? modified at ??? after it was freed"

Но я не могу понять, почему может произойти двойное освобождение???

static unsigned int WINAPI FFT_thread_worker (LPVOID param)
{
        ...
    while(fft->running==TRUE)
    {
        iostatus = GetQueuedCompletionStatus(fft->hIOCP,&byteio,&completion_key (LPOVERLAPPED*)&ovlp,INFINITE);
        if(iostatus == TRUE)
        {
            fftsock  = (FFTSOCKET*)ovlp;
            ffts     = fftsock->session;

            switch(completion_key)
            {
                //-- S E N D
                case FFT_SEND:
                {                   
                                        // io works here                                         
                    FFTSOCKET_ref_inc(fftsock);
                    FFTSOCKET_send(fftsock,&byteio);                    
                    break;
                }

                //-- R E C V
                case FFT_RECV:
                {
                    // io works
                    FFTSOCKET_ref_inc(fftsock);
                    FFTSOCKET_recv(fftsock,&byteio);                    
                    break;
                }
                case FFT_STOPIO:
                {
                  // Just consume complete state
                    break;
                }
                default:break;
            }
            FFTSOCKET_ref_dec(fftsock);  // dec ref counter and if it reasch 0 delete socket
        }
    }
    return 0;
}


// Stop session
int FFT_stop_session (FFTCONTEXT* fft,FFT_session* session)
{
    FFTSOCKET* sock = NULL;
    HANDLE* handles = NULL;

    if (fft != NULL && session != NULL)
    {
        ::EnterCriticalSection (&(fft->cs));

        for(int i = 0; i < session->socks.size(); i++)
        {
            sock = session->socks[i];           
            ::PostQueuedCompletionStatus(fft->hIOCP,0,FFT_STOPIO,&(sock->overlapped));
        }

        ::LeaveCriticalSection (&(fft->cs));
    }
    return 0;
}


int FFTSOCKET_ref_inc (FFTSOCKET* sock)
{
    return ::InterlockedIncrement(&(sock->ref));
}

int FFTSOCKET_ref_dec (FFTSOCKET* sock)
{
    int rst = 0;
    if(sock->ref == 0)
    {
        rst = 1;
    }
    else if(::InterlockedDecrement(&(sock->ref))==0)
    {   
        cout<<endl<<"sock delete:"<<sock->port<<flush;
        if(sock->wsabuf.buf) VirtualFree(sock->wsabuf.buf,0,MEM_RELEASE);
        sock->wsabuf.buf = NULL;
        sock->wsabuf.len = 0;
        WSACloseEvent(sock->overlapped.hEvent);

        if(sock->ai != NULL) freeaddrinfo(sock->ai);

        LINGER lingerStruct;
            lingerStruct.l_onoff  = 1;
                lingerStruct.l_linger = 0;
        ::setsockopt(sock->socket, SOL_SOCKET, SO_LINGER, (char *)&lingerStruct, sizeof(lingerStruct));

        shutdown(sock->socket,SD_BOTH);
        closesocket(sock->socket);
        sock->socket = INVALID_SOCKET;

        delete sock;
        sock = NULL;

        rst = 2;        
    }
    return rst;
}

0 ответов

Другие вопросы по тегам