As of PulseView 0.5.0-git-9d307c6 / libsigrok 0.6.0-git-d7df9dc, there seems to be no facility, that would allow combining captures from .sr files.
For instance, in Audacity, - which is an audio editor, but shares the visualisation of waveform tracks with PulseView, - if the user does File/Import/Audio... once at first, a new session will be opened, and the audio tracks from the file will be added to this session. If the user then does File/Import/Audio... a second time, importing a second file, then the tracks from the second file will be added to the track list of the session - creating a session that contains the combination of the audio tracks from both files.
However, if the user does something similar in PulseView - a new session will be opened for each import, thereby making it impossible to create a session which contains waveforms/data from multiple files.
One of the essential things to note is that, Audacity can handle different sample rates for each track, while also having a sample rate for the entire project/session ( see e.g. https://superuser.com/questions/420531/audacity-resampling ; http://www.dynamicsoflanguage.edu.au/research/data-archives/guides/resampling-audio-using-audacity/ ) .
On the other hand, PulseView (at the time of writing, at least) assumes that there is a single sample rate used in the entire session; thus, in order to start considering merging different tracks/data into a single session in PulseView, we must first consider how to bring all this data under the same sampling rate - that is, we need to consider resampling.
In this example, we consider a simple Arduino application, driven by an analog signal and with output digital signals; this application will be captured by Saleae Logic in mixed mode (both analog and digital capture), which Saleae Logic typically captures with different sampling rates. Then we consider export from Saleae Logic into formats that can be imported into PulseView, from where we will save them as the native .sr
format - which will also inherit the differing sampliing rates. Finally, we look into how can we resample the relevant data, so they are end up under the same sampling rate - which will facilitate merging all of this data into a single session.
The audio waveform is a "chirp", generated by Audacity, using:
- Generate/Chirp: Waveform: Sine, Start: 440 Hz Amplitude 1.0, End: 1320 Hz Amplitude 1.0, Interpolation: Linear, Duration: 1 sec
- At end of the above chirp: Generate/Silence, 0.5 sec
- This is exported as 16-bit, 44100 Hz mono .wav file,
chirp_base.wav
.
When this file is reproduced out of the PC soundcard on max volume, a max ampliture of about ± 1.75 V can be measured on oscilloscope (note that Saleae Logic cannot measure the negative semiperiod below -1V!)
The example Arduino code is taken from Analog Comparator Interrupt - and here it is implemented in sketch_arduino_analog_comp_test/.
The schematic that shows the routing for this test code is shown in img/arduino_test_schematic.png.
The code sets up an analog comparator: the negative input of the comparator (Arduino pin 7) is held on ground, and it represents the comparison threshold at 0V; while the positive input of the comparator is driven by the (mono) channel of the audio, as reproduced by the PC. When the audio channel signal crosses the comparator threshold, an interrupt is raised, where pin 13 of the Arduino is toggled.
On the other hand, the Arduino main loop runs with a 100 ms delay (sleep), and whenever it runs, it copies the state of the comparator to Arduino pin 9.
Finally, with Saleae Logic, we capture the following signals:
- The channel audio, captured as analog signal, as Saleae Logic Ch.0 (usually marked with black color)
- The pin 13 of Arduino (here, as interrupt indicator), captured as digital signal, as Saleae Logic Ch.1 (usually marked with brown color)
- The pin 9 of Arduino (here, as "slow" comparator state indicator), captured as digital signal, as Saleae Logic Ch.2 (usually marked with red color)
When capturing in mixed mode (both analog and digital signals) with Saleae Logic 1.2.18, one cannot arbitrarily choose the sampling rates for analog and digital signals; so the following captures were performed of this test setup:
-
data/6.25MHz_digital_1.5625MHz_analog_3sec.logicdata - 6.25 MHz digital sampling rate, with 1.5625 MHz analog sampling rate, 3-second capture
-
data/500MHz_digital_0.125MHz_analog_5sec.logicdata - 500 MHz digital sampling rate, with 125 kHz analog sampling rate, 5-second capture
So - the goal of this exercise, is to bring all of these captures in a single PulseView session.
Since we made "mixed-mode" (analog & digital) captures in Saleae Logic, we have several options to export the data, so that we can import it in sigrok/PulseView; we can see these options by clicking Options/Export Data in Saleae Logic 1.2.18, once we have loaded a .logicdata
file in it:
- Digital and Analog export - see img/Saleae_Logic_Export_dg_an.png
- Export to Csv
.csv
file - Export to Matlab
.mat
file
- Export to Csv
Note that the Csv import in sigrok/PulseView assumes that there is a single sampling rate; while the sigrok/PulseView .csv importer can interpret timestamps as floating-point seconds, it only does so to calculate the delta between two consecutive timestamps, and to derive the overall sampling rate from them; thus, timestamps are not interpreted "absolutely", and timestamp "gaps" (outside of the assumed sampling rate) in the .csv are not supported (see sigrok / Thread: [sigrok-devel] How is the .csv import with timestamps in PulseView supposed to work?)
However, if we export a "mixed-mode" .csv from Saleae Logic, we get a file like:
Time [s],Channel 0-Analog, Time [s],Channel 1-Digital, Time [s],Channel 2-Digital
0.000000000000000, -0.038281224668026, 0.000000000000000, 0, 0.000000000000000, 0
0.000000640000000, -0.033183418214321, 0.096834720000000, 1, 0.123360000000000, 1
0.000001280000000, -0.033183418214321, 0.096855360000000, 0, 0.223311200000000, 0
0.000001920000000, -0.038281224668026, 0.096867840000000, 1, 0.623114400000000, 1
...
... or in other words, we have several timestamp columns, all of which have "gaps" - which is not likely to import correctly in sigrok/PulseView.
As far as the Matlab file format, sigrok/PulseView does not currently support it natively; there are Python libraries that can load Matlab .mat
files ( using from scipy.io import loadmat
; see https://towardsdatascience.com/how-to-load-matlab-mat-files-in-python-1f200e1287b5 ), however, that means we have to additionally write a converter from Matlab to sigrok format.
So, let's see if there is an easier way to import this data into sigrok/PulseView - we can try exporting digital and analog data from Saleae Logic separately. The options here are (remember to set "Choose which channels to export" in the "Data Export" dialog in Saleae Logic first):
- Analog Only export - see img/Saleae_Logic_Export_an.png
- Export to Binary
.bin
file - Export to Csv
.csv
file - Export to Matlab
.mat
file
- Export to Binary
For the Analog Only export, the Matlab - and the Binary - options present the same problem as earlier - we'd have to write a converter program, to get this data into sigrok/PulseView.
However, using the Csv export option should this time be fine - because the exported .csv
file, for a single channel, looks like this:
Time[s], Channel 0
Time [s],Channel 0-Analog
0.000000000000000, -0.038281224668026
0.000000640000000, -0.033183418214321
0.000001280000000, -0.033183418214321
0.000001920000000, -0.038281224668026
...
... which means the data is represented with timestamps at a constant sampling rate (without gaps) - meaning that sigrok/PulseView should be able to interpret it correctly.
So, let's say we've exported these .csv
files from our source captures (as they are intermediary files, the .csv
files are not saved in this repository):
6.25MHz_digital_1.5625MHz_analog_3sec.logicdata
->6.25MHz_digital_1.5625MHz_analog_3sec_an.csv
500MHz_digital_0.125MHz_analog_5sec.logicdata
->500MHz_digital_0.125MHz_analog_5sec_an.csv
We only have to delete one of the extra header lines (say, the Time [s],Channel 0-Analog
one) manually from the .csv files, and they should be ready for import in sigrok/PulseView.
First, let's check the file sizes (the below snippets have been ran under MSYS2 bash
in MS Windows 10):
$ ls -la *_an.csv
-rw-r--r-- 1 user None 24M Oct 19 07:05 500MHz_digital_0.125MHz_analog_5sec_an.csv
-rw-r--r-- 1 user None 174M Oct 19 07:04 6.25MHz_digital_1.5625MHz_analog_3sec_an.csv
We can see that the file sizes are different - which is expected, since the two analog captures have been made at different sampling rates.
We can also do a quick check with sigrok and logging, to see how these files will be interpreted:
$ "C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" \
-l 3 -I csv:header=yes:column_formats=t,a \
-i 500MHz_digital_0.125MHz_analog_5sec_an.csv \
--show
cli: Received SR_DF_HEADER.
sr: input/csv: Cannot convert timestamp text 0.000000000000000 in line 2 (or zero value).
cli: Received SR_DF_META.
cli: Got samplerate 125000 Hz.
cli: Received SR_DF_ANALOG (626405 samples).
cli: Received SR_DF_END.
Samplerate: 125000
Channels: 1
- Channel 0: analog
Analog sample count: 626405
$ "C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" \
-l 3 -I csv:header=yes:column_formats=t,a \
-i 6.25MHz_digital_1.5625MHz_analog_3sec_an.csv \
--show
cli: Received SR_DF_HEADER.
sr: input/csv: Cannot convert timestamp text 0.000000000000000 in line 2 (or zero value).
cli: Received SR_DF_META.
cli: Got samplerate 1562500 Hz.
cli: Received SR_DF_ANALOG (1048576 samples).
cli: Received SR_DF_ANALOG (1048576 samples).
cli: Received SR_DF_ANALOG (1048576 samples).
cli: Received SR_DF_ANALOG (1048576 samples).
cli: Received SR_DF_ANALOG (523949 samples).
cli: Received SR_DF_END.
Samplerate: 1562500
Channels: 1
- Channel 0: analog
Analog sample count: 4718253
So, we can see that the analog sample rates have been derived correctly from the .csv file timestamps by the sigrok/PulseView .csv importer.
Now, we can start PulseView, and import these .csv's there, and then export the sigrok .sr
session files corresponding to them - again, since upon each import, PulseView will start a new session, we will obtain (in this case) one .sr
file for each .csv
file (each of which contains data for a single analog channel).
In PulseView, just click "Import Comma-separated values...", and then just type t,a
in the subsequent "Import Comma-separated values" dialog (see img/PulseView_csv_import_analog.png). The data will be imported, and then you can click Save As, and choose "Save as type:" "srzip session file format data files (*.sr)".
Or, we can use the command line directly:
"C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" \
-l 3 -I csv:header=yes:column_formats=t,a \
-i 500MHz_digital_0.125MHz_analog_5sec_an.csv \
-o 500MHz_digital_0.125MHz_analog_5sec_an.sr
"C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" \
-l 3 -I csv:header=yes:column_formats=t,a \
-i 6.25MHz_digital_1.5625MHz_analog_3sec_an.csv \
-o 6.25MHz_digital_1.5625MHz_analog_3sec_an.sr
These files are included in this repository:
A quick check of the filesizes:
$ ls -la *_an.sr
-rw-r--r-- 1 user None 490K Oct 19 07:25 500MHz_digital_0.125MHz_analog_5sec_an.sr
-rw-r--r-- 1 user None 1.4M Oct 19 07:27 6.25MHz_digital_1.5625MHz_analog_3sec_an.sr
... and let's restate again, that these two files each have a single channel of analog data - however, at different sampling rates.
Now, let's export the digital channels, from our Saleae Logic captures, into sigrok/PulseView .sr
files. The Saleae Logic options here are:
- Digital Only export - see img/Saleae_Logic_Export_dg.png
- Export to Binary
.bin
file - Export to Csv
.csv
file - Export to Vcd
.vcd
file - Export to Matlab
.mat
file
- Export to Binary
For the Digital Only export, the Matlab - and the Binary - options present the same problem as earlier - we'd have to write a converter program, to get this data into sigrok/PulseView.
The Vcd (see https://en.wikipedia.org/wiki/Value_change_dump) export typically works fine in this case, without having to set any options - and typically sigrok/PulseView can import digital data from a .vcd without a problem.
As far as the Csv export is concerned: if we choose the "Use timestamps [s]" option in the "CSV Settings", then we'll get a file like:
Time[s], Channel 1, Channel 2
0.000000000000000, 1, 1
0.000174486000000, 0, 1
0.000661762000000, 1, 1
0.000674068000000, 0, 1
0.001168152000000, 1, 1
...
... which quite clearly has gaps - which means it will not be imported correctly in sigrok/PulseView.
However, if we use the "Use sample numbers" option instead, we get a file like:
Sample, Channel 1, Channel 2
0, 1, 1
87243, 0, 1
330881, 1, 1
337034, 0, 1
584076, 1, 1
...
... which again is a .csv file with gaps - which means it will not be imported correctly in sigrok/PulseView.
So, our only option here, is to export the digital data from the Saleae Logic captures as a .vcd
file.
So, let's say we've exported these .vcd
files from our source captures (as they are intermediary files, the .vcd
files are not saved in this repository):
6.25MHz_digital_1.5625MHz_analog_3sec.logicdata
->6.25MHz_digital_1.5625MHz_analog_3sec_dg.vcd
500MHz_digital_0.125MHz_analog_5sec.logicdata
->6.25MHz_digital_1.5625MHz_analog_3sec_dg.vcd
Again, let's check the file sizes:
$ ls -la *_dg.vcd
-rw-r--r-- 1 user None 192K Oct 19 07:50 500MHz_digital_0.125MHz_analog_5sec_dg.vcd
-rw-r--r-- 1 user None 105K Oct 19 07:51 6.25MHz_digital_1.5625MHz_analog_3sec_dg.vcd
Since .vcd
is a text file format, it is interesting to see one property of these .vcd files:
$ grep -r 'timescale' . --include='*.vcd'
./500MHz_digital_0.125MHz_analog_5sec_dg.vcd:$timescale 1 ns $end
./6.25MHz_digital_1.5625MHz_analog_3sec_dg.vcd:$timescale 1 ns $end
The $timescale
parameter can be seen as an implied sampling rate of the data in the file; note that, while the original digital captures were done at 500MHz and 6.25MHz respective sampling rates - the data from both, due to the $timescale
VCD parameter, will be intepreted by sigrok/PulseView to have been captured at 1/1e-9 = 1 GHz sampling rate!
We can also inspect the resulting .vcd files using sigrok
(see also: https://sigrok.org/wiki/File_format:Vcd) - which will confirm the 1 GHz sampling rate interpretation:
$ "C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" -i 500MHz_digital_0.125MHz_analog_5sec_dg.vcd --show Samplerate: 1000000000
Channels: 2
- top.Channel_1: logic
- top.Channel_2: logic
Logic unitsize: 1
Logic sample count: 5008344477
$ "C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" -i 6.25MHz_digital_1.5625MHz_analog_3sec_dg.vcd --show
Samplerate: 1000000000
Channels: 2
- top.Channel_1: logic
- top.Channel_2: logic
Logic unitsize: 1
Logic sample count: 2759602561
Again, we can either use PulseView to import the .vcd
files, and to export .sr
files - or, we can use sigrok
from the command like (note, this process takes a while, maybe a couple of minutes - regardless of which approach we choose):
"C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" \
-l 3 \
-i 500MHz_digital_0.125MHz_analog_5sec_dg.vcd \
-o 500MHz_digital_0.125MHz_analog_5sec_dg.sr
"C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" \
-l 3 \
-i 6.25MHz_digital_1.5625MHz_analog_3sec_dg.vcd \
-o 6.25MHz_digital_1.5625MHz_analog_3sec_dg.sr
These files are included in this repository:
A quick check of the filesizes:
$ ls -la *_dg.sr
-rw-r--r-- 1 user None 4.8M Oct 19 08:06 500MHz_digital_0.125MHz_analog_5sec_dg.sr
-rw-r--r-- 1 user None 2.7M Oct 19 08:07 6.25MHz_digital_1.5625MHz_analog_3sec_dg.sr
So, from our two mixed-mode (analog & digital) Saleae Logic captures, we obtained four .sr
files - two with analog data, and two with digital data, all of which having (in principle) differing sampling rates. Let's check them again through the command line:
$ for ifile in *.sr; do echo $ifile; "C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" -i $ifile --show; echo; done
500MHz_digital_0.125MHz_analog_5sec_an.sr
Samplerate: 125000
Channels: 1
- Channel 0: analog
Analog sample count: 626405
500MHz_digital_0.125MHz_analog_5sec_dg.sr
Samplerate: 1000000000
Channels: 2
- top.Channel_1: logic
- top.Channel_2: logic
Logic unitsize: 1
Logic sample count: 5008344477
6.25MHz_digital_1.5625MHz_analog_3sec_an.sr
Samplerate: 1562500
Channels: 1
- Channel 0: analog
Analog sample count: 4718253
6.25MHz_digital_1.5625MHz_analog_3sec_dg.sr
Samplerate: 1000000000
Channels: 2
- top.Channel_1: logic
- top.Channel_2: logic
Logic unitsize: 1
Logic sample count: 2759602561
So, in order to bring all of these .sr
files into the same session, we have to generally handle the following problems:
- All of the data files need to be on the same sampling rate
- The channel names need to be changed so they are unique in the new, merged, session scope
The sigrok .sr
file format is documented at https://sigrok.org/wiki/File_format:Sigrok/v2 - which informs us, that .sr
(of the current version at time of writing, which is V2) files are essentially ZIP archives.
Regarding the sampling rate, the most straightforward solution is to find the largest sampling rate, and then resample all data to this sampling rate. We have already determined that our files use rates of 125000, 1562500 and 1000000000 Hz - thus, the most straightforward solution is to resample the 125000 and 1562500 Hz analog signals to 1000000000 Hz. Clearly, this will result with a massive file size increase - especially since in this case, we are to resample analog data.
Furthermore, we'd need to choose whether we want interpolation of analog samples upon resampling, or not. Thankfully, the relationships between the sampling rates are:
- 1000000000/125000 = 8000
- 1000000000/1562500 = 640
... or in other words, they are integer ratios - so a resampling without interpolation would simply involve repeating a sample an integet number of times.
Note that sigrok (or rather, libsigrok
) has a Python API: https://www.sigrok.org/api/libsigrok/0.5.0/bindings/python/ - and there is a Python example of using this API at https://github.com/martinling/sigrok-cli-python ; however, the Windows installers https://sigrok.org/wiki/Windows#Windows_installers for sigrok/PulseView are static, so no shared (.dll) libsigrok libraries are available from these installers (and consequently, no Python libraries either). So, in order to get these bindings on Windows, we'd have to build libsigrok
from source - however https://sigrok.org/wiki/Windows#Native_build_using_MSYS2 states that:
NOT yet working: libsigrok Python/Ruby/Java bindings [...]
... and indeed, as of time of this writing, there are problems compiling in this setup (see msys2_mingw64_sigrok_build_log.md
So, we need to go few steps back: since the digital data .vcd files are both already on 1 GHz sampling rate, all we need to do is, to upsample the analog data .csv files to 1 GHz as well.
Thankfully, it is relatively easy to resample .csv files, that are in the format in this example, using the pandas Python library. This repository contains one script that does that:
python3 code/resample_csv_pandas.py data/500MHz_digital_0.125MHz_analog_5sec_an.csv
However, note that this script is so memory hungry, it will most likely cause your system to hang or freeze (and likely, eventually fail with MemoryError: Unable to allocate 37.3 GiB for an array with shape (5011232001,) and data type int64
).
So, a trivial use of pandas
to resample (upsample) the analog captures is not really viable at time of writing.
We can take another approach - instead of using pandas, parse the .csv line by line, and emit the upsampled lines accordingly; this approach will definitely not use as much memory. There is a script in this repo that does that; however, here are some results:
# in MINGW64 shell
$ time python3 code/resample_csv_plain.py data/500MHz_digital_0.125MHz_analog_5sec_an.csv
Loading input CSV file: C:/src/sigrok_pulseview_combine_merge_git/data/500MHz_digital_0.125MHz_analog_5sec_an.csv, saving output file C:/src/sigrok_pulseview_combine_merge_git/data/500MHz_digital_0.125MHz_analog_5sec_an_0.000000001.csv
1687912: ['0.001687910', ' -0.247291326522827']
^C
real 836m57.445s
user 566m55.781s
sys 223m27.013s
$ ls -la data/500MHz_digital_0.125MHz_analog_5sec_an_0.000000001.csv
-rw-r--r-- 1 user None 54M Oct 20 06:12 data/500MHz_digital_0.125MHz_analog_5sec_an_0.000000001.csv
# in MSYS2 shell:
$ time python3 code/resample_csv_plain.py data/6.25MHz_digital_1.5625MHz_analog_3sec_an.csv
Loading input CSV file: /d/src/sigrok_pulseview_combine_merge_git/data/6.25MHz_digital_1.5625MHz_analog_3sec_an.csv, saving output file /d/src/sigrok_pulseview_combine_merge_git/data/6.25MHz_digital_1.5625MHz_analog_3sec_an_0.000000001.csv
301662098: ['0.301662096', ' 0.853835046291351']
^C
Traceback (most recent call last):
File "code/resample_csv_plain.py", line 77, in <module>
main(sys.argv[1:])
File "code/resample_csv_plain.py", line 62, in main
sys.stderr.write("\r{}: {}".format(line_count, t_row)); sys.stderr.flush()
KeyboardInterrupt
real 806m5.954s
user 76m33.249s
sys 30m14.046s
$ ls -la data/6.25MHz_digital_1.5625MHz_analog_3sec_an_0.000000001.csv
-rw-r--r-- 1 user None 9.2G Oct 20 06:13 data/6.25MHz_digital_1.5625MHz_analog_3sec_an_0.000000001.csv
So:
- In MINGW64's Python (Python 3.8.6), it took 836 min (13 hours 56 min) to emit enough lines at 1 GHz sampling rate, to cover the capture up to 1.6 ms, which represents 0.032% of the 5-second capture
- In MSYS2's Python (Python 3.8.5), it took 806 min (13 hours 26 min) to emit enough lines at 1 GHz sampling rate, to cover the capture up to 301.6 ms, which represents 10.053% of the 3-second capture
So, while this approach does conserve memory (and avoids related freezes/crashes) - it is, again, not really viable, due to the massive ammount of processing time.
However, there is one more thing we can do - we can load up the "original" analog .csv
files in PulseView, and we can export them as .wav
(the only MS WAV export option in PulseView is "Scale", and we'll keep that at 1.00). Here is what we obtain as a result of that process - again, as intermediary files, these .wav
files are not kept in this repo:
$ ls -la data/*_an.wav
-rw-r--r-- 1 user None 2.4M Oct 20 06:31 data/500MHz_digital_0.125MHz_analog_5sec_an.wav
-rw-r--r-- 1 user None 18M Oct 20 06:30 data/6.25MHz_digital_1.5625MHz_analog_3sec_an.wav
$ mediainfo data/*_an.wav
General
Complete name : data/500MHz_digital_0.125MHz_analog_5sec_an.wav
Format : Wave
File size : 2.39 MiB
Duration : 5 s 11 ms
Overall bit rate mode : Constant
Overall bit rate : 4 000 kb/s
IsTruncated : Yes
Audio
Format : PCM
Format profile : Float
Codec ID : 3
Codec ID/Hint : IEEE
Duration : 5 s 11 ms
Bit rate mode : Constant
Bit rate : 4 000 kb/s
Channel(s) : 1 channel
Sampling rate : 125 kHz
Bit depth : 32 bits
Stream size : 2.39 MiB (100%)
General
Complete name : data/6.25MHz_digital_1.5625MHz_analog_3sec_an.wav
Format : Wave
File size : 18.0 MiB
Duration : 3 s 19 ms
Overall bit rate mode : Constant
Overall bit rate : 50.0 Mb/s
IsTruncated : Yes
Audio
Format : PCM
Format profile : Float
Codec ID : 3
Codec ID/Hint : IEEE
Duration : 3 s 19 ms
Bit rate mode : Constant
Bit rate : 50.0 Mb/s
Channel(s) : 1 channel
Sampling rate : 1 563 kHz
Bit depth : 32 bits
Stream size : 18.0 MiB (100%)
Now, we can use an audio program like SoX - Sound eXchange to perform the resampling:
$ time sox data/6.25MHz_digital_1.5625MHz_analog_3sec_an.wav -r 1000000000 data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.wav
C:\Program Files (x86)\sox-14-4-2\sox.exe WARN wav: Premature EOF on .wav input file
C:\Program Files (x86)\sox-14-4-2\sox.exe WARN rate: rate clipped 200180190 samples; decrease volume?
C:\Program Files (x86)\sox-14-4-2\sox.exe WARN sox: `data/6.25MHz_digital_1.5625MHz_analog_3sec_an.wav' input clipped 403873 samples
C:\Program Files (x86)\sox-14-4-2\sox.exe WARN sox: `data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.wav' output clipped 207161190 samples; decrease volume?
real 3m8.008s
user 0m0.000s
sys 0m0.031s
$ ls -la data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.wav
-rw-r--r-- 1 user None 12G Oct 20 06:40 data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.wav
$ mediainfo data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.wav
General
Complete name : data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.wav
Format : Wave
File size : 11.2 GiB
Duration : 49 s 931 ms
Overall bit rate mode : Constant
Overall bit rate : 1 935 Mb/s
Audio
Format : PCM
Format profile : Float
Codec ID : 3
Codec ID/Hint : IEEE
Duration : 49 s 931 ms
Bit rate mode : Constant
Bit rate : 1 935 Mb/s
Channel(s) : 1 channel
Sampling rate : 1 000 MHz
Bit depth : 32 bits
Stream size : 11.2 GiB (100%)
So, we got an upsampled file in reasonable time, and with (relatively) reasonable file size even - however, by default, sox
performs sample interpolation when resampling ( see: https://stackoverflow.com/questions/64438773/resample-upsample-wav-without-interpolation-sox ), which will definitely (at least partially) mess up our analog measurement data. Note that the duration is, however, reported wrong (but that is likely a limitation of mediainfo
).
Thus, we might as well write our own "upsampler" for .wav
files - and here, it is: code/upsample_wav_nointerp.c. It uses libsndfile to read and write .wav
files, and it can only handle upsampling without interpolation (simply repeating samples), and only for integer ratios between upsampled and original sampling rates, and only for mono files with float samples. The source file contains instructions for compiling it under MINGW64 in Windows (the compiled .exe
is not included in this repository), but it should be possible to compile it for Linux, too.
With this program, we get the following results:
$ time code/upsample_wav_nointerp.exe data/6.25MHz_digital_1.5625MHz_analog_3sec_an.wav 1000000000
Got input file: C:\src\sigrok_pulseview_combine_merge\data\6.25MHz_digital_1.5625MHz_analog_3sec_an.wav
Got input new rate: 1000000000
Opened input file: C:\src\sigrok_pulseview_combine_merge\data\6.25MHz_digital_1.5625MHz_analog_3sec_an.wav with sampling rate: 1562500
Sample rate conversion factor: 640
*** Assuming FLOAT_LE sample format ***
Saving to output file: C:\src\sigrok_pulseview_combine_merge\data\6.25MHz_digital_1.5625MHz_analog_3sec_an_out.wav
Wrote 3019681920 samples. Program finished.
real 1m59.007s
user 0m0.000s
sys 0m0.015s
$ ls -la data/6.25MHz_digital_1.5625MHz_analog_3sec_an_out.wav
-rw-r--r-- 1 user None 12G Oct 20 09:56 data/6.25MHz_digital_1.5625MHz_analog_3sec_an_out.wav
$ time code/upsample_wav_nointerp.exe data/500MHz_digital_0.125MHz_analog_5sec_an.wav 1000000000
Got input file: D:\src\sigrok_pulseview_combine_merge_git\data\500MHz_digital_0.125MHz_analog_5sec_an.wav
Got input new rate: 1000000000
Opened input file: D:\src\sigrok_pulseview_combine_merge_git\data\500MHz_digital_0.125MHz_analog_5sec_an.wav with sampling rate: 125000
Sample rate conversion factor: 8000
*** Assuming FLOAT_LE sample format ***
Saving to output file: D:\src\sigrok_pulseview_combine_merge_git\data\500MHz_digital_0.125MHz_analog_5sec_an_out.wav
Wrote 5011240000 samples. Program finished.
real 3m39.996s
user 0m0.000s
sys 0m0.016s
$ ls -la data/500MHz_digital_0.125MHz_analog_5sec_an_out.wav
-rw-r--r-- 1 user None 19G Oct 20 10:21 data/500MHz_digital_0.125MHz_analog_5sec_an_out.wav
So, we got the right files - upsampled without interpolation, with acceptable time of conversion, and they can be loaded in PulseView fine. Finally, we'd want to use sigrok to generate .sr
files that will contain the upsampled analog captures:
$ time "C:\Program Files (x86)\sigrok\sigrok-cli\sigrok-cli.exe" \
-l 3\
-i 500MHz_digital_0.125MHz_analog_5sec_an_out.wav \
-o 500MHz_digital_0.125MHz_analog_5sec_an_1GHz.sr
cli: Received SR_DF_HEADER.
cli: Received SR_DF_META.
cli: Got samplerate 1000000000 Hz.
cli: Received SR_DF_ANALOG (72000 samples).
cli: Received SR_DF_END.
real 0m0.227s
user 0m0.000s
sys 0m0.015s
Unfortunately, the sigrok-cli
for some reason fails to import the entire .wav
file (only the first 72000 samples are imported) - and so, we must import the .wav
, and export the .sr
from PulseView. These files are included in the repository, and their files sizes are:
$ ls -la data/*_1GHz.sr
-rw-r--r-- 1 user None 22M Oct 22 12:10 data/500MHz_digital_0.125MHz_analog_5sec_an_1GHz.sr
-rw-r--r-- 1 user None 18M Oct 22 11:10 data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.sr
First of all, let us confirm that all of the .sr
files we want to merge into a single session, are at the same sampling rate. As we can see from https://sigrok.org/wiki/File_format:Sigrok/v2 an .sr
file is essentially a zip file, and the information about sample rate is in the metadata
file stored in the .sr
zip file. The parsing of these files is otherwise implemented in libsigrok: session_file.c.
So we can do the following in the bash
command line of MSYS2 on Windows:
$ ls data/*{_dg,_an_1GHz}.sr | while read f; do echo -e "\n$f"; unzip -p $f metadata ; done
data/500MHz_digital_0.125MHz_analog_5sec_an_1GHz.sr
[global]
sigrok version=0.6.0-git-d7df9dc
[device 1]
samplerate=1 GHz
total analog=1
analog1=CH1
data/500MHz_digital_0.125MHz_analog_5sec_dg.sr
[global]
sigrok version=0.6.0-git-d7df9dc
[device 1]
capturefile=logic-1
total probes=2
samplerate=1 GHz
total analog=0
probe1=top.Channel_1
probe2=top.Channel_2
unitsize=1
data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.sr
[global]
sigrok version=0.6.0-git-d7df9dc
[device 1]
samplerate=1 GHz
total analog=1
analog1=CH1
data/6.25MHz_digital_1.5625MHz_analog_3sec_dg.sr
[global]
sigrok version=0.6.0-git-d7df9dc
[device 1]
capturefile=logic-1
total probes=2
samplerate=1 GHz
total analog=0
probe1=top.Channel_1
probe2=top.Channel_2
unitsize=1
So, we've confirmed that all the relevant files are at the same sampling rate - in this case, 1 GHz.
Now, let's take a brief look of the naming of files inside each capture:
$ ls data/*{_dg,_an_1GHz}.sr | while read f; do echo; unzip -l $f | (head -n7; echo "..."; tail -n7) ; done
Archive: data/500MHz_digital_0.125MHz_analog_5sec_an_1GHz.sr
Length Date Time Name
--------- ---------- ----- ----
1 2020-10-22 11:55 version
98 2020-10-22 11:55 metadata
4194304 2020-10-22 11:55 analog-1-1-1
4194304 2020-10-22 11:55 analog-1-1-2
...
4194304 2020-10-22 12:10 analog-1-1-4776
4194304 2020-10-22 12:10 analog-1-1-4777
4194304 2020-10-22 12:10 analog-1-1-4778
4194304 2020-10-22 12:10 analog-1-1-4779
381184 2020-10-22 12:10 analog-1-1-4780
--------- -------
20044960099 4782 files
Archive: data/500MHz_digital_0.125MHz_analog_5sec_dg.sr
Length Date Time Name
--------- ---------- ----- ----
1 2020-10-19 08:04 version
174 2020-10-19 08:04 metadata
4194304 2020-10-19 08:04 logic-1-1
4194304 2020-10-19 08:04 logic-1-2
...
4194304 2020-10-19 08:06 logic-1-1191
4194304 2020-10-19 08:06 logic-1-1192
4194304 2020-10-19 08:06 logic-1-1193
4194304 2020-10-19 08:06 logic-1-1194
345501 2020-10-19 08:06 logic-1-1195
--------- -------
5008344652 1197 files
Archive: data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.sr
Length Date Time Name
--------- ---------- ----- ----
1 2020-10-22 11:05 version
98 2020-10-22 11:05 metadata
4194304 2020-10-22 11:05 analog-1-1-1
4194304 2020-10-22 11:05 analog-1-1-2
...
4194304 2020-10-22 11:10 analog-1-1-2876
4194304 2020-10-22 11:10 analog-1-1-2877
4194304 2020-10-22 11:10 analog-1-1-2878
4194304 2020-10-22 11:10 analog-1-1-2879
3326464 2020-10-22 11:10 analog-1-1-2880
--------- -------
12078727779 2882 files
Archive: data/6.25MHz_digital_1.5625MHz_analog_3sec_dg.sr
Length Date Time Name
--------- ---------- ----- ----
1 2020-10-19 08:07 version
174 2020-10-19 08:07 metadata
4194304 2020-10-19 08:07 logic-1-1
4194304 2020-10-19 08:07 logic-1-2
...
4194304 2020-10-19 08:07 logic-1-654
4194304 2020-10-19 08:07 logic-1-655
4194304 2020-10-19 08:07 logic-1-656
4194304 2020-10-19 08:07 logic-1-657
3944833 2020-10-19 08:07 logic-1-658
--------- -------
2759602736 660 files
So, essentially, we need to unpack each of these .sr files, rename their contents accordingly, and then write in a new metadata
file that will refer to them.
However, there is a problem here: the specification at https://sigrok.org/wiki/File_format:Sigrok/v2 does note that for analog channel data:
There can be one or more binary files in the ZIP file, which contain analog samples. These have a name of
- analog-1--1, analog-1--2, analog-1--3, and so on (where is the analog channel number).
... the channel number is included as part of the file; so we can in principle rearrange analog tracks, by changing the channel index part of the corresponding data analog-1-*
filenames.
However, this is not the case for digital signals:
There can be one or more binary files in the ZIP file, which contain logic analyzer samples. These have a name of
- logic-1 (one large file; deprecated, but still supported), or
- logic-1-1, logic-1-2, logic-1-3, and so on (many smaller files).
Clearly, there is no number for "channel index" in the logic-1-*
filenames; this implies that all of the digital tracks in a session are packed and saved into a single file (saved in many smaller snippets/chunks).
And that means, in order to append digital/logic tracks to a session, we need to somehow add their data to the logic-1-*
file snippets of that session! In fact, comments in srzip.c confirm that:
* Allocate one samples buffer for all logic channels, and * several samples buffers for the analog channels. Allocate * buffers of CHUNK_SIZE size (in bytes), and determine the * sample counts from the respective channel counts and data * type widths. * * These buffers are intended to reduce the number of ZIP * archive update calls, and decouple the srzip output module * from implementation details in other acquisition device * drivers and input modules.
Looking at the code in there, it is hardly trivial to add data to the existing logic-1-*
chunks in a sigrok session file. Possibly, it might have been possible to handle this through the libsigrok Python API - but as we saw earlier, at this time we cannot build it for the platform this document is written on (MSYS2 on Windows).
Thus, we might take another approach: we will export all our .sr
data to .vcd
(which, in principle can handle both analog and digital data); and then we will use a Python .vcd
API to generate a merged file, and finally we will convert the merged .vcd
to a merged .sr
file.
But note that, for Python vcd libraries:
- PyVCD can only write a data structure in memory as .vcd file, but cannot read a .vcd file as data structure in memory
- vcdvcd (fork of Verilog_VCD) can only read a .vcd file as data structure in memory, but cannot write a data structure in memory as .vcd file
- vcd_parsealyze (fork of vcd_parser) - can only read a .vcd file as data structure in memory, but cannot write a data structure in memory as .vcd file
- pyDigitalWaveTools - can apparently both read and write .vcd files, but it is unclear how to pass data from an input to an output file
So, merging via .vcd is a bit of a dead end, for now ....
So, let us go back to the sigrok logic format - it turns out, that whenever we have less that 8 logic channels, they get encoded in a single byte value; thus, for a 2-channel digital signal, we expect to see only byte values 0x00, 0x01, 0x02 and 0x03 in a logic-1-*
file! That can also be quickly checked in PulseView - if you do a "Import Raw binary logic data" of a logic-1-*
file that contains only, say, two channels, you will get the same first two waveforms, regardless of if you enter 2, or 4, or another (larger) number of channels - the remaining channels will simply be zero.
So, if we want to merge two logic/digital captures, where we know that the sum of the number of logic channels from both captures are <= 8, we'd simply read a byte from each capture, then bitshift one byte by the number of channels in the other byte, and add them, in order to get a merged byte; for instance:
byte_A (3-channel capture): 0b00000111
byte_B (2-channel capture): 0b00000011
byte_AB = (byte_B << num_channel_A) + byte_A
= (0b00000011 << 3) + 0b00000111
= 0b00011000 + 0b00000111
= 0b00011111
And finally, this is implemented in the Python script code/merge_sr_sessions.py; and this is how it performs on MSYS2 under Windows:
$ time python3 code/merge_sr_sessions.py data/*_dg.sr data/*_an_1GHz.sr
Assumed output samplerate: 1 GHz
Got .sr input files:
01/04: /d/src/sigrok_pulseview_combine_merge_git/data/500MHz_digital_0.125MHz_analog_5sec_dg.sr
capturefile: logic-1
total probes: 2
samplerate: 1 GHz
total analog: 0
probe1: top.Channel_1
probe2: top.Channel_2
unitsize: 1
02/04: /d/src/sigrok_pulseview_combine_merge_git/data/6.25MHz_digital_1.5625MHz_analog_3sec_dg.sr
capturefile: logic-1
total probes: 2
samplerate: 1 GHz
total analog: 0
probe1: top.Channel_1
probe2: top.Channel_2
unitsize: 1
03/04: /d/src/sigrok_pulseview_combine_merge_git/data/500MHz_digital_0.125MHz_analog_5sec_an_1GHz.sr
samplerate: 1 GHz
total analog: 1
analog1: CH1
04/04: /d/src/sigrok_pulseview_combine_merge_git/data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.sr
samplerate: 1 GHz
total analog: 1
analog1: CH1
Total digital tracks to export: 4
Total analog tracks to export: 2
Output metadata:
[global]
sigrok version = 0.6.0-git-d7df9dc
[device 1]
capturefile = logic-1
total probes = 4
samplerate = 1 GHz
total analog = 2
probe1 = top.Channel_1_0
probe2 = top.Channel_2_0
unitsize = 1
probe3 = top.Channel_1_1
probe4 = top.Channel_2_1
analog5 = CH1_2
analog6 = CH1_3
Extracting data in folder: /tmp/mergedout; output file will be: /tmp/mergedout.sr
Removed existing directory /tmp/mergedout
Created directory /tmp/mergedout
Extracting 01/04: /d/src/sigrok_pulseview_combine_merge_git/data/500MHz_digital_0.125MHz_analog_5sec_dg.sr
... unpacked 1197 files
Extracting 02/04: /d/src/sigrok_pulseview_combine_merge_git/data/6.25MHz_digital_1.5625MHz_analog_3sec_dg.sr
... unpacked 660 files
Extracting 03/04: /d/src/sigrok_pulseview_combine_merge_git/data/500MHz_digital_0.125MHz_analog_5sec_an_1GHz.sr
... unpacked 4782 files
Extracting 04/04: /d/src/sigrok_pulseview_combine_merge_git/data/6.25MHz_digital_1.5625MHz_analog_3sec_an_1GHz.sr
... unpacked 2882 files
Merging digital data...
Packing into sr zip (unfortunately, it also adds .zip extension at end)
real 23m58.196s
user 12m21.687s
sys 2m4.999s
$ mv /tmp/mergedout.sr.zip /tmp/mergedout.sr
$ ls -la /tmp/mergedout.sr
-rw-r--r-- 1 user None 45M Oct 26 18:42 /tmp/mergedout.sr
So, we got a merged file, 45 MB in (compressed) size, which took about 23 minutes to convert.
Considering the amount of data, it is also slow to load in PulseView fully (it takes like half an hour or so); but after it is loaded, and we ungroup the digital channels, and we rearrange them a bit - we're greeted with this sight:
... and thus, the goal of this exercise - to merge several sigrok/PulseView .sr sessions in one - has been achieved.