Add basic support for precompiling plugins
This commit is contained in:
parent
895747476f
commit
a16fc2ba0c
5 changed files with 51 additions and 12 deletions
|
@ -13,3 +13,6 @@ serde_json = "1.0"
|
||||||
bincode = "1.3"
|
bincode = "1.3"
|
||||||
pollster = "0.2.5"
|
pollster = "0.2.5"
|
||||||
smol = "1.2.5"
|
smol = "1.2.5"
|
||||||
|
|
||||||
|
[build-dependencies]
|
||||||
|
wasmtime = "0.38.1"
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
use std::path::Path;
|
use std::{io::Write, path::Path};
|
||||||
|
use wasmtime::{Config, Engine};
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let base = Path::new("../../plugins");
|
let base = Path::new("../../plugins");
|
||||||
|
@ -28,6 +29,8 @@ fn main() {
|
||||||
.expect("Could not find compiled plugins in target");
|
.expect("Could not find compiled plugins in target");
|
||||||
println!("cargo:warning={:?}", binaries);
|
println!("cargo:warning={:?}", binaries);
|
||||||
|
|
||||||
|
let engine = create_engine();
|
||||||
|
|
||||||
for file in binaries {
|
for file in binaries {
|
||||||
let is_wasm = || {
|
let is_wasm = || {
|
||||||
let path = file.ok()?.path();
|
let path = file.ok()?.path();
|
||||||
|
@ -39,11 +42,30 @@ fn main() {
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Some(path) = is_wasm() {
|
if let Some(path) = is_wasm() {
|
||||||
std::fs::copy(&path, base.join("bin").join(path.file_name().unwrap()))
|
let out_path = base.join("bin").join(path.file_name().unwrap());
|
||||||
.expect("Could not copy compiled plugin to bin");
|
std::fs::copy(&path, &out_path).expect("Could not copy compiled plugin to bin");
|
||||||
|
precompile(&out_path, &engine);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// TODO: create .wat versions
|
|
||||||
// TODO: optimize with wasm-opt
|
fn create_engine() -> Engine {
|
||||||
|
let mut config = Config::default();
|
||||||
|
config.async_support(true);
|
||||||
|
// config.epoch_interruption(true);
|
||||||
|
Engine::new(&config).expect("Could not create engine")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn precompile(path: &Path, engine: &Engine) {
|
||||||
|
let bytes = std::fs::read(path).expect("Could not read wasm module");
|
||||||
|
let compiled = engine
|
||||||
|
.precompile_module(&bytes)
|
||||||
|
.expect("Could not precompile module");
|
||||||
|
let out_path = path.parent().unwrap().join(&format!(
|
||||||
|
"{}.pre",
|
||||||
|
path.file_name().unwrap().to_string_lossy()
|
||||||
|
));
|
||||||
|
let mut out_file = std::fs::File::create(out_path)
|
||||||
|
.expect("Could not create output file for precompiled module");
|
||||||
|
out_file.write_all(&compiled).unwrap();
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,7 +51,10 @@ mod tests {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.init(include_bytes!("../../../plugins/bin/test_plugin.wasm"))
|
.init(
|
||||||
|
false,
|
||||||
|
include_bytes!("../../../plugins/bin/test_plugin.wasm"),
|
||||||
|
)
|
||||||
.await
|
.await
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
|
|
@ -231,9 +231,9 @@ impl PluginBuilder {
|
||||||
|
|
||||||
/// Initializes a [`Plugin`] from a given compiled Wasm module.
|
/// Initializes a [`Plugin`] from a given compiled Wasm module.
|
||||||
/// Both binary (`.wasm`) and text (`.wat`) module formats are supported.
|
/// Both binary (`.wasm`) and text (`.wat`) module formats are supported.
|
||||||
pub async fn init<T: AsRef<[u8]>>(self, module: T) -> Result<Plugin, Error> {
|
pub async fn init<T: AsRef<[u8]>>(self, precompiled: bool, module: T) -> Result<Plugin, Error> {
|
||||||
dbg!("initializing plugin");
|
dbg!("initializing plugin");
|
||||||
Plugin::init(module.as_ref().to_vec(), self).await
|
Plugin::init(precompiled, module.as_ref().to_vec(), self).await
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -297,7 +297,11 @@ impl Plugin {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Plugin {
|
impl Plugin {
|
||||||
async fn init(module: Vec<u8>, plugin: PluginBuilder) -> Result<Self, Error> {
|
async fn init(
|
||||||
|
precompiled: bool,
|
||||||
|
module: Vec<u8>,
|
||||||
|
plugin: PluginBuilder,
|
||||||
|
) -> Result<Self, Error> {
|
||||||
dbg!("Initializing new plugin");
|
dbg!("Initializing new plugin");
|
||||||
// initialize the WebAssembly System Interface context
|
// initialize the WebAssembly System Interface context
|
||||||
let engine = plugin.engine;
|
let engine = plugin.engine;
|
||||||
|
@ -314,7 +318,11 @@ impl Plugin {
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
// store.epoch_deadline_async_yield_and_update(todo!());
|
// store.epoch_deadline_async_yield_and_update(todo!());
|
||||||
let module = Module::new(&engine, module)?;
|
let module = if precompiled {
|
||||||
|
unsafe { Module::deserialize(&engine, module)? }
|
||||||
|
} else {
|
||||||
|
Module::new(&engine, module)?
|
||||||
|
};
|
||||||
|
|
||||||
// load the provided module into the asynchronous runtime
|
// load the provided module into the asynchronous runtime
|
||||||
linker.module_async(&mut store, "", &module).await?;
|
linker.module_async(&mut store, "", &module).await?;
|
||||||
|
|
|
@ -21,7 +21,10 @@ pub async fn new_json(executor: Arc<Background>) -> Result<PluginLspAdapter> {
|
||||||
.log_err()
|
.log_err()
|
||||||
.map(|output| output.stdout)
|
.map(|output| output.stdout)
|
||||||
})?
|
})?
|
||||||
.init(include_bytes!("../../../../plugins/bin/json_language.wasm"))
|
.init(
|
||||||
|
true,
|
||||||
|
include_bytes!("../../../../plugins/bin/json_language.wasm.pre"),
|
||||||
|
)
|
||||||
.await?;
|
.await?;
|
||||||
PluginLspAdapter::new(plugin, executor).await
|
PluginLspAdapter::new(plugin, executor).await
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue