Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

auditwheel choose higher priority tag when possible #456

Merged
merged 3 commits into from
Mar 3, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 3 additions & 1 deletion .github/workflows/test-docker.yml
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,8 @@ jobs:
with:
python-version: 3.8
- name: Build Docker container
run: docker build --cache-from konstin2/maturin -t maturin .
run: |
docker pull konstin2/maturin
docker build --cache-from konstin2/maturin -t maturin .
- name: Test the Docker container
run: ./test-dockerfile.sh
3 changes: 1 addition & 2 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -72,8 +72,7 @@ scroll = "0.10.2"
indoc = "1.0.2"

[features]
default = ["auditwheel", "log", "upload", "rustls", "human-panic"]
auditwheel = []
default = ["log", "upload", "rustls", "human-panic"]
upload = ["reqwest", "rpassword", "configparser", "dirs"]
password-storage = ["upload", "keyring"]
log = ["pretty_env_logger"]
Expand Down
286 changes: 286 additions & 0 deletions src/auditwheel/audit.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,286 @@
use crate::Manylinux;
use crate::Target;
use anyhow::Result;
use fs_err::File;
use goblin::elf::{sym::STT_FUNC, Elf};
use goblin::strtab::Strtab;
use regex::Regex;
use scroll::Pread;
use std::collections::{HashMap, HashSet};
use std::io;
use std::io::Read;
use std::path::Path;
use thiserror::Error;

use super::policy::{Policy, POLICIES};

/// Error raised during auditing an elf file for manylinux compatibility
#[derive(Error, Debug)]
#[error("Ensuring manylinux compliance failed")]
pub enum AuditWheelError {
/// The wheel couldn't be read
#[error("Failed to read the wheel")]
IOError(#[source] io::Error),
/// Reexports elfkit parsing errors
#[error("Goblin failed to parse the elf file")]
GoblinError(#[source] goblin::error::Error),
/// The elf file isn't manylinux compatible. Contains the list of offending
/// libraries.
#[error(
"Your library links libpython ({0}), which libraries must not do. Have you forgotten to activate the extension-module feature?",
)]
LinksLibPythonError(String),
/// The elf file isn't manylinux compatible. Contains the list of offending
/// libraries.
#[error(
"Your library is not manylinux compliant because it links the following forbidden libraries: {0:?}",
)]
ManylinuxValidationError(Vec<String>),
/// The elf file isn't manylinux compaible. Contains unsupported architecture
#[error(
"Your library is not manylinux compliant because it has unsupported architecture: {0}"
)]
UnsupportedArchitecture(String),
}

/// Structure of "version needed" entries is documented in
/// https://refspecs.linuxfoundation.org/LSB_3.0.0/LSB-PDA/LSB-PDA.junk/symversion.html
#[derive(Clone, Copy, Debug, Pread)]
#[repr(C)]
struct GnuVersionNeed {
/// Version of structure. This value is currently set to 1,
/// and will be reset if the versioning implementation is incompatibly altered.
version: u16,
/// Number of associated verneed array entries.
cnt: u16,
/// Offset to the file name string in the section header, in bytes.
file: u32,
/// Offset to a corresponding entry in the vernaux array, in bytes.
aux: u32,
/// Offset to the next verneed entry, in bytes.
next: u32,
}

/// Version Needed Auxiliary Entries
#[derive(Clone, Copy, Debug, Pread)]
#[repr(C)]
struct GnuVersionNeedAux {
/// Dependency name hash value (ELF hash function).
hash: u32,
/// Dependency information flag bitmask.
flags: u16,
/// Object file version identifier used in the .gnu.version symbol version array.
/// Bit number 15 controls whether or not the object is hidden; if this bit is set,
/// the object cannot be used and the static linker will ignore the symbol's presence in the object.
other: u16,
/// Offset to the dependency name string in the section header, in bytes.
name: u32,
/// Offset to the next vernaux entry, in bytes.
next: u32,
}

#[derive(Clone, Debug)]
struct VersionedLibrary {
/// library name
name: String,
/// versions needed
versions: HashSet<String>,
}

/// Find required dynamic linked libraries with version information
fn find_versioned_libraries(
elf: &Elf,
buffer: &[u8],
) -> Result<Vec<VersionedLibrary>, AuditWheelError> {
let mut symbols = Vec::new();
let section = elf
.section_headers
.iter()
.find(|h| &elf.shdr_strtab[h.sh_name] == ".gnu.version_r");
if let Some(section) = section {
let linked_section = &elf.section_headers[section.sh_link as usize];
linked_section
.check_size(buffer.len())
.map_err(AuditWheelError::GoblinError)?;
let strtab = Strtab::parse(
buffer,
linked_section.sh_offset as usize,
linked_section.sh_size as usize,
0x0,
)
.map_err(AuditWheelError::GoblinError)?;
let num_versions = section.sh_info as usize;
let mut offset = section.sh_offset as usize;
for _ in 0..num_versions {
let ver = buffer
.gread::<GnuVersionNeed>(&mut offset)
.map_err(goblin::error::Error::Scroll)
.map_err(AuditWheelError::GoblinError)?;
let mut versions = HashSet::new();
for _ in 0..ver.cnt {
let ver_aux = buffer
.gread::<GnuVersionNeedAux>(&mut offset)
.map_err(goblin::error::Error::Scroll)
.map_err(AuditWheelError::GoblinError)?;
let aux_name = &strtab[ver_aux.name as usize];
versions.insert(aux_name.to_string());
}
let name = &strtab[ver.file as usize];
// Skip dynamic linker/loader
if name.starts_with("ld-linux") || name == "ld64.so.2" || name == "ld64.so.1" {
continue;
}
symbols.push(VersionedLibrary {
name: name.to_string(),
versions,
});
}
}
Ok(symbols)
}

/// Find incompliant symbols from symbol versions
fn find_incompliant_symbols(
elf: &Elf,
symbol_versions: &[String],
) -> Result<Vec<String>, AuditWheelError> {
let mut symbols = Vec::new();
let strtab = &elf.strtab;
for sym in &elf.syms {
if sym.st_type() == STT_FUNC {
let name = strtab
.get(sym.st_name)
.unwrap_or(Ok("BAD NAME"))
.map_err(AuditWheelError::GoblinError)?;
for symbol_version in symbol_versions {
if name.ends_with(&format!("@{}", symbol_version)) {
symbols.push(name.to_string());
}
}
}
}
Ok(symbols)
}

fn policy_is_satisfied(
policy: &Policy,
elf: &Elf,
arch: &str,
deps: &[String],
versioned_libraries: &[VersionedLibrary],
) -> Result<i64, AuditWheelError> {
let arch_versions = &policy
.symbol_versions
.get(arch)
.ok_or_else(|| AuditWheelError::UnsupportedArchitecture(arch.to_string()))?;
let mut offenders = HashSet::new();
for dep in deps {
// Skip dynamic linker/loader
if dep.starts_with("ld-linux") || dep == "ld64.so.2" || dep == "ld64.so.1" {
continue;
}
if !policy.lib_whitelist.contains(dep) {
offenders.insert(dep.clone());
}
}
for library in versioned_libraries {
if !policy.lib_whitelist.contains(&library.name) {
offenders.insert(library.name.clone());
continue;
}
let mut versions: HashMap<String, HashSet<String>> = HashMap::new();
for v in &library.versions {
let mut parts = v.splitn(2, '_');
let name = parts.next().unwrap();
let version = parts.next().unwrap();
versions
.entry(name.to_string())
.or_default()
.insert(version.to_string());
}
for (name, versions_needed) in versions.iter() {
let versions_allowed = &arch_versions[name];
if !versions_needed.is_subset(versions_allowed) {
let offending_versions: Vec<&str> = versions_needed
.difference(versions_allowed)
.map(|v| v.as_ref())
.collect();
let offending_symbol_versions: Vec<String> = offending_versions
.iter()
.map(|v| format!("{}_{}", name, v))
.collect();
let offending_symbols = find_incompliant_symbols(&elf, &offending_symbol_versions)?;
let offender = if offending_symbols.is_empty() {
format!(
"{} offending versions: {}",
library.name,
offending_symbol_versions.join(", ")
)
} else {
format!(
"{} offending symbols: {}",
library.name,
offending_symbols.join(", ")
)
};
offenders.insert(offender);
}
}
}
// Checks if we can give a more helpful error message
let is_libpython = Regex::new(r"^libpython3\.\d+\.so\.\d+\.\d+$").unwrap();
let offenders: Vec<String> = offenders.into_iter().collect();
match offenders.as_slice() {
[] => Ok(policy.priority),
[lib] if is_libpython.is_match(lib) => {
Err(AuditWheelError::LinksLibPythonError(lib.clone()))
}
offenders => Err(AuditWheelError::ManylinuxValidationError(
offenders.to_vec(),
)),
}
}

/// An reimplementation of auditwheel, which checks elf files for
/// manylinux compliance. Returns an error for non compliant elf files
///
/// Checks for the libraries marked as NEEDED and symbol versions
pub fn auditwheel_rs(
path: &Path,
target: &Target,
manylinux: &Manylinux,
) -> Result<Option<Policy>, AuditWheelError> {
if !target.is_linux() || matches!(manylinux, Manylinux::Off) {
return Ok(None);
}
let arch = target.target_arch().to_string();
let mut file = File::open(path).map_err(AuditWheelError::IOError)?;
let mut buffer = Vec::new();
file.read_to_end(&mut buffer)
.map_err(AuditWheelError::IOError)?;
let elf = Elf::parse(&buffer).map_err(AuditWheelError::GoblinError)?;
// This returns essentially the same as ldd
let deps: Vec<String> = elf.libraries.iter().map(ToString::to_string).collect();
let versioned_libraries = find_versioned_libraries(&elf, &buffer)?;

let policy = POLICIES
.iter()
.find(|p| p.name == manylinux.to_string())
.unwrap();
let mut match_policies = Vec::new();
match_policies.push(policy_is_satisfied(
policy,
&elf,
&arch,
&deps,
&versioned_libraries,
)?);
for policy in policy.higher_priority_policies() {
if let Ok(priority) = policy_is_satisfied(policy, &elf, &arch, &deps, &versioned_libraries)
{
match_policies.push(priority);
}
}
let max_priority = match_policies.into_iter().max().unwrap_or_default();
Ok(Policy::from_priority(max_priority))
}
Loading