roc/crates/repl_wasm/build.rs

66 lines
1.9 KiB
Rust

use std::env;
use std::path::PathBuf;
use std::process::Command;
use roc_builtins::bitcode;
use wasi_libc_sys::{WASI_COMPILER_RT_PATH, WASI_LIBC_PATH};
const PLATFORM_FILENAME: &str = "repl_platform";
const PRE_LINKED_BINARY: [&str; 2] = ["src", "pre_linked_binary.o"];
fn main() {
println!("cargo:rerun-if-changed=build.rs");
let source_path = format!("src/{}.c", PLATFORM_FILENAME);
println!("cargo:rerun-if-changed={}", source_path);
// Zig can produce *either* an object containing relocations OR an object containing libc code
// But we want both, so we have to compile twice with different flags, then link them
// Create an object file with relocations
let out_dir = env::var("OUT_DIR").unwrap();
let platform_obj = build_wasm_platform(&out_dir, &source_path);
let pre_linked_binary_path: PathBuf = PRE_LINKED_BINARY.iter().collect();
Command::new(&zig_executable())
.args([
"wasm-ld",
bitcode::BUILTINS_WASM32_OBJ_PATH,
platform_obj.to_str().unwrap(),
WASI_COMPILER_RT_PATH,
WASI_LIBC_PATH,
"-o",
pre_linked_binary_path.to_str().unwrap(),
"--export-all",
"--no-entry",
"--relocatable",
])
.output()
.unwrap();
}
fn zig_executable() -> String {
match std::env::var("ROC_ZIG") {
Ok(path) => path,
Err(_) => "zig".into(),
}
}
fn build_wasm_platform(out_dir: &str, source_path: &str) -> PathBuf {
let mut platform_obj = PathBuf::from(out_dir).join(PLATFORM_FILENAME);
platform_obj.set_extension("o");
Command::new(&zig_executable())
.args([
"build-lib",
"-target",
"wasm32-wasi",
"-lc",
source_path,
&format!("-femit-bin={}", platform_obj.to_str().unwrap()),
])
.output()
.unwrap();
platform_obj
}