Add cmake compatibility to c-api (#4369)
* Add cmake compatibility to c-api * Add CMake documentation to wasmtime.h * Add CMake instructions in examples * Modify CI for CMake support * Use correct rust in CI * Trigger build * Refactor run-examples * Reintroduce example_to_run in run-examples * Replace run-examples crate with cmake * Fix markdown formatting in examples readme * Fix cmake test quotes * Build rust wasm before cmake tests * Pass CTEST_OUTPUT_ON_FAILURE * Another cmake test * Handle os differences in cmake test * Fix bugs in memory and multimemory examples
This commit is contained in:
64
crates/c-api/CMakeLists.txt
Normal file
64
crates/c-api/CMakeLists.txt
Normal file
@@ -0,0 +1,64 @@
|
||||
cmake_minimum_required(VERSION 3.10)
|
||||
|
||||
option(BUILD_SHARED_LIBS "Build using shared libraries" OFF)
|
||||
|
||||
if (CMAKE_BUILD_TYPE STREQUAL "Release")
|
||||
set(WASMTIME_BUILD_TYPE_FLAG "--release")
|
||||
set(WASMTIME_BUILD_TYPE "release")
|
||||
else()
|
||||
set(WASMTIME_BUILD_TYPE "debug")
|
||||
endif()
|
||||
|
||||
if (BUILD_SHARED_LIBS)
|
||||
# Copy shared library into build directory
|
||||
if(WIN32)
|
||||
set(WASMTIME_INSTALL_COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||
${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/wasmtime.dll
|
||||
${CMAKE_BINARY_DIR})
|
||||
elseif(APPLE)
|
||||
set(WASMTIME_INSTALL_COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||
${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/libwasmtime.dylib
|
||||
${CMAKE_BINARY_DIR})
|
||||
else()
|
||||
set(WASMTIME_INSTALL_COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||
${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/libwasmtime.so
|
||||
${CMAKE_BINARY_DIR})
|
||||
endif()
|
||||
endif()
|
||||
|
||||
include(ExternalProject)
|
||||
ExternalProject_Add(
|
||||
wasmtime-crate
|
||||
DOWNLOAD_COMMAND ""
|
||||
CONFIGURE_COMMAND ""
|
||||
INSTALL_COMMAND "${WASMTIME_INSTALL_COMMAND}"
|
||||
BUILD_COMMAND cargo build ${WASMTIME_BUILD_TYPE_FLAG}
|
||||
BINARY_DIR ${CMAKE_CURRENT_SOURCE_DIR}
|
||||
BUILD_ALWAYS ON)
|
||||
add_library(wasmtime INTERFACE)
|
||||
add_dependencies(wasmtime wasmtime-crate)
|
||||
|
||||
if (BUILD_SHARED_LIBS)
|
||||
if(WIN32)
|
||||
target_link_libraries(wasmtime INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/wasmtime.dll.lib)
|
||||
elseif(APPLE)
|
||||
target_link_libraries(wasmtime INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/libwasmtime.dylib)
|
||||
set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -Wl,-rpath='$ORIGIN'")
|
||||
else()
|
||||
target_link_libraries(wasmtime INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/libwasmtime.so)
|
||||
set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -Wl,-rpath='$ORIGIN'")
|
||||
endif()
|
||||
else()
|
||||
if(WIN32)
|
||||
target_compile_options(wasmtime INTERFACE -DWASM_API_EXTERN= -DWASI_API_EXTERN=)
|
||||
target_link_libraries(wasmtime INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/wasmtime.lib
|
||||
ws2_32 advapi32 userenv ntdll shell32 ole32 bcrypt)
|
||||
elseif(APPLE)
|
||||
target_link_libraries(wasmtime INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/libwasmtime.a)
|
||||
else()
|
||||
target_link_libraries(wasmtime INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/../../target/${WASMTIME_BUILD_TYPE}/libwasmtime.a
|
||||
pthread dl m)
|
||||
endif()
|
||||
endif()
|
||||
|
||||
target_include_directories(wasmtime INTERFACE ${CMAKE_CURRENT_SOURCE_DIR}/include ${CMAKE_CURRENT_SOURCE_DIR}/wasm-c-api/include)
|
||||
@@ -28,6 +28,22 @@
|
||||
* as a `lib` directory with both a static archive and a dynamic library of
|
||||
* Wasmtime. You can link to either of them as you see fit.
|
||||
*
|
||||
* ## Installing the C API through CMake
|
||||
*
|
||||
* CMake can be used to make the process of linking and compiling easier. An
|
||||
* example of this if you have wasmtime as a git submodule at
|
||||
* `third_party/wasmtime`:
|
||||
* ```
|
||||
* add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/third_party/wasmtime/crates/c-api
|
||||
* ${CMAKE_CURRENT_BINARY_DIR}/wasmtime)
|
||||
* ...
|
||||
* target_include_directories(YourProject PUBLIC wasmtime)
|
||||
* target_link_libraries(YourProject PUBLIC wasmtime)
|
||||
* ```
|
||||
* `BUILD_SHARED_LIBS` is provided as a define if you would like to build a
|
||||
* shared library instead. You must distribute the appropriate shared library
|
||||
* for your platform if you do this.
|
||||
*
|
||||
* ## Linking against the C API
|
||||
*
|
||||
* You'll want to arrange the `include` directory of the C API to be in your
|
||||
@@ -166,8 +182,8 @@
|
||||
|
||||
#include <wasi.h>
|
||||
#include <wasmtime/config.h>
|
||||
#include <wasmtime/error.h>
|
||||
#include <wasmtime/engine.h>
|
||||
#include <wasmtime/error.h>
|
||||
#include <wasmtime/extern.h>
|
||||
#include <wasmtime/func.h>
|
||||
#include <wasmtime/global.h>
|
||||
|
||||
@@ -1,10 +0,0 @@
|
||||
[package]
|
||||
name = "run-examples"
|
||||
version = "0.19.0"
|
||||
authors = ["The Wasmtime Project Developers"]
|
||||
edition = "2021"
|
||||
publish = false
|
||||
|
||||
[dependencies]
|
||||
anyhow = "1.0.31"
|
||||
cc = "1.0"
|
||||
@@ -1,7 +0,0 @@
|
||||
fn main() {
|
||||
println!(
|
||||
"cargo:rustc-env=TARGET={}",
|
||||
std::env::var("TARGET").unwrap()
|
||||
);
|
||||
println!("cargo:rerun-if-changed=build.rs");
|
||||
}
|
||||
@@ -1,121 +0,0 @@
|
||||
use anyhow::Context;
|
||||
use std::collections::BTreeSet;
|
||||
use std::process::Command;
|
||||
|
||||
fn main() -> anyhow::Result<()> {
|
||||
let example_to_run = std::env::args().nth(1);
|
||||
let mut examples = BTreeSet::new();
|
||||
for e in std::fs::read_dir("examples")? {
|
||||
let e = e?;
|
||||
let path = e.path();
|
||||
let dir = e.metadata()?.is_dir();
|
||||
if let Some("wat") = path.extension().and_then(|s| s.to_str()) {
|
||||
continue;
|
||||
}
|
||||
|
||||
examples.insert((path.file_stem().unwrap().to_str().unwrap().to_owned(), dir));
|
||||
}
|
||||
|
||||
println!("======== Building libwasmtime.a ===========");
|
||||
run(Command::new("cargo")
|
||||
.args(&["build"])
|
||||
.current_dir("crates/c-api"))?;
|
||||
|
||||
for (example, is_dir) in examples {
|
||||
if example == "README" {
|
||||
continue;
|
||||
}
|
||||
if let Some(example_to_run) = &example_to_run {
|
||||
if !example.contains(&example_to_run[..]) {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
if is_dir {
|
||||
println!("======== Rust wasm file `{}` ============", example);
|
||||
let target = if example == "fib-debug" {
|
||||
"wasm32-unknown-unknown"
|
||||
} else {
|
||||
"wasm32-wasi"
|
||||
};
|
||||
run(Command::new("cargo")
|
||||
.arg("build")
|
||||
.arg("-p")
|
||||
.arg(format!("example-{}-wasm", example))
|
||||
.arg("--target")
|
||||
.arg(target))?;
|
||||
}
|
||||
println!("======== Rust example `{}` ============", example);
|
||||
let mut cargo_cmd = Command::new("cargo");
|
||||
cargo_cmd.arg("run").arg("--example").arg(&example);
|
||||
|
||||
if example.contains("tokio") {
|
||||
cargo_cmd.arg("--features").arg("wasmtime-wasi/tokio");
|
||||
}
|
||||
run(&mut cargo_cmd)?;
|
||||
|
||||
println!("======== C/C++ example `{}` ============", example);
|
||||
for extension in ["c", "cc"].iter() {
|
||||
let mut cmd = cc::Build::new()
|
||||
.opt_level(0)
|
||||
.cargo_metadata(false)
|
||||
.target(env!("TARGET"))
|
||||
.host(env!("TARGET"))
|
||||
.include("crates/c-api/include")
|
||||
.include("crates/c-api/wasm-c-api/include")
|
||||
.define("WASM_API_EXTERN", Some("")) // static linkage, not dynamic
|
||||
.warnings(false)
|
||||
.get_compiler()
|
||||
.to_command();
|
||||
|
||||
let file = if is_dir {
|
||||
format!("examples/{}/main.{}", example, extension)
|
||||
} else {
|
||||
format!("examples/{}.{}", example, extension)
|
||||
};
|
||||
|
||||
if !std::path::Path::new(&file).exists() {
|
||||
// C and C++ files are optional so we can skip them.
|
||||
continue;
|
||||
}
|
||||
|
||||
cmd.arg(file);
|
||||
let exe = if cfg!(windows) {
|
||||
cmd.arg("target/debug/wasmtime.lib")
|
||||
.arg("ws2_32.lib")
|
||||
.arg("advapi32.lib")
|
||||
.arg("userenv.lib")
|
||||
.arg("ntdll.lib")
|
||||
.arg("shell32.lib")
|
||||
.arg("ole32.lib")
|
||||
.arg("bcrypt.lib");
|
||||
if is_dir {
|
||||
"./main.exe".to_string()
|
||||
} else {
|
||||
format!("./{}.exe", example)
|
||||
}
|
||||
} else {
|
||||
cmd.arg("target/debug/libwasmtime.a").arg("-o").arg("foo");
|
||||
"./foo".to_string()
|
||||
};
|
||||
if cfg!(target_os = "linux") {
|
||||
cmd.arg("-lpthread").arg("-ldl").arg("-lm");
|
||||
}
|
||||
run(&mut cmd)?;
|
||||
|
||||
run(&mut Command::new(exe))?;
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn run(cmd: &mut Command) -> anyhow::Result<()> {
|
||||
(|| -> anyhow::Result<()> {
|
||||
let s = cmd.status()?;
|
||||
if !s.success() {
|
||||
anyhow::bail!("Exited with failure status: {}", s);
|
||||
}
|
||||
Ok(())
|
||||
})()
|
||||
.with_context(|| format!("failed to run `{:?}`", cmd))
|
||||
}
|
||||
Reference in New Issue
Block a user