Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add gate calls and the AST items it depends on to the new QASM3 compiler #2246

Merged
merged 29 commits into from
Mar 24, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
1c2743a
add qubit declaration
orpuente-MS Mar 20, 2025
55ac0f3
compile MeasureExpr and Bitarrays
orpuente-MS Mar 21, 2025
ecd9d57
update syntax::ast expect tests
orpuente-MS Mar 21, 2025
244fb0e
fix typo in display_utils docstring
orpuente-MS Mar 21, 2025
0c8ae4f
add measure arrow stmt
orpuente-MS Mar 21, 2025
fa5ac92
lower gate-call
orpuente-MS Mar 21, 2025
15aee6f
remove unnecessary import
orpuente-MS Mar 21, 2025
f564768
compile gates
orpuente-MS Mar 21, 2025
9be86ba
rebase
orpuente-MS Mar 21, 2025
8c84f53
PR feedback fixes
orpuente-MS Mar 21, 2025
e47da76
wrap gate-call in expr-stmt
orpuente-MS Mar 21, 2025
6326692
fix constness bugs
orpuente-MS Mar 21, 2025
7f106fd
update qreg test
orpuente-MS Mar 21, 2025
33679ab
implement barrier and reset
orpuente-MS Mar 21, 2025
74628ed
update span visitior
orpuente-MS Mar 21, 2025
32e37f3
fix duplicated error on decl and assignment
orpuente-MS Mar 21, 2025
b61f134
fix duplicated errors and constness issues
orpuente-MS Mar 21, 2025
bb4efec
fix typo
orpuente-MS Mar 21, 2025
892f1bc
add U and gphase gate
orpuente-MS Mar 21, 2025
c4ad6db
update expect-tests
orpuente-MS Mar 23, 2025
98ee9b6
add const eval unit tests (work in progress)
orpuente-MS Mar 23, 2025
b984293
rework const evaluator to match the table of allowed operations
orpuente-MS Mar 24, 2025
b898c0f
PR feedback fix
orpuente-MS Mar 24, 2025
e221567
add const_eval unit tests
orpuente-MS Mar 24, 2025
279e5f0
format test file
orpuente-MS Mar 24, 2025
f98f2fd
fix const_eval unit tests
orpuente-MS Mar 24, 2025
3359df0
add missing tests
orpuente-MS Mar 24, 2025
1fb8983
resolve PR comments
orpuente-MS Mar 24, 2025
c6e7928
format const_eval.rs
orpuente-MS Mar 24, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
312 changes: 255 additions & 57 deletions compiler/qsc_qasm3/src/compiler.rs

Large diffs are not rendered by default.

71 changes: 49 additions & 22 deletions compiler/qsc_qasm3/src/parser/ast.rs
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,7 @@ impl WithSpan for Path {
#[derive(Clone, Debug)]
pub struct MeasureExpr {
pub span: Span,
pub measure_token_span: Span,
pub operand: GateOperand,
}

Expand Down Expand Up @@ -248,29 +249,51 @@ impl Display for UnaryOp {
}

#[derive(Clone, Debug, Default)]
pub enum GateOperand {
pub struct GateOperand {
pub span: Span,
pub kind: GateOperandKind,
}

impl WithSpan for GateOperand {
fn with_span(self, span: Span) -> Self {
Self {
span,
kind: self.kind.with_span(span),
}
}
}

impl Display for GateOperand {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
writeln_header(f, "GateOperand", self.span)?;
write_field(f, "kind", &self.kind)
}
}

#[derive(Clone, Debug, Default)]
pub enum GateOperandKind {
IndexedIdent(Box<IndexedIdent>),
HardwareQubit(Box<HardwareQubit>),
#[default]
Err,
}

impl Display for GateOperand {
impl Display for GateOperandKind {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
match self {
GateOperand::IndexedIdent(ident) => write!(f, "{ident}"),
GateOperand::HardwareQubit(qubit) => write!(f, "{qubit}"),
GateOperand::Err => write!(f, "Error"),
Self::IndexedIdent(ident) => write!(f, "{ident}"),
Self::HardwareQubit(qubit) => write!(f, "{qubit}"),
Self::Err => write!(f, "Error"),
}
}
}

impl WithSpan for GateOperand {
impl WithSpan for GateOperandKind {
fn with_span(self, span: Span) -> Self {
match self {
GateOperand::IndexedIdent(ident) => GateOperand::IndexedIdent(ident.with_span(span)),
GateOperand::HardwareQubit(qubit) => GateOperand::HardwareQubit(qubit.with_span(span)),
GateOperand::Err => GateOperand::Err,
Self::IndexedIdent(ident) => Self::IndexedIdent(ident.with_span(span)),
Self::HardwareQubit(qubit) => Self::HardwareQubit(qubit.with_span(span)),
Self::Err => Self::Err,
}
}
}
Expand Down Expand Up @@ -335,7 +358,7 @@ pub enum StmtKind {
GPhase(GPhase),
Include(IncludeStmt),
IODeclaration(IODeclaration),
Measure(MeasureStmt),
Measure(MeasureArrowStmt),
Pragma(Pragma),
QuantumGateDefinition(QuantumGateDefinition),
QuantumDecl(QubitDeclaration),
Expand Down Expand Up @@ -445,12 +468,14 @@ impl Display for BarrierStmt {
#[derive(Clone, Debug)]
pub struct ResetStmt {
pub span: Span,
pub reset_token_span: Span,
pub operand: Box<GateOperand>,
}

impl Display for ResetStmt {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
writeln_header(f, "ResetStmt", self.span)?;
writeln_field(f, "reset_token_span", &self.reset_token_span)?;
write_field(f, "operand", &self.operand)
}
}
Expand Down Expand Up @@ -991,13 +1016,15 @@ impl Display for IncludeStmt {
#[derive(Clone, Debug)]
pub struct QubitDeclaration {
pub span: Span,
pub ty_span: Span,
pub qubit: Ident,
pub size: Option<Expr>,
}

impl Display for QubitDeclaration {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
writeln_header(f, "QubitDeclaration", self.span)?;
writeln_field(f, "ty_span", &self.ty_span)?;
writeln_field(f, "ident", &self.qubit)?;
write_opt_field(f, "size", self.size.as_ref())
}
Expand Down Expand Up @@ -1110,15 +1137,15 @@ impl Display for BoxStmt {
}

#[derive(Clone, Debug)]
pub struct MeasureStmt {
pub struct MeasureArrowStmt {
pub span: Span,
pub measurement: MeasureExpr,
pub target: Option<Box<IndexedIdent>>,
}

impl Display for MeasureStmt {
impl Display for MeasureArrowStmt {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
writeln_header(f, "MeasureStmt", self.span)?;
writeln_header(f, "MeasureArrowStmt", self.span)?;
writeln_field(f, "measurement", &self.measurement)?;
write_opt_field(f, "target", self.target.as_ref())
}
Expand All @@ -1129,7 +1156,7 @@ pub struct ClassicalDeclarationStmt {
pub span: Span,
pub ty: Box<TypeDef>,
pub identifier: Ident,
pub init_expr: Option<Box<ValueExpression>>,
pub init_expr: Option<Box<ValueExpr>>,
}

impl Display for ClassicalDeclarationStmt {
Expand All @@ -1142,16 +1169,16 @@ impl Display for ClassicalDeclarationStmt {
}

#[derive(Clone, Debug)]
pub enum ValueExpression {
pub enum ValueExpr {
Expr(Expr),
Measurement(MeasureExpr),
}

impl Display for ValueExpression {
impl Display for ValueExpr {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
match self {
ValueExpression::Expr(expr) => write!(f, "{expr}"),
ValueExpression::Measurement(measure) => write!(f, "{measure}"),
Self::Expr(expr) => write!(f, "{expr}"),
Self::Measurement(measure) => write!(f, "{measure}"),
}
}
}
Expand All @@ -1178,7 +1205,7 @@ pub struct ConstantDeclStmt {
pub span: Span,
pub ty: TypeDef,
pub identifier: Box<Ident>,
pub init_expr: Expr,
pub init_expr: ValueExpr,
}

impl Display for ConstantDeclStmt {
Expand Down Expand Up @@ -1339,7 +1366,7 @@ impl Display for DefStmt {
#[derive(Clone, Debug)]
pub struct ReturnStmt {
pub span: Span,
pub expr: Option<Box<ValueExpression>>,
pub expr: Option<Box<ValueExpr>>,
}

impl Display for ReturnStmt {
Expand Down Expand Up @@ -1470,7 +1497,7 @@ impl Display for ExprKind {
pub struct AssignStmt {
pub span: Span,
pub lhs: IndexedIdent,
pub rhs: Expr,
pub rhs: ValueExpr,
}

impl Display for AssignStmt {
Expand All @@ -1486,7 +1513,7 @@ pub struct AssignOpStmt {
pub span: Span,
pub op: BinOp,
pub lhs: IndexedIdent,
pub rhs: Expr,
pub rhs: ValueExpr,
}

impl Display for AssignOpStmt {
Expand Down
4 changes: 2 additions & 2 deletions compiler/qsc_qasm3/src/parser/ast/display_utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -129,7 +129,7 @@ pub(crate) fn writeln_opt_field<T: Display>(
writeln!(f)
}

/// Writes an field of a structure to the given buffer
/// Writes a field of a structure to the given buffer
/// or stream with an additional indententation level.
/// The field must be an iterable.
pub(crate) fn write_list_field<'write, 'itemref, 'item, T, I>(
Expand All @@ -148,7 +148,7 @@ where
write_list(&mut indent, vals)
}

/// Writes an field of a structure to the given buffer
/// Writes a field of a structure to the given buffer
/// or stream with an additional indententation level.
/// The field must be an iterable.
/// Inserts a newline afterwards.
Expand Down
44 changes: 33 additions & 11 deletions compiler/qsc_qasm3/src/parser/expr.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,9 +24,9 @@ use crate::parser::Result;
use super::{
ast::{
list_from_iter, BinOp, BinaryOpExpr, Cast, DiscreteSet, Expr, ExprKind, FunctionCall,
GateOperand, HardwareQubit, Ident, IndexElement, IndexExpr, IndexSet, IndexSetItem,
IndexedIdent, List, Lit, LiteralKind, MeasureExpr, RangeDefinition, TimeUnit, TypeDef,
UnaryOp, UnaryOpExpr, ValueExpression, Version,
GateOperand, GateOperandKind, HardwareQubit, Ident, IndexElement, IndexExpr, IndexSet,
IndexSetItem, IndexedIdent, List, Lit, LiteralKind, MeasureExpr, RangeDefinition, TimeUnit,
TypeDef, UnaryOp, UnaryOpExpr, ValueExpr, Version,
},
completion::WordKinds,
error::{Error, ErrorKind},
Expand Down Expand Up @@ -708,9 +708,11 @@ fn lit_array_element(s: &mut ParserContext) -> Result<Expr> {
lit_array(s)
}

pub(super) fn value_expr(s: &mut ParserContext) -> Result<Box<ValueExpression>> {
/// These are expressions allowed in classical declarations.
/// Grammar: `arrayLiteral | expression | measureExpression`.
pub(super) fn declaration_expr(s: &mut ParserContext) -> Result<ValueExpr> {
if let Some(measurement) = opt(s, measure_expr)? {
return Ok(Box::new(ValueExpression::Measurement(measurement)));
return Ok(ValueExpr::Measurement(measurement));
}

let expr = if let Some(expr) = opt(s, expr)? {
Expand All @@ -719,7 +721,17 @@ pub(super) fn value_expr(s: &mut ParserContext) -> Result<Box<ValueExpression>>
lit_array(s)?
};

Ok(Box::new(ValueExpression::Expr(expr)))
Ok(ValueExpr::Expr(expr))
}

/// These are expressions allowed in `Assign`, `AssignOp`, and return stmts.
/// Grammar: `expression | measureExpression`.
pub(super) fn expr_or_measurement(s: &mut ParserContext) -> Result<ValueExpr> {
if let Some(measurement) = opt(s, measure_expr)? {
return Ok(ValueExpr::Measurement(measurement));
}

Ok(ValueExpr::Expr(expr(s)?))
}

pub(crate) fn expr_list(s: &mut ParserContext) -> Result<Vec<Expr>> {
Expand All @@ -729,18 +741,28 @@ pub(crate) fn expr_list(s: &mut ParserContext) -> Result<Vec<Expr>> {
pub(crate) fn measure_expr(s: &mut ParserContext) -> Result<MeasureExpr> {
let lo = s.peek().span.lo;
token(s, TokenKind::Measure)?;
let measure_token_span = s.span(lo);
let operand = gate_operand(s)?;

Ok(MeasureExpr {
span: s.span(lo),
operand: gate_operand(s)?,
measure_token_span,
operand,
})
}

pub(crate) fn gate_operand(s: &mut ParserContext) -> Result<GateOperand> {
if let Some(indexed_ident) = opt(s, indexed_identifier)? {
return Ok(GateOperand::IndexedIdent(Box::new(indexed_ident)));
}
Ok(GateOperand::HardwareQubit(Box::new(hardware_qubit(s)?)))
let lo = s.peek().span.lo;
let kind = if let Some(indexed_ident) = opt(s, indexed_identifier)? {
GateOperandKind::IndexedIdent(Box::new(indexed_ident))
} else {
GateOperandKind::HardwareQubit(Box::new(hardware_qubit(s)?))
};

Ok(GateOperand {
span: s.span(lo),
kind,
})
}

fn hardware_qubit(s: &mut ParserContext) -> Result<HardwareQubit> {
Expand Down
28 changes: 15 additions & 13 deletions compiler/qsc_qasm3/src/parser/expr/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1146,8 +1146,9 @@ fn measure_hardware_qubit() {
super::measure_expr,
"measure $12",
&expect![[r#"
MeasureExpr [0-7]:
operand: HardwareQubit [8-11]: 12"#]],
MeasureExpr [0-11]:
operand: GateOperand [8-11]:
kind: HardwareQubit [8-11]: 12"#]],
);
}

Expand All @@ -1157,17 +1158,18 @@ fn measure_indexed_identifier() {
super::measure_expr,
"measure qubits[1][2]",
&expect![[r#"
MeasureExpr [0-7]:
operand: IndexedIdent [8-20]:
name: Ident [8-14] "qubits"
index_span: [14-20]
indices:
IndexSet [15-16]:
values:
Expr [15-16]: Lit: Int(1)
IndexSet [18-19]:
values:
Expr [18-19]: Lit: Int(2)"#]],
MeasureExpr [0-20]:
operand: GateOperand [8-20]:
kind: IndexedIdent [8-20]:
name: Ident [8-14] "qubits"
index_span: [14-20]
indices:
IndexSet [15-16]:
values:
Expr [15-16]: Lit: Int(1)
IndexSet [18-19]:
values:
Expr [18-19]: Lit: Int(2)"#]],
);
}

Expand Down
Loading
Loading