Skip to content

Commit 66695a7

Browse files
authored
Merge pull request #26 from crstnbr/fftfreq
Move Frequencies, fftfreq, rfftfreq from DSP.jl to AbstractFFTs.jl
2 parents 162c162 + 5f11309 commit 66695a7

File tree

4 files changed

+71
-2
lines changed

4 files changed

+71
-2
lines changed

docs/src/api.md

+2
Original file line numberDiff line numberDiff line change
@@ -21,4 +21,6 @@ AbstractFFTs.plan_brfft
2121
AbstractFFTs.plan_irfft
2222
AbstractFFTs.fftshift
2323
AbstractFFTs.ifftshift
24+
AbstractFFTs.fftfreq
25+
AbstractFFTs.rfftfreq
2426
```

src/AbstractFFTs.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ module AbstractFFTs
33
export fft, ifft, bfft, fft!, ifft!, bfft!,
44
plan_fft, plan_ifft, plan_bfft, plan_fft!, plan_ifft!, plan_bfft!,
55
rfft, irfft, brfft, plan_rfft, plan_irfft, plan_brfft,
6-
fftshift, ifftshift
6+
fftshift, ifftshift, Frequencies, fftfreq, rfftfreq
77

88
include("definitions.jl")
99

src/definitions.jl

+49-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
using LinearAlgebra
44
using LinearAlgebra: BlasReal
55
import Base: show, summary, size, ndims, length, eltype,
6-
*, inv, \
6+
*, inv, \, size, step, getindex, iterate
77

88
# DFT plan where the inputs are an array of eltype T
99
abstract type Plan{T} end
@@ -396,6 +396,54 @@ function ifftshift(x,dim)
396396
circshift(x, s)
397397
end
398398

399+
##############################################################################
400+
401+
402+
struct Frequencies{T<:Number} <: AbstractVector{T}
403+
n_nonnegative::Int
404+
n::Int
405+
multiplier::T
406+
407+
Frequencies(n_nonnegative::Int, n::Int, multiplier::T) where {T<:Number} = begin
408+
1 n_nonnegative n || throw(ArgumentError("Condition 1 ≤ n_nonnegative ≤ n isn't satisfied."))
409+
return new{T}(n_nonnegative, n, multiplier)
410+
end
411+
end
412+
413+
unsafe_getindex(x::Frequencies, i::Int) =
414+
(i-1+ifelse(i <= x.n_nonnegative, 0, -x.n))*x.multiplier
415+
@inline function Base.getindex(x::Frequencies, i::Int)
416+
@boundscheck Base.checkbounds(x, i)
417+
unsafe_getindex(x, i)
418+
end
419+
420+
function Base.iterate(x::Frequencies, i::Int=1)
421+
i > x.n ? nothing : (unsafe_getindex(x,i), i + 1)
422+
end
423+
Base.size(x::Frequencies) = (x.n,)
424+
Base.step(x::Frequencies) = x.multiplier
425+
426+
"""
427+
fftfreq(n, fs=1)
428+
Return the discrete Fourier transform (DFT) sample frequencies for a DFT of length `n`. The returned
429+
`Frequencies` object is an `AbstractVector` containing the frequency
430+
bin centers at every sample point. `fs` is the sample rate of the
431+
input signal.
432+
"""
433+
fftfreq(n::Int, fs::Number=1) = Frequencies((n+1) >> 1, n, fs/n)
434+
435+
"""
436+
rfftfreq(n, fs=1)
437+
Return the discrete Fourier transform (DFT) sample frequencies for a real DFT of length `n`.
438+
The returned `Frequencies` object is an `AbstractVector`
439+
containing the frequency bin centers at every sample point. `fs`
440+
is the sample rate of the input signal.
441+
"""
442+
rfftfreq(n::Int, fs::Number=1) = Frequencies((n >> 1)+1, (n >> 1)+1, fs/n)
443+
444+
fftshift(x::Frequencies) = (x.n_nonnegative-x.n:x.n_nonnegative-1)*x.multiplier
445+
446+
399447
##############################################################################
400448

401449
"""

test/runtests.jl

+19
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,25 @@ end
8181
@test AbstractFFTs.ifftshift([1 2 3; 4 5 6], 1:2) == [5 6 4; 2 3 1]
8282
end
8383

84+
@testset "FFT Frequencies" begin
85+
# N even
86+
@test fftfreq(8) == [0.0, 0.125, 0.25, 0.375, -0.5, -0.375, -0.25, -0.125]
87+
@test rfftfreq(8) == [0.0, 0.125, 0.25, 0.375, 0.5]
88+
@test fftshift(fftfreq(8)) == -0.5:0.125:0.375
89+
90+
# N odd
91+
@test fftfreq(5) == [0.0, 0.2, 0.4, -0.4, -0.2]
92+
@test rfftfreq(5) == [0.0, 0.2, 0.4]
93+
@test fftshift(fftfreq(5)) == -0.4:0.2:0.4
94+
95+
# Sampling Frequency
96+
@test fftfreq(5, 2) == [0.0, 0.4, 0.8, -0.8, -0.4]
97+
# <:Number type compatibility
98+
@test eltype(fftfreq(5, ComplexF64(2))) == ComplexF64
99+
100+
@test_throws ArgumentError Frequencies(12, 10, 1)
101+
end
102+
84103
@testset "normalization" begin
85104
# normalization should be inferable even if region is only inferred as ::Any,
86105
# need to wrap in another function to test this (note that p.region::Any for

0 commit comments

Comments
 (0)