Skip to content

Commit dced819

Browse files
authored
Add array and measurement declaration statements (#2180)
1 parent f6f848d commit dced819

File tree

4 files changed

+219
-12
lines changed

4 files changed

+219
-12
lines changed

compiler/qsc_qasm3/src/ast.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -300,14 +300,14 @@ impl Display for UnaryOp {
300300

301301
#[derive(Clone, Debug)]
302302
pub enum GateOperand {
303-
Ident(Box<Identifier>),
303+
IndexedIdent(Box<IndexedIdent>),
304304
HardwareQubit(Box<HardwareQubit>),
305305
}
306306

307307
impl Display for GateOperand {
308308
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
309309
match self {
310-
GateOperand::Ident(ident) => write!(f, "GateOperand {ident}"),
310+
GateOperand::IndexedIdent(ident) => write!(f, "GateOperand {ident}"),
311311
GateOperand::HardwareQubit(qubit) => write!(f, "GateOperand {qubit}"),
312312
}
313313
}
@@ -1333,7 +1333,7 @@ impl Display for ClassicalDeclarationStmt {
13331333
#[derive(Clone, Debug)]
13341334
pub enum ValueExpression {
13351335
Expr(ExprStmt),
1336-
Measurement(QuantumMeasurement),
1336+
Measurement(MeasureExpr),
13371337
}
13381338

13391339
impl Display for ValueExpression {
@@ -1876,7 +1876,7 @@ impl Display for Lit {
18761876

18771877
#[derive(Clone, Debug)]
18781878
pub enum LiteralKind {
1879-
Array(List<ExprStmt>),
1879+
Array(List<Expr>),
18801880
Bitstring(BigInt, usize),
18811881
Bool(bool),
18821882
Duration { value: f64, unit: TimeUnit },

compiler/qsc_qasm3/src/parser/expr.rs

+88-8
Original file line numberDiff line numberDiff line change
@@ -16,9 +16,10 @@ use qsc_data_structures::span::Span;
1616

1717
use crate::{
1818
ast::{
19-
self, AssignExpr, AssignOpExpr, BinOp, BinaryOpExpr, Cast, DiscreteSet, Expr, ExprKind,
20-
ExprStmt, FunctionCall, IndexElement, IndexExpr, IndexSetItem, Lit, LiteralKind,
21-
RangeDefinition, TypeDef, UnaryOp, ValueExpression, Version,
19+
self, list_from_iter, AssignExpr, AssignOpExpr, BinOp, BinaryOpExpr, Cast, DiscreteSet,
20+
Expr, ExprKind, ExprStmt, FunctionCall, GateOperand, HardwareQubit, Ident, IndexElement,
21+
IndexExpr, IndexSetItem, IndexedIdent, Lit, LiteralKind, MeasureExpr, RangeDefinition,
22+
TypeDef, UnaryOp, ValueExpression, Version,
2223
},
2324
keyword::Keyword,
2425
lex::{
@@ -36,7 +37,7 @@ use crate::parser::Result;
3637

3738
use super::{
3839
error::{Error, ErrorKind},
39-
prim::{ident, opt, seq, FinalSep},
40+
prim::{ident, many, opt, recovering_token, seq, FinalSep},
4041
stmt::scalar_or_array_type,
4142
};
4243

@@ -458,7 +459,7 @@ fn cast_op(s: &mut ParserContext, r#type: TypeDef) -> Result<ExprKind> {
458459
fn index_expr(s: &mut ParserContext, lhs: Expr) -> Result<ExprKind> {
459460
let lo = s.span(0).hi - 1;
460461
let index = index_element(s)?;
461-
token(s, TokenKind::Close(Delim::Bracket))?;
462+
recovering_token(s, TokenKind::Close(Delim::Bracket));
462463
Ok(ExprKind::IndexExpr(IndexExpr {
463464
span: s.span(lo),
464465
collection: lhs,
@@ -662,17 +663,96 @@ pub(super) fn designator(s: &mut ParserContext) -> Result<ExprStmt> {
662663
})
663664
}
664665

666+
/// A literal array is a list of literal array elements.
667+
fn lit_array(s: &mut ParserContext) -> Result<Expr> {
668+
let lo = s.peek().span.lo;
669+
token(s, TokenKind::Open(Delim::Brace))?;
670+
let elements = seq(s, lit_array_element).map(|pair| pair.0)?;
671+
recovering_token(s, TokenKind::Close(Delim::Brace));
672+
Ok(Expr {
673+
span: s.span(lo),
674+
kind: Box::new(ExprKind::Lit(Lit {
675+
span: s.span(lo),
676+
kind: LiteralKind::Array(list_from_iter(elements)),
677+
})),
678+
})
679+
}
680+
681+
/// A literal array element can be an expression, or a literal array element.
682+
fn lit_array_element(s: &mut ParserContext) -> Result<Expr> {
683+
if let Some(elt) = opt(s, expr)? {
684+
return Ok(elt);
685+
}
686+
lit_array(s)
687+
}
688+
665689
pub(super) fn value_expr(s: &mut ParserContext) -> Result<Box<ValueExpression>> {
666690
let lo = s.peek().span.lo;
667-
let expr = expr_stmt(s)?;
691+
if let Some(measurement) = opt(s, measure_expr)? {
692+
return Ok(Box::new(ValueExpression::Measurement(measurement)));
693+
}
694+
695+
let expr = if let Some(expr) = opt(s, expr_stmt)? {
696+
expr
697+
} else {
698+
lit_array(s)?
699+
};
700+
668701
let stmt = ExprStmt {
669702
span: s.span(lo),
670703
expr,
671704
};
672-
// todo: measurement
705+
673706
Ok(Box::new(ValueExpression::Expr(stmt)))
674707
}
675708

676-
pub(crate) fn expr_list(s: &mut ParserContext<'_>) -> Result<Vec<Expr>> {
709+
pub(crate) fn expr_list(s: &mut ParserContext) -> Result<Vec<Expr>> {
677710
seq(s, expr).map(|pair| pair.0)
678711
}
712+
713+
fn measure_expr(s: &mut ParserContext) -> Result<MeasureExpr> {
714+
let lo = s.peek().span.lo;
715+
token(s, TokenKind::Measure)?;
716+
717+
Ok(MeasureExpr {
718+
span: s.span(lo),
719+
operand: gate_operand(s)?,
720+
})
721+
}
722+
723+
fn gate_operand(s: &mut ParserContext) -> Result<GateOperand> {
724+
if let Some(indexed_ident) = opt(s, indexed_identifier)? {
725+
return Ok(GateOperand::IndexedIdent(Box::new(indexed_ident)));
726+
}
727+
Ok(GateOperand::HardwareQubit(Box::new(hardware_qubit(s)?)))
728+
}
729+
730+
fn hardware_qubit(s: &mut ParserContext) -> Result<HardwareQubit> {
731+
let lo = s.peek().span.lo;
732+
let hardware_qubit = s.read();
733+
token(s, TokenKind::HardwareQubit)?;
734+
735+
Ok(HardwareQubit {
736+
span: s.span(lo),
737+
name: hardware_qubit[1..].into(),
738+
})
739+
}
740+
741+
fn indexed_identifier(s: &mut ParserContext) -> Result<IndexedIdent> {
742+
let lo = s.peek().span.lo;
743+
let name: Ident = ident(s)?;
744+
let indices = list_from_iter(many(s, index_operand)?);
745+
746+
Ok(IndexedIdent {
747+
span: s.span(lo),
748+
name,
749+
indices,
750+
})
751+
}
752+
753+
fn index_operand(s: &mut ParserContext) -> Result<IndexElement> {
754+
token(s, TokenKind::Open(Delim::Bracket))?;
755+
let index = index_element(s)?;
756+
recovering_token(s, TokenKind::Close(Delim::Bracket));
757+
Ok(index)
758+
}

compiler/qsc_qasm3/src/parser/expr/tests.rs

+58
Original file line numberDiff line numberDiff line change
@@ -1141,6 +1141,18 @@ fn set_expr() {
11411141
);
11421142
}
11431143

1144+
#[test]
1145+
fn lit_array() {
1146+
check(
1147+
super::lit_array,
1148+
"{{2, {5}}, 1 + z}",
1149+
&expect![[r#"
1150+
Expr [0-17]: Lit: Array:
1151+
Expr { span: Span { lo: 1, hi: 9 }, kind: Lit(Lit { span: Span { lo: 1, hi: 9 }, kind: Array([Expr { span: Span { lo: 2, hi: 3 }, kind: Lit(Lit { span: Span { lo: 2, hi: 3 }, kind: Int(2) }) }, Expr { span: Span { lo: 5, hi: 8 }, kind: Lit(Lit { span: Span { lo: 5, hi: 8 }, kind: Array([Expr { span: Span { lo: 6, hi: 7 }, kind: Lit(Lit { span: Span { lo: 6, hi: 7 }, kind: Int(5) }) }]) }) }]) }) }
1152+
Expr { span: Span { lo: 11, hi: 16 }, kind: BinaryOp(BinaryOpExpr { op: Add, lhs: Expr { span: Span { lo: 11, hi: 12 }, kind: Lit(Lit { span: Span { lo: 11, hi: 12 }, kind: Int(1) }) }, rhs: Expr { span: Span { lo: 15, hi: 16 }, kind: Ident(Ident { span: Span { lo: 15, hi: 16 }, name: "z" }) } }) }"#]],
1153+
);
1154+
}
1155+
11441156
#[test]
11451157
fn assignment_and_unop() {
11461158
check(
@@ -1168,3 +1180,49 @@ fn assignment_unop_and() {
11681180
Expr [15-16]: Ident [15-16] "b""#]],
11691181
);
11701182
}
1183+
1184+
#[test]
1185+
fn hardware_qubit() {
1186+
check(
1187+
super::hardware_qubit,
1188+
"$12",
1189+
&expect!["HardwareQubit [0-3]: 12"],
1190+
);
1191+
}
1192+
1193+
#[test]
1194+
fn indexed_identifier() {
1195+
check(
1196+
super::indexed_identifier,
1197+
"arr[1][2]",
1198+
&expect![[r#"
1199+
IndexedIdent [0-9]: Ident [0-3] "arr"[
1200+
IndexElement:
1201+
IndexSetItem Expr [4-5]: Lit: Int(1)
1202+
IndexElement:
1203+
IndexSetItem Expr [7-8]: Lit: Int(2)]"#]],
1204+
);
1205+
}
1206+
1207+
#[test]
1208+
fn measure_hardware_qubit() {
1209+
check(
1210+
super::measure_expr,
1211+
"measure $12",
1212+
&expect!["MeasureExpr [0-7]: GateOperand HardwareQubit [8-11]: 12"],
1213+
);
1214+
}
1215+
1216+
#[test]
1217+
fn measure_indexed_identifier() {
1218+
check(
1219+
super::measure_expr,
1220+
"measure qubits[1][2]",
1221+
&expect![[r#"
1222+
MeasureExpr [0-7]: GateOperand IndexedIdent [8-20]: Ident [8-14] "qubits"[
1223+
IndexElement:
1224+
IndexSetItem Expr [15-16]: Lit: Int(1)
1225+
IndexElement:
1226+
IndexSetItem Expr [18-19]: Lit: Int(2)]"#]],
1227+
);
1228+
}

compiler/qsc_qasm3/src/parser/stmt/tests/classical_decl.rs

+69
Original file line numberDiff line numberDiff line change
@@ -668,3 +668,72 @@ fn stretch_decl() {
668668
StmtKind: ClassicalDeclarationStmt [0-10]: ClassicalType [0-7]: Stretch, Ident [8-9] "s""#]],
669669
);
670670
}
671+
672+
#[test]
673+
fn empty_array_decl() {
674+
check(
675+
parse,
676+
"array[int, 0] arr = {};",
677+
&expect![[r#"
678+
Stmt [0-23]
679+
StmtKind: ClassicalDeclarationStmt [0-23]: ArrayType [0-13]: ArrayBaseTypeKind IntType [6-9]
680+
Expr [11-12]: Lit: Int(0), Ident [14-17] "arr", ValueExpression ExprStmt [20-22]: Expr [20-22]: Lit: Array:"#]],
681+
);
682+
}
683+
684+
#[test]
685+
fn simple_array_decl() {
686+
check(
687+
parse,
688+
"array[int[32], 3] arr = {1, 2, 3};",
689+
&expect![[r#"
690+
Stmt [0-34]
691+
StmtKind: ClassicalDeclarationStmt [0-34]: ArrayType [0-17]: ArrayBaseTypeKind IntType[ExprStmt [9-13]: Expr [10-12]: Lit: Int(32)]: [6-13]
692+
Expr [15-16]: Lit: Int(3), Ident [18-21] "arr", ValueExpression ExprStmt [24-33]: Expr [24-33]: Lit: Array:
693+
Expr { span: Span { lo: 25, hi: 26 }, kind: Lit(Lit { span: Span { lo: 25, hi: 26 }, kind: Int(1) }) }
694+
Expr { span: Span { lo: 28, hi: 29 }, kind: Lit(Lit { span: Span { lo: 28, hi: 29 }, kind: Int(2) }) }
695+
Expr { span: Span { lo: 31, hi: 32 }, kind: Lit(Lit { span: Span { lo: 31, hi: 32 }, kind: Int(3) }) }"#]],
696+
);
697+
}
698+
699+
#[test]
700+
fn nested_array_decl() {
701+
check(
702+
parse,
703+
"array[int[32], 3, 2] arr = {{1, 2}, {3, 4}, {5, 6}};",
704+
&expect![[r#"
705+
Stmt [0-52]
706+
StmtKind: ClassicalDeclarationStmt [0-52]: ArrayType [0-20]: ArrayBaseTypeKind IntType[ExprStmt [9-13]: Expr [10-12]: Lit: Int(32)]: [6-13]
707+
Expr [15-16]: Lit: Int(3)
708+
Expr [18-19]: Lit: Int(2), Ident [21-24] "arr", ValueExpression ExprStmt [27-51]: Expr [27-51]: Lit: Array:
709+
Expr { span: Span { lo: 28, hi: 34 }, kind: Lit(Lit { span: Span { lo: 28, hi: 34 }, kind: Array([Expr { span: Span { lo: 29, hi: 30 }, kind: Lit(Lit { span: Span { lo: 29, hi: 30 }, kind: Int(1) }) }, Expr { span: Span { lo: 32, hi: 33 }, kind: Lit(Lit { span: Span { lo: 32, hi: 33 }, kind: Int(2) }) }]) }) }
710+
Expr { span: Span { lo: 36, hi: 42 }, kind: Lit(Lit { span: Span { lo: 36, hi: 42 }, kind: Array([Expr { span: Span { lo: 37, hi: 38 }, kind: Lit(Lit { span: Span { lo: 37, hi: 38 }, kind: Int(3) }) }, Expr { span: Span { lo: 40, hi: 41 }, kind: Lit(Lit { span: Span { lo: 40, hi: 41 }, kind: Int(4) }) }]) }) }
711+
Expr { span: Span { lo: 44, hi: 50 }, kind: Lit(Lit { span: Span { lo: 44, hi: 50 }, kind: Array([Expr { span: Span { lo: 45, hi: 46 }, kind: Lit(Lit { span: Span { lo: 45, hi: 46 }, kind: Int(5) }) }, Expr { span: Span { lo: 48, hi: 49 }, kind: Lit(Lit { span: Span { lo: 48, hi: 49 }, kind: Int(6) }) }]) }) }"#]],
712+
);
713+
}
714+
715+
#[test]
716+
fn measure_hardware_qubit_decl() {
717+
check(
718+
parse,
719+
"bit res = measure $12;",
720+
&expect![[r#"
721+
Stmt [0-22]
722+
StmtKind: ClassicalDeclarationStmt [0-22]: ClassicalType [0-3]: BitType, Ident [4-7] "res", ValueExpression MeasureExpr [10-17]: GateOperand HardwareQubit [18-21]: 12"#]],
723+
);
724+
}
725+
726+
#[test]
727+
fn measure_register_decl() {
728+
check(
729+
parse,
730+
"bit res = measure qubits[2][3];",
731+
&expect![[r#"
732+
Stmt [0-31]
733+
StmtKind: ClassicalDeclarationStmt [0-31]: ClassicalType [0-3]: BitType, Ident [4-7] "res", ValueExpression MeasureExpr [10-17]: GateOperand IndexedIdent [18-30]: Ident [18-24] "qubits"[
734+
IndexElement:
735+
IndexSetItem Expr [25-26]: Lit: Int(2)
736+
IndexElement:
737+
IndexSetItem Expr [28-29]: Lit: Int(3)]"#]],
738+
);
739+
}

0 commit comments

Comments
 (0)