mwalib 2.0.3

A library to simplify reading Murchison Widefield Array (MWA) raw visibilities, voltages and metadata.
Documentation
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.

/// Given gpubox files, provide a way to output/dump visibilities.
use anyhow::*;
use clap::Parser;
use mwalib::*;
use std::fs::File;
use std::io::Write;

#[derive(Parser, Debug)]
#[clap(name = "mwalib-data-dump", author)]
struct Opt {
    /// timestep number (0 indexed)
    #[clap(short, long)]
    timestep: usize,

    /// baseline number (0 indexed)
    #[clap(short, long)]
    baseline: usize,

    /// Fine channel to start with
    #[clap(long)]
    fine_chan1: usize,
    /// Fine channel to end with
    #[clap(long)]
    fine_chan2: usize,

    /// Coarse channel
    #[clap(long)]
    coarse_chan: usize,

    /// Path to the metafits file.
    #[clap(short, long)]
    metafits: std::path::PathBuf,

    /// Paths to the gpubox files.
    #[clap(name = "GPUBOX FILE")]
    files: Vec<std::path::PathBuf>,

    // Dump filename
    #[clap(short, long)]
    dump_filename: std::path::PathBuf,
}

fn dump_data<T: AsRef<std::path::Path>>(
    metafits: T,
    files: &[T],
    timestep: usize,
    baseline: usize,
    fine_chan_range: (usize, usize),
    coarse_chan: usize,
    dump_filename: T,
) -> Result<(), anyhow::Error> {
    let mut dump_file = File::create(dump_filename)?;
    println!("Dumping data via mwalib...");
    let context = CorrelatorContext::new(metafits, files)?;
    let coarse_chan_array = &context.coarse_chans;
    let timestep_array = &context.timesteps;

    println!("MWA version: {}", context.mwa_version);

    let floats_per_finechan = context.metafits_context.num_visibility_pols * 2;
    let floats_per_baseline =
        context.metafits_context.num_corr_fine_chans_per_coarse * floats_per_finechan;

    let (ant1, ant2) =
        get_antennas_from_baseline(baseline, context.metafits_context.num_ants).unwrap();
    let ant1_name: String = context.metafits_context.antennas[ant1]
        .tile_name
        .to_string();
    let ant2_name: String = context.metafits_context.antennas[ant2]
        .tile_name
        .to_string();

    let baseline_index = baseline * floats_per_baseline;

    let ch1 = fine_chan_range.0;
    let ch2 = fine_chan_range.1;

    let ch_start_index = baseline_index + (ch1 * floats_per_finechan);
    let ch_end_index = baseline_index + (ch2 * floats_per_finechan) + floats_per_finechan;

    let mut sum: f64 = 0.;
    let mut float_count: u64 = 0;
    println!(
        "Dumping t={} coarse chan: {} ({}) {:.3} Mhz, fine ch: {}-{}, ant {} vs {}",
        timestep,
        coarse_chan,
        coarse_chan_array[coarse_chan].rec_chan_number,
        (coarse_chan_array[coarse_chan].chan_centre_hz as f32 / 1.0e6),
        ch1,
        ch2,
        ant1_name,
        ant2_name
    );
    for (t, _) in timestep_array.iter().enumerate() {
        if t == timestep {
            println!("timestep: {}", t);
            for (c, _) in coarse_chan_array.iter().enumerate() {
                if c == coarse_chan {
                    println!("Reading timestep {}, coarse channel {}...", t, c);
                    let data = context.read_by_baseline(timestep, coarse_chan)?;
                    let mut fine_chan_counter = 0;
                    for v in (0..data.len()).step_by(floats_per_finechan) {
                        if v >= ch_start_index && v < ch_end_index {
                            writeln!(
                                &mut dump_file,
                                "{},{},{},{},{},{},{},{},{}",
                                ch1 + fine_chan_counter,
                                data[v],
                                data[v + 1],
                                data[v + 2],
                                data[v + 3],
                                data[v + 4],
                                data[v + 5],
                                data[v + 6],
                                data[v + 7],
                            )?;

                            sum = sum
                                + (data[v] as f64)
                                + (data[v + 1] as f64)
                                + (data[v + 2] as f64)
                                + (data[v + 3] as f64)
                                + (data[v + 4] as f64)
                                + (data[v + 5] as f64)
                                + (data[v + 6] as f64)
                                + (data[v + 7] as f64);
                            float_count += 8;

                            fine_chan_counter += 1;
                        }
                    }
                }
            }
        }
    }

    println!("Sum was {}, count was {} floats", sum, float_count);

    Ok(())
}

fn main() -> Result<(), anyhow::Error> {
    env_logger::try_init().unwrap_or(());
    let opts = Opt::parse();

    dump_data(
        opts.metafits,
        &opts.files,
        opts.timestep,
        opts.baseline,
        (opts.fine_chan1, opts.fine_chan2),
        opts.coarse_chan,
        opts.dump_filename,
    )?;
    Ok(())
}