3

pthread_cond_timedwait() で奇妙な問題が発生しました。POSIX 仕様によると、これはキャンセル ポイントです。ただし、スレッドで pthread_cancel() を呼び出すと、キャンセルされません! 代わりに、pthread_cond_timedwait() は引き続き正常に実行されます。ロックも何もせず、 pthread_cancel() が呼び出されなかったかのように実行を続けます。しかし、pthread_testcancel() 呼び出しを挿入するとすぐに、スレッドは正しくキャンセルされます! pthread_testcancel() を呼び出さないと、常に pthread_cond_timedwait() を呼び出していますが、スレッドがキャンセルされることはありません。

ここで何がうまくいかないのか、誰にも分かりますか?どうもありがとう!

編集:ここにコードがあります:

#include <stdio.h>
#include <pthread.h>
#include <time.h>
#include <sys/time.h>

// replacement function because OS X doesn't seem to have clock_gettime()
static int clock_gettime(int clk_id, struct timespec* t)
{
    struct timeval now;
    int rv = gettimeofday(&now, NULL);

        if(rv) return rv;

    t->tv_sec = now.tv_sec;
        t->tv_nsec = now.tv_usec * 1000;

    return 0;
}

static void *threadproc(void *data)
{
    pthread_mutex_t mutex;
    pthread_mutexattr_t attr;               
    pthread_cond_t cond;

    pthread_mutexattr_init(&attr);
    pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);

    pthread_mutex_init(&mutex, &attr);      
    pthread_mutexattr_destroy(&attr);

    pthread_cond_init(&cond, NULL);

    for(;;) {

        struct timespec ts;

        clock_gettime(0, &ts);

        // wait 60ms
        ts.tv_nsec += 60 * 1000000;

        pthread_mutex_lock(&mutex);
        pthread_cond_timedwait(&cond, &mutex, &ts);
        pthread_mutex_unlock(&mutex);   

#if 0
        pthread_testcancel();
#endif      
    }

    return NULL;    
}

int main(int argc, char *argv[])
{
    pthread_t pThread;

    pthread_create(&pThread, NULL, threadproc, NULL);

    printf("Waiting...\n");
    sleep(5);
    printf("Killing thread...\n");

    pthread_cancel(pThread);
    pthread_join(pThread, NULL);

    printf("Ok!\n");

    return 0;
}
4

2 に答える 2

4

コードがどのように動作するかについてのあなたの期待は正しく、実際、私がテストした他のシステムでも期待どおりに動作します。OSX で (さらに別の) バグを発見したと思います。

于 2012-05-14T15:19:23.947 に答える
0

pthread_cancelOS X 10.11.4 (および場合によってはそれ以前のバージョン) で正しく機能します。さらに、予想どおり、returnのvalue_ptr引数。pthread_joinPTHREAD_CANCELED

于 2016-04-19T20:42:51.093 に答える