File: rms_image.cc

package info (click to toggle)
wsclean 3.6-3
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 16,296 kB
  • sloc: cpp: 129,246; python: 22,066; sh: 360; ansic: 230; makefile: 185
file content (125 lines) | stat: -rw-r--r-- 4,774 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
// SPDX-License-Identifier: LGPL-3.0-only

#include "math/rms_image.h"

#include <aocommon/image.h>
#include <aocommon/logger.h>
#include <aocommon/staticfor.h>
#include <aocommon/units/fluxdensity.h>

#include <schaapcommon/math/restoreimage.h>

using aocommon::Image;

namespace radler::math::rms_image {

void Make(Image& rms_output, const Image& input_image, double window_size,
          long double beam_major, long double beam_minor, long double beam_pa,
          long double pixel_scale_l, long double pixel_scale_m) {
  Image image(input_image);
  image.Square();
  rms_output = Image(image.Width(), image.Height(), 0.0);

  schaapcommon::math::RestoreImage(
      rms_output.Data(), image.Data(), image.Width(), image.Height(),
      beam_major * window_size, beam_minor * window_size, beam_pa,
      pixel_scale_l, pixel_scale_m);

  const double s = std::sqrt(2.0 * M_PI);
  const long double sigmaMaj = beam_major / (2.0L * sqrtl(2.0L * logl(2.0L)));
  const long double sigmaMin = beam_minor / (2.0L * sqrtl(2.0L * logl(2.0L)));
  const double norm = 1.0 / (s * sigmaMaj / pixel_scale_l * window_size * s *
                             sigmaMin / pixel_scale_l * window_size);
  for (auto& val : rms_output) val = std::sqrt(val * norm);
}

void SlidingMinimum(Image& output, const Image& input, size_t window_size) {
  const size_t width = input.Width();
  output = Image(width, input.Height());
  Image temp(output);

  aocommon::StaticFor<size_t> loop;

  loop.Run(0, input.Height(), [&](size_t yStart, size_t yEnd) {
    for (size_t y = yStart; y != yEnd; ++y) {
      float* outRowptr = &temp[y * width];
      const float* inRowptr = &input[y * width];
      for (size_t x = 0; x != width; ++x) {
        size_t left = std::max(x, window_size / 2) - window_size / 2;
        size_t right = std::min(x, width - window_size / 2) + window_size / 2;
        outRowptr[x] = *std::min_element(inRowptr + left, inRowptr + right);
      }
    }
  });

  loop.Run(0, width, [&](size_t xStart, size_t xEnd) {
    aocommon::UVector<float> vals;
    for (size_t x = xStart; x != xEnd; ++x) {
      for (size_t y = 0; y != input.Height(); ++y) {
        size_t top = std::max(y, window_size / 2) - window_size / 2;
        size_t bottom =
            std::min(y, input.Height() - window_size / 2) + window_size / 2;
        vals.clear();
        for (size_t winY = top; winY != bottom; ++winY) {
          vals.push_back(temp[winY * width + x]);
        }
        output[y * width + x] = *std::min_element(vals.begin(), vals.end());
      }
    }
  });
}

void SlidingMaximum(Image& output, const Image& input, size_t window_size) {
  Image flipped(input);
  flipped.Negate();
  SlidingMinimum(output, flipped, window_size);
  output.Negate();
}

void MakeWithNegativityLimit(Image& rms_output, const Image& input_image,
                             double window_size, long double beam_major,
                             long double beam_minor, long double beam_pa,
                             long double pixel_scale_l,
                             long double pixel_scale_m) {
  Make(rms_output, input_image, window_size, beam_major, beam_minor, beam_pa,
       pixel_scale_l, pixel_scale_m);
  Image slidingMinimum(input_image.Width(), input_image.Height());
  double beamInPixels = std::max(beam_major / pixel_scale_l, 1.0L);
  SlidingMinimum(slidingMinimum, input_image, window_size * beamInPixels);
  for (size_t i = 0; i != rms_output.Size(); ++i) {
    rms_output[i] = std::max<float>(rms_output[i],
                                    std::abs(slidingMinimum[i]) * (1.5 / 5.0));
  }
}

double MakeRmsFactorImage(Image& rms_image, double local_rms_strength) {
  const double stddev = rms_image.Min();
  aocommon::Logger::Info << "Lowest RMS in image: "
                         << aocommon::units::FluxDensity::ToNiceString(stddev)
                         << '\n';
  if (stddev < 0.0) {
    throw std::runtime_error(
        "RMS image can only contain values >= 0, but contains values < "
        "0.0");
  }
  // Convert the RMS image to a "factor" that can be multiplied with the
  // image. The factor will be 1 at the minimum RMS such that Jy remains
  // somewhat interpretable.
  if (local_rms_strength == 1.0) {
    // Optimization of generic case to avoid unnecessary std::pow evaluation.
    for (float& value : rms_image) {
      if (value != 0.0) value = stddev / value;
    }
  } else if (local_rms_strength == 0.0) {
    // This special case is needed to make sure that zeros in the image are
    // still converted to one.
    rms_image = 1.0;
  } else {
    for (float& value : rms_image) {
      if (value != 0.0) value = std::pow(stddev / value, local_rms_strength);
    }
  }
  return stddev;
}

}  // namespace radler::math::rms_image