diff --git a/.github/workflows/llama.yml b/.github/workflows/llama.yml index c0b9cd3..44bb1ea 100644 --- a/.github/workflows/llama.yml +++ b/.github/workflows/llama.yml @@ -25,7 +25,7 @@ jobs: strategy: matrix: runner: [ubuntu-20.04, macos-m1] - wasmedge: ["0.13.5", "0.14.0"] + wasmedge: ["0.14.1"] plugin: [wasi_nn-ggml] job: - name: "Tiny Llama" @@ -301,6 +301,23 @@ jobs: default \ $'[INST] <>\nYou are a helpful, respectful and honest assistant. Always output JSON format string.\n<>\nGive me a JSON array of Apple products.[/INST]' + - name: Qwen2-VL + run: | + test -f ~/.wasmedge/env && source ~/.wasmedge/env + cd wasmedge-ggml/qwen2vl + curl -LO https://huggingface.co/second-state/Qwen2-VL-2B-Instruct-GGUF/resolve/main/Qwen2-VL-2B-Instruct-vision-encoder.gguf + curl -LO https://huggingface.co/second-state/Qwen2-VL-2B-Instruct-GGUF/resolve/main/Qwen2-VL-2B-Instruct-Q5_K_M.gguf + curl -LO https://llava-vl.github.io/static/images/monalisa.jpg + cargo build --target wasm32-wasi --release + time wasmedge --dir .:. \ + --env n_gpu_layers="$NGL" \ + --nn-preload default:GGML:AUTO:Qwen2-VL-2B-Instruct-Q5_K_M.gguf \ + --env mmproj=Qwen2-VL-2B-Instruct-vision-encoder.gguf \ + --env image=monalisa.jpg \ + target/wasm32-wasi/release/wasmedge-ggml-qwen2vl.wasm \ + default \ + $'<|im_start|>system\nYou are a helpful assistant.<|im_end|>\n<|im_start|>user\n<|vision_start|><|vision_end|>what is in this picture?<|im_end|>\n<|im_start|>assistant\n' + - name: Build llama-stream run: | cd wasmedge-ggml/llama-stream diff --git a/wasmedge-ggml/qwen2vl/Cargo.toml b/wasmedge-ggml/qwen2vl/Cargo.toml new file mode 100644 index 0000000..122469d --- /dev/null +++ b/wasmedge-ggml/qwen2vl/Cargo.toml @@ -0,0 +1,8 @@ +[package] +name = "wasmedge-ggml-qwen2vl" +version = "0.1.0" +edition = "2021" + +[dependencies] +serde_json = "1.0" +wasmedge-wasi-nn = "0.7.1" diff --git a/wasmedge-ggml/qwen2vl/README.md b/wasmedge-ggml/qwen2vl/README.md new file mode 100644 index 0000000..1143a4e --- /dev/null +++ b/wasmedge-ggml/qwen2vl/README.md @@ -0,0 +1,47 @@ +# Qwen-2VL Example For WASI-NN with GGML Backend + +> [!NOTE] +> Please refer to the [wasmedge-ggml/README.md](../README.md) for the general introduction and the setup of the WASI-NN plugin with GGML backend. This document will focus on the specific example of the Qwen2-VL model. + +## Get Qwen2-VL Model + +In this example, we are going to use the pre-converted [Qwen2-VL-2B](https://huggingface.co/second-state/Qwen2-VL-2B-Instruct-GGUF/tree/main) model. + +Download the model: + +```bash +curl -LO https://huggingface.co/second-state/Qwen2-VL-2B-Instruct-GGUF/resolve/main/Qwen2-VL-2B-Instruct-vision-encoder.gguf +curl -LO https://huggingface.co/second-state/Qwen2-VL-2B-Instruct-GGUF/resolve/main/Qwen2-VL-2B-Instruct-Q5_K_M.gguf +``` + +## Prepare the Image + +Download the image you want to perform inference on: + +```bash +curl -LO https://llava-vl.github.io/static/images/monalisa.jpg +``` + +## Parameters + +> [!NOTE] +> Please check the parameters section of [wasmedge-ggml/README.md](https://github.com/second-state/WasmEdge-WASINN-examples/tree/master/wasmedge-ggml#parameters) first. + +In Qwen2-VL inference, we recommend to use the `ctx-size` at least `4096` for better results. + +```rust +options.insert("ctx-size", Value::from(4096)); +``` + +## Execute + +Execute the WASM with the `wasmedge` using the named model feature to preload a large model: + +```bash +wasmedge --dir .:. \ + --nn-preload default:GGML:AUTO:Qwen2-VL-2B-Instruct-Q5_K_M.gguf \ + --env mmproj=Qwen2-VL-2B-Instruct-vision-encoder.gguf \ + --env image=monalisa.jpg \ + --env ctx_size=4096 \ + wasmedge-ggml-qwen2vl.wasm default +``` diff --git a/wasmedge-ggml/qwen2vl/src/main.rs b/wasmedge-ggml/qwen2vl/src/main.rs new file mode 100644 index 0000000..73356dd --- /dev/null +++ b/wasmedge-ggml/qwen2vl/src/main.rs @@ -0,0 +1,197 @@ +use serde_json::Value; +use std::collections::HashMap; +use std::env; +use std::io; +use wasmedge_wasi_nn::{ + self, BackendError, Error, ExecutionTarget, GraphBuilder, GraphEncoding, GraphExecutionContext, + TensorType, +}; + +fn read_input() -> String { + loop { + let mut answer = String::new(); + io::stdin() + .read_line(&mut answer) + .expect("Failed to read line"); + if !answer.is_empty() && answer != "\n" && answer != "\r\n" { + return answer.trim().to_string(); + } + } +} + +fn get_options_from_env() -> HashMap<&'static str, Value> { + let mut options = HashMap::new(); + + // Required parameters for llava + if let Ok(val) = env::var("mmproj") { + options.insert("mmproj", Value::from(val.as_str())); + } else { + eprintln!("Failed to get mmproj model."); + std::process::exit(1); + } + if let Ok(val) = env::var("image") { + options.insert("image", Value::from(val.as_str())); + } else { + eprintln!("Failed to get the target image."); + std::process::exit(1); + } + + // Optional parameters + if let Ok(val) = env::var("enable_log") { + options.insert("enable-log", serde_json::from_str(val.as_str()).unwrap()); + } else { + options.insert("enable-log", Value::from(false)); + } + if let Ok(val) = env::var("ctx_size") { + options.insert("ctx-size", serde_json::from_str(val.as_str()).unwrap()); + } else { + options.insert("ctx-size", Value::from(4096)); + } + if let Ok(val) = env::var("n_gpu_layers") { + options.insert("n-gpu-layers", serde_json::from_str(val.as_str()).unwrap()); + } else { + options.insert("n-gpu-layers", Value::from(0)); + } + options +} + +fn set_data_to_context(context: &mut GraphExecutionContext, data: Vec) -> Result<(), Error> { + context.set_input(0, TensorType::U8, &[1], &data) +} + +fn get_data_from_context(context: &GraphExecutionContext, index: usize) -> String { + // Preserve for 4096 tokens with average token length 6 + const MAX_OUTPUT_BUFFER_SIZE: usize = 4096 * 6; + let mut output_buffer = vec![0u8; MAX_OUTPUT_BUFFER_SIZE]; + let mut output_size = context + .get_output(index, &mut output_buffer) + .expect("Failed to get output"); + output_size = std::cmp::min(MAX_OUTPUT_BUFFER_SIZE, output_size); + + String::from_utf8_lossy(&output_buffer[..output_size]).to_string() +} + +fn get_output_from_context(context: &GraphExecutionContext) -> String { + get_data_from_context(context, 0) +} + +fn get_metadata_from_context(context: &GraphExecutionContext) -> Value { + serde_json::from_str(&get_data_from_context(context, 1)).expect("Failed to get metadata") +} + +fn main() { + let args: Vec = env::args().collect(); + let model_name: &str = &args[1]; + + // Set options for the graph. Check our README for more details: + // https://github.com/second-state/WasmEdge-WASINN-examples/tree/master/wasmedge-ggml#parameters + let options = get_options_from_env(); + // You could also set the options manually like this: + + // Create graph and initialize context. + let graph = GraphBuilder::new(GraphEncoding::Ggml, ExecutionTarget::AUTO) + .config(serde_json::to_string(&options).expect("Failed to serialize options")) + .build_from_cache(model_name) + .expect("Failed to build graph"); + let mut context = graph + .init_execution_context() + .expect("Failed to init context"); + + // If there is a third argument, use it as the prompt and enter non-interactive mode. + // This is mainly for the CI workflow. + if args.len() >= 3 { + let prompt = &args[2]; + // Set the prompt. + println!("Prompt:\n{}", prompt); + let tensor_data = prompt.as_bytes().to_vec(); + context + .set_input(0, TensorType::U8, &[1], &tensor_data) + .expect("Failed to set input"); + println!("Response:"); + + // Get the number of input tokens and llama.cpp versions. + let input_metadata = get_metadata_from_context(&context); + println!("[INFO] llama_commit: {}", input_metadata["llama_commit"]); + println!( + "[INFO] llama_build_number: {}", + input_metadata["llama_build_number"] + ); + println!( + "[INFO] Number of input tokens: {}", + input_metadata["input_tokens"] + ); + + // Get the output. + context.compute().expect("Failed to compute"); + let output = get_output_from_context(&context); + println!("{}", output.trim()); + + // Retrieve the output metadata. + let metadata = get_metadata_from_context(&context); + println!( + "[INFO] Number of input tokens: {}", + metadata["input_tokens"] + ); + println!( + "[INFO] Number of output tokens: {}", + metadata["output_tokens"] + ); + std::process::exit(0); + } + + let mut saved_prompt = String::new(); + let system_prompt = String::from("You are a helpful assistant."); + let image_placeholder = ""; + + loop { + println!("USER:"); + let input = read_input(); + + // Qwen2VL prompt format: <|im_start|>system\n{system_prompt}<|im_end|>\n<|im_start|>user\n<|vision_start|>{image_placeholder}<|vision_end|>{user_prompt}<|im_end|>\n<|im_start|>assistant\n"; + if saved_prompt.is_empty() { + saved_prompt = format!( + "<|im_start|>system\n{}<|im_end|>\n<|im_start|>user\n<|vision_start|>{}<|vision_end|>{}<|im_end|>\n<|im_start|>assistant\n", + system_prompt, image_placeholder, input + ); + } else { + saved_prompt = format!( + "{}<|im_start|>user\n{}<|im_end|>\n<|im_start|>assistant\n", + saved_prompt, input + ); + } + + // Set prompt to the input tensor. + set_data_to_context(&mut context, saved_prompt.as_bytes().to_vec()) + .expect("Failed to set input"); + + // Execute the inference. + let mut reset_prompt = false; + match context.compute() { + Ok(_) => (), + Err(Error::BackendError(BackendError::ContextFull)) => { + println!("\n[INFO] Context full, we'll reset the context and continue."); + reset_prompt = true; + } + Err(Error::BackendError(BackendError::PromptTooLong)) => { + println!("\n[INFO] Prompt too long, we'll reset the context and continue."); + reset_prompt = true; + } + Err(err) => { + println!("\n[ERROR] {}", err); + std::process::exit(1); + } + } + + // Retrieve the output. + let mut output = get_output_from_context(&context); + println!("ASSISTANT:\n{}", output.trim()); + + // Update the saved prompt. + if reset_prompt { + saved_prompt.clear(); + } else { + output = output.trim().to_string(); + saved_prompt = format!("{}{}<|im_end|>\n", saved_prompt, output); + } + } +} diff --git a/wasmedge-ggml/qwen2vl/wasmedge-ggml-qwen2vl.wasm b/wasmedge-ggml/qwen2vl/wasmedge-ggml-qwen2vl.wasm new file mode 100755 index 0000000..e713ff0 Binary files /dev/null and b/wasmedge-ggml/qwen2vl/wasmedge-ggml-qwen2vl.wasm differ