Skip to content

Commit

Permalink
Fix objcopy and nm on ARM platforms (0.2.x) (#30)
Browse files Browse the repository at this point in the history
* 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.
  • Loading branch information
gmartin82 authored Oct 29, 2024
1 parent 700d028 commit b0a271c
Show file tree
Hide file tree
Showing 7 changed files with 147 additions and 30 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/rust.yml
Original file line number Diff line number Diff line change
Expand Up @@ -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:
Expand Down
4 changes: 2 additions & 2 deletions Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "cyclors"
version = "0.2.2"
version = "0.2.3"
authors = ["kydos <[email protected]>"]
license = "Apache-2.0"
readme = "README.md"
Expand All @@ -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]
Expand Down
62 changes: 35 additions & 27 deletions build.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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<HashSet<String>, 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<String> = HashSet::new();
for line in stdout.lines() {
let mut result: HashSet<String> = 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)),
}
}

Expand Down Expand Up @@ -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 {}",
Expand Down
39 changes: 39 additions & 0 deletions nm/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -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..."
)
10 changes: 10 additions & 0 deletions nm/check_errors.sh
Original file line number Diff line number Diff line change
@@ -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
50 changes: 50 additions & 0 deletions objcopy/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -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..."
)
10 changes: 10 additions & 0 deletions objcopy/check_errors.sh
Original file line number Diff line number Diff line change
@@ -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

0 comments on commit b0a271c

Please sign in to comment.