Skip to content

Commit

Permalink
feat(hydroflow): prototype a functional surface syntax using staging
Browse files Browse the repository at this point in the history
  • Loading branch information
shadaj committed Sep 9, 2023
1 parent 6e8fba7 commit 917a19d
Show file tree
Hide file tree
Showing 10 changed files with 585 additions and 0 deletions.
38 changes: 38 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 4 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,10 @@ members = [
"hydroflow_datalog_core",
"hydroflow_lang",
"hydroflow_macro",
"hydroflow_plus",
"hydroflow_plus_macro",
"hydroflow_plus_example_flow",
"hydroflow_plus_example_runtime",
"lattices",
"multiplatform_test",
"pusherator",
Expand Down
24 changes: 24 additions & 0 deletions hydroflow_plus/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
[package]
name = "hydroflow_plus"
publish = true
version = "0.4.0"
edition = "2021"
license = "Apache-2.0"
documentation = "https://docs.rs/hydroflow_plus/"
description = "Functional programming API for hydroflow"

[lib]
path = "src/lib.rs"

[features]
default = []
diagnostics = [ "hydroflow_lang/diagnostics" ]

[dependencies]
quote = "1.0.0"
syn = { version = "2.0.0", features = [ "parsing", "extra-traits" ] }
proc-macro2 = "1.0.57"
proc-macro-crate = "1.1.0"
hydroflow = { path = "../hydroflow", version = "^0.4.0" }
hydroflow_lang = { path = "../hydroflow_lang", version = "^0.4.0" }
hydroflow_plus_macro = { path = "../hydroflow_plus_macro", version = "^0.4.0" }
229 changes: 229 additions & 0 deletions hydroflow_plus/src/lib.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,229 @@
use std::cell::RefCell;
use std::marker::PhantomData;
use std::mem::MaybeUninit;
use std::ops::Deref;

use hydroflow_lang::graph::{partition_graph, propegate_flow_props, FlatGraphBuilder};
pub use hydroflow_plus_macro::{flow, q};
use proc_macro2::{Span, TokenStream};
pub use quote::quote;
use quote::ToTokens;
use syn::parse_quote;
pub use {hydroflow, syn};

pub trait ParseFromLiteral {
fn parse_from_literal(literal: &syn::Expr) -> Self;
}

impl ParseFromLiteral for u32 {
fn parse_from_literal(literal: &syn::Expr) -> Self {
match literal {
syn::Expr::Lit(syn::ExprLit {
lit: syn::Lit::Int(lit_int),
..
}) => lit_int.base10_parse().unwrap(),
_ => panic!("Expected literal"),
}
}
}

pub trait FreeVariable<O> {
fn to_tokens(&self) -> TokenStream;
fn uninitialized(&self) -> O {
#[allow(clippy::uninit_assumed_init)]
unsafe {
MaybeUninit::uninit().assume_init()
}
}
}

impl FreeVariable<u32> for u32 {
fn to_tokens(&self) -> TokenStream {
quote!(#self)
}
}

pub struct RuntimeData<T> {
ident: String,
_phantom: PhantomData<T>,
}

impl<T> RuntimeData<T> {
pub fn new(ident: &str) -> RuntimeData<T> {
RuntimeData {
ident: ident.to_string(),
_phantom: PhantomData,
}
}
}

impl<T> Deref for RuntimeData<T> {
type Target = T;
fn deref(&self) -> &T {
panic!("RuntimeData should not be dereferenced.")
}
}

impl<T> FreeVariable<T> for RuntimeData<T> {
fn to_tokens(&self) -> TokenStream {
let ident = syn::Ident::new(&self.ident, Span::call_site());
quote!(#ident)
}
}

thread_local! {
static HYDROFLOW_NEXT_ID: RefCell<usize> = RefCell::new(0);
static HYDROFLOW_BUILDER: RefCell<Option<FlatGraphBuilder>> = RefCell::new(None);
}

pub fn hydroflow_build(f: impl Fn()) -> TokenStream {
let hydroflow_crate = proc_macro_crate::crate_name("hydroflow_plus")
.expect("hydroflow_plus should be present in `Cargo.toml`");
let root = match hydroflow_crate {
proc_macro_crate::FoundCrate::Itself => quote! { hydroflow_plus::hydroflow },
proc_macro_crate::FoundCrate::Name(name) => {
let ident = syn::Ident::new(&name, Span::call_site());
quote! { #ident::hydroflow }
}
};

HYDROFLOW_NEXT_ID.with(|next_id| {
*next_id.borrow_mut() = 0;
HYDROFLOW_BUILDER.with(|builder| {
*builder.borrow_mut() = Some(FlatGraphBuilder::new());
f();

let (flat_graph, _, _) = builder.borrow_mut().take().unwrap().build();
let mut partitioned_graph =
partition_graph(flat_graph).expect("Failed to partition (cycle detected).");

let mut diagnostics = Vec::new();
// Propgeate flow properties throughout the graph.
// TODO(mingwei): Should this be done at a flat graph stage instead?
let _ = propegate_flow_props::propegate_flow_props(
&mut partitioned_graph,
&mut diagnostics,
);

partitioned_graph.as_code(&root, true, quote::quote!(), &mut diagnostics)
})
})
}

pub struct QuotedExpr<T> {
expr: syn::Expr,
free_variables: Vec<(String, TokenStream)>,
_phantom: PhantomData<T>,
}

impl<T> QuotedExpr<T> {
pub fn create(
expr: &str,
free_variables: Vec<(String, TokenStream)>,
_unused_type_check: T,
) -> QuotedExpr<T> {
let expr = syn::parse_str(expr).unwrap();
QuotedExpr {
expr,
free_variables,
_phantom: PhantomData,
}
}
}

impl<T> ToTokens for QuotedExpr<T> {
fn to_tokens(&self, tokens: &mut TokenStream) {
let instantiated_free_variables = self.free_variables.iter().map(|(ident, value)| {
let ident = syn::Ident::new(ident, Span::call_site());
quote!(let #ident = #value;)
});

let expr = &self.expr;
tokens.extend(quote!({
#(#instantiated_free_variables)*
#expr
}));
}
}

pub struct HydroflowNode<T> {
ident: syn::Ident,
_phantom: PhantomData<T>,
}

impl<T> HydroflowNode<T> {
pub fn source_iter(e: QuotedExpr<impl IntoIterator<Item = T>>) -> HydroflowNode<T> {
let next_id = HYDROFLOW_NEXT_ID.with(|next_id| {
let mut next_id = next_id.borrow_mut();
let id = *next_id;
*next_id += 1;
id
});

let ident = syn::Ident::new(&format!("source_{}", next_id), Span::call_site());

HYDROFLOW_BUILDER.with(|builder| {
builder
.borrow_mut()
.as_mut()
.unwrap()
.add_statement(parse_quote! {
#ident = source_iter(#e) -> tee();
});
});

HydroflowNode {
ident,
_phantom: PhantomData,
}
}

pub fn map<U>(&self, f: QuotedExpr<impl Fn(T) -> U>) -> HydroflowNode<U> {
let next_id = HYDROFLOW_NEXT_ID.with(|next_id| {
let mut next_id = next_id.borrow_mut();
let id = *next_id;
*next_id += 1;
id
});

let self_ident = &self.ident;
let ident = syn::Ident::new(&format!("map_{}", next_id), Span::call_site());

HYDROFLOW_BUILDER.with(|builder| {
builder
.borrow_mut()
.as_mut()
.unwrap()
.add_statement(parse_quote! {
#ident = #self_ident -> map(#f) -> tee();
});
});

HydroflowNode {
ident,
_phantom: PhantomData,
}
}

pub fn for_each(&self, f: QuotedExpr<impl Fn(T)>) {
let next_id = HYDROFLOW_NEXT_ID.with(|next_id| {
let mut next_id = next_id.borrow_mut();
let id = *next_id;
*next_id += 1;
id
});

let self_ident = &self.ident;
let ident = syn::Ident::new(&format!("for_each_{}", next_id), Span::call_site());

HYDROFLOW_BUILDER.with(|builder| {
builder
.borrow_mut()
.as_mut()
.unwrap()
.add_statement(parse_quote! {
#ident = #self_ident -> for_each(#f);
});
});
}
}
12 changes: 12 additions & 0 deletions hydroflow_plus_example_flow/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
[package]
name = "hydroflow_plus_example_flow"
publish = false
version = "0.0.0"
edition = "2021"

[lib]
proc-macro = true
path = "src/lib.rs"

[dependencies]
hydroflow_plus = { path = "../hydroflow_plus", version = "^0.4.0" }
15 changes: 15 additions & 0 deletions hydroflow_plus_example_flow/src/lib.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
use hydroflow_plus::{q, HydroflowNode, RuntimeData};

#[hydroflow_plus::flow]
pub fn my_example_flow(
number_of_foreach: u32,
multiplier: RuntimeData<u32>,
text: RuntimeData<&str>,
) {
let source = HydroflowNode::source_iter(q!(vec![1, 2, 3, number_of_foreach]));
let mapped = source.map(q!(move |x| x * multiplier));

for _ in 0..number_of_foreach {
mapped.for_each(q!(move |x| println!("{} {}", text, x)));
}
}
9 changes: 9 additions & 0 deletions hydroflow_plus_example_runtime/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
[package]
name = "hydroflow_plus_example_runtime"
publish = false
version = "0.0.0"
edition = "2021"

[dependencies]
hydroflow_plus = { path = "../hydroflow_plus", version = "^0.4.0" }
hydroflow_plus_example_flow = { path = "../hydroflow_plus_example_flow" }
11 changes: 11 additions & 0 deletions hydroflow_plus_example_runtime/src/main.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
use hydroflow_plus_example_flow::my_example_flow;

fn main() {
let multiplier = std::env::args()
.nth(1)
.expect("Expected multiplier as first argument")
.parse::<u32>()
.expect("Expected multiplier to be a number");
let mut flow = my_example_flow!(1, multiplier, "hi");
flow.run_tick();
}
18 changes: 18 additions & 0 deletions hydroflow_plus_macro/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
[package]
name = "hydroflow_plus_macro"
publish = true
version = "0.4.0"
edition = "2021"
license = "Apache-2.0"
documentation = "https://docs.rs/hydroflow_plus_macro/"
description = "Helper macros for the hydroflow_plus crate"

[lib]
proc-macro = true
path = "src/lib.rs"

[dependencies]
quote = "1.0.0"
syn = { version = "2.0.0", features = [ "parsing", "extra-traits", "visit" ] }
proc-macro2 = "1.0.57"
proc-macro-crate = "1.1.0"
Loading

0 comments on commit 917a19d

Please sign in to comment.