Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
P
pulse_lib
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Build
Pipelines
Jobs
Pipeline schedules
Artifacts
Deploy
Releases
Package Registry
Model registry
Operate
Environments
Terraform modules
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
CI/CD analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Terms and privacy
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
QuTech QDLabs
pulse_lib
Commits
6579e849
Commit
6579e849
authored
1 year ago
by
Sander Snoo
Browse files
Options
Downloads
Patches
Plain Diff
Committed missing Qblox conditional file
parent
f6e8dcd4
No related branches found
Branches containing commit
No related tags found
Tags containing commit
No related merge requests found
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
pulse_lib/qblox/qblox_conditional.py
+272
-0
272 additions, 0 deletions
pulse_lib/qblox/qblox_conditional.py
with
272 additions
and
0 deletions
pulse_lib/qblox/qblox_conditional.py
0 → 100644
+
272
−
0
View file @
6579e849
import
time
import
numpy
as
np
import
logging
from
dataclasses
import
dataclass
,
field
from
typing
import
List
from
collections.abc
import
Iterable
from
pulse_lib.segments.conditional_segment
import
conditional_segment
from
pulse_lib.segments.segment_base
import
segment_base
from
pulse_lib.segments.segment_acquisition
import
segment_acquisition
from
pulse_lib.segments.segment_IQ
import
segment_IQ
from
pulse_lib.segments.segment_markers
import
segment_marker
from
pulse_lib.segments.utility.measurement_ref
import
MeasurementRef
logger
=
logging
.
getLogger
(
__name__
)
err_acqs
=
None
class
ConditionalAcquisition
:
class
_AcquisitionData
:
def
__init__
(
self
,
data
):
self
.
data
=
data
def
get_data
(
self
):
return
self
.
data
def
__init__
(
self
,
seg_channels
:
List
[
segment_acquisition
]):
self
.
seg_channels
=
seg_channels
def
_get_data_all_at
(
self
,
index
):
ch_data
=
[
ch
.
_get_data_all_at
(
index
).
get_data
()
for
ch
in
self
.
seg_channels
]
# acquisitions in conditional branch must all be equal
for
data
in
ch_data
[
1
:]:
if
data
!=
ch_data
[
0
]:
global
err_acqs
err_acqs
=
ch_data
raise
Exception
(
'
Non-sequenced channels must be equal for all branches.
'
)
return
ConditionalAcquisition
.
_AcquisitionData
(
ch_data
[
0
])
class
ConditionalMarker
:
class
_MarkerData
:
def
__init__
(
self
):
self
.
my_marker_data
=
[]
def
__init__
(
self
,
seg_channels
:
List
[
segment_marker
]):
self
.
seg_channels
=
seg_channels
def
_get_data_all_at
(
self
,
index
):
# NOTE: marker are not conditional in hardware. Merge markers of all branches.
# merge markers of all branches
data
=
ConditionalMarker
.
_MarkerData
()
for
ch
in
self
.
seg_channels
:
data
.
my_marker_data
+=
ch
.
_get_data_all_at
(
index
).
my_marker_data
return
data
def
integrate
(
self
,
index
,
sample_rate
=
1e9
):
# Marker channels are not connected to bias-T. No charge accumulation.
return
0
err_wvfs
=
None
class
ConditionalWaveform
:
def
__init__
(
self
,
seg_channels
:
List
[
segment_base
]):
self
.
seg_channels
=
seg_channels
def
_get_data_all_at
(
self
,
index
):
data
=
self
.
seg_channels
[
0
].
_get_data_all_at
(
index
)
for
branch
in
self
.
seg_channels
[
1
:]:
branch_data
=
branch
.
_get_data_all_at
(
index
)
if
branch_data
!=
data
:
raise
Exception
(
'
Pulse channels must be equal for all branches.
'
)
return
data
def
integrate
(
self
,
index
,
sample_rate
=
1e9
):
integrals
=
[
seg_ch
.
integrate
(
index
,
sample_rate
)
for
seg_ch
in
self
.
seg_channels
]
return
integrals
[
0
]
def
get_acquisition_names
(
conditional
:
conditional_segment
):
condition
=
conditional
.
condition
refs
=
condition
if
isinstance
(
condition
,
Iterable
)
else
[
condition
]
acquisition_names
=
set
()
for
ref
in
refs
:
acquisition_names
.
update
(
ref
.
keys
)
acquisition_names
=
list
(
acquisition_names
)
logger
.
info
(
f
'
acquisitions:
{
acquisition_names
}
'
)
return
acquisition_names
class
QsConditionalSegment
:
def
__init__
(
self
,
conditional
:
conditional_segment
):
self
.
conditional
=
conditional
self
.
n_branches
=
len
(
conditional
.
branches
)
condition
=
conditional
.
condition
start
=
time
.
perf_counter
()
refs
=
condition
if
isinstance
(
condition
,
Iterable
)
else
[
condition
]
# Lookup acquistions for condition
self
.
acquisition_names
=
self
.
get_acquisition_names
(
refs
)
self
.
order
=
self
.
get_branch_order
(
refs
)
duration
=
time
.
perf_counter
()
-
start
logger
.
debug
(
f
'
duration
{
duration
*
1000
:
6.3
f
}
ms
'
)
def
get_acquisition_names
(
self
,
refs
:
List
[
MeasurementRef
]):
acquisition_names
=
set
()
for
ref
in
refs
:
acquisition_names
.
update
(
ref
.
keys
)
acquisition_names
=
list
(
acquisition_names
)
logger
.
info
(
f
'
acquisitions:
{
acquisition_names
}
'
)
return
acquisition_names
def
get_branch_order
(
self
,
refs
):
# Assumes max 4 branches
# special case: 1 measurement, 2 acquisitions (and 2 options) => expand to 4 options
# this is handled gracefully by this code:
# 1 measurement: result contains only 0 and 1
# 2 measurements: result contains 0,1,2,3
# 0, 1, 2, 3 in binary representation on 2 acquisitions
all_values
=
np
.
array
([[
0
,
1
,
0
,
1
],[
0
,
0
,
1
,
1
]])
values
=
{
key
:
all_values
[
i
]
for
i
,
key
in
enumerate
(
self
.
acquisition_names
)}
order
=
np
.
zeros
(
4
,
dtype
=
int
)
for
ref
in
refs
:
order
=
2
*
order
+
ref
.
evaluate
(
values
)
logger
.
info
(
f
'
reordered branches:
{
order
}
'
)
return
order
class
QsConditionalMW
():
# sequencer: find common offset per sequencer, generate waveforms
# when uploading, generate extra entries in index table for conditional waveforms
# upload waveforms as usual. Store start/stop
# generate index table entries for conditionals. Start at 248-251.
# ASSUME: only 1 pulse per sequencer.
# Phase shift can shift within segment between pulses.
# Phase shift can be combined in pre-phase or post-phase
# sequencer:
# combine branch to pre, mw_pulse, post.
# set MW pulse
@dataclass
class
BranchPulse
:
mw_pulse
:
any
=
None
prephase
:
float
=
0.0
postphase
:
float
=
0.0
@dataclass
class
ConditionalInstruction
:
start
:
float
end
:
float
pulses
:
List
[
'
QsConditionalMW.BranchPulse
'
]
=
field
(
default_factory
=
list
)
def
__init__
(
self
,
seg_channels
:
List
[
segment_IQ
],
index
):
self
.
seg_channels
=
seg_channels
self
.
n_branches
=
len
(
seg_channels
)
self
.
index
=
index
self
.
conditional_instructions
:
List
[
'
QsConditionalMW.ConditionalInstruction
'
]
=
[]
self
.
combine_branches
()
def
add_pulse
(
self
,
pulse
,
ibranch
):
start
=
pulse
.
start
end
=
pulse
.
stop
for
instr
in
self
.
conditional_instructions
:
# pulse overlaps with instruction
if
start
<
instr
.
end
and
end
>
instr
.
start
:
instr
.
start
=
min
(
start
,
instr
.
start
)
instr
.
end
=
max
(
end
,
instr
.
end
)
if
instr
.
pulses
[
ibranch
]
is
not
None
:
raise
Exception
(
f
'
overlapping pulses in conditional (branch:
{
ibranch
}
)
'
)
instr
.
pulses
[
ibranch
]
=
QsConditionalMW
.
BranchPulse
(
pulse
)
return
# add new instruction
instr
=
QsConditionalMW
.
ConditionalInstruction
(
start
,
end
,
[
None
]
*
self
.
n_branches
)
instr
.
pulses
[
ibranch
]
=
QsConditionalMW
.
BranchPulse
(
pulse
)
self
.
conditional_instructions
.
append
(
instr
)
def
add_phase
(
self
,
phase_shift
,
ibranch
):
for
instr
in
self
.
conditional_instructions
:
if
instr
.
end
>
phase_shift
.
time
:
logger
.
debug
(
f
'
Instr:
{
instr
}
Phase:
{
phase_shift
}
'
)
pulse
=
instr
.
pulses
[
ibranch
]
# try to add phase shift to existing pulse
if
not
pulse
:
instr
.
pulses
[
ibranch
]
=
QsConditionalMW
.
BranchPulse
(
prephase
=
phase_shift
.
phase_shift
)
elif
not
pulse
.
mw_pulse
or
pulse
.
mw_pulse
.
start
>=
phase_shift
.
time
:
pulse
.
prephase
+=
phase_shift
.
phase_shift
elif
pulse
.
mw_pulse
.
stop
<=
phase_shift
.
time
:
pulse
.
postphase
+=
phase_shift
.
phase_shift
else
:
raise
Exception
(
f
'
Phase overlaps with pulse. Branch:
{
ibranch
}
{
phase_shift
}
,
{
pulse
}
'
)
return
# add new instruction
start
=
phase_shift
.
time
end
=
start
+
2
# phase shift requires 2 ns
instr
=
QsConditionalMW
.
ConditionalInstruction
(
start
,
end
,
[
None
]
*
self
.
n_branches
)
instr
.
pulses
[
ibranch
]
=
QsConditionalMW
.
BranchPulse
(
prephase
=
phase_shift
.
phase_shift
)
self
.
conditional_instructions
.
append
(
instr
)
def
combine_branches
(
self
):
# find time + duration of MW pulses
for
ibranch
,
branch
in
enumerate
(
self
.
seg_channels
):
pulse_data
=
branch
.
_get_data_all_at
(
self
.
index
).
MW_pulse_data
# logger.debug(f'Adding MW pulses branch {ibranch} {pulse_data}')
for
pulse
in
pulse_data
:
self
.
add_pulse
(
pulse
,
ibranch
)
self
.
conditional_instructions
.
sort
(
key
=
lambda
x
:
x
.
start
)
# logger.debug(f'Conditional instructions: {self.conditional_instructions}')
# add phase shifts to pulses, pre-phase of post-phase. Sum phase-shifts
for
ibranch
,
branch
in
enumerate
(
self
.
seg_channels
):
phase_data
=
branch
.
_get_data_all_at
(
self
.
index
).
phase_shifts
# logger.debug(f'Adding phase shifts branch {ibranch} {phase_data}')
for
phase_shift
in
phase_data
:
if
phase_shift
.
phase_shift
!=
0.0
:
self
.
add_phase
(
phase_shift
,
ibranch
)
# logger.debug(f'Conditional instructions: {self.conditional_instructions}')
# check pulse overlaps.
last_end
=
-
1
for
instr
in
self
.
conditional_instructions
:
if
instr
.
start
<
last_end
:
raise
Exception
(
f
'
Overlapping conditional instructions
'
)
last_end
=
instr
.
end
def
integrate
(
self
,
index
,
sample_rate
=
1e9
):
# MW channels are not connected to bias-T. No charge accumulation.
return
0
def
get_conditional_channel
(
conditional
:
conditional_segment
,
channel_name
:
str
,
index
=
None
,
sequenced
:
bool
=
False
):
## create Conditional channels
seg_channels
=
[
branch
[
channel_name
]
for
branch
in
conditional
.
branches
]
if
isinstance
(
seg_channels
[
0
],
segment_marker
):
return
ConditionalMarker
(
seg_channels
)
if
isinstance
(
seg_channels
[
0
],
segment_acquisition
):
return
ConditionalAcquisition
(
seg_channels
)
if
sequenced
and
isinstance
(
seg_channels
[
0
],
segment_IQ
):
return
QsConditionalMW
(
seg_channels
,
index
)
if
isinstance
(
seg_channels
[
0
],
segment_base
):
return
ConditionalWaveform
(
seg_channels
)
raise
Exception
(
f
'
Oops:
{
type
(
seg_channels
[
0
])
}
'
)
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment