From 1eb1fa4c9e5b771caf1f79773df4937c533099de Mon Sep 17 00:00:00 2001 From: Bobby Jackson Date: Wed, 4 Sep 2024 10:40:51 -0500 Subject: [PATCH 1/4] FIX: Ensure bias is float64 upon init --- pysp2/util/deadtime.py | 111 ++++++++++++++++++++++------------------- 1 file changed, 61 insertions(+), 50 deletions(-) diff --git a/pysp2/util/deadtime.py b/pysp2/util/deadtime.py index 5888529..ab399e3 100644 --- a/pysp2/util/deadtime.py +++ b/pysp2/util/deadtime.py @@ -1,21 +1,22 @@ import numpy as np import xarray as xr + def deadtime(my_binary, my_ini): """ Function to calculate Deadtime Bias in the SP2 data as published in - Joshua. P. Schwarz, J. M. Katich, S. L. Lee, D. S. Thomson & L. A. Watts - (2022) “Invisible bias” in the single particle soot photometer due to - trigger deadtime, Aerosol Science and Technology, 56:7, 623-635, - DOI: 10.1080/02786826.2022.2064265 - + Joshua. P. Schwarz, J. M. Katich, S. L. Lee, D. S. Thomson & L. A. Watts + (2022) “Invisible bias” in the single particle soot photometer due to + trigger deadtime, Aerosol Science and Technology, 56:7, 623-635, + DOI: 10.1080/02786826.2022.2064265 + Parameters ---------- my_binary : xarrray Dataset xarray Dataset that is the output of pysp2.util.gaussian_fit(my_sp2b, my_ini) Dataset with gaussian fits, peak heights etc. - + my_ini : dict The .ini file loaded as a dict. my_ini=pysp2.io.read_config(ini_file) @@ -23,57 +24,67 @@ def deadtime(my_binary, my_ini): ------- my_binary : xarrray Dataset Returns the input xarray dataset with one additional variable: - DeadtimeRelativeBias. This can be used to correct for Deadtime Bias - in the SP2 at high concentrations. + DeadtimeRelativeBias. This can be used to correct for Deadtime Bias + in the SP2 at high concentrations. """ - #numpy array to store the relative bias in - bias = np.zeros(my_binary.sizes['event_index'], ) - #scattering triggered - scatter_triggered = np.any(np.isfinite(np.vstack((my_binary['PkHt_ch0'].values, - my_binary['PkHt_ch4'].values))), axis=0) - #incandesence triggered - incandesence_triggered = np.any(np.isfinite(np.vstack((my_binary['PkHt_ch1'].values, - my_binary['PkHt_ch5'].values))), axis=0) - digitization_rate = int(my_ini['Acquisition']['Samples/Sec']) - buffer_length = int(my_ini['Acquisition']['Scan Length']) #how many data points in one buffer - buffer_length_seconds = buffer_length/digitization_rate #in seconds - event_length_points = int(my_ini['Acquisition']['Points per Event']) - pretrigger_length_points = int(my_ini['Acquisition']['Pre-Trig Points']) - #intentialanny skipped particles, notation from Eq. 2 in doi:10.1080/02786826.2022.2064265 - S_S = int(my_ini['Acquisition']['1 of every']) - #find locations where the buffer changes (i.e. where the TimeWave values changes) - #get unique time stamps - unique_buffer_time_stamps = np.unique(my_binary['TimeWave'].values) - #find locations where those unique time stamps should be inserted - ind = np.searchsorted(my_binary['TimeWave'].values, unique_buffer_time_stamps) - #add the end to the list of indexes if needed - if ind[-1] < my_binary.sizes['event_index']: - ind = np.append(ind, my_binary.sizes['event_index']) - for i in range(len(ind)-1): - #from index + # numpy array to store the relative bias in + bias = np.zeros(my_binary.sizes["event_index"], dtype="float64") + # scattering triggered + scatter_triggered = np.any( + np.isfinite( + np.vstack((my_binary["PkHt_ch0"].values, my_binary["PkHt_ch4"].values)) + ), + axis=0, + ) + # incandesence triggered + incandesence_triggered = np.any( + np.isfinite( + np.vstack((my_binary["PkHt_ch1"].values, my_binary["PkHt_ch5"].values)) + ), + axis=0, + ) + digitization_rate = int(my_ini["Acquisition"]["Samples/Sec"]) + buffer_length = int( + my_ini["Acquisition"]["Scan Length"] + ) # how many data points in one buffer + buffer_length_seconds = buffer_length / digitization_rate # in seconds + event_length_points = int(my_ini["Acquisition"]["Points per Event"]) + pretrigger_length_points = int(my_ini["Acquisition"]["Pre-Trig Points"]) + # intentialanny skipped particles, notation from Eq. 2 in doi:10.1080/02786826.2022.2064265 + S_S = int(my_ini["Acquisition"]["1 of every"]) + # find locations where the buffer changes (i.e. where the TimeWave values changes) + # get unique time stamps + unique_buffer_time_stamps = np.unique(my_binary["TimeWave"].values) + # find locations where those unique time stamps should be inserted + ind = np.searchsorted(my_binary["TimeWave"].values, unique_buffer_time_stamps) + # add the end to the list of indexes if needed + if ind[-1] < my_binary.sizes["event_index"]: + ind = np.append(ind, my_binary.sizes["event_index"]) + for i in range(len(ind) - 1): + # from index fr = ind[i] - #to index - to = ind[i+1] - #number of scattering particle windows saved in the buffer + # to index + to = ind[i + 1] + # number of scattering particle windows saved in the buffer N_S = scatter_triggered[fr:to].sum() - #number of incandesence particle windows saved in the buffer + # number of incandesence particle windows saved in the buffer N_I = incandesence_triggered[fr:to].sum() - #length of time, in seconds, for a single measurement of the digitizer - t_b = 1. / digitization_rate - #"PW is the total number of data points (for each channel of data) in the window" + # length of time, in seconds, for a single measurement of the digitizer + t_b = 1.0 / digitization_rate + # "PW is the total number of data points (for each channel of data) in the window" P_W = event_length_points - #Buffer length in seconds - T_B = buffer_length_seconds - #Fraction triggered (Eq. 2) in doi:10.1080/02786826.2022.2064265 - F_T = (N_S * S_S+N_I) * P_W * t_b / T_B - #pretrigger length data points + # Buffer length in seconds + T_B = buffer_length_seconds + # Fraction triggered (Eq. 2) in doi:10.1080/02786826.2022.2064265 + F_T = (N_S * S_S + N_I) * P_W * t_b / T_B + # pretrigger length data points P_PT = pretrigger_length_points - #T_P is the total points in each window + # T_P is the total points in each window T_P = event_length_points - #relative bias as in Eq. 3 in doi:10.1080/02786826.2022.2064265 + # relative bias as in Eq. 3 in doi:10.1080/02786826.2022.2064265 B_rel = -(P_PT / T_P) * F_T - #save the bias value to the whole buffer + # save the bias value to the whole buffer bias[fr:to] = B_rel - #add the bias containing array to the xarray - my_binary['DeadtimeRelativeBias'] = xr.DataArray(bias, dims=['event_index']) + # add the bias containing array to the xarray + my_binary["DeadtimeRelativeBias"] = xr.DataArray(bias, dims=["event_index"]) return my_binary From 74a768749c582407c64ce9be7b76c65e51d4cc78 Mon Sep 17 00:00:00 2001 From: Bobby Jackson Date: Wed, 4 Sep 2024 10:45:50 -0500 Subject: [PATCH 2/4] Convert num of scattered and incandescence to float64 --- pysp2/util/deadtime.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/pysp2/util/deadtime.py b/pysp2/util/deadtime.py index ab399e3..cf82f2a 100644 --- a/pysp2/util/deadtime.py +++ b/pysp2/util/deadtime.py @@ -66,9 +66,9 @@ def deadtime(my_binary, my_ini): # to index to = ind[i + 1] # number of scattering particle windows saved in the buffer - N_S = scatter_triggered[fr:to].sum() + N_S = scatter_triggered[fr:to].sum().astype('float64') # number of incandesence particle windows saved in the buffer - N_I = incandesence_triggered[fr:to].sum() + N_I = incandesence_triggered[fr:to].sum().astype('float64') # length of time, in seconds, for a single measurement of the digitizer t_b = 1.0 / digitization_rate # "PW is the total number of data points (for each channel of data) in the window" From 355205e12d01e7cf7bca4d2b4fc4f23ef17ab459 Mon Sep 17 00:00:00 2001 From: Bobby Jackson Date: Wed, 4 Sep 2024 10:49:57 -0500 Subject: [PATCH 3/4] Convert ints to floats --- pysp2/util/deadtime.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/pysp2/util/deadtime.py b/pysp2/util/deadtime.py index cf82f2a..c853743 100644 --- a/pysp2/util/deadtime.py +++ b/pysp2/util/deadtime.py @@ -43,15 +43,15 @@ def deadtime(my_binary, my_ini): ), axis=0, ) - digitization_rate = int(my_ini["Acquisition"]["Samples/Sec"]) - buffer_length = int( + digitization_rate = float(my_ini["Acquisition"]["Samples/Sec"]) + buffer_length = float( my_ini["Acquisition"]["Scan Length"] ) # how many data points in one buffer buffer_length_seconds = buffer_length / digitization_rate # in seconds - event_length_points = int(my_ini["Acquisition"]["Points per Event"]) - pretrigger_length_points = int(my_ini["Acquisition"]["Pre-Trig Points"]) + event_length_points = float(my_ini["Acquisition"]["Points per Event"]) + pretrigger_length_points = float(my_ini["Acquisition"]["Pre-Trig Points"]) # intentialanny skipped particles, notation from Eq. 2 in doi:10.1080/02786826.2022.2064265 - S_S = int(my_ini["Acquisition"]["1 of every"]) + S_S = float(my_ini["Acquisition"]["1 of every"]) # find locations where the buffer changes (i.e. where the TimeWave values changes) # get unique time stamps unique_buffer_time_stamps = np.unique(my_binary["TimeWave"].values) From 79a033d8246c4ba2862c053f653ce4bd321effe8 Mon Sep 17 00:00:00 2001 From: Bobby Jackson Date: Wed, 4 Sep 2024 11:37:11 -0500 Subject: [PATCH 4/4] Float32? --- pysp2/util/deadtime.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pysp2/util/deadtime.py b/pysp2/util/deadtime.py index c853743..1474faf 100644 --- a/pysp2/util/deadtime.py +++ b/pysp2/util/deadtime.py @@ -28,7 +28,7 @@ def deadtime(my_binary, my_ini): in the SP2 at high concentrations. """ # numpy array to store the relative bias in - bias = np.zeros(my_binary.sizes["event_index"], dtype="float64") + bias = np.zeros(my_binary.sizes["event_index"], dtype="float32") # scattering triggered scatter_triggered = np.any( np.isfinite(