use std::path::{Path, PathBuf};
use eyre::bail;
use crate::result::Result;
pub struct Replacement {
pub placeholder: &'static [u8],
pub value: Vec<u8>,
}
pub fn standard_replacements() -> Vec<Replacement> {
let prefix_buf = super::prefix::prefix();
let prefix = prefix_buf.to_string_lossy();
let repository_buf = super::prefix::repository();
let repository = repository_buf.to_string_lossy();
let macos = cfg!(target_os = "macos");
vec![
Replacement {
placeholder: b"@@HOMEBREW_PREFIX@@",
value: prefix.as_bytes().to_vec(),
},
Replacement {
placeholder: b"@@HOMEBREW_CELLAR@@",
value: format!("{prefix}/Cellar").into_bytes(),
},
Replacement {
placeholder: b"@@HOMEBREW_REPOSITORY@@",
value: repository.as_bytes().to_vec(),
},
Replacement {
placeholder: b"@@HOMEBREW_LIBRARY@@",
value: format!("{repository}/Library").into_bytes(),
},
Replacement {
placeholder: b"@@HOMEBREW_PERL@@",
value: if macos {
b"/usr/bin/perl".to_vec()
} else {
format!("{prefix}/opt/perl/bin/perl").into_bytes()
},
},
Replacement {
placeholder: b"@@HOMEBREW_JAVA@@",
value: if macos {
format!("{prefix}/opt/openjdk/libexec/openjdk.jdk/Contents/Home").into_bytes()
} else {
format!("{prefix}/opt/openjdk/libexec").into_bytes()
},
},
]
}
#[derive(Debug, Default)]
pub struct RelocationReport {
pub changed_files: Vec<PathBuf>,
pub changed_machos: Vec<PathBuf>,
}
fn is_macho(content: &[u8]) -> bool {
if content.len() < 4 {
return false;
}
matches!(
u32::from_be_bytes([content[0], content[1], content[2], content[3]]),
0xfeedface | 0xcefaedfe | 0xfeedfacf | 0xcffaedfe | 0xcafebabe | 0xbebafeca
)
}
fn contains_any_placeholder(content: &[u8], replacements: &[Replacement]) -> bool {
replacements
.iter()
.any(|r| memmem(content, r.placeholder).is_some())
}
fn memmem(haystack: &[u8], needle: &[u8]) -> Option<usize> {
haystack
.windows(needle.len())
.position(|window| window == needle)
}
fn replace_text(content: &[u8], replacements: &[Replacement]) -> Vec<u8> {
let mut out = content.to_vec();
for r in replacements {
let mut result = Vec::with_capacity(out.len());
let mut rest: &[u8] = &out;
while let Some(pos) = memmem(rest, r.placeholder) {
result.extend_from_slice(&rest[..pos]);
result.extend_from_slice(&r.value);
rest = &rest[pos + r.placeholder.len()..];
}
result.extend_from_slice(rest);
out = result;
}
out
}
fn replace_in_binary(
content: &mut [u8],
replacements: &[Replacement],
path: &Path,
) -> Result<bool> {
let mut changed = false;
for r in replacements {
let mut search_from = 0;
while let Some(rel_pos) = memmem(&content[search_from..], r.placeholder) {
let start = search_from + rel_pos;
let str_end = content[start..]
.iter()
.position(|&b| b == 0)
.map(|p| start + p)
.unwrap_or(content.len());
let slot_end = content[str_end..]
.iter()
.position(|&b| b != 0)
.map(|p| str_end + p)
.unwrap_or(content.len());
let old = content[start..str_end].to_vec();
let mut new = r.value.clone();
new.extend_from_slice(&old[r.placeholder.len()..]);
let slot = slot_end.saturating_sub(start);
if new.len() + 1 > slot {
bail!(
"cannot relocate {}: replacement for {} does not fit ({} > {} bytes)",
path.display(),
String::from_utf8_lossy(r.placeholder),
new.len() + 1,
slot,
);
}
content[start..start + new.len()].copy_from_slice(&new);
for b in &mut content[start + new.len()..slot_end] {
*b = 0;
}
changed = true;
search_from = start + new.len();
}
}
Ok(changed)
}
pub fn relocate_keg(keg: &Path, formula_name: &str) -> Result<RelocationReport> {
let replacements = standard_replacements();
let elf_opts = super::elf::LinkageOpts::for_formula(formula_name);
let patch_elf = formula_name != "glibc" && !formula_name.starts_with("glibc@");
let mut report = RelocationReport::default();
for entry in walkdir::WalkDir::new(keg).follow_links(false) {
let entry = entry?;
if !entry.file_type().is_file() {
continue;
}
let path = entry.path();
let content = crate::file::read(path)?;
if !contains_any_placeholder(&content, &replacements) {
continue;
}
let perms = path.metadata()?.permissions();
let mut writable = perms.clone();
std::os::unix::fs::PermissionsExt::set_mode(
&mut writable,
std::os::unix::fs::PermissionsExt::mode(&perms) | 0o200,
);
std::fs::set_permissions(path, writable)?;
let macho = is_macho(&content);
let elf = cfg!(target_os = "linux") && super::elf::is_elf(&content);
if macho || (!elf && content.contains(&0)) {
let mut content = content;
let mut changed = macho && super::macho::patch(&mut content, &replacements, path)?;
changed |= replace_in_binary(&mut content, &replacements, path)?;
if changed {
crate::file::write(path, &content)?;
if macho {
report.changed_machos.push(path.to_path_buf());
}
report.changed_files.push(path.to_path_buf());
}
} else if elf {
if patch_elf {
let mut content = content;
if super::elf::patch(&mut content, &elf_opts, path)? {
crate::file::write(path, &content)?;
report.changed_files.push(path.to_path_buf());
}
}
} else {
let new_content = replace_text(&content, &replacements);
if new_content != content {
crate::file::write(path, &new_content)?;
report.changed_files.push(path.to_path_buf());
}
}
std::fs::set_permissions(path, perms)?;
}
Ok(report)
}
pub fn codesign(files: &[PathBuf]) -> Result<()> {
for file in files {
let res = crate::cmd::cmd(
"/usr/bin/codesign",
[
"--sign",
"-",
"--force",
"--preserve-metadata=entitlements,requirements,flags,runtime",
&file.to_string_lossy(),
],
)
.stderr_capture()
.stdout_capture()
.unchecked()
.run()?;
if !res.status.success() {
bail!(
"codesign failed for {}: {}",
file.display(),
String::from_utf8_lossy(&res.stderr).trim()
);
}
}
Ok(())
}
#[cfg(test)]
pub(super) mod tests {
use super::*;
pub(in super::super) fn test_replacements() -> Vec<Replacement> {
vec![
Replacement {
placeholder: b"@@HOMEBREW_PREFIX@@",
value: b"/opt/homebrew".to_vec(),
},
Replacement {
placeholder: b"@@HOMEBREW_CELLAR@@",
value: b"/opt/homebrew/Cellar".to_vec(),
},
]
}
#[test]
fn test_replace_text() {
let replacements = test_replacements();
let content = b"#!@@HOMEBREW_PREFIX@@/bin/bash\nCELLAR=@@HOMEBREW_CELLAR@@/foo\n";
let out = replace_text(content, &replacements);
assert_eq!(
String::from_utf8_lossy(&out),
"#!/opt/homebrew/bin/bash\nCELLAR=/opt/homebrew/Cellar/foo\n"
);
}
#[test]
fn test_replace_in_binary_shrinking() {
let replacements = test_replacements();
let mut content = b"@@HOMEBREW_PREFIX@@/lib/libx.dylib\0\0\0\0after".to_vec();
let changed = replace_in_binary(&mut content, &replacements, Path::new("test")).unwrap();
assert!(changed);
assert_eq!(
&content[..],
b"/opt/homebrew/lib/libx.dylib\0\0\0\0\0\0\0\0\0\0after"
);
}
#[test]
fn test_replace_in_binary_growing_fits_in_padding() {
let replacements = test_replacements();
let mut content = b"@@HOMEBREW_CELLAR@@/foo\0\0\0after".to_vec();
let changed = replace_in_binary(&mut content, &replacements, Path::new("test")).unwrap();
assert!(changed);
assert_eq!(&content[..], b"/opt/homebrew/Cellar/foo\0\0after");
}
#[test]
fn test_replace_in_binary_growing_does_not_fit() {
let replacements = test_replacements();
let mut content = b"@@HOMEBREW_CELLAR@@/foo\0after".to_vec();
let res = replace_in_binary(&mut content, &replacements, Path::new("test"));
assert!(res.is_err());
}
#[test]
fn test_is_macho() {
assert!(is_macho(&0xfeedfacf_u32.to_be_bytes()));
assert!(is_macho(&0xcafebabe_u32.to_be_bytes()));
assert!(!is_macho(b"#!/bin/bash"));
}
}