6ac126e05c
User uploaded 3 high-amplitude events (PPV 6-7 in/s — shook the geophone
hard) to decode-re/5-11-26/. These cracked the Tran codec:
- Preamble bytes [3:5] and [5:7] = Tran[0] and Tran[1] as int16 BE
in 16-count units (LSB = 0.005 in/s). Confirmed across all 7
fixtures.
- First data block carries Tran deltas from sample 2 onward:
* 10 NN block: NN/2 bytes of payload, each byte = two 4-bit signed
nibble deltas (high nibble first)
* 20 NN block: NN int8 signed deltas
Verified 22+42+46 = 110 Tran samples across SP0/SS0/SV0 with 0 errors
against BW's ASCII export.
Why the earlier 96-combination brute force failed: the quiet 5-8
events all had T[0] = T[1] ≈ 0 so the preamble's per-channel encoding
was undetectable. Loud events made the encoding obvious.
What's solved:
- minimateplus.waveform_codec.decode_tran_initial: returns first
N Tran samples in 16-count units for any body.
- Walker length formula for in-data 30 NN blocks (NN*2 instead of NN*4).
- Walker now handles bodies that start with 20 NN (in addition to 10 NN).
What's still open:
- Tran past the first data block (multi-block channel switching).
- Vert / Long / MicL channel encodings.
- Walker correctness past offset ~427 in event-b.
Tests: 36 pass. decode_waveform_v2 still returns None — the full
multi-channel decoder is not wired up. decode_tran_initial is the
new verified entry point.
Files: minimateplus/waveform_codec.py, tests/test_waveform_codec.py
(adds 5-11-26 fixtures + decode_tran_initial tests), and
docs/instantel_protocol_reference.md §7.6.1 (Tran codec spec).
51 lines
2.0 KiB
Python
51 lines
2.0 KiB
Python
"""Quick inspection of the new high-amplitude events."""
|
|
import os, re, sys
|
|
sys.path.insert(0, ".")
|
|
from analysis.load_bundle import _parse_txt
|
|
from minimateplus.waveform_codec import walk_body, find_data_start
|
|
|
|
ROOT = "decode-re/5-11-26"
|
|
|
|
|
|
def main():
|
|
for stem in ("M529LL1A.SP0", "M529LL1A.SS0", "M529LL1A.SV0"):
|
|
bin_path = os.path.join(ROOT, stem)
|
|
txt_path = bin_path + ".TXT"
|
|
with open(bin_path, "rb") as f:
|
|
raw = f.read()
|
|
body = raw[43:-26]
|
|
meta, samples = _parse_txt(txt_path)
|
|
n = len(samples["Tran"])
|
|
|
|
print(f"\n=== {stem} ===")
|
|
print(f" file={len(raw)}, body={len(body)}, N_samples={n}")
|
|
print(f" rectime={meta.get('Record Time')} pretrig={meta.get('Pre-trigger Length')}")
|
|
print(f" PPV(T,V,L)={meta.get('Tran PPV')} / {meta.get('Vert PPV')} / {meta.get('Long PPV')}")
|
|
# Show first few non-trivial samples
|
|
print(f" First 5 truth samples (in/s):")
|
|
for i in range(5):
|
|
print(f" T={samples['Tran'][i]:8.3f} V={samples['Vert'][i]:8.3f} "
|
|
f"L={samples['Long'][i]:8.3f} M={samples['MicL'][i]:8.3f}")
|
|
# Peak sample positions
|
|
for ch in ("Tran", "Vert", "Long"):
|
|
vals = samples[ch]
|
|
peak_i = max(range(n), key=lambda i: abs(vals[i]))
|
|
print(f" {ch}: peak {vals[peak_i]:.3f} at sample {peak_i} (t={peak_i/1024:.3f}s)")
|
|
# Body structure
|
|
start = find_data_start(body)
|
|
blocks = walk_body(body, start)
|
|
types = {}
|
|
for b in blocks:
|
|
types[b.tag_hi] = types.get(b.tag_hi, 0) + 1
|
|
print(f" body start={start}, total blocks walked: {len(blocks)}")
|
|
print(f" block tag counts: {types}")
|
|
# How far the walker got
|
|
if blocks:
|
|
last = blocks[-1]
|
|
walked = last.offset + last.length
|
|
print(f" walker stopped at offset {walked}/{len(body)} ({100*walked/len(body):.0f}%)")
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|