From b0a271cb9d3d491b13a16c0a081e7861f261f780 Mon Sep 17 00:00:00 2001 From: Geoff Martin Date: Tue, 29 Oct 2024 03:07:40 -0700 Subject: [PATCH] Fix objcopy and nm on ARM platforms (0.2.x) (#30) * objcopy and nm commands now executed via cmake to use architecture specific tool. * Adding ARM build to CI. * Updated build script functions to use cmake for nm and objcopy. * Switched check_errors.sh script to use the sh interpreter rather than bash. --- .github/workflows/rust.yml | 2 +- Cargo.toml | 4 +-- build.rs | 62 +++++++++++++++++++++----------------- nm/CMakeLists.txt | 39 ++++++++++++++++++++++++ nm/check_errors.sh | 10 ++++++ objcopy/CMakeLists.txt | 50 ++++++++++++++++++++++++++++++ objcopy/check_errors.sh | 10 ++++++ 7 files changed, 147 insertions(+), 30 deletions(-) create mode 100644 nm/CMakeLists.txt create mode 100755 nm/check_errors.sh create mode 100644 objcopy/CMakeLists.txt create mode 100755 objcopy/check_errors.sh diff --git a/.github/workflows/rust.yml b/.github/workflows/rust.yml index c1d7e23..71a87fc 100644 --- a/.github/workflows/rust.yml +++ b/.github/workflows/rust.yml @@ -14,7 +14,7 @@ jobs: strategy: fail-fast: true matrix: - os: [ubuntu-latest, windows-latest, macos-latest] + os: [ubuntu-latest, windows-latest, macos-latest, [ubuntu-22.04, ARM64]] runs-on: ${{ matrix.os }} steps: diff --git a/Cargo.toml b/Cargo.toml index 9ad9647..badfd31 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "cyclors" -version = "0.2.2" +version = "0.2.3" authors = ["kydos "] license = "Apache-2.0" readme = "README.md" @@ -25,7 +25,7 @@ serde = { version = "1.0.154", features = ["derive"] } serde_json = "1.0.94" [build-dependencies] -bindgen = "0.68" +bindgen = "0.69" cmake = "0.1" [features] diff --git a/build.rs b/build.rs index 4310032..1015e67 100644 --- a/build.rs +++ b/build.rs @@ -219,37 +219,42 @@ fn main() { #[cfg(all(target_os = "linux", not(feature = "iceoryx")))] fn get_defined_symbols(lib_dir: &Path, lib_name: &str) -> Result, String> { + use std::io::{BufRead, BufReader}; + let lib_path = lib_dir.to_path_buf().join(lib_name); + let mut nm_file_name = lib_name.to_owned(); + nm_file_name.push_str(".nm"); + let symbol_file_path = lib_dir.to_path_buf().join(nm_file_name); + + let mut nm = cmake::Config::new("nm"); + nm.build_target("all") + .define("LIB_PATH", lib_path.clone()) + .build(); + + match File::open(symbol_file_path.clone()) { + Ok(symbol_file) => { + let reader = BufReader::new(symbol_file); - let rc = Command::new("nm") - .arg("--defined-only") - .arg("--print-file-name") - .arg(lib_path) - .output(); - - match rc { - Ok(output) => { - let stdout = String::from_utf8_lossy(&output.stdout); - let stderr = String::from_utf8_lossy(&output.stderr); - - match stderr.is_empty() { - true => { - let mut result: HashSet = HashSet::new(); - for line in stdout.lines() { + let mut result: HashSet = HashSet::new(); + for line in reader.lines() { + match line { + Ok(line) => { let tokens: Vec<&str> = line.split_whitespace().collect(); let symbol = *tokens.last().unwrap(); result.insert(String::from(symbol)); } - Ok(result) + Err(_) => return Err(format!("Failed to run nm on library {}", lib_name)), } - false => Err(format!( - "Failed to run nm on library {} (stderr: {})", - lib_name, - String::from(stderr) - )), } + Ok(result) + } + Err(_) => { + println!( + "nm file open problem: {}", + symbol_file_path.to_str().unwrap() + ); + Err(format!("Failed to run nm on library {}", lib_name)) } - Err(_) => Err(format!("Failed to run nm on library {}", lib_name)), } } @@ -290,11 +295,14 @@ fn prefix_symbols( lib_name )); } - let arg = format!("--redefine-syms={}", symbol_file_path.to_str().unwrap()); - match Command::new("objcopy").arg(arg).arg(lib_file_path).output() { - Ok(_) => Ok(()), - Err(_) => Err(format!("Failed to run objcopy on library {}", lib_name)), - } + + let mut objcopy = cmake::Config::new("objcopy"); + objcopy + .build_target("all") + .define("LIB_PATH", lib_file_path.clone()) + .define("SYMBOL_FILE_PATH", symbol_file_path.clone()) + .build(); + Ok(()) } Err(_) => Err(format!( "Failed to create symbol file for library {}", diff --git a/nm/CMakeLists.txt b/nm/CMakeLists.txt new file mode 100644 index 0000000..300d5ce --- /dev/null +++ b/nm/CMakeLists.txt @@ -0,0 +1,39 @@ +cmake_minimum_required(VERSION 3.12) +project(NM) + +# Include the CMakeFindBinUtils module to find nm +include(CMakeFindBinUtils) + +# Ensure nm is available +if(NOT CMAKE_NM) + message(FATAL_ERROR "CMAKE_NM is not defined. Ensure nm is installed on your system!") +else() + message(STATUS "CMAKE_NM found: ${CMAKE_NM}") +endif() + +# Ensure LIB_PATH is defined and exists +if(NOT DEFINED LIB_PATH) + message(FATAL_ERROR "LIB_PATH not specified!") +else() + if(NOT EXISTS ${LIB_PATH}) + message(FATAL_ERROR "Library not found: ${LIB_PATH}") + else() + message(STATUS "LIB_PATH: ${LIB_PATH}") + endif() +endif() + +# Custom target to run nm on the library +add_custom_target(read_symbols ALL + COMMAND ${CMAKE_COMMAND} -E echo "Running nm on ${LIB_PATH}..." + + # Run nm and redirect stderr to a file + COMMAND ${CMAKE_NM} --defined-only --print-file-name ${LIB_PATH} 1> ${LIB_PATH}.nm 2> ${LIB_PATH}.stderr + + # Check if stderr is empty (i.e., no errors were produced) + COMMAND ${CMAKE_COMMAND} -E echo "Checking for nm errors..." + COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/check_errors.sh ${LIB_PATH}.stderr + + # Clean up stderr.txt after checking for errors + COMMAND ${CMAKE_COMMAND} -E remove ${LIB_PATH}.stderr + COMMENT "Reading library symbols with nm..." +) diff --git a/nm/check_errors.sh b/nm/check_errors.sh new file mode 100755 index 0000000..f5b1166 --- /dev/null +++ b/nm/check_errors.sh @@ -0,0 +1,10 @@ +#!/bin/sh + +if [ ! -s $1 ]; then + echo 'Command succeeded with no stderr output.' + exit 0 +else + echo 'Command failed with errors:' + cat stderr.txt + exit 1 +fi diff --git a/objcopy/CMakeLists.txt b/objcopy/CMakeLists.txt new file mode 100644 index 0000000..5589699 --- /dev/null +++ b/objcopy/CMakeLists.txt @@ -0,0 +1,50 @@ +cmake_minimum_required(VERSION 3.12) +project(Objcopy) + +# Include the CMakeFindBinUtils module to find objcopy +include(CMakeFindBinUtils) + +# Ensure objcopy is available +if(NOT CMAKE_OBJCOPY) + message(FATAL_ERROR "CMAKE_OBJCOPY is not defined. Ensure objcopy is installed on your system!") +else() + message(STATUS "CMAKE_OBJCOPY found: ${CMAKE_OBJCOPY}") +endif() + +# Ensure LIB_PATH is defined and exists +if(NOT DEFINED LIB_PATH) + message(FATAL_ERROR "LIB_PATH not specified!") +else() + if(NOT EXISTS ${LIB_PATH}) + message(FATAL_ERROR "Library not found: ${LIB_PATH}") + else() + message(STATUS "LIB_PATH: ${LIB_PATH}") + endif() +endif() + +# Ensure SYMBOL_FILE_PATH is defined and exists +if(NOT DEFINED SYMBOL_FILE_PATH) + message(FATAL_ERROR "SYMBOL_FILE_PATH not specified!") +else() + if(NOT EXISTS ${SYMBOL_FILE_PATH}) + message(FATAL_ERROR "Symbol file not found: ${SYMBOL_FILE_PATH}") + else() + message(STATUS "SYMBOL_FILE_PATH: ${SYMBOL_FILE_PATH}") + endif() +endif() + +# Custom target to mangle the library +add_custom_target(mangle_library ALL + COMMAND ${CMAKE_COMMAND} -E echo "Running objcopy --redefine-syms on ${LIB_PATH} with symbols from ${SYMBOL_FILE_PATH}..." + + # Run objcopy and redirect stderr to a file + COMMAND ${CMAKE_OBJCOPY} --redefine-syms=${SYMBOL_FILE_PATH} ${LIB_PATH} 2> ${LIB_PATH}.stderr + + # Check if stderr is empty (i.e., no errors were produced) + COMMAND ${CMAKE_COMMAND} -E echo "Checking for objcopy errors..." + COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/check_errors.sh ${LIB_PATH}.stderr + + # Clean up stderr.txt after checking for errors + COMMAND ${CMAKE_COMMAND} -E remove ${LIB_PATH}.stderr + COMMENT "Mangling library with objcopy..." +) diff --git a/objcopy/check_errors.sh b/objcopy/check_errors.sh new file mode 100755 index 0000000..f5b1166 --- /dev/null +++ b/objcopy/check_errors.sh @@ -0,0 +1,10 @@ +#!/bin/sh + +if [ ! -s $1 ]; then + echo 'Command succeeded with no stderr output.' + exit 0 +else + echo 'Command failed with errors:' + cat stderr.txt + exit 1 +fi