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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
use super::ffi::*;
use once_cell::sync::Lazy;
use std::marker::PhantomData;

pub struct Cluster {
    /// Index of the first logical processor in the cluster
    pub id: u32,
    /// Clock rate (non-Turbo) of the cores in the cluster, in Hz
    pub frequency: u64,
    /// Either value of CPUID leaf 1 EAX register on x86 OR value of Main ID Register
    /// on ARM
    pub cpuid: u32,
}

pub struct Package {
    pub name: String,
    pub clusters: Vec<Cluster>,
}

struct Wrapper {
    pub packages: Vec<Package>,
    pub phantom: PhantomData<Vec<Package>>,
}

impl Drop for Wrapper {
    fn drop(&mut self) {
        unsafe { cpuinfo_deinitialize() };
    }
}

static GLOBAL_SOCKETS: Lazy<Wrapper> = Lazy::new(|| {
    unsafe { cpuinfo_initialize() };

    let mut sockets: Vec<Package> = vec![];

    let packages_count = unsafe { cpuinfo_get_packages_count() };

    let packages_ptr = unsafe { cpuinfo_get_packages() };

    let packages = unsafe { std::slice::from_raw_parts(packages_ptr, packages_count as usize) };

    for p in packages {
        let name = unsafe { std::ffi::CStr::from_ptr(p.name.as_ptr()) };

        let mut clusters = vec![];

        for cid in p.cluster_start..(p.cluster_start + p.cluster_count) {
            let maybe_cluster = unsafe { cpuinfo_get_cluster(cid).as_ref() };
            match maybe_cluster {
                Some(c) => {
                    #[allow(unused_assignments)]
                    let mut cpuid = 0 as u32;
                    cfg_if::cfg_if! {
                        if #[cfg(any(target_arch = "x86", target_arch = "x86_64"))] {
                            cpuid = c.cpuid;
                        } else if #[cfg(any(target_arch = "arm", target_arch = "aarch64"))] {
                            cpuid = c.midr;
                        }
                    };
                    let cluster = Cluster {
                        id: c.cluster_id,
                        frequency: c.frequency,
                        cpuid,
                    };

                    clusters.push(cluster);
                }
                None => {}
            }
        }

        let package = Package {
            name: name
                .to_owned()
                .to_str()
                .expect("failed to construct string")
                .to_owned(),
            clusters,
        };
        sockets.push(package);
    }

    return Wrapper {
        packages: sockets,
        phantom: PhantomData::default(),
    };
});

pub fn get_packages() -> &'static [Package] {
    return &GLOBAL_SOCKETS.packages;
}

#[cfg(test)]
mod test {
    use crate::get_packages;

    #[test]
    fn basic_info() {
        let packages = get_packages();
        assert!(!packages.is_empty());
        assert!(!packages.first().unwrap().name.is_empty());

        let package = packages.first().unwrap();

        assert!(!package.clusters.is_empty());

        let cluster = package.clusters.first().unwrap();

        assert_ne!(cluster.cpuid, 0);
        assert_eq!(cluster.frequency, 0);
    }
}