Lowering I
parent
e4e044e733
commit
09952b5193
|
@ -89,7 +89,15 @@
|
|||
0,
|
||||
0
|
||||
],
|
||||
"interface": {
|
||||
"interface": null,
|
||||
"interfaces": [
|
||||
{
|
||||
"name": "asset_delivery",
|
||||
"version": [
|
||||
0,
|
||||
0,
|
||||
0
|
||||
],
|
||||
"methods": [
|
||||
{
|
||||
"name": "fetchById",
|
||||
|
@ -200,8 +208,8 @@
|
|||
"ret": null
|
||||
}
|
||||
]
|
||||
},
|
||||
"interfaces": [],
|
||||
}
|
||||
],
|
||||
"types": [
|
||||
{
|
||||
"name": "Asset",
|
||||
|
@ -241,7 +249,15 @@
|
|||
0,
|
||||
0
|
||||
],
|
||||
"interface": {
|
||||
"interface": null,
|
||||
"interfaces": [
|
||||
{
|
||||
"name": "world",
|
||||
"version": [
|
||||
0,
|
||||
0,
|
||||
0
|
||||
],
|
||||
"methods": [],
|
||||
"events": [
|
||||
{
|
||||
|
@ -288,10 +304,13 @@
|
|||
}
|
||||
]
|
||||
},
|
||||
"interfaces": [
|
||||
{
|
||||
"name": "Entity",
|
||||
"version": null,
|
||||
"version": [
|
||||
0,
|
||||
0,
|
||||
0
|
||||
],
|
||||
"methods": [
|
||||
{
|
||||
"name": "interact",
|
||||
|
|
|
@ -13,7 +13,7 @@ pub struct Namespace {
|
|||
pub struct Extension {
|
||||
pub name: String,
|
||||
pub version: Version,
|
||||
pub interface: ImplicitInterface,
|
||||
pub interface: Option<ImplicitInterface>,
|
||||
pub interfaces: Vec<ExtensionInterface>,
|
||||
pub types: Vec<TypeDef>,
|
||||
}
|
||||
|
|
|
@ -5,6 +5,7 @@ fn main() -> Result<()> {
|
|||
let tree = fs::read_to_string("hypercosm.hidl")?;
|
||||
|
||||
let tree = hidl::grammar::NamespaceParser::new().parse(&tree).unwrap();
|
||||
let tree = hidl::hir::lower_namespace(tree);
|
||||
|
||||
debug2::dbg!(&tree);
|
||||
|
||||
|
|
|
@ -1,10 +0,0 @@
|
|||
use std::fs;
|
||||
|
||||
use hidl::IDL;
|
||||
|
||||
fn main() {
|
||||
let file = fs::read_to_string("spec.yaml").unwrap();
|
||||
let spec: IDL = serde_yaml::from_str(&file).unwrap();
|
||||
|
||||
debug2::dbg!(spec);
|
||||
}
|
|
@ -21,7 +21,7 @@ pub Namespace: Namespace = {
|
|||
Extension: Extension = {
|
||||
"extension" <name:Ident> "{"
|
||||
<version:Version?>
|
||||
<interface:ImplicitInterface>
|
||||
<interface:ImplicitInterface?>
|
||||
<interfaces:List<ExtensionInterface>>
|
||||
<types:List<TypeDef>>
|
||||
"}"
|
||||
|
|
3821
hidl/src/grammar.rs
3821
hidl/src/grammar.rs
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,115 @@
|
|||
//! Hir has the same types as ast, but with the following invarients:
|
||||
//!
|
||||
//! - All enum variants have a number
|
||||
//! - The implicit interface is None
|
||||
//! - All interfaces have a version
|
||||
|
||||
use std::collections::BTreeSet;
|
||||
|
||||
use crate::ast::{
|
||||
Enum, EnumField, Extension, ExtensionInterface, ImplicitInterface, Namespace, TypeDef,
|
||||
TypeKind, Version,
|
||||
};
|
||||
|
||||
pub fn lower_namespace(
|
||||
Namespace {
|
||||
name,
|
||||
interfaces,
|
||||
types,
|
||||
extensions,
|
||||
}: Namespace,
|
||||
) -> Namespace {
|
||||
Namespace {
|
||||
name,
|
||||
interfaces,
|
||||
types: types.into_iter().map(lower_type_def).collect(),
|
||||
extensions: extensions.into_iter().map(lower_extension).collect(),
|
||||
}
|
||||
}
|
||||
|
||||
fn lower_extension(
|
||||
Extension {
|
||||
name,
|
||||
version,
|
||||
interface,
|
||||
interfaces,
|
||||
types,
|
||||
}: Extension,
|
||||
) -> Extension {
|
||||
let mut new_interfaces = interface
|
||||
.into_iter()
|
||||
.map(|l| lower_implicit_interface(l, &name, version))
|
||||
.collect::<Vec<_>>();
|
||||
|
||||
new_interfaces.extend(
|
||||
interfaces
|
||||
.into_iter()
|
||||
.map(|l| lower_extension_interface(l, version)),
|
||||
);
|
||||
|
||||
Extension {
|
||||
name,
|
||||
version,
|
||||
interface: None,
|
||||
interfaces: new_interfaces,
|
||||
types,
|
||||
}
|
||||
}
|
||||
|
||||
fn lower_implicit_interface(
|
||||
i: ImplicitInterface,
|
||||
name: &str,
|
||||
version: Version,
|
||||
) -> ExtensionInterface {
|
||||
ExtensionInterface {
|
||||
name: name.to_owned(),
|
||||
version: Some(version),
|
||||
methods: i.methods,
|
||||
events: i.events,
|
||||
}
|
||||
}
|
||||
|
||||
fn lower_extension_interface(i: ExtensionInterface, version: Version) -> ExtensionInterface {
|
||||
ExtensionInterface {
|
||||
name: i.name,
|
||||
version: Some(i.version.unwrap_or(version)),
|
||||
methods: i.methods,
|
||||
events: i.events,
|
||||
}
|
||||
}
|
||||
fn lower_type_def(TypeDef { name, kind }: TypeDef) -> TypeDef {
|
||||
TypeDef {
|
||||
name,
|
||||
kind: match kind {
|
||||
TypeKind::Struct(_) => kind,
|
||||
TypeKind::Enum(e) => TypeKind::Enum(lower_enum(e)),
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
fn lower_enum(Enum { backing, fields }: Enum) -> Enum {
|
||||
let mut pos = 0;
|
||||
let mut seen = BTreeSet::new();
|
||||
let mut new_fields = Vec::with_capacity(fields.len());
|
||||
|
||||
for EnumField { name, value } in fields {
|
||||
let value = value.unwrap_or(pos);
|
||||
|
||||
if seen.contains(&value) {
|
||||
panic!("Enum {} has duplicate value {}", name, value);
|
||||
}
|
||||
|
||||
seen.insert(value);
|
||||
pos += 1;
|
||||
|
||||
new_fields.push(EnumField {
|
||||
name,
|
||||
value: Some(value),
|
||||
});
|
||||
}
|
||||
|
||||
Enum {
|
||||
fields: new_fields,
|
||||
backing,
|
||||
}
|
||||
}
|
157
hidl/src/lib.rs
157
hidl/src/lib.rs
|
@ -1,160 +1,5 @@
|
|||
use debug2::Debug as Debug2;
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
pub mod ast;
|
||||
pub mod hir;
|
||||
|
||||
#[allow(clippy::all)]
|
||||
pub mod grammar;
|
||||
|
||||
#[derive(Debug, Debug2, Default, Clone, Serialize, Deserialize, PartialEq)]
|
||||
|
||||
pub struct Module {
|
||||
interfaces: Vec<Interface>,
|
||||
|
||||
types: Vec<CustomType>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Default, Clone, Serialize, Deserialize, PartialEq)]
|
||||
|
||||
pub struct IDL {
|
||||
name: String,
|
||||
root: Module,
|
||||
|
||||
extenstions: Vec<Extension>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Default, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct Extension {
|
||||
name: String,
|
||||
version: Version,
|
||||
module: Module,
|
||||
}
|
||||
|
||||
type Version = (u8, u8, u8);
|
||||
|
||||
#[derive(Debug, Debug2, Default, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct Interface {
|
||||
name: String,
|
||||
version: Version,
|
||||
|
||||
methods: Vec<Method>,
|
||||
|
||||
events: Vec<Method>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Default, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct Method {
|
||||
name: String,
|
||||
|
||||
ret: Type,
|
||||
|
||||
args: Vec<(String, Type)>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Clone, Serialize, Deserialize, PartialEq)]
|
||||
#[serde(untagged)]
|
||||
pub enum Type {
|
||||
NotIntType(NotIntType),
|
||||
IntType(IntType),
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Clone, Serialize, Deserialize, PartialEq)]
|
||||
|
||||
pub enum NotIntType {
|
||||
/// TODO: Should we special case Array of u8?
|
||||
Array(Box<Type>),
|
||||
Custom(String),
|
||||
String,
|
||||
Object,
|
||||
Uuid,
|
||||
/// Unit type
|
||||
Void,
|
||||
}
|
||||
impl Default for Type {
|
||||
fn default() -> Self {
|
||||
Self::NotIntType(NotIntType::Void)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub enum IntType {
|
||||
U8,
|
||||
U16,
|
||||
U32,
|
||||
U64,
|
||||
I8,
|
||||
I16,
|
||||
I32,
|
||||
I64,
|
||||
|
||||
VU16,
|
||||
VU8,
|
||||
VU32,
|
||||
VU64,
|
||||
VI8,
|
||||
VI16,
|
||||
VI32,
|
||||
VI64,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct CustomType {
|
||||
name: String,
|
||||
kind: CustomTypeKind,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Clone, Serialize, Deserialize, PartialEq)]
|
||||
|
||||
enum CustomTypeKind {
|
||||
Struct(CustomStruct),
|
||||
Enum(Enum),
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Clone, Serialize, Deserialize, PartialEq)]
|
||||
|
||||
pub struct Enum {
|
||||
storage: IntType,
|
||||
values: Vec<(String, i64)>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Default, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct CustomStruct {
|
||||
fields: Vec<(String, Type)>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Debug2, Default, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct Event {}
|
||||
|
||||
fn is_default<T: Default + PartialEq>(t: &T) -> bool {
|
||||
t == &T::default()
|
||||
}
|
||||
|
||||
mod ty {
|
||||
use super::IntType::*;
|
||||
use super::NotIntType::*;
|
||||
use super::Type::{self, *};
|
||||
|
||||
pub fn void() -> Type {
|
||||
NotIntType(Void)
|
||||
}
|
||||
|
||||
pub fn array(t: Type) -> Type {
|
||||
NotIntType(Array(Box::new(t)))
|
||||
}
|
||||
|
||||
pub fn string() -> Type {
|
||||
NotIntType(String)
|
||||
}
|
||||
|
||||
pub fn object() -> Type {
|
||||
NotIntType(Object)
|
||||
}
|
||||
|
||||
pub fn uuid() -> Type {
|
||||
NotIntType(Uuid)
|
||||
}
|
||||
|
||||
pub fn u8() -> Type {
|
||||
IntType(U8)
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue