Программа Xlib Muti-thread работает только под strace

Я пишу программу с несколькими потоками, используя xlib, pthread и cairo. Эта программа создает поток для рисования десяти точек после события щелчка.

Проблема в:

После того, как программа набрала три балла, произошел сбой и жалоба на 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, например "strace ./a.out".

Вот мои кодовые клипы:

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;
}

У кого-нибудь есть идеи об этой странной проблеме? Большое спасибо!


person user2854481    schedule 07.10.2013    source источник


Ответы (1)


Проблема вызвана использованием многопоточности с двумя потоками, пытающимися одновременно получить доступ к дисплею.

strace изменит синхронизацию, поэтому потоки будут обращаться к дисплею в разное время.

В Xlib есть функции для предотвращения конфликтов. Lookup XInitThreads, который включает поддержку потоков, и XLockDisplay и XUnlockDisplay, которые вам нужно будет вызывать из каждого потока до и после доступа к дисплею.

person parkydr    schedule 07.10.2013