ADI Reduction of $\beta$ Pictoris

This example will walk through a full reduction and some analysis of $\beta$ Pictoris in order to show the basic usage of ADI.jl for high-contrast imaging. $\beta$ Pictoris is a known exoplanet host with many publications regarding the direct images of its substellar companion and circumstellar disk.

What will not be covered in this example are the basics of Julia, the fine details of ADI post-processing, or any reference documentation.


Setup

Let's begin by importing the necessary libraries. You may need to add these packages if they are not already on your system.

(@v1.5) pkg> add DataFrames HCIDatasets Plots PSFModels

In addition, a Project.toml file exists in the examples/ folder with the necessary dependencies. Start the REPL in the base ADI.jl folder, then from Pkg mode

(@v1.5) pkg> activate examples
(examples) pkg> instantiate
julia> include("examples/betapictoris.jl")

using ADI
using DataFrames
using HCIDatasets: BetaPictoris
using Plots

# set up plotting
function imshow(img; kwargs...)
    xs, ys = axes(img)
    heatmap(xs, ys, transpose(img); aspect_ratio=1,
            xlim=extrema(xs), ylim=extrema(ys), kwargs...)
end;

Data Reduction

Here we load the data for $\beta$ Pictoris from NaCo at the VLT. You may be prompted to download the data; see HCIDatasets.jl for more details.

cube, angles = BetaPictoris[:cube, :pa];

To reduce the data, we need an algorithm. In ADI.jl we currently have median subtraction, PCA, NMF, and fixed-point GreeDS. These algorithms are treated as "objects" in the sense that we initialize them with options and then pass them around inside the ADI.jl framework to retrieve the results we want.

The usage for fitting the speckle estimate, projecting and subtracting this estimate from the target cube, and derotating and collapsing the residual all are encompassed by calling the algorithm as a function.

alg = PCA(10) # 10 components
reduced = alg(cube, angles)
imshow(reduced)

To try out different algorithms, all you should have to do is change one line and re-run the remaining code. Let's briefly explore a few different algorithms

algs = (alg, NMF(10), Classic(), GreeDS(10))
reduced_frames = [alg(cube, angles) for alg in algs];
figs = (imshow(reduced, ticks=false) for reduced in reduced_frames)
plot(
    figs...,
    title=["PCA(10)" "NMF(10)" "Classic(median)" "GreeDS(10)"],
    layout=(2, 2),
    size=(900, 800),
    dpi=75
)

You may want to mask out an interior angle since there is an inner limit for our signal to be a real planet (as opposed to systematics from the optical system or noise). We can mask out an interior circle either before processing with the algorithm or afterwards using HCIToolbox.mask_circle (note: HCIToolbox is re-exported by ADI.jl, so all its features are usable without importing it directly).

mask_cube = mask_circle(cube, 10)
mask_reduced = alg(mask_cube, angles)
imshow(mask_reduced)

S/N and Significance Maps

Now that we have our reduced frame, let's look at the signal-to-noise ratio (SNR, S/N). We use the exact S/N calculation here, implemented in a fast, multi-threaded framework using detectionmap. In order to measure the S/N, though, we need the effective FWHM of our instrument. Normally, we would measure this from an off-axis (or non-coronagraphic) PSF, but for simplicity I'll hard-code a value.

fwhm = 4.6
snrmap = detectionmap(snr, reduced, fwhm)
imshow(snrmap)

If we want to get the statistical significance, we need to convert from the Student-t confidence interval derived in the S/N to the Gaussian significance. We can accomplish this by calling

sigmap = detectionmap(significance, reduced, fwhm)
imshow(sigmap)

Now, let's do some very basic frequentist planet detection by thresholding this significance

sigmap_cutoff = @. sigmap > 5
imshow(sigmap_cutoff)

looks like we've successfully pulled out the companion $\beta$ Pictoris b from the data!

Contrast Curve

We are also interested in analyzing how the algorithm affects our data, especially calculating the throughput and the contrast curve. These measure, effectively, how much signal is lost during the subtraction step of the algorithm and give us an idea of what the limits of our algorithm are with our data.

Before we move on, we need to create a PSF model for our data. PSFModels.jl contains some simple functional PSFs, or we can use an empirical PSF. We will use the empirical PSF provided by HCIDatasets for our calculations

using PSFModels
psf = BetaPictoris[:psf] ./ maximum(BetaPictoris[:psf])
synthpsf = gaussian(eltype(psf); x=0, y=0, fwhm)
#15 (generic function with 1 method)
plot(
    imshow(psf),
    psfplot(synthpsf, -19:19, -19:19),
    layout=2,
    size=(500, 250),
    cbar=false,
    ticks=false
)

and now we can calculate the 5σ contrast curve using contrast_curve. Contrast is defined by the ratio of astrophysical flux between the host and the companion. Therefore, we need the flux of the star; by default ADI.jl finds this flux by measuring the flux with a circular aperture in the central fwhm of the median-combined cube. Be careful, if you are using a masked cube as input, you will need to calculate this manually, otherwise the stellar flux will appear to be 0!

cc = contrast_curve(alg, cube, angles, psf; fwhm=fwhm, nbranch=6) |> DataFrame
first(filter(row -> isfinite(row.contrast_corr), cc), 5)

5 rows × 5 columns

distancethroughputcontrastcontrast_corrnoise
Float64Float64Float64Float64Float64
19.60.003419920.5061110.95231610.3531
210.60.004717640.3660970.65450310.3307
311.60.006256450.2591590.4435399.69851
412.60.008254050.1746070.2785148.62062
513.60.01085990.126130.1954798.19323

and lets plot it

plot(
    cc.distance,
    [cc.contrast_corr cc.contrast],
    yscale=:log10,
    xlim=(0, NaN),
    label=["Student-t" "Gaussian"],
    ylabel="5-sigma contrast",
    xlabel="radius [px]"
)

The contrast uses a robust estimator for the noise, which means the bright companion doesn't overly bias the contrast measurement. Nonetheless, it is good form to remove the companion signal in a maximum likelihood framework. For convenience here, let's use the :cube_empty entry for BetaPictoris, which already has the companion removed.

cube_empty = BetaPictoris[:cube_empty]
reduced_empty = alg(cube_empty, angles)
imshow(reduced_empty)
cc_empty = contrast_curve(alg, cube_empty, angles, psf; fwhm=fwhm, nbranch=6) |> DataFrame
first(filter(row -> isfinite(row.contrast_corr), cc_empty), 5)

5 rows × 5 columns

distancethroughputcontrastcontrast_corrnoise
Float64Float64Float64Float64Float64
19.60.003593350.4998050.94045210.7421
210.60.004795120.370770.66285710.6339
311.60.006214610.2720860.46566110.1137
412.60.008600460.171280.2732088.81085
513.60.01246650.102170.1583467.61833
plot(
    cc_empty.distance,
    [cc_empty.contrast_corr cc_empty.contrast],
    yscale=:log10,
    xlim=(0, NaN),
    label=["Student-t" "Gaussian"],
    ylabel="5-sigma contrast",
    xlabel="radius [px]"
)

This page was generated using Literate.jl.