initial parser for morphism-base using chumsky
This commit is contained in:
parent
f65dfee2b3
commit
d66dc0e78b
2 changed files with 101 additions and 0 deletions
8
Cargo.toml
Normal file
8
Cargo.toml
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
[package]
|
||||||
|
name = "ldmc"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2024"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
chumsky = "0.9.0"
|
||||||
|
ariadne = "0.2"
|
93
src/main.rs
Normal file
93
src/main.rs
Normal file
|
@ -0,0 +1,93 @@
|
||||||
|
use {
|
||||||
|
ariadne::{Color, Label, Report, ReportKind, Source},
|
||||||
|
chumsky::{
|
||||||
|
prelude::*, text::*
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
struct Morphism {
|
||||||
|
symbol: String,
|
||||||
|
src_type: String,
|
||||||
|
dst_type: String,
|
||||||
|
type_args: Vec<(String, String)>,
|
||||||
|
locations: Vec<String>
|
||||||
|
}
|
||||||
|
|
||||||
|
/* morphism-base text format:
|
||||||
|
* NAME '(' [TYPE-ARG-NAME ':' KIND] ')'
|
||||||
|
* SRC-TYPE
|
||||||
|
* '-->' DST-TYPE
|
||||||
|
* '@' [ LOCATION ':' ]
|
||||||
|
*/
|
||||||
|
fn parser() -> impl Parser<char, Vec<Morphism>, Error = Simple<char>> {
|
||||||
|
ident().padded()
|
||||||
|
.then(
|
||||||
|
ident().padded()
|
||||||
|
.then_ignore(just(':').padded())
|
||||||
|
.then(none_of(",)").repeated().padded())
|
||||||
|
.separated_by(just(',').padded())
|
||||||
|
.delimited_by(just('('), just(')'))
|
||||||
|
)
|
||||||
|
.then_ignore(just('\n'))
|
||||||
|
.then(
|
||||||
|
take_until(just("-->").ignored())
|
||||||
|
.then(take_until(just('@').ignored()))
|
||||||
|
)
|
||||||
|
.then(
|
||||||
|
none_of(":\n").repeated().separated_by(just(':'))
|
||||||
|
)
|
||||||
|
.map(|(((symbol, type_args), ((src_type, _), (dst_type, _))), locations)| {
|
||||||
|
Morphism {
|
||||||
|
symbol,
|
||||||
|
src_type: src_type.iter().collect(),
|
||||||
|
dst_type: dst_type.iter().collect(),
|
||||||
|
type_args: type_args.into_iter().map(|(v,k)| (v,k.into_iter().collect())).collect(),
|
||||||
|
locations: locations.into_iter().map(|l| l.into_iter().collect()).collect()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.separated_by(text::newline())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
println!("Hello, world!");
|
||||||
|
|
||||||
|
let src = "
|
||||||
|
morph_digit_as_char_to_uint8 (Radix:ℤ_16)
|
||||||
|
<Digit Radix> ~ Char ~ Ascii ~ Byte
|
||||||
|
--> <Digit Radix> ~ x86.UInt8 ~ Byte
|
||||||
|
@lib/libmorph_posint.so:src/posint.c
|
||||||
|
|
||||||
|
morph_string_as_nullterm_to_length_prefix ()
|
||||||
|
[~<ValueDelim '\0'> Char ~ Ascii]
|
||||||
|
--> [~<LengthPrefix x86.UInt64> Char ~ Ascii]
|
||||||
|
@lib/libmorph_length-prefix.so:src/length_prefix.c
|
||||||
|
|
||||||
|
morph_string_as_length_prefix_to_nullterm ()
|
||||||
|
[~<LengthPrefix x86.UInt64> Char ~ Ascii]
|
||||||
|
--> [~<ValueDelim '\0'> Char ~ Ascii]
|
||||||
|
@lib/libmorph_length-prefix.so:src/length_prefix.c
|
||||||
|
";
|
||||||
|
|
||||||
|
let result = parser().parse(src);
|
||||||
|
|
||||||
|
match result {
|
||||||
|
Ok(morphisms) => {
|
||||||
|
println!("parse ok.\n{:?}", morphisms);
|
||||||
|
}
|
||||||
|
Err(errs) => {
|
||||||
|
errs.into_iter().for_each(|e| {
|
||||||
|
Report::build(ReportKind::Error, (), e.span().start)
|
||||||
|
.with_message(e.to_string())
|
||||||
|
.with_label(
|
||||||
|
Label::new(e.span())
|
||||||
|
.with_message(e)
|
||||||
|
.with_color(Color::Red),
|
||||||
|
)
|
||||||
|
.finish()
|
||||||
|
.print(Source::from(&src))
|
||||||
|
.unwrap()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue