From 0aabd502d7ff61cacb54d4131085ecd74bf92a80 Mon Sep 17 00:00:00 2001 From: Sander de Snoo <59472150+sldesnoo-Delft@users.noreply.github.com> Date: Mon, 10 Jul 2023 11:46:33 +0200 Subject: [PATCH] Render wave of add_sin() with phase starting at start of wave, i.e. independent of start time of pulse --- pulse_lib/keysight/qs_sequence.py | 8 +++---- pulse_lib/qblox/pulsar_uploader.py | 8 +++++-- pulse_lib/segments/data_classes/data_IQ.py | 9 +++++++- pulse_lib/segments/data_classes/data_pulse.py | 22 ++++++++++++------- pulse_lib/segments/segment_pulse.py | 8 ++++--- pulse_lib/tests/utils/last_upload.py | 2 +- 6 files changed, 38 insertions(+), 19 deletions(-) diff --git a/pulse_lib/keysight/qs_sequence.py b/pulse_lib/keysight/qs_sequence.py index 357af09d..914a750f 100644 --- a/pulse_lib/keysight/qs_sequence.py +++ b/pulse_lib/keysight/qs_sequence.py @@ -205,8 +205,8 @@ class IQSequenceBuilder: if abs(frequency) > 450e6: raise Exception(f'Waveform NCO frequency {frequency/1e6:5.1f} MHz is out of range') - prephase += mw_pulse_data.start_phase - postphase -= mw_pulse_data.start_phase + prephase += mw_pulse_data.phase_offset + postphase -= mw_pulse_data.phase_offset return Waveform(mw_pulse_data.amplitude, amp_envelope, frequency, pm_envelope, prephase, postphase, duration) @@ -216,8 +216,8 @@ class IQSequenceBuilder: if abs(frequency) > 450e6: raise Exception(f'Waveform NCO frequency {frequency/1e6:5.1f} MHz is out of range') - prephase = mw_pulse_data.start_phase - postphase = -mw_pulse_data.start_phase + prephase = mw_pulse_data.phase_offset + postphase = -mw_pulse_data.phase_offset start_wvf = Waveform(mw_pulse_data.amplitude, 1.0, frequency, 0.0, prephase=prephase, diff --git a/pulse_lib/qblox/pulsar_uploader.py b/pulse_lib/qblox/pulsar_uploader.py index 8b8774fc..ea51f290 100644 --- a/pulse_lib/qblox/pulsar_uploader.py +++ b/pulse_lib/qblox/pulsar_uploader.py @@ -620,7 +620,11 @@ class UploadAggregator: t_end = e.stop + seg_start wave_duration = iround(e.stop) - iround(e.start) # 1 ns resolution amod, phmod = get_modulation(e.envelope, wave_duration) - sinewave = SineWaveform(wave_duration, e.frequency, e.start_phase, amod, phmod) + if e.coherent_pulsing: + phase = e.phase_offset + 2*np.pi*e.frequency*t*1e-9 + else: + phase = e.phase_offset + sinewave = SineWaveform(wave_duration, e.frequency, phase, amod, phmod) seq.add_sin(t, t_end, e.amplitude*scaling, sinewave) elif isinstance(e, PhaseShift): raise Exception('Phase shift not supported for AWG channel') @@ -690,7 +694,7 @@ class UploadAggregator: wave_duration = iround(e.stop - e.start) # 1 ns resolution for waveform amod, phmod = get_modulation(e.envelope, wave_duration) sinewave = SineWaveform(wave_duration, e.frequency-lo_freq, - e.start_phase, amod, phmod) + e.phase_offset, amod, phmod) seq.pulse(t_start, t_end, e.amplitude*scaling, sinewave) elif isinstance(e, PhaseShift): t_start = e.start + seg_start diff --git a/pulse_lib/segments/data_classes/data_IQ.py b/pulse_lib/segments/data_classes/data_IQ.py index 9b49fff2..673bf4e0 100644 --- a/pulse_lib/segments/data_classes/data_IQ.py +++ b/pulse_lib/segments/data_classes/data_IQ.py @@ -111,9 +111,16 @@ class IQ_data_single: stop : float = 0 amplitude : float = 1 frequency : float = 0 - start_phase : float = 0 + phase_offset : float = 0 + ''' offset from coherent pulse ''' envelope : envelope_generator = None ref_channel : str = None + coherent_pulsing : bool = True + + @property + def start_phase(self): + ''' Old name for phase_offset ''' + return self.phase_offset @dataclass class Chirp: diff --git a/pulse_lib/segments/data_classes/data_pulse.py b/pulse_lib/segments/data_classes/data_pulse.py index f3f9a51a..13a1cad2 100644 --- a/pulse_lib/segments/data_classes/data_pulse.py +++ b/pulse_lib/segments/data_classes/data_pulse.py @@ -356,7 +356,7 @@ class pulse_data(parent_data): return for mw_pulse in self.MW_pulse_data: - mw_pulse.start_phase += phase_shift + mw_pulse.phase_offset += phase_shift for chirp in self.chirp_data: chirp.phase += phase_shift @@ -715,7 +715,7 @@ class pulse_data(parent_data): if abs(freq) > sample_rate*1e9/2: raise Exception(f'Frequency {freq*1e-6:5.1f} MHz is above Nyquist frequency ({sample_rate*1e3/2} MHz)') # TODO add check on configurable bandwidth. - phase = mw_pulse.start_phase + phase = mw_pulse.phase_offset if ref_channel_states and mw_pulse.ref_channel in ref_channel_states.start_phase: ref_start_time = ref_channel_states.start_time ref_start_phase = ref_channel_states.start_phase[mw_pulse.ref_channel] @@ -745,10 +745,16 @@ class pulse_data(parent_data): start_pt = iround(start_pulse * sample_rate) stop_pt = start_pt + n_pt - # add the sin pulse - total_phase = phase_shift + phase + phase_envelope + ref_start_phase - t = start_pt+ref_start_time/sample_rate + np.arange(n_pt) - wvf[start_pt:stop_pt] += amp*amp_envelope*np.sin(2*np.pi*freq/sample_rate*1e-9*t + total_phase) + # add the sine wave + if mw_pulse.coherent_pulsing: + total_phase = phase_shift + phase + phase_envelope + ref_start_phase + t = start_pt+ref_start_time/sample_rate + np.arange(n_pt) + wvf[start_pt:stop_pt] += amp*amp_envelope*np.sin(2*np.pi*freq/sample_rate*1e-9*t + total_phase) + else: + # it's not an IQ pulse, but a sine wave on voltage channel. + total_phase = phase + phase_envelope + t = np.arange(n_pt) + wvf[start_pt:stop_pt] += amp*amp_envelope*np.sin(2*np.pi*freq/sample_rate*1e-9*t + total_phase) for custom_pulse in self.custom_pulse_data: data = custom_pulse.render(sample_rate*1e9) @@ -830,7 +836,7 @@ class pulse_data(parent_data): # max amp, freq and phase. amp = mw_pulse.amplitude freq = mw_pulse.frequency - phase = mw_pulse.start_phase + phase = mw_pulse.phase_offset if ref_channel_states and mw_pulse.ref_channel in ref_channel_states.start_phase: ref_start_time = ref_channel_states.start_time ref_start_phase = ref_channel_states.start_phase[mw_pulse.ref_channel] @@ -910,7 +916,7 @@ class pulse_data(parent_data): pd['stop'] = pulse.stop pd['amplitude'] = pulse.amplitude pd['frequency'] = pulse.frequency - pd['start_phase'] = pulse.start_phase + phase_shift + pd['start_phase'] = pulse.phase_offset + phase_shift envelope = pulse.envelope if envelope is None: envelope = envelope_generator() diff --git a/pulse_lib/segments/segment_pulse.py b/pulse_lib/segments/segment_pulse.py index 9883476b..2c888877 100644 --- a/pulse_lib/segments/segment_pulse.py +++ b/pulse_lib/segments/segment_pulse.py @@ -113,8 +113,9 @@ class segment_pulse(segment_base): @loop_controller def add_sin(self, start, stop, amp, freq, phase_offset=0): ''' - add a sinus to the current segment, parameters should be self exlenatory. - The pulse will have a not have a relative phase phase. + Adds a sine wave to the current segment. + The pulse does not have a coherent phase with other pulses, + unlike add_MW_pulse on IQ channels. Args: start (double) : start time in ns of the pulse stop (double) : stop time in ns of the pulse @@ -127,7 +128,8 @@ class segment_pulse(segment_base): amp, freq, phase_offset, None, # no envelope - self.name)) + self.name, + coherent_pulsing=False)) return self.data_tmp @loop_controller diff --git a/pulse_lib/tests/utils/last_upload.py b/pulse_lib/tests/utils/last_upload.py index 0cb41b0a..861b01b7 100644 --- a/pulse_lib/tests/utils/last_upload.py +++ b/pulse_lib/tests/utils/last_upload.py @@ -64,5 +64,5 @@ class LastUpload: elif isinstance(e, PhaseShift): print(f'shift phase {e.phase_shift:+6.3f} rad ({e.channel_name})') elif isinstance(e, IQ_data_single): - print(f'MW pulse {e.amplitude:6.2f} {e.frequency/1e6:6.1f} MHz {e.start_phase:+6.3f} rad ({e.ref_channel})') + print(f'MW pulse {e.amplitude:6.2f} {e.frequency/1e6:6.1f} MHz {e.phase_offset:+6.3f} rad ({e.ref_channel})') -- GitLab