use regex::Regex;
use std::env;
use std::fs::File;
use std::io::Write;
use std::path::{Path, PathBuf};
type DynError = Box<dyn std::error::Error>;
#[derive(Clone, Default)]
struct MyError(String);
impl MyError {
pub fn new<S: AsRef<str>>(msg: S) -> Self {
Self {
0: String::from(msg.as_ref()),
}
}
}
impl std::fmt::Debug for MyError {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
write!(f, "MyError {{ {} }}", self.0)
}
}
impl std::fmt::Display for MyError {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
write!(f, "{}", self.0)
}
}
impl std::error::Error for MyError {}
macro_rules! MyErr {
($msg: expr) => {
Err(Box::new(MyError::new($msg)))
};
}
#[derive(Copy, Clone, Debug, Default)]
struct MyParseCallbacks;
impl bindgen::callbacks::ParseCallbacks for MyParseCallbacks {
#[allow(clippy::single_match)]
fn item_name(&self, original_item_name: &str) -> Option<String> {
match original_item_name {
"hifbDYNAMIC_RANGE_E" => {
return Some(String::from("HIFB_DYNAMIC_RANGE_E"));
}
_ => {}
}
let re = Regex::new(r"^(hi|hifb)([^a-z]+)$").unwrap();
if let Some(cap) = re.captures_iter(original_item_name).next() {
return Some(cap[2].to_string());
}
let re = Regex::new(r"^(hi|hifb)([^a-z]+)__bindgen_ty_(\d)$").unwrap();
if let Some(cap) = re.captures_iter(original_item_name).next() {
return Some(format!("{}_U{}", &cap[2], &cap[3]));
}
None
}
}
fn detect_mpp_path(mpp_dir: &str) -> Result<PathBuf, DynError> {
let dir = env::var("CARGO_MANIFEST_DIR").unwrap();
let mut base_path = Path::new(&dir);
for _a in 0..9 {
let np = base_path.join(mpp_dir);
let path = Path::new(&np);
if path.exists() {
return Ok(path.to_path_buf());
}
match base_path.parent() {
Some(v) => base_path = v,
None => break,
}
}
MyErr!(format!("The `MPP_DIR={}` does not detected!", mpp_dir))
}
fn setup_envir() -> Result<(), DynError> {
if let Ok(val) = env::var("TARGET") {
if val == "x86_64-unknown-linux-gnu" {
return MyErr!("Target not supported!");
}
}
if env::var("MPP_DIR").is_err() {
#[cfg(any(
feature = "hi3516ev200",
feature = "hi3516ev300",
feature = "hi3518ev200",
feature = "hi3518ev300"
))]
env::set_var(
"MPP_DIR",
detect_mpp_path("vendor/mpp-lib-Hi3516EV200_V1.0.1.0").unwrap(),
);
#[cfg(feature = "hi3531v100")]
env::set_var(
"MPP_DIR",
detect_mpp_path("vendor/mpp-lib-Hi3531V100_V1.0.D.0").unwrap(),
);
#[cfg(feature = "hi3559av100")]
env::set_var(
"MPP_DIR",
detect_mpp_path("vendor/mpp-lib-Hi3559AV100_V2.0.2.0").unwrap(),
);
}
if env::var("SYS_INCLUDE").is_err() {
#[cfg(any(
feature = "hi3516ev200",
feature = "hi3516ev300",
feature = "hi3518ev200",
feature = "hi3518ev300"
))]
env::set_var(
"SYS_INCLUDE",
"/opt/hisi-linux/x86-arm/arm-himix100-linux/target/usr/include",
);
#[cfg(feature = "hi3531v100")]
env::set_var(
"SYS_INCLUDE",
"/opt/hisi-linux-nptl/arm-hisiv100-linux/target/usr/include",
);
#[cfg(feature = "hi3559av100")]
env::set_var(
"SYS_INCLUDE",
"/opt/hisi-linux/x86-arm/aarch64-himix100-linux/aarch64-linux-gnu/sys-include",
);
}
Ok(())
}
fn main() -> Result<(), DynError> {
if cfg!(not(any(
feature = "hi3516ev200",
feature = "hi3516ev300",
feature = "hi3518ev200",
feature = "hi3518ev300",
feature = "hi3519av100",
feature = "hi3531v100",
feature = "hi3559av100",
))) {
return MyErr!("The target board does not specified!");
}
println!("cargo:rerun-if-env-changed=MPP_DIR");
println!("cargo:rerun-if-env-changed=SYS_INCLUDE");
println!("cargo:rerun-if-changed=build.rs");
setup_envir()?;
let mpp_dir = env::var("MPP_DIR").unwrap();
if !Path::new(&mpp_dir).exists() {
return MyErr!(format!("The `MPP_DIR={}` does not exists", mpp_dir));
}
println!("cargo:rustc-link-search=native={}/lib", mpp_dir);
let wrapper_path = PathBuf::from(env::var("OUT_DIR").unwrap()).join("wrapper.h");
let wrapper_path = wrapper_path.to_str().unwrap();
let mut wrapper = File::create(wrapper_path).unwrap();
writeln!(wrapper, "#include <hifb.h>")?;
let bindings = bindgen::Builder::default()
.header(wrapper_path)
.default_enum_style(bindgen::EnumVariation::Rust {
non_exhaustive: false,
})
.anon_fields_prefix("un")
.derive_debug(true)
.impl_debug(false)
.derive_default(true)
.derive_partialeq(true)
.derive_eq(true)
.impl_partialeq(true)
.whitelist_function("^HI_.*")
.whitelist_type("fb_bitfield|fb_fix_screeninfo|fb_var_screeninfo")
.whitelist_type("HIFB_LAYER_INFO_MASKBIT")
.whitelist_type("^HI_.*$")
.whitelist_type("HIFB.*_[ES]$")
.whitelist_var("^FBIO.*|^HIFB.*")
.whitelist_var("^HI_.*")
.use_core()
.clang_arg(format!("-I{}/include", env::var("MPP_DIR").unwrap()))
.clang_arg(format!("-I{}", env::var("SYS_INCLUDE").unwrap()))
.parse_callbacks(Box::new(MyParseCallbacks::default()))
.generate()
.expect("Unable to generate bindings");
let out_path = PathBuf::from(env::var("OUT_DIR").unwrap());
bindings
.write_to_file(out_path.join("bindings.rs"))
.expect("Couldn't write bindings!");
Ok(())
}