extern crate wasmi; extern crate parity_wasm; use std::env; use std::fmt; use std::fs::File; use wasmi::{ Error as InterpreterError, ModuleInstance, ModuleRef, Externals, RuntimeValue, FuncRef, ModuleImportResolver, FuncInstance, HostError, ImportsBuilder, Signature, ValueType, RuntimeArgs, Trap, }; #[derive(Debug)] pub enum Error { OutOfRange, AlreadyOccupied, Interpreter(InterpreterError), } impl fmt::Display for Error { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { write!(f, "{:?}", self) } } impl From<InterpreterError> for Error { fn from(e: InterpreterError) -> Self { Error::Interpreter(e) } } impl HostError for Error {} mod tictactoe { use super::Error; #[derive(Copy, Clone, Debug, PartialEq, Eq)] pub enum Player { X, O, } #[derive(Copy, Clone, Debug, PartialEq, Eq)] pub enum GameResult { Draw, Won(Player), } impl Player { pub fn into_i32(maybe_player: Option<Player>) -> i32 { match maybe_player { None => 0, Some(Player::X) => 1, Some(Player::O) => 2, } } } #[derive(Debug)] pub struct Game { board: [Option<Player>; 9], } impl Game { pub fn new() -> Game { Game { board: [None; 9], } } pub fn set(&mut self, idx: i32, player: Player) -> Result<(), Error> { if idx < 0 || idx > 9 { return Err(Error::OutOfRange); } if self.board[idx as usize] != None { return Err(Error::AlreadyOccupied); } self.board[idx as usize] = Some(player); Ok(()) } pub fn get(&self, idx: i32) -> Result<Option<Player>, Error> { if idx < 0 || idx > 9 { return Err(Error::OutOfRange); } Ok(self.board[idx as usize]) } pub fn game_result(&self) -> Option<GameResult> { // 0, 1, 2 // 3, 4, 5 // 6, 7, 8 let patterns = &[ // Rows (0, 1, 2), (3, 4, 5), (6, 7, 8), // Columns (0, 3, 6), (1, 4, 7), (2, 5, 8), // Diagonals (0, 4, 8), (2, 4, 6), ]; // Returns Some(player) if all cells contain same Player. let all_same = |i1: usize, i2: usize, i3: usize| -> Option<Player> { if self.board[i1].is_none() { return None; } if self.board[i1] == self.board[i2] && self.board[i2] == self.board[i3] { return self.board[i1]; } None }; for &(i1, i2, i3) in patterns { if let Some(player) = all_same(i1, i2, i3) { return Some(GameResult::Won(player)); } } // Ok, there is no winner. Check if it's draw. let all_occupied = self.board.iter().all(|&cell| cell.is_some()); if all_occupied { Some(GameResult::Draw) } else { // Nah, there are still empty cells left. None } } } } struct Runtime<'a> { player: tictactoe::Player, game: &'a mut tictactoe::Game, } const SET_FUNC_INDEX: usize = 0; const GET_FUNC_INDEX: usize = 1; impl<'a> Externals for Runtime<'a> { fn invoke_index( &mut self, index: usize, args: RuntimeArgs, ) -> Result<Option<RuntimeValue>, Trap> { match index { SET_FUNC_INDEX => { let idx: i32 = args.nth(0); self.game.set(idx, self.player)?; Ok(None) } GET_FUNC_INDEX => { let idx: i32 = args.nth(0); let val: i32 = tictactoe::Player::into_i32(self.game.get(idx)?); Ok(Some(val.into())) } _ => panic!("unknown function index") } } } struct RuntimeModuleImportResolver; impl<'a> ModuleImportResolver for RuntimeModuleImportResolver { fn resolve_func( &self, field_name: &str, _signature: &Signature, ) -> Result<FuncRef, InterpreterError> { let func_ref = match field_name { "set" => { FuncInstance::alloc_host(Signature::new(&[ValueType::I32][..], None), SET_FUNC_INDEX) }, "get" => FuncInstance::alloc_host(Signature::new(&[ValueType::I32][..], Some(ValueType::I32)), GET_FUNC_INDEX), _ => return Err( InterpreterError::Function( format!("host module doesn't export function with name {}", field_name) ) ) }; Ok(func_ref) } } fn instantiate(path: &str) -> Result<ModuleRef, Error> { let module = { use std::io::prelude::*; let mut file = File::open(path).unwrap(); let mut wasm_buf = Vec::new(); file.read_to_end(&mut wasm_buf).unwrap(); wasmi::Module::from_buffer(&wasm_buf)? }; let mut imports = ImportsBuilder::new(); imports.push_resolver("env", &RuntimeModuleImportResolver); let instance = ModuleInstance::new(&module, &imports)? .assert_no_start(); Ok(instance) } fn play( x_instance: ModuleRef, o_instance: ModuleRef, game: &mut tictactoe::Game, ) -> Result<tictactoe::GameResult, Error> { let mut turn_of = tictactoe::Player::X; let game_result = loop { let (instance, next_turn_of) = match turn_of { tictactoe::Player::X => (&x_instance, tictactoe::Player::O), tictactoe::Player::O => (&o_instance, tictactoe::Player::X), }; { let mut runtime = Runtime { player: turn_of, game: game, }; let _ = instance.invoke_export("mk_turn", &[], &mut runtime)?; } match game.game_result() { Some(game_result) => break game_result, None => {} } turn_of = next_turn_of; }; Ok(game_result) } fn main() { let mut game = tictactoe::Game::new(); let args: Vec<_> = env::args().collect(); if args.len() < 3 { println!("Usage: {} <x player module> <y player module>", args[0]); return; } // Instantiate modules of X and O players. let x_instance = instantiate(&args[1]).expect("X player module to load"); let o_instance = instantiate(&args[2]).expect("Y player module to load"); let result = play(x_instance, o_instance, &mut game); println!("result = {:?}, game = {:#?}", result, game); }