1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
//! Management of pending errors for the XCB connection.
//!
//! We add our own error tracking ontop of what XCB already provides. For this reason, this module
//! contains a struct for tracking requests that were send and also tracking errors that were
//! received, but not yet given to the user of this library.

use std::cmp::Reverse;
use std::collections::{BinaryHeap, VecDeque};
use std::sync::Mutex;

use super::{Buffer, XCBConnection};
use x11rb_protocol::SequenceNumber;

#[derive(Debug, Default)]
struct PendingErrorsInner {
    in_flight: BinaryHeap<Reverse<SequenceNumber>>,
    pending: VecDeque<(SequenceNumber, Buffer)>,
}

/// A management struct for pending X11 errors
#[derive(Debug, Default)]
pub(crate) struct PendingErrors {
    inner: Mutex<PendingErrorsInner>,
}

impl PendingErrors {
    pub(crate) fn append_error(&self, error: (SequenceNumber, Buffer)) {
        self.inner.lock().unwrap().pending.push_back(error)
    }

    pub(crate) fn discard_reply(&self, sequence: SequenceNumber) {
        self.inner.lock().unwrap().in_flight.push(Reverse(sequence));
    }

    pub(crate) fn get(&self, conn: &XCBConnection) -> Option<(SequenceNumber, Buffer)> {
        let mut inner = self.inner.lock().unwrap();

        // Check if we already have an element at hand
        let err = inner.pending.pop_front();
        if err.is_some() {
            return err;
        }

        // Check if any of the still in-flight responses got a reply/error
        while let Some(&Reverse(seqno)) = inner.in_flight.peek() {
            let result = match conn.poll_for_reply(seqno) {
                Err(()) => {
                    // This request was not answered/errored yet, so later request will not
                    // have answers as well.
                    return None;
                }
                Ok(reply) => reply,
            };

            let seqno2 = inner.in_flight.pop();
            assert_eq!(Some(Reverse(seqno)), seqno2);

            if let Some(result) = result {
                // Is this an error?
                if result[0] == 0 {
                    return Some((seqno, result));
                } else {
                    // It's a reply, just ignore it
                }
            }
        }

        None
    }
}