diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 000000000..70fac3b60 --- /dev/null +++ b/.gitmodules @@ -0,0 +1,80 @@ +[submodule "helix-syntax/languages/tree-sitter-cpp"] + path = helix-syntax/languages/tree-sitter-cpp + url = https://github.com/tree-sitter/tree-sitter-cpp + shallow = true +[submodule "helix-syntax/languages/tree-sitter-javascript"] + path = helix-syntax/languages/tree-sitter-javascript + url = https://github.com/tree-sitter/tree-sitter-javascript + shallow = true +[submodule "helix-syntax/languages/tree-sitter-julia"] + path = helix-syntax/languages/tree-sitter-julia + url = https://github.com/tree-sitter/tree-sitter-julia + shallow = true +[submodule "helix-syntax/languages/tree-sitter-python"] + path = helix-syntax/languages/tree-sitter-python + url = https://github.com/tree-sitter/tree-sitter-python + shallow = true +[submodule "helix-syntax/languages/tree-sitter-typescript"] + path = helix-syntax/languages/tree-sitter-typescript + url = https://github.com/tree-sitter/tree-sitter-typescript + shallow = true +[submodule "helix-syntax/languages/tree-sitter-agda"] + path = helix-syntax/languages/tree-sitter-agda + url = https://github.com/tree-sitter/tree-sitter-agda + shallow = true +[submodule "helix-syntax/languages/tree-sitter-go"] + path = helix-syntax/languages/tree-sitter-go + url = https://github.com/tree-sitter/tree-sitter-go + shallow = true +[submodule "helix-syntax/languages/tree-sitter-ruby"] + path = helix-syntax/languages/tree-sitter-ruby + url = https://github.com/tree-sitter/tree-sitter-ruby + shallow = true +[submodule "helix-syntax/languages/tree-sitter-java"] + path = helix-syntax/languages/tree-sitter-java + url = https://github.com/tree-sitter/tree-sitter-java + shallow = true +[submodule "helix-syntax/languages/tree-sitter-php"] + path = helix-syntax/languages/tree-sitter-php + url = https://github.com/tree-sitter/tree-sitter-php + shallow = true +[submodule "helix-syntax/languages/tree-sitter-html"] + path = helix-syntax/languages/tree-sitter-html + url = https://github.com/tree-sitter/tree-sitter-html + shallow = true +[submodule "helix-syntax/languages/tree-sitter-scala"] + path = helix-syntax/languages/tree-sitter-scala + url = https://github.com/tree-sitter/tree-sitter-scala + shallow = true +[submodule "helix-syntax/languages/tree-sitter-bash"] + path = helix-syntax/languages/tree-sitter-bash + url = https://github.com/tree-sitter/tree-sitter-bash + shallow = true +[submodule "helix-syntax/languages/tree-sitter-rust"] + path = helix-syntax/languages/tree-sitter-rust + url = https://github.com/tree-sitter/tree-sitter-rust + shallow = true +[submodule "helix-syntax/languages/tree-sitter-json"] + path = helix-syntax/languages/tree-sitter-json + url = https://github.com/tree-sitter/tree-sitter-json + shallow = true +[submodule "helix-syntax/languages/tree-sitter-css"] + path = helix-syntax/languages/tree-sitter-css + url = https://github.com/tree-sitter/tree-sitter-css + shallow = true +[submodule "helix-syntax/languages/tree-sitter-c-sharp"] + path = helix-syntax/languages/tree-sitter-c-sharp + url = https://github.com/tree-sitter/tree-sitter-c-sharp + shallow = true +[submodule "helix-syntax/languages/tree-sitter-c"] + path = helix-syntax/languages/tree-sitter-c + url = https://github.com/tree-sitter/tree-sitter-c + shallow = true +[submodule "helix-syntax/languages/tree-sitter-haskell"] + path = helix-syntax/languages/tree-sitter-haskell + url = https://github.com/tree-sitter/tree-sitter-haskell + shallow = true +[submodule "helix-syntax/languages/tree-sitter-swift"] + path = helix-syntax/languages/tree-sitter-swift + url = https://github.com/tree-sitter/tree-sitter-swift + shallow = true diff --git a/Cargo.lock b/Cargo.lock index c44a788a4..0385b9d3d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1,5 +1,14 @@ # This file is automatically @generated by Cargo. # It is not intended for manual editing. +[[package]] +name = "aho-corasick" +version = "0.7.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "043164d8ba5c4c3035fec9bbee8647c0261d788f3474306f93bb65901cae0e86" +dependencies = [ + "memchr", +] + [[package]] name = "anyhow" version = "1.0.32" @@ -223,6 +232,9 @@ name = "cc" version = "1.0.59" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "66120af515773fb005778dc07c261bd201ec8ce50bd6e7144c927753fe013381" +dependencies = [ + "jobserver", +] [[package]] name = "cfg-if" @@ -274,6 +286,26 @@ dependencies = [ "winapi", ] +[[package]] +name = "enum-iterator" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c79a6321a1197d7730510c7e3f6cb80432dfefecb32426de8cea0aa19b4bb8d7" +dependencies = [ + "enum-iterator-derive", +] + +[[package]] +name = "enum-iterator-derive" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e94aa31f7c0dc764f57896dc615ddd76fc13b0d5dca7eb6cc5e018a5a09ec06" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "event-listener" version = "2.4.0" @@ -344,6 +376,15 @@ dependencies = [ "unicode-width", ] +[[package]] +name = "helix-syntax" +version = "0.1.0" +dependencies = [ + "cc", + "enum-iterator", + "tree-sitter", +] + [[package]] name = "helix-term" version = "0.1.0" @@ -366,6 +407,15 @@ dependencies = [ "libc", ] +[[package]] +name = "jobserver" +version = "0.1.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c71313ebb9439f74b00d9d2dcec36440beaf57a6aa0623068441dd7cd81a7f2" +dependencies = [ + "libc", +] + [[package]] name = "lazy_static" version = "1.4.0" @@ -536,6 +586,24 @@ version = "0.1.57" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "41cc0f7e4d5d4544e8861606a285bb08d3e70712ccc7d2b84d7c0ccfaf4b05ce" +[[package]] +name = "regex" +version = "1.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c3780fcf44b193bc4d09f36d2a3c87b251da4a046c87795a0d35f4f927ad8e6" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", + "thread_local", +] + +[[package]] +name = "regex-syntax" +version = "0.6.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26412eb97c6b088a6997e05f69403a802a92d520de2f8e63c2b65f9e0f47c4e8" + [[package]] name = "ropey" version = "1.2.0" @@ -629,6 +697,25 @@ dependencies = [ "utf-8", ] +[[package]] +name = "thread_local" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d40c6d1b69745a6ec6fb1ca717914848da4b44ae29d9b3080cbee91d72a69b14" +dependencies = [ + "lazy_static", +] + +[[package]] +name = "tree-sitter" +version = "0.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1df33680edb07e4fb76edcbdd9c7b849b96709fb878afcf0ada678d6bda167af" +dependencies = [ + "cc", + "regex", +] + [[package]] name = "tui" version = "0.10.0" diff --git a/Cargo.toml b/Cargo.toml index d84aef1de..e63518fd6 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -2,4 +2,5 @@ members = [ "helix-core", "helix-term", + "helix-syntax", ] diff --git a/helix-syntax/Cargo.toml b/helix-syntax/Cargo.toml new file mode 100644 index 000000000..ef8ed8631 --- /dev/null +++ b/helix-syntax/Cargo.toml @@ -0,0 +1,14 @@ +[package] +name = "helix-syntax" +version = "0.1.0" +authors = ["Blaž Hrastnik "] +edition = "2018" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +tree-sitter = "0.16" +enum-iterator = "0.6" + +[build-dependencies] +cc = { version = "1", features = ["parallel"] } diff --git a/helix-syntax/build.rs b/helix-syntax/build.rs new file mode 100644 index 000000000..217ba9fa6 --- /dev/null +++ b/helix-syntax/build.rs @@ -0,0 +1,126 @@ +use cc::Build; + +use std::io::{BufWriter, Read, Write}; +use std::path::{Path, PathBuf}; +use std::{env, fs}; + +fn get_opt_level() -> u32 { + env::var("OPT_LEVEL").unwrap().parse::().unwrap() +} + +fn get_debug() -> bool { + env::var("DEBUG").unwrap() == "true" +} + +fn collect_tree_sitter_dirs(ignore: Vec) -> Vec { + let mut dirs = Vec::new(); + for entry in fs::read_dir("languages").unwrap() { + if let Ok(entry) = entry { + let path = entry.path(); + let dir = path.file_name().unwrap().to_str().unwrap().to_string(); + if !ignore.contains(&dir) { + dirs.push(dir); + } + } + } + dirs +} + +fn collect_src_files(dir: &str) -> (Vec, Vec) { + eprintln!("Collect files for {}", dir); + + let mut c_files = Vec::new(); + let mut cpp_files = Vec::new(); + let path = PathBuf::from("languages").join(&dir).join("src"); + for entry in fs::read_dir(path).unwrap() { + if let Ok(entry) = entry { + let path = entry.path(); + if path + .file_stem() + .unwrap() + .to_str() + .unwrap() + .starts_with("binding") + { + continue; + } + if let Some(ext) = path.extension() { + if ext == "c" { + c_files.push(path.to_str().unwrap().to_string()); + } else if ext == "cc" || ext == "cpp" || ext == "cxx" { + cpp_files.push(path.to_str().unwrap().to_string()); + } + } + } + } + (c_files, cpp_files) +} + +fn build_c(files: Vec, language: &str) { + let mut build = cc::Build::new(); + for file in files { + build + .file(&file) + .include(PathBuf::from(file).parent().unwrap()) + .pic(true) + .opt_level(get_opt_level()) + .debug(get_debug()) + .warnings(false) + .flag_if_supported("-std=c99"); + } + build.compile(&format!("tree-sitter-{}-c", language)); +} + +fn build_cpp(files: Vec, language: &str) { + let mut build = cc::Build::new(); + for file in files { + build + .file(&file) + .include(PathBuf::from(file).parent().unwrap()) + .pic(true) + .opt_level(get_opt_level()) + .debug(get_debug()) + .warnings(false) + .cpp(true); + } + build.compile(&format!("tree-sitter-{}-cpp", language)); +} + +fn build_dir(dir: &str, language: &str) { + println!("Build language {}", language); + if PathBuf::from("languages") + .join(dir) + .read_dir() + .unwrap() + .next() + .is_none() + { + eprintln!( + "The directory {} is empty, did you use 'git clone --recursive'?", + dir + ); + eprintln!("You can fix in using 'git submodule init && git submodule update --recursive'."); + std::process::exit(1); + } + let (c, cpp) = collect_src_files(&dir); + if !c.is_empty() { + build_c(c, &language); + } + if !cpp.is_empty() { + build_cpp(cpp, &language); + } +} + +fn main() { + let ignore = vec![ + "tree-sitter-typescript".to_string(), + "tree-sitter-cpp".to_string(), + ]; + let dirs = collect_tree_sitter_dirs(ignore); + for dir in dirs { + let language = &dir[12..]; // skip tree-sitter- prefix + build_dir(&dir, &language); + } + build_dir("tree-sitter-typescript/tsx", "tsx"); + build_dir("tree-sitter-typescript/typescript", "typescript"); +} diff --git a/helix-syntax/languages/tree-sitter-agda b/helix-syntax/languages/tree-sitter-agda new file mode 160000 index 000000000..7fcba5a1b --- /dev/null +++ b/helix-syntax/languages/tree-sitter-agda @@ -0,0 +1 @@ +Subproject commit 7fcba5a1b9f83f52a3812e8cdaf16cb60b069aff diff --git a/helix-syntax/languages/tree-sitter-bash b/helix-syntax/languages/tree-sitter-bash new file mode 160000 index 000000000..8ece09ca4 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-bash @@ -0,0 +1 @@ +Subproject commit 8ece09ca4c0b5e59b124cd19fa92c76b1a9e9dd4 diff --git a/helix-syntax/languages/tree-sitter-c b/helix-syntax/languages/tree-sitter-c new file mode 160000 index 000000000..99151b1e9 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-c @@ -0,0 +1 @@ +Subproject commit 99151b1e9293c9e025498fee7e6691e1a52e1d03 diff --git a/helix-syntax/languages/tree-sitter-c-sharp b/helix-syntax/languages/tree-sitter-c-sharp new file mode 160000 index 000000000..075a1b2ff --- /dev/null +++ b/helix-syntax/languages/tree-sitter-c-sharp @@ -0,0 +1 @@ +Subproject commit 075a1b2ff5fae3142a9318d9479d568843d2fe5d diff --git a/helix-syntax/languages/tree-sitter-cpp b/helix-syntax/languages/tree-sitter-cpp new file mode 160000 index 000000000..5e7476bd0 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-cpp @@ -0,0 +1 @@ +Subproject commit 5e7476bd014445abdae879661e9caf299215478a diff --git a/helix-syntax/languages/tree-sitter-css b/helix-syntax/languages/tree-sitter-css new file mode 160000 index 000000000..23f2cb97d --- /dev/null +++ b/helix-syntax/languages/tree-sitter-css @@ -0,0 +1 @@ +Subproject commit 23f2cb97d47860c517f67f03e1f4b621d5bd2085 diff --git a/helix-syntax/languages/tree-sitter-go b/helix-syntax/languages/tree-sitter-go new file mode 160000 index 000000000..34181774b --- /dev/null +++ b/helix-syntax/languages/tree-sitter-go @@ -0,0 +1 @@ +Subproject commit 34181774b3e86b7801c939c79c7b80a82df91a2b diff --git a/helix-syntax/languages/tree-sitter-haskell b/helix-syntax/languages/tree-sitter-haskell new file mode 160000 index 000000000..2a0aa1cb5 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-haskell @@ -0,0 +1 @@ +Subproject commit 2a0aa1cb5f1b787a4056a29fa0791e87846e33fb diff --git a/helix-syntax/languages/tree-sitter-html b/helix-syntax/languages/tree-sitter-html new file mode 160000 index 000000000..7f442e1c6 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-html @@ -0,0 +1 @@ +Subproject commit 7f442e1c6163d450c69c75c7a621badc3a0ea98f diff --git a/helix-syntax/languages/tree-sitter-java b/helix-syntax/languages/tree-sitter-java new file mode 160000 index 000000000..ee0a2a076 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-java @@ -0,0 +1 @@ +Subproject commit ee0a2a076785145e350fbd413775d1e003f79315 diff --git a/helix-syntax/languages/tree-sitter-javascript b/helix-syntax/languages/tree-sitter-javascript new file mode 160000 index 000000000..feca6ec5e --- /dev/null +++ b/helix-syntax/languages/tree-sitter-javascript @@ -0,0 +1 @@ +Subproject commit feca6ec5e577fa30766f0c0a1e03d32c073027f9 diff --git a/helix-syntax/languages/tree-sitter-json b/helix-syntax/languages/tree-sitter-json new file mode 160000 index 000000000..d3976b27d --- /dev/null +++ b/helix-syntax/languages/tree-sitter-json @@ -0,0 +1 @@ +Subproject commit d3976b27df8622ed17bef6dd5e358b398e73c676 diff --git a/helix-syntax/languages/tree-sitter-julia b/helix-syntax/languages/tree-sitter-julia new file mode 160000 index 000000000..6a0863f1c --- /dev/null +++ b/helix-syntax/languages/tree-sitter-julia @@ -0,0 +1 @@ +Subproject commit 6a0863f1ce3fcf6f99dc0addb7886dcbd27c5a48 diff --git a/helix-syntax/languages/tree-sitter-php b/helix-syntax/languages/tree-sitter-php new file mode 160000 index 000000000..b0c0367d4 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-php @@ -0,0 +1 @@ +Subproject commit b0c0367d4b7058921fdc4ba11e257441a64ab809 diff --git a/helix-syntax/languages/tree-sitter-python b/helix-syntax/languages/tree-sitter-python new file mode 160000 index 000000000..58f572408 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-python @@ -0,0 +1 @@ +Subproject commit 58f57240834d6b88624e32ad0ab9531d55fb7a5d diff --git a/helix-syntax/languages/tree-sitter-ruby b/helix-syntax/languages/tree-sitter-ruby new file mode 160000 index 000000000..14a5e56a6 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-ruby @@ -0,0 +1 @@ +Subproject commit 14a5e56a6fff1f2d40c151ae38b5581fc5c44574 diff --git a/helix-syntax/languages/tree-sitter-rust b/helix-syntax/languages/tree-sitter-rust new file mode 160000 index 000000000..40620bf40 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-rust @@ -0,0 +1 @@ +Subproject commit 40620bf4097cbc9cea79504d7e877865df43a19e diff --git a/helix-syntax/languages/tree-sitter-scala b/helix-syntax/languages/tree-sitter-scala new file mode 160000 index 000000000..211bb726b --- /dev/null +++ b/helix-syntax/languages/tree-sitter-scala @@ -0,0 +1 @@ +Subproject commit 211bb726bb5857f872247b600c7c1808e641a8d4 diff --git a/helix-syntax/languages/tree-sitter-swift b/helix-syntax/languages/tree-sitter-swift new file mode 160000 index 000000000..a22fa5e19 --- /dev/null +++ b/helix-syntax/languages/tree-sitter-swift @@ -0,0 +1 @@ +Subproject commit a22fa5e19bae50098e2252ea96cba3aba43f4c58 diff --git a/helix-syntax/languages/tree-sitter-typescript b/helix-syntax/languages/tree-sitter-typescript new file mode 160000 index 000000000..220ae17fa --- /dev/null +++ b/helix-syntax/languages/tree-sitter-typescript @@ -0,0 +1 @@ +Subproject commit 220ae17fad029f86513498648c90198e8bed872e diff --git a/helix-syntax/src/lib.rs b/helix-syntax/src/lib.rs new file mode 100644 index 000000000..1ca36ca6c --- /dev/null +++ b/helix-syntax/src/lib.rs @@ -0,0 +1,87 @@ +use enum_iterator::IntoEnumIterator; +use tree_sitter::Language; + +#[macro_export] +macro_rules! mk_extern { + ( $( $name:ident ),* ) => { + $( + extern "C" { pub fn $name() -> Language; } + )* + }; +} + +#[macro_export] +macro_rules! mk_enum { + ( $( $camel:ident ),* ) => { + #[derive(Clone, Debug, IntoEnumIterator, PartialEq)] + pub enum LANG { + $( + $camel, + )* + } + }; +} + +#[macro_export] +macro_rules! mk_get_language { + ( $( ($camel:ident, $name:ident) ),* ) => { + pub fn get_language(lang: &LANG) -> Language { + unsafe { + match lang { + $( + LANG::$camel => $name(), + )* + } + } + } + }; +} + +#[macro_export] +macro_rules! mk_get_language_name { + ( $( $camel:ident ),* ) => { + pub fn get_language_name(lang: &LANG) -> &'static str { + match lang { + $( + LANG::$camel => stringify!($camel), + )* + } + } + }; +} + +#[macro_export] +macro_rules! mk_langs { + ( $( ($camel:ident, $name:ident) ),* ) => { + mk_extern!($( $name ),*); + mk_enum!($( $camel ),*); + mk_get_language!($( ($camel, $name) ),*); + mk_get_language_name!($( $camel ),*); + }; +} + +mk_langs!( + // 1) Name for enum + // 2) tree-sitter function to call to get a Language + (Agda, tree_sitter_agda), + (Bash, tree_sitter_bash), + (C, tree_sitter_c), + (CSharp, tree_sitter_c_sharp), + // (Cpp, tree_sitter_cpp), + (Css, tree_sitter_css), + (Go, tree_sitter_go), + (Haskell, tree_sitter_haskell), + (Html, tree_sitter_html), + (Java, tree_sitter_java), + (Javascript, tree_sitter_javascript), + (Json, tree_sitter_json), + (Julia, tree_sitter_julia), + (Php, tree_sitter_php), + (Python, tree_sitter_python), + (Ruby, tree_sitter_ruby), + (Rust, tree_sitter_rust), + (Scala, tree_sitter_scala), + (Swift, tree_sitter_swift), + (Tsx, tree_sitter_tsx), + (Typescript, tree_sitter_typescript) +); diff --git a/helix-term/Cargo.toml b/helix-term/Cargo.toml index e6aa68d3b..ce3add5c3 100644 --- a/helix-term/Cargo.toml +++ b/helix-term/Cargo.toml @@ -14,6 +14,8 @@ path = "src/main.rs" # termwiz = { git = "https://github.com/wez/wezterm", features = ["widgets"] } # termwiz = { path = "../../wezterm/termwiz", default-features = false, features = ["widgets"] } +helix-syntax = { path = "../helix-syntax" } + anyhow = "1" argh = "0.1.3" helix-core = { path = "../helix-core" }