From 961550ba19e7119e7af3f818237d2059ce98a834 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Wed, 12 Apr 2023 13:13:15 -0400 Subject: [PATCH] Improve Rust enum translations --- .vscode/settings.json | 1 + .vscode/tasks.json | 23 ++++++++++++++++ Cargo.toml | 2 +- README.md | 2 +- build/bind.rs | 40 ++++++++++++++++++++++------ build/wrap/enums.rs | 61 +++++++++++++++++++++++++++++++++++++++++-- 6 files changed, 117 insertions(+), 12 deletions(-) create mode 100644 .vscode/tasks.json diff --git a/.vscode/settings.json b/.vscode/settings.json index a9c7b58..00e2976 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -4,4 +4,5 @@ "${workspaceFolder}/third_party/raylib/src", ], "python.formatting.provider": "yapf", + "liveServer.settings.root": "/target/doc", } \ No newline at end of file diff --git a/.vscode/tasks.json b/.vscode/tasks.json new file mode 100644 index 0000000..a3522a5 --- /dev/null +++ b/.vscode/tasks.json @@ -0,0 +1,23 @@ +{ + "version": "2.0.0", + "tasks": [ + { + "type": "cargo", + "command": "build", + "problemMatcher": [ + "$rustc" + ], + "group": "build", + "label": "Compile Library" + }, + { + "type": "cargo", + "command": "doc", + "problemMatcher": [ + "$rustc" + ], + "group": "build", + "label": "Build Docs" + } + ] +} \ No newline at end of file diff --git a/Cargo.toml b/Cargo.toml index 84d2d32..5d3f1dd 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "raylib-ffi" -version = "4.5.2" +version = "4.5.3" authors = ["Evan Pratten "] edition = "2021" description = "Automatic raw Rust bindings to raylib" diff --git a/README.md b/README.md index 1d89f20..2c48272 100644 --- a/README.md +++ b/README.md @@ -23,6 +23,6 @@ cargo run --example basic ## Versioning policy -`raylib-ffi` follow [SemVer](https://semver.org/). +`raylib-ffi` follows [SemVer](https://semver.org/). The major and minor version numbers of a `raylib-ffi` will always match the version of raylib it was built against. The patch version may be incremented if a rust-only fix is needed at any point. diff --git a/build/bind.rs b/build/bind.rs index 9b3675c..ea41907 100644 --- a/build/bind.rs +++ b/build/bind.rs @@ -1,5 +1,7 @@ use std::path::PathBuf; +use crate::wrap::enums::get_blocked_enum_names; + /// Compiles raylib pub fn compile_raylib(raylib_path: &str) { // Construct a config for running cmake @@ -23,9 +25,18 @@ pub fn compile_raylib(raylib_path: &str) { let destination = cmake_config.build(); // Tell cargo where the libraries might be - println!("cargo:rustc-link-search=native={}", destination.join("lib").display()); - println!("cargo:rustc-link-search=native={}", destination.join("lib32").display()); - println!("cargo:rustc-link-search=native={}", destination.join("lib64").display()); + println!( + "cargo:rustc-link-search=native={}", + destination.join("lib").display() + ); + println!( + "cargo:rustc-link-search=native={}", + destination.join("lib32").display() + ); + println!( + "cargo:rustc-link-search=native={}", + destination.join("lib64").display() + ); } /// Links libraries @@ -62,16 +73,29 @@ pub fn link_libs() { /// Generates `bindings.rs` file pub fn generate_bindings(header_file: &str) { - // Generate the data - let bindings = bindgen::Builder::default() + // Construct a builder for generating bindings + let mut builder = bindgen::Builder::default() .header(header_file) .parse_callbacks(Box::new(bindgen::CargoCallbacks)) - .generate() - .expect("Unable to generate bindings"); + .blocklist_item("DEG2RAD") + .blocklist_item("PI") + .blocklist_item("RAD2DEG") + .blocklist_item("__GNUC_VA_LIST") + .blocklist_item("__bool_true_false_are_defined") + .blocklist_item("false_") + .blocklist_item("true_"); + + // Deny all blocked enums + for enum_name in get_blocked_enum_names() { + builder = builder.blocklist_type(format!("{}.*", enum_name)); + } + + // Generate the bindings + let bindings = builder.generate().expect("Unable to generate bindings"); // Write `src/bindings.rs` let out_path = PathBuf::from(std::env::var("OUT_DIR").unwrap()); bindings .write_to_file(out_path.join("bindings.rs")) .expect("Couldn't write bindings!"); -} \ No newline at end of file +} diff --git a/build/wrap/enums.rs b/build/wrap/enums.rs index 9af05a1..41c4333 100644 --- a/build/wrap/enums.rs +++ b/build/wrap/enums.rs @@ -2,6 +2,55 @@ use std::path::PathBuf; use super::raylib_api::RayLibApiDefinition; +/// A mapping of enum names to the number of tokens to pop off the variant prefix +/// (C enums tend to have prefixes to avoid name collisions) +fn get_prefix_len(name: &str) -> usize { + // Some enums have more than one prefix token + match name { + "CubemapLayout" => 2, + "GamepadAxis" => 2, + "GamepadButton" => 2, + "MaterialMapIndex" => 2, + "MouseButton" => 2, + "MouseCursor" => 2, + "PixelFormat" => 2, + "ShaderAttributeDataType" => 2, + "ShaderLocationIndex" => 2, + "ShaderUniformDataType" => 2, + "TextureFilter" => 2, + "TextureWrap" => 2, + _ => 1, + } +} + +/// Gets a list of enum names for bindgen to skip. +/// We are manually binding them to Rust here, so there is no need to double-translate +pub fn get_blocked_enum_names() -> Vec { + vec![ + "BlendMode", + "CameraMode", + "CameraProjection", + "ConfigFlags", + "CubemapLayout", + "FontType", + "GamepadAxis", + "GamepadButton", + "Gesture", + "KeyboardKey", + "MaterialMapIndex", + "MouseButton", + "MouseCursor", + "NPatchLayout", + "PixelFormat", + "ShaderAttributeDataType", + "ShaderLocationIndex", + "ShaderUniformDataType", + "TextureFilter", + "TextureWrap", + "TraceLogLevel" + ].into_iter().map(|s| s.to_string()).collect() +} + pub fn wrap_exposed_enums(api_defs: RayLibApiDefinition) { // Allocate an output buffer for lines let mut lines = Vec::new(); @@ -29,11 +78,14 @@ pub fn wrap_exposed_enums(api_defs: RayLibApiDefinition) { // Transform the name into a valid Rust identifier let name_parts = variant.name.split('_').collect::>(); + // Ignore the prefix tokens + let name_parts = &name_parts[get_prefix_len(&en.name)..]; + // Capitalize and join let mut name = String::new(); for part in name_parts.iter() { let mut chars = part.chars(); name.push(chars.next().unwrap_or(' ').to_ascii_uppercase()); - name.push_str(chars.as_str()); + name.push_str(chars.as_str().to_ascii_lowercase().as_str()); } // Write the variant declaration @@ -43,11 +95,16 @@ pub fn wrap_exposed_enums(api_defs: RayLibApiDefinition) { // Close the enum declaration lines.push("}".to_string()); + // Write a conversion function + lines.push(format!("impl Into for {} {{", en.name)); + lines.push(format!("\tfn into(self) -> usize {{")); + lines.push(format!("\t\tself as usize")); + lines.push(format!("\t}}")); + lines.push(format!("}}")); } // Write the output file let out_path = PathBuf::from(std::env::var("OUT_DIR").unwrap()); let output = lines.join("\n"); std::fs::write(out_path.join("enums.rs"), output).unwrap(); - }