Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
111 changes: 108 additions & 3 deletions source/compiler/qsc/src/codegen/tests/adaptive_profile.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1756,9 +1756,9 @@ fn controlled_specialization_inlines() {
);
}

/// A recursive operation inlines (recursion is excluded from eligibility).
/// A recursive operation can be emitted into an IR function.
#[test]
fn recursive_operation_inlines() {
fn recursive_operation_emits_to_ir_function() {
let source = "namespace Test {
operation Recurse(n : Int, q : Qubit) : Unit {
if n > 0 {
Expand All @@ -1773,7 +1773,53 @@ fn recursive_operation_inlines() {
}
}";
let qir = compile_source_to_qir(source, *CAPABILITIES);
assert_inlined(&qir, "Recurse");
expect![[r#"
@0 = internal constant [4 x i8] c"0_t\00"

define i64 @ENTRYPOINT__main() #0 {
block_0:
call void @__quantum__rt__initialize(ptr null)
call void @Recurse(i64 3, ptr inttoptr (i64 0 to ptr))
call void @__quantum__rt__tuple_record_output(i64 0, ptr @0)
ret i64 0
}

declare void @__quantum__rt__initialize(ptr)

define void @Recurse(i64 %var_0, ptr %var_1) {
block_1:
%var_2 = icmp sgt i64 %var_0, 0
br i1 %var_2, label %block_2, label %block_3
block_2:
call void @__quantum__qis__x__body(ptr %var_1)
%var_3 = sub i64 %var_0, 1
call void @Recurse(i64 %var_3, ptr %var_1)
br label %block_3
block_3:
ret void
}

declare void @__quantum__qis__x__body(ptr)

declare void @__quantum__rt__tuple_record_output(i64, ptr)

attributes #0 = { "entry_point" "output_labeling_schema" "qir_profiles"="adaptive_profile" "required_num_qubits"="1" "required_num_results"="0" }
attributes #1 = { "irreversible" }

; module flags

!llvm.module.flags = !{!0, !1, !2, !3, !4, !5, !6, !7, !8}

!0 = !{i32 1, !"qir_major_version", i32 2}
!1 = !{i32 7, !"qir_minor_version", i32 1}
!2 = !{i32 1, !"dynamic_qubit_management", i1 false}
!3 = !{i32 1, !"dynamic_result_management", i1 false}
!4 = !{i32 5, !"int_computations", !{!"i64"}}
!5 = !{i32 5, !"float_computations", !{!"double"}}
!6 = !{i32 7, !"backwards_branching", i2 3}
!7 = !{i32 1, !"arrays", i1 true}
!8 = !{i32 1, !"ir_functions", i1 true}
"#]].assert_eq(&qir);
}

/// A call into a stdlib/library operation (cross-package) still inlines.
Expand Down Expand Up @@ -2097,3 +2143,62 @@ fn value_returning_ir_function_rir_reloads_after_same_block_store() {
"#]]
.assert_eq(ssa);
}

#[test]
fn preparepurestated_cyclic_library_calls_generate_correct_qir() {
let source = "
operation Main() : Result {
use q = Qubit();
Std.StatePreparation.PreparePureStateD([0.0, 1.0], [q]);
MResetZ(q)
}
";
let qir = compile_source_to_qir(source, *CAPABILITIES);
expect![[r#"
@0 = internal constant [4 x i8] c"0_r\00"

define i64 @ENTRYPOINT__main() #0 {
block_0:
call void @__quantum__rt__initialize(ptr null)
call void @__quantum__qis__s__adj(ptr inttoptr (i64 0 to ptr))
call void @__quantum__qis__h__body(ptr inttoptr (i64 0 to ptr))
call void @__quantum__qis__rz__body(double 3.141592653589793, ptr inttoptr (i64 0 to ptr))
call void @__quantum__qis__h__body(ptr inttoptr (i64 0 to ptr))
call void @__quantum__qis__s__body(ptr inttoptr (i64 0 to ptr))
call void @__quantum__qis__mresetz__body(ptr inttoptr (i64 0 to ptr), ptr inttoptr (i64 0 to ptr))
call void @__quantum__rt__result_record_output(ptr inttoptr (i64 0 to ptr), ptr @0)
ret i64 0
}

declare void @__quantum__rt__initialize(ptr)

declare void @__quantum__qis__s__adj(ptr)

declare void @__quantum__qis__h__body(ptr)

declare void @__quantum__qis__rz__body(double, ptr)

declare void @__quantum__qis__s__body(ptr)

declare void @__quantum__qis__mresetz__body(ptr, ptr) #1

declare void @__quantum__rt__result_record_output(ptr, ptr)

attributes #0 = { "entry_point" "output_labeling_schema" "qir_profiles"="adaptive_profile" "required_num_qubits"="1" "required_num_results"="1" }
attributes #1 = { "irreversible" }

; module flags

!llvm.module.flags = !{!0, !1, !2, !3, !4, !5, !6, !7, !8}

!0 = !{i32 1, !"qir_major_version", i32 2}
!1 = !{i32 7, !"qir_minor_version", i32 1}
!2 = !{i32 1, !"dynamic_qubit_management", i1 false}
!3 = !{i32 1, !"dynamic_result_management", i1 false}
!4 = !{i32 5, !"int_computations", !{!"i64"}}
!5 = !{i32 5, !"float_computations", !{!"double"}}
!6 = !{i32 7, !"backwards_branching", i2 3}
!7 = !{i32 1, !"arrays", i1 true}
!8 = !{i32 1, !"ir_functions", i1 true}
"#]].assert_eq(&qir);
}
58 changes: 58 additions & 0 deletions source/compiler/qsc/src/codegen/tests/adaptive_ri_profile.rs
Original file line number Diff line number Diff line change
Expand Up @@ -610,3 +610,61 @@ fn terminal_result_return_with_qubit_cleanup_generates_rir() {
assert_terminal_result_return_with_qubit_cleanup_rir(raw, "raw");
assert_terminal_result_return_with_qubit_cleanup_rir(ssa, "ssa");
}

#[test]
fn preparepurestated_cyclic_library_calls_generate_correct_qir() {
let source = "
operation Main() : Result {
use q = Qubit();
Std.StatePreparation.PreparePureStateD([0.0, 1.0], [q]);
MResetZ(q)
}
";
let qir = compile_source_to_qir(source, *CAPABILITIES);
expect![[r#"
%Result = type opaque
%Qubit = type opaque
@0 = internal constant [4 x i8] c"0_r\00"
define i64 @ENTRYPOINT__main() #0 {
block_0:
call void @__quantum__rt__initialize(i8* null)
call void @__quantum__qis__s__adj(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__h__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__rz__body(double 3.141592653589793, %Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__h__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__s__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__mresetz__body(%Qubit* inttoptr (i64 0 to %Qubit*), %Result* inttoptr (i64 0 to %Result*))
call void @__quantum__rt__result_record_output(%Result* inttoptr (i64 0 to %Result*), i8* getelementptr inbounds ([4 x i8], [4 x i8]* @0, i64 0, i64 0))
ret i64 0
}
declare void @__quantum__rt__initialize(i8*)
declare void @__quantum__qis__s__adj(%Qubit*)
declare void @__quantum__qis__h__body(%Qubit*)
declare void @__quantum__qis__rz__body(double, %Qubit*)
declare void @__quantum__qis__s__body(%Qubit*)
declare void @__quantum__qis__mresetz__body(%Qubit*, %Result*) #1
declare void @__quantum__rt__result_record_output(%Result*, i8*)
attributes #0 = { "entry_point" "output_labeling_schema" "qir_profiles"="adaptive_profile" "required_num_qubits"="1" "required_num_results"="1" }
attributes #1 = { "irreversible" }
; module flags
!llvm.module.flags = !{!0, !1, !2, !3, !4}
!0 = !{i32 1, !"qir_major_version", i32 1}
!1 = !{i32 7, !"qir_minor_version", i32 0}
!2 = !{i32 1, !"dynamic_qubit_management", i1 false}
!3 = !{i32 1, !"dynamic_result_management", i1 false}
!4 = !{i32 5, !"int_computations", !{!"i64"}}
"#]].assert_eq(&qir);
}
59 changes: 59 additions & 0 deletions source/compiler/qsc/src/codegen/tests/adaptive_rif_profile.rs
Original file line number Diff line number Diff line change
Expand Up @@ -707,3 +707,62 @@ fn dynamic_double_intrinsic() {
!5 = !{i32 5, !"float_computations", !{!"double"}}
"#]].assert_eq(&qir);
}

#[test]
fn preparepurestated_cyclic_library_calls_generate_correct_qir() {
let source = "
operation Main() : Result {
use q = Qubit();
Std.StatePreparation.PreparePureStateD([0.0, 1.0], [q]);
MResetZ(q)
}
";
let qir = compile_source_to_qir(source, *CAPABILITIES);
expect![[r#"
%Result = type opaque
%Qubit = type opaque

@0 = internal constant [4 x i8] c"0_r\00"

define i64 @ENTRYPOINT__main() #0 {
block_0:
call void @__quantum__rt__initialize(i8* null)
call void @__quantum__qis__s__adj(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__h__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__rz__body(double 3.141592653589793, %Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__h__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__s__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__mresetz__body(%Qubit* inttoptr (i64 0 to %Qubit*), %Result* inttoptr (i64 0 to %Result*))
call void @__quantum__rt__result_record_output(%Result* inttoptr (i64 0 to %Result*), i8* getelementptr inbounds ([4 x i8], [4 x i8]* @0, i64 0, i64 0))
ret i64 0
}

declare void @__quantum__rt__initialize(i8*)

declare void @__quantum__qis__s__adj(%Qubit*)

declare void @__quantum__qis__h__body(%Qubit*)

declare void @__quantum__qis__rz__body(double, %Qubit*)

declare void @__quantum__qis__s__body(%Qubit*)

declare void @__quantum__qis__mresetz__body(%Qubit*, %Result*) #1

declare void @__quantum__rt__result_record_output(%Result*, i8*)

attributes #0 = { "entry_point" "output_labeling_schema" "qir_profiles"="adaptive_profile" "required_num_qubits"="1" "required_num_results"="1" }
attributes #1 = { "irreversible" }

; module flags

!llvm.module.flags = !{!0, !1, !2, !3, !4, !5}

!0 = !{i32 1, !"qir_major_version", i32 1}
!1 = !{i32 7, !"qir_minor_version", i32 0}
!2 = !{i32 1, !"dynamic_qubit_management", i1 false}
!3 = !{i32 1, !"dynamic_result_management", i1 false}
!4 = !{i32 5, !"int_computations", !{!"i64"}}
!5 = !{i32 5, !"float_computations", !{!"double"}}
"#]].assert_eq(&qir);
}
57 changes: 57 additions & 0 deletions source/compiler/qsc/src/codegen/tests/base_profile.rs
Original file line number Diff line number Diff line change
Expand Up @@ -351,3 +351,60 @@ fn noise_intrinsic_generates_correct_qir() {
!3 = !{i32 1, !"dynamic_result_management", i1 false}
"#]].assert_eq(&qir);
}

#[test]
fn preparepurestated_cyclic_library_calls_generate_correct_qir() {
let source = "
operation Main() : Result {
use q = Qubit();
Std.StatePreparation.PreparePureStateD([0.0, 1.0], [q]);
MResetZ(q)
}
";
let qir = compile_source_to_qir(source, *CAPABILITIES);
expect![[r#"
%Result = type opaque
%Qubit = type opaque

@0 = internal constant [4 x i8] c"0_r\00"

define i64 @ENTRYPOINT__main() #0 {
block_0:
call void @__quantum__rt__initialize(i8* null)
call void @__quantum__qis__s__adj(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__h__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__rz__body(double 3.141592653589793, %Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__h__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__s__body(%Qubit* inttoptr (i64 0 to %Qubit*))
call void @__quantum__qis__m__body(%Qubit* inttoptr (i64 0 to %Qubit*), %Result* inttoptr (i64 0 to %Result*))
call void @__quantum__rt__result_record_output(%Result* inttoptr (i64 0 to %Result*), i8* getelementptr inbounds ([4 x i8], [4 x i8]* @0, i64 0, i64 0))
ret i64 0
}

declare void @__quantum__rt__initialize(i8*)

declare void @__quantum__qis__s__adj(%Qubit*)

declare void @__quantum__qis__h__body(%Qubit*)

declare void @__quantum__qis__rz__body(double, %Qubit*)

declare void @__quantum__qis__s__body(%Qubit*)

declare void @__quantum__rt__result_record_output(%Result*, i8*)

declare void @__quantum__qis__m__body(%Qubit*, %Result*) #1

attributes #0 = { "entry_point" "output_labeling_schema" "qir_profiles"="base_profile" "required_num_qubits"="1" "required_num_results"="1" }
attributes #1 = { "irreversible" }

; module flags

!llvm.module.flags = !{!0, !1, !2, !3}

!0 = !{i32 1, !"qir_major_version", i32 1}
!1 = !{i32 7, !"qir_minor_version", i32 0}
!2 = !{i32 1, !"dynamic_qubit_management", i1 false}
!3 = !{i32 1, !"dynamic_result_management", i1 false}
"#]].assert_eq(&qir);
}
4 changes: 2 additions & 2 deletions source/compiler/qsc_partial_eval/src/evaluation_context.rs
Original file line number Diff line number Diff line change
Expand Up @@ -323,7 +323,7 @@ fn map_eval_value_to_value_kind(value: &Value) -> ValueKind {
}
ValueKind::Constant
}
Value::Result(Result::Id(_) | Result::Loss) | Value::Var(_) => ValueKind::Variable,
Value::Result(Result::Loss) | Value::Var(_) => ValueKind::Variable,
Value::BigInt(_)
| Value::Bool(_)
| Value::Closure(_)
Expand All @@ -333,7 +333,7 @@ fn map_eval_value_to_value_kind(value: &Value) -> ValueKind {
| Value::Pauli(_)
| Value::Qubit(_)
| Value::Range(_)
| Value::Result(Result::Val(_))
| Value::Result(Result::Val(_) | Result::Id(_))
| Value::String(_) => ValueKind::Constant,
}
}
Loading
Loading