summaryrefslogtreecommitdiff
path: root/keystore2/src/vintf/lib.rs
blob: 8730a3e31d239e0992bb0f8b4faa7221d74fbd07 (plain)
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
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// Copyright 2021, The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

//! Bindings for getting the list of HALs.

use keystore2_vintf_bindgen::{
    freeNames, getAidlInstances, getHalNames, getHalNamesAndVersions, getHidlInstances,
};
use std::ffi::{CStr, CString};
use std::os::raw::c_char;
use std::str::Utf8Error;

/// A struct that contains a list of HALs (optionally with version numbers).
/// To use it, call as_vec to get a Vec view of the data it contains.
pub struct HalNames {
    data: *mut *mut c_char,
    len: usize,
}

impl Drop for HalNames {
    fn drop(&mut self) {
        // Safety: The memory is allocated by our C shim so it must free it as well.
        unsafe { freeNames(self.data, self.len) }
    }
}

impl<'a> HalNames {
    /// Get a Vec view of the list of HALs.
    pub fn as_vec(&'a self) -> Result<Vec<&'a str>, Utf8Error> {
        // Safety: self.data contains self.len C strings.
        // The lifetimes ensure that the HalNames (and hence the strings) live
        // at least as long as the returned vector.
        unsafe { (0..self.len).map(|i| CStr::from_ptr(*self.data.add(i)).to_str()) }.collect()
    }
}

/// Gets all HAL names.
/// Note that this is not a zero-cost shim: it will make copies of the strings.
pub fn get_hal_names() -> HalNames {
    let mut len: usize = 0;
    // Safety: We'll wrap this in HalNames to free the memory it allocates.
    // It stores the size of the array it returns in len.
    let raw_strs = unsafe { getHalNames(&mut len) };
    HalNames { data: raw_strs, len }
}

/// Gets all HAL names and versions.
/// Note that this is not a zero-cost shim: it will make copies of the strings.
pub fn get_hal_names_and_versions() -> HalNames {
    let mut len: usize = 0;
    // Safety: We'll wrap this in HalNames to free the memory it allocates.
    // It stores the size of the array it returns in len.
    let raw_strs = unsafe { getHalNamesAndVersions(&mut len) };
    HalNames { data: raw_strs, len }
}

/// Gets the instances of the given package, version, and interface tuple.
/// Note that this is not a zero-cost shim: it will make copies of the strings.
pub fn get_hidl_instances(
    package: &str,
    major_version: usize,
    minor_version: usize,
    interface_name: &str,
) -> HalNames {
    let mut len: usize = 0;
    let packages = CString::new(package).expect("Failed to make CString from package.");
    let interface_name =
        CString::new(interface_name).expect("Failed to make CString from interface_name.");
    // Safety: We'll wrap this in HalNames to free the memory it allocates.
    // It stores the size of the array it returns in len.
    let raw_strs = unsafe {
        getHidlInstances(
            &mut len,
            packages.as_ptr(),
            major_version,
            minor_version,
            interface_name.as_ptr(),
        )
    };
    HalNames { data: raw_strs, len }
}

/// Gets the instances of the given package, version, and interface tuple.
/// Note that this is not a zero-cost shim: it will make copies of the strings.
pub fn get_aidl_instances(package: &str, version: usize, interface_name: &str) -> HalNames {
    let mut len: usize = 0;
    let packages = CString::new(package).expect("Failed to make CString from package.");
    let interface_name =
        CString::new(interface_name).expect("Failed to make CString from interface_name.");
    // Safety: We'll wrap this in HalNames to free the memory it allocates.
    // It stores the size of the array it returns in len.
    let raw_strs =
        unsafe { getAidlInstances(&mut len, packages.as_ptr(), version, interface_name.as_ptr()) };
    HalNames { data: raw_strs, len }
}

#[cfg(test)]
mod tests {

    use super::*;

    #[test]
    fn test() -> Result<(), Utf8Error> {
        let result = get_hal_names();
        let names = result.as_vec()?;
        assert_ne!(names.len(), 0);

        let result = get_hal_names_and_versions();
        let names_and_versions = result.as_vec()?;
        assert_ne!(names_and_versions.len(), 0);

        assert!(names_and_versions.len() >= names.len());

        Ok(())
    }
}