2

ゲームエンジンを作っています。Windows では、(wglShareLists を使用して) 共有された 2 つのコンテキストを持つ 2 つのスレッドがありました。うまくいきました。1 つのスレッドがリソースをロードしている間、もう 1 つのスレッドは単純なロード画面をレンダリングしていました。Linux には WGL はなく、glX のみです。正しい使い方がわかりません。現在、私のスレッドコードは次のようになっています。

LinuxThread::LinuxThread() :
    handle_(0),
    running_(false),
    task_(0),
    useGraphicsContext_(0),
    threadContext_(0),
    threadDrawable_(0),
    dsp_(0)
{
}

LinuxThread::~LinuxThread() {
    finishTask();
    running_ = false;
    glXDestroyContext(dsp_, threadContext_);
}

ULONG LinuxThread::getId() {
    return static_cast<ULONG>(handle_);
}

void LinuxThread::start() {
    running_ = true;
    pthread_create(&handle_, 0, &LinuxThread::staticRun, (void*) this);
}

bool LinuxThread::isRunning() {
    return running_;
}

void LinuxThread::setGraphicsContext(bool state) {
    if (state) {
        Display* dsp = XOpenDisplay(0);
        threadDrawable_ = glXGetCurrentDrawable();
        GLint att[] = { GLX_RGBA, None };
        XVisualInfo* vi = glXChooseVisual(dsp, 0, att);
        GLXContext glc = glXGetCurrentContext();
        bool directlyToScreen = true; // False for x-server display.
        threadContext_ = glXCreateContext(dsp, vi, glc, directlyToScreen);
    }
    useGraphicsContext_ = state;
}

void LinuxThread::setTask(Task* task) {
//  if (useGraphicsContext_) {
//      task->run();
//      delete task;
//      return;
//  }
    finishTask();
    task_ = task;
}

bool LinuxThread::hasTask() {
    return task_ != 0;
}

void LinuxThread::finishTask() {
    while (task_ != 0) {
        usleep(1000);
    }
}

void LinuxThread::stop() {
    running_ = false;
}

void* LinuxThread::staticRun(void* thread) {
    return (void*) ((LinuxThread*) thread)->run();
}

int LinuxThread::run() {
    while (running_) {
        usleep(10000);
        if (task_ != 0) {
            if (useGraphicsContext_) {
                glXMakeCurrent(dsp_, threadDrawable_, threadContext_);
            }
            task_->run();
            if (useGraphicsContext_) {
                glFinish();
                glXMakeCurrent(dsp_, 0, 0);
            }
            delete task_;
            task_ = 0;
        }
    }
    return 1;
}

次のようなエラー メッセージでクラッシュします。

nouveau: kernel rejected pushbuf: No such file or directory
nouveau: ch6: krec 0 pushes 1 bufs 14 relocs 0
nouveau: ch6: buf 00000000 00000002 00000006 00000006 00000000
nouveau: ch6: buf 00000001 00000010 00000002 00000002 00000000
nouveau: ch6: buf 00000002 0000000f 00000002 00000002 00000002
nouveau: ch6: buf 00000003 0000000e 00000002 00000000 00000002
nouveau: ch6: buf 00000004 00000007 00000002 00000002 00000002
nouveau: ch6: buf 00000005 0000000a 00000002 00000002 00000000
nouveau: ch6: buf 00000006 0000000b 00000002 00000002 00000000

私が間違っていることは何ですか?

4

1 に答える 1

1

特定の問題の原因はわかりませんが、スレッド化されたコードが非常に複雑になっています。あなたの立場で私は:

  1. LIBGL_ALWAYS_SOFTWARE=ynouveau ドライバーで問題が発生していないことを確認するために、実行する前に設定を試してください。(これは、内部の Mesa Software レンダラーを使用します。)
  2. 共有コードを単一のスレッドに抽出し、それが機能することを確認してから、既知の機能するコードをできるだけ単純に複数のスレッドに戻します。
于 2013-05-28T18:50:04.830 に答える