4

Mac OS X で非同期ファイル IO に aio_* 関数を使用しようとしていますが、何らかの形式のユーザー データをシグナル ハンドラに取り込む際に問題が発生しています。

これは、操作を設定するコードです。

class aio_context {
public:
    aio_context(int fildes, boost::uint64_t offset,
        const MyBufferClassPtr &buffer)
    {
        // The aiocb struct must be zeroed
        memset(&m_aiocb, 0, sizeof(struct aiocb));

        // Set what to do
        m_aiocb.aio_fildes = fildes;
        m_aiocb.aio_buf = buffer->data();
        m_aiocb.aio_nbytes = buffer->size();
        m_aiocb.aio_offset = offset;

        // Set notification
        m_aiocb.aio_sigevent.sigev_notify = SIGEV_SIGNAL;
        m_aiocb.aio_sigevent.sigev_signo = SIGUSR1;

        // ATTEMPT TO SET A VALUE THAT CAN BE READ IN THE HANDLER
        m_aiocb.aio_sigevent.sigev_value.sival_ptr = this;
    }

    struct aiocb* GetAiocbp()
    {
        return &m_aiocb;
    }

private:
    struct aiocb m_aiocb;
    // Some more context here
};

これは、次のように別の場所から呼び出されます。

aio_context *ctx = new aio_context(file_descriptor, offset, data);
// set some more context here
int ret = aio_write(ctx->GetAiocbp());
if (0 != ret) {
    // throw something
}

私のシグナル処理設定は次のようになります。

sigemptyset(&m_CurrentSIGHandler.sa_mask);
m_CurrentSIGHandler.sa_sigaction = aio_completion_handler;
m_CurrentSIGHandler.sa_flags = SA_SIGINFO;
sigaction(SIGUSR1, &m_CurrentSIGHandler, &m_PreviousSIGHandler);

実際のハンドラーは次のようになります。

void aio_completion_handler(int signo, siginfo_t *info, void *context)
{
    if (info->si_signo == SIGUSR1) {
        // Get the aio operation
        aio_context *ctx = static_cast<aio_context *>(info->si_value.sival_ptr);

        // THIS ASSERT ALWAYS FAILS - ctx IS NULL
        assert(ctx);

        // next check aio_error and aio_return using the aicb member of the ctx
        // ...
    }
}

したがって、問題は、si_value.sival_ptr が、aiocb 構造体で設定した aio_context ポインターではなく、シグナル ハンドラーで常に NULL であることです。これを行う方法について何か誤解しているに違いないので、誰かが私が間違っていることを教えてもらえますか?

私は MacOSX 10.6 で実行していますが、問題があれば 10.5 用にコンパイルしています (少なくともそうしようとしています)。

また、この質問への回答は、AIO を完全に無視する必要があることを示しているようです。これは本当に事実ですか?

アップデート:

http://lists.apple.com/archives/darwin-dev/2008/Oct/msg00054.htmlで同じ問題を抱えている他の人を見つけました。

http://www.opensource.apple.com/source/xnu/xnu-1504.9.26/bsd/kern/kern_aio.cでカーネル コードも確認しましたが、正しく理解できれば、sigev_value は完全に無視されます。ここで、Mac OS X で aio_* 関数の予想される使用法がどうなるかについて、私は本当に途方に暮れています。私は何かを誤解していますか、それとも aio_* 関数は私のユースケースの行き止まりですか?

4

1 に答える 1

5

Mac OS X は、シグナルにユーザーデータ ポインターを追加する POSIX のセクションであるリアルタイム シグナル( posix 仕様の [RTS] ) をサポートしていません。これにより、Mac OS X を AIO で効率的に使用することが非常に難しくなります。唯一のオプションは、すべての未処理のジョブをループして、完了したジョブを処理することです。

于 2011-03-15T04:40:55.997 に答える