This is a full-featured Julia interface to FINUFFT, which is a lightweight and fast parallel nonuniform fast Fourier transform (NUFFT) library released by the Flatiron Institute. This interface stands at v3.x, and it uses FINUFFT version 2.1.0 (note that the interface version number is distinct from the version of the wrapped binary FINUFFT library).
FINUFFT.jl requires Julia v1.3 or later, and has been tested up to v1.7.1. From the Pkg REPL mode (hit ]
in REPL to enter), run
add FINUFFT
This installs the stable registered version and its dependencies, including our multi-platform precompiled
binaries finufft_jll.jl, which are now microarchitecture-specific (including avx2
) for better performance.
You may be able to squeeze a little more performance via locally-compiled binaries; see below.
This module now provides the functions nufft1d1
, nufft1d2
, ..., nufft3d3
, nufft1d1!
, nufft1d2!
, ..., nufft3d3!
, that now wrap the
simple and vectorized interfaces in a unified way,
as well as
finufft_makeplan
, finufft_setpts!
, finufft_exec
, finufft_exec!
and finufft_destroy!
that wrap the guru interface.
This brings the Julia interface up to the standards of the
FINUFFT's MATLAB/Octave
and Python interfaces.
The underlying C++ routines that are called have full documentation
here.
An auto-generated reference for all provided Julia functions is here.
Warning: On 10/28/21 (v2.1.0) and 1/5/22 (v3.0.0), the interface has changed (improved) significantly, breaking some backward compatibility, as follows. Please also read the documentation.
- Function calls mimic the C/C++ interface, with the exception that you don't need to pass the dimensions of any arrays in the argument (they are inferred using
size()
). - A vectorized call (performing multiple transforms, each with different coefficient vectors but the same set of nonuniform points) can now be performed using the same functions as the single-transform interface, detected from the size of the input arrays.
- Both 64-bit and 32-bit precision calls are now supported using a single
set of function names. Which precision to use is inferred from the type of the input arrays, except for in the guru interface where the
dtype
argument is required forfinufft_makeplan
. (NOTE: The use of thedtype
argument in the simple interface is deprecated as of v3.1.0) - The functions named
nufftDdN
return the output array. - In contrast, the functions named
nufftDdN!
take the output array as an argument. This needs to be preallocated with the correct size. - Likewise, in the guru interface,
finufft_exec
returns the output array, whilefinufft_exec!
takes the output array as an argument that needs to be preallocated. The methodsfinufft_setpts!
andfinufft_destroy!
now include explamation points in Julian style, since they both change the plan. - Options differing from their default values are now set using keyword arguments both in the simple interfaces, or in
finufft_makeplan
for the guru interface.
using FINUFFT
# Here we demo a Float64 1D type 1 transform
nj = 1000000
x = pi*(1.0 .- 2.0*rand(nj)) # nonuniform points
c = rand(nj) + 1im*rand(nj) # their strengths
ms = 2000000 # output size (number of Fourier modes)
tol = 1e-9 # requested relative tolerance
# Output as return value (1e6 pts to 2e6 modes takes about 0.1 sec)...
fk = nufft1d1(x, c, 1, tol, ms)
# Or, output into preallocated array, whose size determines ms...
out = Array{ComplexF64}(undef, ms)
nufft1d1!(x, c, 1, tol, out)
# Demo using keyword args to change options from defaults...
fk_fftord = nufft1d1(x, c, 1, tol, ms, debug=1, modeord=1, nthreads=4)
The above code may be found in examples/demo1d1.jl
For a 2D type 1 with timing benchmark, see examples/time2d1.jl
Finally, the more involved code test/test_nufft.jl
tests dtype=Float64
and dtype=Float32
precisions
for all nine transform types.
The outputs are tested there for mathematical correctness.
In the 1D type 1 it also tests a vectorized simple, a guru call and
a vectorized guru call.
The help documentation for each function will also gradually be populated
with examples.
To get the latest version of this interface do add FINUFFT#master
, but note this still uses the precompiled binaries from finufft_jll
.
You may get a little more performance by locally compiling binaries as follows. This has only been tested on ubuntu linux, so YMMV. First install the source
FINUFFT,
cd
to its top directory (which we'll call YOURFINUFFT
),
make test
and check that gives no errors. You may need to create a
make.inc
for your system, as per its documentation.
Now start Julia and install the latest interface in develop mode:
pkg> dev https://github.com/ludvigak/FINUFFT.jl
This should create ~/.julia/dev/FINUFFT/src/FINUFFT.jl
which you should edit,
following the simple instructions to set
const libfinufft = "YOURFINUFFT/lib/libfinufft.so"
Restart Julia, and pkg> test FINUFFT
to check it worked.
You may find that julia> include("examples/time2d1.jl")
runs faster
than before (however, since we included avx2
in our binaries, it is
unlikely to run faster on an x86_64 CPU).
Now proceed by using FINUFFT
as usual.
You may do pkg> free FINUFFT
and restart to return to the registered package
with generic binaries.
Here's general info about packages.
Finally, older versions of the package are available also for Julia v1.0-v1.2, but the user needs to have a recent version of GCC installed.
Main authors:
- Ludvig af Klinteberg (old interface)
- Libin Lu (new full-featured interface)
- Jonas Krimmer (many contributions to full-featured interface)
Additional authors:
- Alex Barnett (guidance/tweaks/docs/examples)
- Mose Giordano (packaging, binaries)
- populate the docstrings each with a working example
- add more
examples/
with math tests - more extensive tests, including more "dumb inputs" as in C++