Skip to content

Commit

Permalink
feat(dsp): add decompose function to perform OMP
Browse files Browse the repository at this point in the history
  • Loading branch information
mchitre committed Nov 23, 2024
1 parent 76ad5b8 commit 78c6e57
Show file tree
Hide file tree
Showing 3 changed files with 78 additions and 2 deletions.
2 changes: 1 addition & 1 deletion Project.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
name = "SignalAnalysis"
uuid = "df1fea92-c066-49dd-8b36-eace3378ea47"
authors = ["Mandar Chitre <[email protected]>"]
version = "0.9.2"
version = "0.10.0"

[deps]
DSP = "717857b8-e6f2-59f4-9121-6e50c889abd2"
Expand Down
59 changes: 58 additions & 1 deletion src/dsp.jl
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ export fir, removedc, removedc!, demon
export upconvert, downconvert, rrcosfir, rcosfir
export mseq, gmseq, circconv, circcorr, goertzel, pll, hadamard
export mfilter, findsignal, dzt, idzt
export istft, whiten, filt, filtfilt, resample, delay, delay!, compose
export istft, whiten, filt, filtfilt, resample, delay, delay!, compose, decompose

"""
$(SIGNATURES)
Expand Down Expand Up @@ -813,6 +813,63 @@ function compose(r, t, a; duration=duration(r)+maximum(t), fs=framerate(r))
signal(isanalytic(r) ? x : 2 * real(x), fs)
end

"""
$(SIGNATURES)
Decompose a signal as a sum of reference signal with some arrival times and amplitudes.
If the number of signals `n` is non-zero, the function will limit the decomposition
to the strongest `n` signals.
The resolution for arrival time is one sample. The `threshold` parameter controls
the stopping criterion for the decomposition. If the relative change in the residual
energy is less than `threshold`, the decomposition stops. The `refine` parameter
controls whether the amplitudes are refined using an optimization algorithm. Without
refinement, the amplitudes are simply the matched filter outputs at the arrival times.
# Examples:
```julia-repl
julia> x = compose(mseq(12), [0.1, 0.2], [1.0, 0.7]; duration=1.0, fs=8000)
julia> x += 0.1 * randn(size(x))
julia> decompose(mseq(12), x)
(Float32[0.1, 0.2], [1.0020050686753323, 0.7006076693773022])
```
"""
function decompose(r::AbstractVector, x::AbstractVector, n=0; threshold=0.1, refine=true)
# use OMP algorithm for sparse signal decomposition
Λ = Int[]
a = Array{eltype(x)}(undef, 0)
xᵣ = x
while n == 0 || length(Λ) < n
mfo = mfilter(r, xᵣ) / energy(r)
absmfo = abs.(mfo)
absmfo[Λ] .= 0
λ = argmax(absmfo)
absmfo[λ] == 0 && break
push!(Λ, λ)
push!(a, mfo[λ])
if refine
sol = optimize(a, BFGS()) do a
= compose(r, time(Λ, x), a; duration=duration(x), fs=framerate(x))
sum(abs2, x - x̂)
end
next_a = minimizer(sol)
else
next_a = a
end
= compose(r, time(Λ, x), next_a; duration=duration(x), fs=framerate(x))
prev_rms = rms(xᵣ[λ:min+length(r)-1, end)])
xᵣ = x -
if (prev_rms - rms(xᵣ[λ:min+length(r)-1, end)])) / prev_rms < threshold
# remove last entry, since it does not meet threshold
resize!(Λ, length(Λ)-1)
resize!(a, length(a)-1)
break
end
a = next_a
end
ndx = sortperm(Λ)
(time(Λ[ndx], x), a[ndx])
end

"""
dzt(x, L, K)
dzt(x, L)
Expand Down
19 changes: 19 additions & 0 deletions test/tests-core.jl
Original file line number Diff line number Diff line change
Expand Up @@ -730,6 +730,25 @@ function test_dsp()
@test vec(Zx)' * vec(Zy) x' * y
@test dzt(4.2x + 2.7y, 64) 4.2Zx + 2.7Zy

x = compose(mseq(12), [0.1, 0.2], [1.0, 0.7]; duration=1.0, fs=8000)
x += 0.1 * randn(size(x))
t, a = decompose(mseq(12), x)
@test t [0.1, 0.2]
@test eltype(a) == Float64
@test a [1.0, 0.7] atol=0.01
t, a = decompose(mseq(12), x; refine=false)
@test t [0.1, 0.2]
@test eltype(a) == Float64
@test a [1.0, 0.7] atol=0.01
t, a = decompose(mseq(12), 2 * analytic(x))
@test t [0.1, 0.2]
@test eltype(a) == ComplexF64
@test a [1.0, 0.7] atol=0.01
t, a = decompose(ComplexF64.(mseq(12)), 2 * analytic(x), 2)
@test t [0.1, 0.2]
@test eltype(a) == ComplexF64
@test a [1.0, 0.7] atol=0.1

end

function test_rand()
Expand Down

0 comments on commit 78c6e57

Please sign in to comment.