Files
wifi-densepose/rust-port/wifi-densepose-rs/crates/wifi-densepose-ruvector/src/signal/bvp.rs
Claude ed3261fbcb feat(ruvector): implement ADR-017 as wifi-densepose-ruvector crate + fix MAT warnings
New crate `wifi-densepose-ruvector` implements all 7 ruvector v2.0.4
integration points from ADR-017 (signal processing + MAT disaster detection):

signal::subcarrier   — mincut_subcarrier_partition (ruvector-mincut)
signal::spectrogram  — gate_spectrogram (ruvector-attn-mincut)
signal::bvp          — attention_weighted_bvp (ruvector-attention)
signal::fresnel      — solve_fresnel_geometry (ruvector-solver)
mat::triangulation   — solve_triangulation TDoA (ruvector-solver)
mat::breathing       — CompressedBreathingBuffer 50-75% mem reduction (ruvector-temporal-tensor)
mat::heartbeat       — CompressedHeartbeatSpectrogram tiered compression (ruvector-temporal-tensor)

16 tests, 0 compilation errors. Workspace grows from 14 → 15 crates.

MAT crate: fix all 54 warnings (0 remaining in wifi-densepose-mat):
- Remove unused imports (Arc, HashMap, RwLock, mpsc, Mutex, ConfidenceScore, etc.)
- Prefix unused variables with _ (timestamp_low, agc, perm)
- Add #![allow(unexpected_cfgs)] for onnx feature gates in ML files
- Move onnx-conditional imports under #[cfg(feature = "onnx")] guards

README: update crate count 14→15, ADR count 24→26, add ruvector crate
table with 7-row integration summary.

Total tests: 939 → 955 (16 new). All passing, 0 regressions.

https://claude.ai/code/session_0164UZu6rG6gA15HmVyLZAmU
2026-03-01 15:50:05 +00:00

96 lines
3.2 KiB
Rust

//! Attention-weighted BVP aggregation (ruvector-attention).
//!
//! [`attention_weighted_bvp`] combines per-subcarrier STFT rows using
//! scaled dot-product attention, weighted by per-subcarrier sensitivity
//! scores, to produce a single robust BVP (body velocity profile) vector.
use ruvector_attention::attention::ScaledDotProductAttention;
use ruvector_attention::traits::Attention;
/// Compute attention-weighted BVP aggregation across subcarriers.
///
/// `stft_rows`: one row per subcarrier, each row is `[n_velocity_bins]`.
/// `sensitivity`: per-subcarrier weight.
/// Returns weighted aggregation of length `n_velocity_bins`.
///
/// # Arguments
///
/// - `stft_rows`: one STFT row per subcarrier; each row has `n_velocity_bins`
/// elements representing the Doppler velocity spectrum.
/// - `sensitivity`: per-subcarrier sensitivity weight (same length as
/// `stft_rows`). Higher values cause the corresponding subcarrier to
/// contribute more to the initial query vector.
/// - `n_velocity_bins`: number of Doppler velocity bins in each STFT row.
///
/// # Returns
///
/// Attention-weighted aggregation vector of length `n_velocity_bins`.
/// Returns all-zeros on empty input or zero velocity bins.
pub fn attention_weighted_bvp(
stft_rows: &[Vec<f32>],
sensitivity: &[f32],
n_velocity_bins: usize,
) -> Vec<f32> {
if stft_rows.is_empty() || n_velocity_bins == 0 {
return vec![0.0; n_velocity_bins];
}
let sens_sum: f32 = sensitivity.iter().sum::<f32>().max(f32::EPSILON);
// Build the weighted-mean query vector across all subcarriers.
let query: Vec<f32> = (0..n_velocity_bins)
.map(|v| {
stft_rows
.iter()
.zip(sensitivity.iter())
.map(|(row, &s)| row[v] * s)
.sum::<f32>()
/ sens_sum
})
.collect();
let attn = ScaledDotProductAttention::new(n_velocity_bins);
let keys: Vec<&[f32]> = stft_rows.iter().map(|r| r.as_slice()).collect();
let values: Vec<&[f32]> = stft_rows.iter().map(|r| r.as_slice()).collect();
attn.compute(&query, &keys, &values)
.unwrap_or_else(|_| vec![0.0; n_velocity_bins])
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn attention_bvp_output_length() {
let n_subcarriers = 3;
let n_velocity_bins = 8;
let stft_rows: Vec<Vec<f32>> = (0..n_subcarriers)
.map(|sc| (0..n_velocity_bins).map(|v| (sc * n_velocity_bins + v) as f32 * 0.1).collect())
.collect();
let sensitivity = vec![0.5_f32, 0.3, 0.8];
let result = attention_weighted_bvp(&stft_rows, &sensitivity, n_velocity_bins);
assert_eq!(
result.len(),
n_velocity_bins,
"output must have length n_velocity_bins = {n_velocity_bins}"
);
}
#[test]
fn attention_bvp_empty_input_returns_zeros() {
let result = attention_weighted_bvp(&[], &[], 8);
assert_eq!(result, vec![0.0_f32; 8]);
}
#[test]
fn attention_bvp_zero_bins_returns_empty() {
let stft_rows = vec![vec![1.0_f32, 2.0]];
let sensitivity = vec![1.0_f32];
let result = attention_weighted_bvp(&stft_rows, &sensitivity, 0);
assert!(result.is_empty());
}
}