Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SPR1-1009: Hetero beams #57

Open
wants to merge 8 commits into
base: master
Choose a base branch
from
Open

SPR1-1009: Hetero beams #57

wants to merge 8 commits into from

Conversation

KimMcAlpine
Copy link
Contributor

Enable the pipeline to use primary beam models when predicting model visibilities. This will allow it to use an intrinsic sky model for calibrator fields, which can then be modified by the appropriate primary beam model to produce different apparent sky models for MeerKAT and SKA dishes.

This cannot be properly tested or merged until we have an appropriate intrinsic sky model for the calibrator fields. The currently used models are all apparent sky models (for MeerKAT antennas)

KimMcAlpine added 4 commits November 10, 2021 11:29
Retrieve the appropriate primary beam models and store as a parameter to
pass to the pipeline for use when predicting visibilities
To address SPR1-1009 this commit uses the primary beam models retireved
from telstate to predict model visibilities when a complex
(multi-component) sky model is provided.

The prediction makes a number of simplifying assumptions to spead up the
prediction. These are:
1.) Calibrator field sources are unpolarised (Q,U,V = 0)
2.) Antennas of the same design have the same primary beam
3.) H and V primary beams are identical.
@bennahugo
Copy link
Contributor

bennahugo commented Nov 10, 2021 via email

@ludwigschwardt ludwigschwardt changed the title Hetero beams SPR1-1009: Hetero beams Nov 23, 2021
Copy link
Contributor

@ludwigschwardt ludwigschwardt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some small corrections but mostly questions for now 🙂

@@ -239,6 +246,27 @@ def _check_blank_freqwin(parameters):
return blank_freqwin


def _get_primary_beam_model(telstate_l0, antenna):
with katsdpmodels.fetch.requests.Fetcher() as fetcher:
Copy link
Contributor

@ludwigschwardt ludwigschwardt Nov 23, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I thought it would be more convenient to use a TelescopeStateFetcher... but then I realised (as you probably also did) that indexed keys are not so nice in this regard. 🤦🏾

I wonder whether we should extend telstate keys in future to make this more convenient 🙂 We could e.g. base keys on pathlib and then treat the index key as a kind of suffix, or just support tuples of strings.

Then we could pass primary_beam_key + antenna.name to TelescopeStateFetcher.get.

Just a thought, not for now...

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed, it is a bit of a clumsy workaround as it stands now :(

Copy link
Contributor

@ludwigschwardt ludwigschwardt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry for the long wait... Just saw this again 😅

pol1 = beam_pol_idx[pol_order[1]]
beam_phand = np.stack([beam_sample[..., pol0, pol0], beam_sample[..., pol1, pol1]], axis=-1)
# select only the hh and vv beams, order doesn't matter as we average the two pols
beam_phand = np.stack([beam_sample[..., 0, 0], beam_sample[..., 1, 1]], axis=-1)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A small suggestion: maybe it's better to just add the HH and VV parts together, instead of first assembling them into a new array of twice the size and then averaging that down again in a second step. I'm not sure exactly how large beam_sample is, but this seems more memory efficient and also a bit more readable. E.g.:

beam_phand = 0.5 * (beam_sample[..., 0, 0] + beam_sample[..., 1, 1])

Another thought: how does this differ from the UNPOLARIZED_POWER OutputType? Why not use that instead?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good suggestion, the current setup it probably a hangover of when I was still doing separate polarisations.

I think the difference here from UNPOLARIZED_POWER is that you retain some phase information, while power is a scalar. Ideally you want to retain this phase information so that for the SKA - MKAT baselines you have an approximately correct phase for the combined beam. For MKAT - MKAT baselines and SKA - SKA baselines the phases of beams will cancel each other out and then it should be indistinguishable from using the UNPOLARIZED_POWER type.


def _complex_interp(fp, xp, x):
return complex_interp(xp, x, fp)
beam = np.apply_along_axis(_complex_interp, 0, beam_phand,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cool trick - I must remember this!

Since you are already wrapping the function, why not populate the x and xp parameters upfront? For example:

def _complex_interp_over_freq(fp):
    return complex_interp(chan_freqs.value, chan_freqs.value[::freq_stride], fp)
beam = np.apply_along_axis(_complex_interp_over_freq, 0, beam_phand)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

# sample beam at close to its frequency resolution
channel_width = chan_freqs[1] - chan_freqs[0]
freq_stride = int(np.floor(beam_model.frequency_resolution() / channel_width))

Copy link
Contributor

@ludwigschwardt ludwigschwardt May 25, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe add

sampled_freqs = chan_freqs[::freq_stride]

for more clarity, and use below (also with complex_interp).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

This is to address SPR1-1009
Copy link
Contributor

@ludwigschwardt ludwigschwardt left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Long live the branch!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants