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
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
use crate::counter::Counter;
use crate::PAPI_destroy_eventset;
use crate::{check_error, PAPI_create_eventset};
use crate::{PAPI_accum, PAPI_read, PAPI_start, PAPI_stop, PAPI_NULL};
use crate::{PAPI_add_event, PapiError};
use std::os::raw::{c_int, c_longlong};
use std::sync::Arc;

pub struct EventsSet {
    counter_defs: Arc<Vec<Counter>>,
    counter_values: Vec<c_longlong>,
    evt_set_id: c_int,
}

impl EventsSet {
    fn create_event(counters: &[Counter]) -> Result<c_int, PapiError> {
        let mut evt_set_id = PAPI_NULL;

        unsafe { check_error(PAPI_create_eventset(&mut evt_set_id as *mut c_int))? };

        for evt in counters {
            unsafe {
                check_error(PAPI_add_event(evt_set_id, evt.code))?;
            }
        }

        Ok(evt_set_id)
    }

    pub fn new(counters: &[Counter]) -> Result<Self, PapiError> {
        Ok(Self {
            counter_defs: Arc::new(counters.to_vec()),
            counter_values: vec![0; counters.len()],
            evt_set_id: Self::create_event(counters)?,
        })
    }

    pub fn len(&self) -> usize {
        self.counter_defs.len()
    }

    pub fn try_clone(&self) -> Result<Self, PapiError> {
        Ok(Self {
            counter_defs: self.counter_defs.clone(),
            counter_values: vec![0; self.counter_defs.len()],
            evt_set_id: Self::create_event(&self.counter_defs)?,
        })
    }

    pub fn start(&mut self) -> Result<(), PapiError> {
        unsafe { check_error(PAPI_start(self.evt_set_id)) }
    }

    pub fn read(&mut self) -> Result<&[c_longlong], PapiError> {
        unsafe {
            check_error(PAPI_read(self.evt_set_id, self.counter_values.as_mut_ptr()))?;
        }
        Ok(&self.counter_values)
    }

    pub fn read_into<'a>(
        &self,
        counters: &'a mut [c_longlong],
    ) -> Result<&'a [c_longlong], PapiError> {
        assert_eq!(counters.len(), self.counter_defs.len());
        unsafe {
            check_error(PAPI_read(self.evt_set_id, counters.as_mut_ptr()))?;
        }
        Ok(counters)
    }

    pub fn accum(&mut self) -> Result<&[c_longlong], PapiError> {
        unsafe {
            check_error(PAPI_accum(
                self.evt_set_id,
                self.counter_values.as_mut_ptr(),
            ))?;
        }
        Ok(&self.counter_values)
    }

    pub fn stop(&mut self) -> Result<&[c_longlong], PapiError> {
        unsafe {
            check_error(PAPI_stop(self.evt_set_id, self.counter_values.as_mut_ptr()))?;
        }
        Ok(&self.counter_values)
    }
}

impl Drop for EventsSet {
    fn drop(&mut self) {
        let _ = self.stop();
        unsafe {
            PAPI_destroy_eventset(&mut self.evt_set_id as *mut c_int);
        }
    }
}