use std::ffi::OsStr; use std::path::Path; use kecc::*; fn test_dir(path: &Path, ext: &OsStr, f: F) where F: Fn(&Path), { let dir = path.read_dir().expect("read_dir call failed"); for entry in dir { let entry = ok_or!(entry, continue); let path = entry.path(); if !(path.is_file() && path.extension() == Some(ext)) { continue; } println!("[testing {:?}]", path); f(&path); } } fn test_opt_between_dirs>(from: &Path, to: &Path, opt: &mut O) { let from_dir = from.read_dir().expect("read_dir call failed"); for entry in from_dir { let entry = ok_or!(entry, continue); let from_file_path = entry.path(); let file_name = from_file_path .strip_prefix(from) .expect("`from_file _path` must have file name"); let to_file_path = to.join(file_name); assert_eq!(from_file_path.is_file(), true); assert_eq!(to_file_path.exists(), true); assert_eq!(to_file_path.is_file(), true); // TODO: `ignore_list` is a list of examples that ir parser does not support. // We will delete `ignore_list` after implementing complete ir parser. let ignore_list = vec![ "array5.ir", "foo3.ir", "minus_constant.ir", "struct.ir", "struct2.ir", "struct3.ir", "temp2.ir", "typecast.ir", ]; if ignore_list.contains( &file_name .to_str() .expect("`file_name` must be transformed to `&str`"), ) { return; } println!("[testing {:?} to {:?}]", from_file_path, to_file_path); test_opt(&from_file_path, &to_file_path, opt); } } const IRGEN_SMALL_TEST_IGNORE_LIST: [&str; 10] = [ "examples/c/array.c", "examples/c/array2.c", "examples/c/array3.c", "examples/c/array4.c", "examples/c/array5.c", "examples/c/float.c", "examples/c/struct.c", "examples/c/struct2.c", "examples/c/struct3.c", "examples/c/temp2.c", ]; const ASMGEN_SMALL_TEST_IGNORE_LIST: [&str; 10] = [ "examples/asmgen/array.ir", "examples/asmgen/array2.ir", "examples/asmgen/array3.ir", "examples/asmgen/array4.ir", "examples/asmgen/array5.ir", "examples/asmgen/float.ir", "examples/asmgen/struct.ir", "examples/asmgen/struct2.ir", "examples/asmgen/struct3.ir", "examples/asmgen/temp2.ir", ]; #[test] fn test_examples_write_c() { test_dir(Path::new("examples/c"), &OsStr::new("c"), test_write_c); } #[test] fn test_examples_irgen_small() { test_dir(Path::new("examples/c"), &OsStr::new("c"), |path| { let path_str = &path.to_str().expect("`path` must be transformed to `&str`"); if IRGEN_SMALL_TEST_IGNORE_LIST.contains(path_str) { return; } test_irgen(path) }); } #[test] fn test_examples_irgen_large() { test_dir(Path::new("examples/c"), &OsStr::new("c"), |path| { let path_str = &path.to_str().expect("`path` must be transformed to `&str`"); if IRGEN_SMALL_TEST_IGNORE_LIST.contains(path_str) { test_irgen(path) } }); } // TODO: make it work! #[test] fn test_examples_irparse() { test_dir(Path::new("examples/c"), &OsStr::new("c"), test_irparse); } #[test] fn test_examples_simplify_cfg() { test_opt( &Path::new("examples/simplify_cfg/const_prop.input.ir"), &Path::new("examples/simplify_cfg/const_prop.output.ir"), &mut FunctionPass::::default(), ); test_opt( &Path::new("examples/simplify_cfg/reach.input.ir"), &Path::new("examples/simplify_cfg/reach.output.ir"), &mut FunctionPass::::default(), ); test_opt( &Path::new("examples/simplify_cfg/merge.input.ir"), &Path::new("examples/simplify_cfg/merge.output.ir"), &mut FunctionPass::::default(), ); test_opt( &Path::new("examples/simplify_cfg/empty.input.ir"), &Path::new("examples/simplify_cfg/empty.output.ir"), &mut FunctionPass::::default(), ); test_opt_between_dirs( &Path::new("examples/ir0"), &Path::new("examples/ir1"), &mut SimplifyCfg::default(), ); } #[test] fn test_examples_mem2reg() { test_opt( &Path::new("examples/mem2reg/mem2reg.input.ir"), &Path::new("examples/mem2reg/mem2reg.output.ir"), &mut Mem2reg::default(), ); test_opt_between_dirs( &Path::new("examples/ir1"), &Path::new("examples/ir2"), &mut Mem2reg::default(), ); } #[test] fn test_examples_deadcode() { test_opt( &Path::new("examples/deadcode/deadcode.input.ir"), &Path::new("examples/deadcode/deadcode.output.ir"), &mut Deadcode::default(), ); test_opt_between_dirs( &Path::new("examples/ir2"), &Path::new("examples/ir3"), &mut Deadcode::default(), ); } #[test] fn test_examples_gvn() { test_opt( &Path::new("examples/gvn/gvn.input.ir"), &Path::new("examples/gvn/gvn.output.ir"), &mut Gvn::default(), ); test_opt_between_dirs( &Path::new("examples/ir3"), &Path::new("examples/ir4"), &mut Gvn::default(), ); } #[test] fn test_examples_asmgen_small() { test_dir(Path::new("examples/asmgen"), &OsStr::new("ir"), |path| { let path_str = &path.to_str().expect("`path` must be transformed to `&str`"); if ASMGEN_SMALL_TEST_IGNORE_LIST.contains(path_str) { return; } test_asmgen(path) }); } #[test] fn test_examples_asmgen_large() { test_dir(Path::new("examples/asmgen"), &OsStr::new("ir"), |path| { let path_str = &path.to_str().expect("`path` must be transformed to `&str`"); if ASMGEN_SMALL_TEST_IGNORE_LIST.contains(path_str) { test_asmgen(path) } }); } #[test] fn test_examples_end_to_end() { test_dir(Path::new("examples/c"), &OsStr::new("c"), test_end_to_end); }