mirror of
https://github.com/ProvableHQ/leo.git
synced 2024-12-23 10:12:21 +03:00
constraint structs and function definitions
This commit is contained in:
parent
1903858e7a
commit
2f9e4b31e0
@ -2,4 +2,6 @@ struct Foo {
|
|||||||
field a
|
field a
|
||||||
bool b
|
bool b
|
||||||
}
|
}
|
||||||
return 1
|
def main() -> (field) :
|
||||||
|
a = 1 + 1
|
||||||
|
return a
|
@ -1,5 +1,5 @@
|
|||||||
use crate::aleo_program::{
|
use crate::aleo_program::{
|
||||||
BooleanExpression, Expression, FieldExpression, Program, Statement, Variable,
|
BooleanExpression, Expression, FieldExpression, Function, Program, Statement, Struct, Variable,
|
||||||
};
|
};
|
||||||
|
|
||||||
use snarkos_models::curves::{Field, PrimeField};
|
use snarkos_models::curves::{Field, PrimeField};
|
||||||
@ -14,6 +14,8 @@ use std::collections::HashMap;
|
|||||||
pub enum ResolvedValue {
|
pub enum ResolvedValue {
|
||||||
Boolean(Boolean),
|
Boolean(Boolean),
|
||||||
FieldElement(UInt32),
|
FieldElement(UInt32),
|
||||||
|
Struct(Struct),
|
||||||
|
Function(Function),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct ResolvedProgram {
|
pub struct ResolvedProgram {
|
||||||
@ -315,115 +317,141 @@ impl ResolvedProgram {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn enforce_statement<F: Field + PrimeField, CS: ConstraintSystem<F>>(
|
||||||
|
&mut self,
|
||||||
|
cs: &mut CS,
|
||||||
|
statement: Statement,
|
||||||
|
) {
|
||||||
|
match statement {
|
||||||
|
Statement::Definition(variable, expression) => match expression {
|
||||||
|
Expression::Boolean(boolean_expression) => {
|
||||||
|
let res = self.enforce_boolean_expression(cs, boolean_expression);
|
||||||
|
println!(
|
||||||
|
" variable boolean result: {} = {}",
|
||||||
|
variable.0,
|
||||||
|
res.get_value().unwrap()
|
||||||
|
);
|
||||||
|
self.insert(variable, ResolvedValue::Boolean(res));
|
||||||
|
}
|
||||||
|
Expression::FieldElement(field_expression) => {
|
||||||
|
let res = self.enforce_field_expression(cs, field_expression);
|
||||||
|
println!(
|
||||||
|
" variable field result: {} = {}",
|
||||||
|
variable.0,
|
||||||
|
res.value.unwrap()
|
||||||
|
);
|
||||||
|
self.insert(variable, ResolvedValue::FieldElement(res));
|
||||||
|
}
|
||||||
|
Expression::Variable(unresolved_variable) => {
|
||||||
|
if self.resolved_variables.contains_key(&unresolved_variable) {
|
||||||
|
// Reassigning variable to another variable
|
||||||
|
let already_assigned = self
|
||||||
|
.resolved_variables
|
||||||
|
.get_mut(&unresolved_variable)
|
||||||
|
.unwrap()
|
||||||
|
.clone();
|
||||||
|
self.insert(variable, already_assigned);
|
||||||
|
} else {
|
||||||
|
// The type of the unassigned variable depends on what is passed in
|
||||||
|
if std::env::args()
|
||||||
|
.nth(1)
|
||||||
|
.expect("variable declaration not passed in")
|
||||||
|
.parse::<bool>()
|
||||||
|
.is_ok()
|
||||||
|
{
|
||||||
|
let resolved_boolean = self.bool_from_variable(cs, unresolved_variable);
|
||||||
|
println!(
|
||||||
|
"variable boolean result: {} = {}",
|
||||||
|
variable.0,
|
||||||
|
resolved_boolean.get_value().unwrap()
|
||||||
|
);
|
||||||
|
self.insert(variable, ResolvedValue::Boolean(resolved_boolean));
|
||||||
|
} else {
|
||||||
|
let resolved_field_element =
|
||||||
|
self.u32_from_variable(cs, unresolved_variable);
|
||||||
|
println!(
|
||||||
|
" variable field result: {} = {}",
|
||||||
|
variable.0,
|
||||||
|
resolved_field_element.value.unwrap()
|
||||||
|
);
|
||||||
|
self.insert(
|
||||||
|
variable,
|
||||||
|
ResolvedValue::FieldElement(resolved_field_element),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Statement::Return(statements) => {
|
||||||
|
statements
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|expression| match expression {
|
||||||
|
Expression::Boolean(boolean_expression) => {
|
||||||
|
let res = self.enforce_boolean_expression(cs, boolean_expression);
|
||||||
|
println!("\n Boolean result = {}", res.get_value().unwrap());
|
||||||
|
}
|
||||||
|
Expression::FieldElement(field_expression) => {
|
||||||
|
let res = self.enforce_field_expression(cs, field_expression);
|
||||||
|
println!("\n Field result = {}", res.value.unwrap());
|
||||||
|
}
|
||||||
|
Expression::Variable(variable) => {
|
||||||
|
match self.resolved_variables.get_mut(&variable).unwrap().clone() {
|
||||||
|
ResolvedValue::Boolean(boolean) => println!(
|
||||||
|
"\n Variable result = {}",
|
||||||
|
boolean.get_value().unwrap()
|
||||||
|
),
|
||||||
|
ResolvedValue::FieldElement(field_element) => println!(
|
||||||
|
"\n Variable field result = {}",
|
||||||
|
field_element.value.unwrap()
|
||||||
|
),
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
pub fn generate_constraints<F: Field + PrimeField, CS: ConstraintSystem<F>>(
|
pub fn generate_constraints<F: Field + PrimeField, CS: ConstraintSystem<F>>(
|
||||||
cs: &mut CS,
|
cs: &mut CS,
|
||||||
program: Program,
|
program: Program,
|
||||||
) {
|
) {
|
||||||
let mut resolved_program = ResolvedProgram::new();
|
let mut resolved_program = ResolvedProgram::new();
|
||||||
|
|
||||||
|
program
|
||||||
|
.structs
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|(variable, struct_def)| {
|
||||||
|
resolved_program
|
||||||
|
.resolved_variables
|
||||||
|
.insert(variable, ResolvedValue::Struct(struct_def));
|
||||||
|
});
|
||||||
|
program
|
||||||
|
.functions
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|(variable, function)| {
|
||||||
|
resolved_program
|
||||||
|
.resolved_variables
|
||||||
|
.insert(variable, ResolvedValue::Function(function));
|
||||||
|
});
|
||||||
|
|
||||||
|
let main = resolved_program
|
||||||
|
.resolved_variables
|
||||||
|
.get_mut(&Variable("main".into()))
|
||||||
|
.expect("main function not defined");
|
||||||
|
match main {
|
||||||
|
ResolvedValue::Function(function) => function
|
||||||
|
.statements
|
||||||
|
.clone()
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|statement| resolved_program.enforce_statement(cs, statement)),
|
||||||
|
_ => unimplemented!("main must be a function"),
|
||||||
|
}
|
||||||
|
|
||||||
program
|
program
|
||||||
.statements
|
.statements
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.for_each(|statement| match statement {
|
.for_each(|statement| resolved_program.enforce_statement(cs, statement));
|
||||||
Statement::Definition(variable, expression) => match expression {
|
|
||||||
Expression::Boolean(boolean_expression) => {
|
|
||||||
let res =
|
|
||||||
resolved_program.enforce_boolean_expression(cs, boolean_expression);
|
|
||||||
println!(
|
|
||||||
" variable boolean result: {} = {}",
|
|
||||||
variable.0,
|
|
||||||
res.get_value().unwrap()
|
|
||||||
);
|
|
||||||
resolved_program.insert(variable, ResolvedValue::Boolean(res));
|
|
||||||
}
|
|
||||||
Expression::FieldElement(field_expression) => {
|
|
||||||
let res = resolved_program.enforce_field_expression(cs, field_expression);
|
|
||||||
println!(
|
|
||||||
" variable field result: {} = {}",
|
|
||||||
variable.0,
|
|
||||||
res.value.unwrap()
|
|
||||||
);
|
|
||||||
resolved_program.insert(variable, ResolvedValue::FieldElement(res));
|
|
||||||
}
|
|
||||||
Expression::Variable(unresolved_variable) => {
|
|
||||||
if resolved_program
|
|
||||||
.resolved_variables
|
|
||||||
.contains_key(&unresolved_variable)
|
|
||||||
{
|
|
||||||
// Reassigning variable to another variable
|
|
||||||
let already_assigned = resolved_program
|
|
||||||
.resolved_variables
|
|
||||||
.get_mut(&unresolved_variable)
|
|
||||||
.unwrap()
|
|
||||||
.clone();
|
|
||||||
resolved_program.insert(variable, already_assigned);
|
|
||||||
} else {
|
|
||||||
// The type of the unassigned variable depends on what is passed in
|
|
||||||
if std::env::args()
|
|
||||||
.nth(1)
|
|
||||||
.expect("variable declaration not passed in")
|
|
||||||
.parse::<bool>()
|
|
||||||
.is_ok()
|
|
||||||
{
|
|
||||||
let resolved_boolean =
|
|
||||||
resolved_program.bool_from_variable(cs, unresolved_variable);
|
|
||||||
println!(
|
|
||||||
"variable boolean result: {} = {}",
|
|
||||||
variable.0,
|
|
||||||
resolved_boolean.get_value().unwrap()
|
|
||||||
);
|
|
||||||
resolved_program
|
|
||||||
.insert(variable, ResolvedValue::Boolean(resolved_boolean));
|
|
||||||
} else {
|
|
||||||
let resolved_field_element =
|
|
||||||
resolved_program.u32_from_variable(cs, unresolved_variable);
|
|
||||||
println!(
|
|
||||||
" variable field result: {} = {}",
|
|
||||||
variable.0,
|
|
||||||
resolved_field_element.value.unwrap()
|
|
||||||
);
|
|
||||||
resolved_program.insert(
|
|
||||||
variable,
|
|
||||||
ResolvedValue::FieldElement(resolved_field_element),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Statement::Return(statements) => {
|
|
||||||
statements
|
|
||||||
.into_iter()
|
|
||||||
.for_each(|expression| match expression {
|
|
||||||
Expression::Boolean(boolean_expression) => {
|
|
||||||
let res = resolved_program
|
|
||||||
.enforce_boolean_expression(cs, boolean_expression);
|
|
||||||
println!("\n Boolean result = {}", res.get_value().unwrap());
|
|
||||||
}
|
|
||||||
Expression::FieldElement(field_expression) => {
|
|
||||||
let res =
|
|
||||||
resolved_program.enforce_field_expression(cs, field_expression);
|
|
||||||
println!("\n Field result = {}", res.value.unwrap());
|
|
||||||
}
|
|
||||||
Expression::Variable(variable) => {
|
|
||||||
match resolved_program
|
|
||||||
.resolved_variables
|
|
||||||
.get_mut(&variable)
|
|
||||||
.unwrap()
|
|
||||||
.clone()
|
|
||||||
{
|
|
||||||
ResolvedValue::Boolean(boolean) => println!(
|
|
||||||
"\n Variable result = {}",
|
|
||||||
boolean.get_value().unwrap()
|
|
||||||
),
|
|
||||||
ResolvedValue::FieldElement(field_element) => println!(
|
|
||||||
"\n Variable field result = {}",
|
|
||||||
field_element.value.unwrap()
|
|
||||||
),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4,29 +4,11 @@
|
|||||||
//! @author Collin Chin <collin@aleo.org>
|
//! @author Collin Chin <collin@aleo.org>
|
||||||
//! @date 2020
|
//! @date 2020
|
||||||
|
|
||||||
// id == 0 for field values
|
use std::collections::HashMap;
|
||||||
// id < 0 for boolean values
|
|
||||||
/// A variable in a constraint system.
|
/// A variable in a constraint system.
|
||||||
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
||||||
pub struct Variable(pub String);
|
pub struct Variable(pub String);
|
||||||
//
|
|
||||||
// /// Linear combination of variables in a program. (a + b + c)
|
|
||||||
// #[derive(Debug, Clone)]
|
|
||||||
// pub struct LinearCombination (pub Vec<Variable>);
|
|
||||||
//
|
|
||||||
// impl LinearCombination {
|
|
||||||
// pub fn one() -> Self {
|
|
||||||
// LinearCombination(vec![Variable{ id: 0, value: "1".into() }])
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// pub fn value(&self) -> String {
|
|
||||||
// self.0[0].value.clone()
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// /// Quadratic combination of variables in a program (a * b)
|
|
||||||
// #[derive(Debug, Clone)]
|
|
||||||
// pub struct QuadraticCombination (pub LinearCombination, pub LinearCombination);
|
|
||||||
|
|
||||||
/// Expression that evaluates to a field value
|
/// Expression that evaluates to a field value
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
@ -109,11 +91,33 @@ pub struct Struct {
|
|||||||
pub fields: Vec<StructField>,
|
pub fields: Vec<StructField>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub enum Visibility {
|
||||||
|
Public,
|
||||||
|
Private,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct Parameter {
|
||||||
|
pub visibility: Option<Visibility>,
|
||||||
|
pub ty: Type,
|
||||||
|
pub variable: Variable,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct Function {
|
||||||
|
pub variable: Variable,
|
||||||
|
pub parameters: Vec<Parameter>,
|
||||||
|
pub returns: Vec<Type>,
|
||||||
|
pub statements: Vec<Statement>,
|
||||||
|
}
|
||||||
|
|
||||||
/// A simple program with statement expressions, program arguments and program returns.
|
/// A simple program with statement expressions, program arguments and program returns.
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct Program {
|
pub struct Program {
|
||||||
pub id: String,
|
pub id: String,
|
||||||
pub structs: Vec<Struct>,
|
pub structs: HashMap<Variable, Struct>,
|
||||||
|
pub functions: HashMap<Variable, Function>,
|
||||||
pub statements: Vec<Statement>,
|
pub statements: Vec<Statement>,
|
||||||
pub arguments: Vec<Variable>,
|
pub arguments: Vec<Variable>,
|
||||||
pub returns: Vec<Variable>,
|
pub returns: Vec<Variable>,
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
//! @date 2020
|
//! @date 2020
|
||||||
|
|
||||||
use crate::{aleo_program::types, ast};
|
use crate::{aleo_program::types, ast};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
impl<'ast> From<ast::Field<'ast>> for types::FieldExpression {
|
impl<'ast> From<ast::Field<'ast>> for types::FieldExpression {
|
||||||
fn from(field: ast::Field<'ast>) -> Self {
|
fn from(field: ast::Field<'ast>) -> Self {
|
||||||
@ -341,8 +342,8 @@ impl<'ast> From<ast::Statement<'ast>> for types::Statement {
|
|||||||
impl<'ast> From<ast::BasicType<'ast>> for types::Type {
|
impl<'ast> From<ast::BasicType<'ast>> for types::Type {
|
||||||
fn from(basic_type: ast::BasicType<'ast>) -> Self {
|
fn from(basic_type: ast::BasicType<'ast>) -> Self {
|
||||||
match basic_type {
|
match basic_type {
|
||||||
ast::BasicType::Field(ty) => types::Type::FieldElement,
|
ast::BasicType::Field(_ty) => types::Type::FieldElement,
|
||||||
ast::BasicType::Boolean(ty) => types::Type::Boolean,
|
ast::BasicType::Boolean(_ty) => types::Type::Boolean,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -396,20 +397,82 @@ impl<'ast> From<ast::Struct<'ast>> for types::Struct {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<ast::Visibility> for types::Visibility {
|
||||||
|
fn from(visibility: ast::Visibility) -> Self {
|
||||||
|
match visibility {
|
||||||
|
ast::Visibility::Private(_private) => types::Visibility::Private,
|
||||||
|
ast::Visibility::Public(_public) => types::Visibility::Public,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'ast> From<ast::Parameter<'ast>> for types::Parameter {
|
||||||
|
fn from(parameter: ast::Parameter<'ast>) -> Self {
|
||||||
|
let ty = types::Type::from(parameter.ty);
|
||||||
|
let variable = types::Variable::from(parameter.variable);
|
||||||
|
|
||||||
|
if parameter.visibility.is_some() {
|
||||||
|
let visibility = Some(types::Visibility::from(parameter.visibility.unwrap()));
|
||||||
|
types::Parameter {
|
||||||
|
visibility,
|
||||||
|
ty,
|
||||||
|
variable,
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
types::Parameter {
|
||||||
|
visibility: None,
|
||||||
|
ty,
|
||||||
|
variable,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'ast> From<ast::Function<'ast>> for types::Function {
|
||||||
|
fn from(function_definition: ast::Function<'ast>) -> Self {
|
||||||
|
let variable = types::Variable::from(function_definition.variable);
|
||||||
|
let parameters = function_definition
|
||||||
|
.parameters
|
||||||
|
.into_iter()
|
||||||
|
.map(|parameter| types::Parameter::from(parameter))
|
||||||
|
.collect();
|
||||||
|
let returns = function_definition
|
||||||
|
.returns
|
||||||
|
.into_iter()
|
||||||
|
.map(|return_type| types::Type::from(return_type))
|
||||||
|
.collect();
|
||||||
|
let statements = function_definition
|
||||||
|
.statements
|
||||||
|
.into_iter()
|
||||||
|
.map(|statement| types::Statement::from(statement))
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
types::Function {
|
||||||
|
variable,
|
||||||
|
parameters,
|
||||||
|
returns,
|
||||||
|
statements,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<'ast> From<ast::File<'ast>> for types::Program {
|
impl<'ast> From<ast::File<'ast>> for types::Program {
|
||||||
fn from(file: ast::File<'ast>) -> Self {
|
fn from(file: ast::File<'ast>) -> Self {
|
||||||
// 1. compile ast -> aleo program representation
|
// 1. compile ast -> aleo program representation
|
||||||
let structs = file
|
let mut structs = HashMap::new();
|
||||||
.structs
|
let mut functions = HashMap::new();
|
||||||
.into_iter()
|
|
||||||
.map(|struct_def| {
|
file.structs.into_iter().for_each(|struct_def| {
|
||||||
println!("{:#?}", struct_def);
|
println!("{:#?}", struct_def);
|
||||||
types::Struct::from(struct_def)
|
let struct_definition = types::Struct::from(struct_def);
|
||||||
})
|
structs.insert(struct_definition.variable.clone(), struct_definition);
|
||||||
.collect();
|
});
|
||||||
file.functions
|
file.functions.into_iter().for_each(|function_def| {
|
||||||
.into_iter()
|
println!("{:#?}", function_def);
|
||||||
.for_each(|function_def| println!("{:#?}", function_def));
|
let function_definition = types::Function::from(function_def);
|
||||||
|
functions.insert(function_definition.variable.clone(), function_definition);
|
||||||
|
});
|
||||||
|
|
||||||
let statements: Vec<types::Statement> = file
|
let statements: Vec<types::Statement> = file
|
||||||
.statements
|
.statements
|
||||||
.into_iter()
|
.into_iter()
|
||||||
@ -419,6 +482,7 @@ impl<'ast> From<ast::File<'ast>> for types::Program {
|
|||||||
types::Program {
|
types::Program {
|
||||||
id: "main".into(),
|
id: "main".into(),
|
||||||
structs,
|
structs,
|
||||||
|
functions,
|
||||||
statements,
|
statements,
|
||||||
arguments: vec![],
|
arguments: vec![],
|
||||||
returns: vec![],
|
returns: vec![],
|
||||||
|
16
src/ast.rs
16
src/ast.rs
@ -408,20 +408,20 @@ pub struct ArrayAccess<'ast> {
|
|||||||
pub span: Span<'ast>,
|
pub span: Span<'ast>,
|
||||||
}
|
}
|
||||||
|
|
||||||
// #[derive(Clone, Debug, FromPest, PartialEq)]
|
#[derive(Clone, Debug, FromPest, PartialEq)]
|
||||||
// #[pest_ast(rule(Rule::member_access))]
|
#[pest_ast(rule(Rule::access_member))]
|
||||||
// pub struct MemberAccess<'ast> {
|
pub struct MemberAccess<'ast> {
|
||||||
// pub id: IdentifierExpression<'ast>,
|
pub variable: Variable<'ast>,
|
||||||
// #[pest_ast(outer())]
|
#[pest_ast(outer())]
|
||||||
// pub span: Span<'ast>,
|
pub span: Span<'ast>,
|
||||||
// }
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, FromPest, PartialEq)]
|
#[derive(Clone, Debug, FromPest, PartialEq)]
|
||||||
#[pest_ast(rule(Rule::access))]
|
#[pest_ast(rule(Rule::access))]
|
||||||
pub enum Access<'ast> {
|
pub enum Access<'ast> {
|
||||||
// Call(CallAccess<'ast>),
|
// Call(CallAccess<'ast>),
|
||||||
Select(ArrayAccess<'ast>),
|
Select(ArrayAccess<'ast>),
|
||||||
// Member(MemberAccess<'ast>),
|
Member(MemberAccess<'ast>),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug, FromPest, PartialEq)]
|
#[derive(Clone, Debug, FromPest, PartialEq)]
|
||||||
|
Loading…
Reference in New Issue
Block a user