0

xlib、pthread、cairo を使用してマルチスレッド プログラムを作成しています。このプログラムは、クリック イベントの後に 10 個のポイントを描画するためにスレッドを作成します。

問題は:

プログラムが 3 点を引いた後、クラッシュし、xlib に苦情が寄せられました

X Error of failed request:  BadRequest (invalid request code or no such operation)
  Major opcode of failed request:  0 ()
  Serial number of failed request:  67
  Current serial number in output stream:  97

ただし、「 strace ./a.out 」のような strace を使用している場合は、正常に動作します。

ここに私のコードクリップがあります:

void *draw_point(void *arg) { //paint random-postion point
    int i = 0;
    int seed;
    double x ,y;
    srand(seed);
    cairo_set_source_rgba (cairo, 1, 0.2, 0.2, 0.6);
    for(i = 0; i< 10; i++) {
        x = rand() % 200;
        y = rand() % 200;
        if(candraw) {
            cairo_arc (cairo, x, y, 10.0, 0, 2*M_PI);
            cairo_fill (cairo);
        }
        hasdraw = true;
        sleep(1);
    }
    return NULL;
}

bool win_main(void)
{
    int clickx = 0, clicky = 0;
    unsigned long valuemask;
    XEvent event;

    valuemask = ButtonPressMask | ButtonReleaseMask | ButtonMotionMask | PointerMotionMask;

    XSelectInput(display, win, valuemask);

    pthread_t thread;

    while (1) {
        while (XPending(display) == 0) {
            candraw = true;
            if(hasdraw)
            XFlush(display);
            candraw = false;
        }
        XNextEvent(display, &event);
        candraw = false;
        switch (event.type) {
        case MotionNotify:
            //...
            break;
        case ButtonPress:
            clickx = event.xbutton.x;
            clicky = event.xbutton.y;
            if(clicky < 50)
                pthread_create(&thread, NULL, draw_point, NULL);
            break;
        case ButtonRelease:
            //...
            break;
        default:
            break;
        }
    }
    return 0;
}

誰かがこの奇妙な問題について考えを持っていますか? どうもありがとう!

4

1 に答える 1

1

この問題は、2 つのスレッドが同時にディスプレイにアクセスしようとするマルチスレッドを使用することによって発生します。

strace はタイミングを変更するため、スレッドは異なる時間にディスプレイにアクセスしています。

Xlib には競合を防ぐ機能があります。スレッド サポートを有効にする XInitThreads と、ディスプレイにアクセスする前後に各スレッドから呼び出す必要がある XLockDisplay および XUnlockDisplay を参照します。

于 2013-10-07T16:46:32.493 に答える