1
0
mirror of https://github.com/f4exb/sdrangel.git synced 2024-11-28 18:58:48 -05:00
sdrangel/sdrbase/util/fits.cpp
2024-06-10 14:01:57 +01:00

189 lines
5.8 KiB
C++

///////////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2021-2022 Jon Beniston, M7RCE <jon@beniston.com> //
// Copyright (C) 2022 Edouard Griffiths, F4EXB <f4exb06@gmail.com> //
// //
// This program is free software; you can redistribute it and/or modify //
// it under the terms of the GNU General Public License as published by //
// the Free Software Foundation as version 3 of the License, or //
// (at your option) any later version. //
// //
// This program is distributed in the hope that it will be useful, //
// but WITHOUT ANY WARRANTY; without even the implied warranty of //
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //
// GNU General Public License V3 for more details. //
// //
// You should have received a copy of the GNU General Public License //
// along with this program. If not, see <http://www.gnu.org/licenses/>. //
///////////////////////////////////////////////////////////////////////////////////
#include <algorithm>
#include <cmath>
#include <QtGlobal>
#include <QRegularExpression>
#include <QDebug>
#include <QResource>
#include "fits.h"
FITS::FITS(QString resourceName) :
m_valid(false)
{
QResource m_res(resourceName);
if (!m_res.isValid()) {
qWarning() << "FITS: - " << resourceName << " is not a valid resource";
return;
}
int m_headerSize = 2880;
qint64 m_fileSize;
#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
m_data = m_res.uncompressedData();
m_fileSize = m_res.uncompressedSize();
#else
m_data = QByteArray::fromRawData((const char *)m_res.data(), m_res.size());
if (m_res.isCompressed()) {
m_data = qUncompress(m_data);
}
m_fileSize = m_res.size();
#endif
int hLen = std::min((qint64)m_headerSize * 3, m_fileSize); // Could possibly be bigger
QByteArray headerBytes = m_data.left(hLen);
QString header = QString::fromLatin1(headerBytes);
QRegularExpression widthRE("NAXIS1 *= *([0-9]+)");
QRegularExpression heightRE("NAXIS2 *= *([0-9]+)");
QRegularExpression bitsPerPixelRE("BITPIX *= *(-?[0-9]+)");
QRegularExpression bzeroRE("BZERO *= *([0-9]+)");
QRegularExpression bscaleRE("BSCALE *= *(-?[0-9]+(.[0-9]+)?)");
QRegularExpression buintRE("BUNIT *= *\\'([A-Z ]+)\\'");
QRegularExpression cdelt1RE("CDELT1 *= *(-?[0-9]+(.[0-9]+)?)");
QRegularExpression cdelt2RE("CDELT2 *= *(-?[0-9]+(.[0-9]+)?)");
QRegularExpression endRE("END {77}");
QRegularExpressionMatch match;
match = widthRE.match(header);
if (match.hasMatch())
m_width = match.capturedTexts()[1].toInt();
else
{
qWarning() << "FITS: NAXIS1 missing";
return;
}
match = heightRE.match(header);
if (match.hasMatch())
m_height = match.capturedTexts()[1].toInt();
else
{
qWarning() << "FITS: NAXIS2 missing";
return;
}
match = bitsPerPixelRE.match(header);
if (match.hasMatch())
m_bitsPerPixel = match.capturedTexts()[1].toInt();
else
{
qWarning() << "FITS: BITPIX missing";
return;
}
m_bytesPerPixel = abs(m_bitsPerPixel)/8;
match = bzeroRE.match(header);
if (match.hasMatch())
m_bzero = match.capturedTexts()[1].toInt();
else
m_bzero = 0;
match = bscaleRE.match(header);
if (match.hasMatch())
m_bscale = match.capturedTexts()[1].toDouble();
else
m_bscale = 1.0;
match = cdelt1RE.match(header);
if (match.hasMatch())
m_cdelta1 = match.capturedTexts()[1].toDouble();
else
m_cdelta1 = 0.0;
match = cdelt2RE.match(header);
if (match.hasMatch())
m_cdelta2 = match.capturedTexts()[1].toDouble();
else
m_cdelta2 = 0.0;
match = buintRE.match(header);
if (match.hasMatch())
{
m_buint = match.capturedTexts()[1].trimmed();
if (m_buint.contains("MILLI"))
m_uintScale = 0.001f;
else
m_uintScale = 1.0f;
}
else
m_uintScale = 1.0f;
match = endRE.match(header);
int endIdx = match.capturedStart(0);
if (!match.hasMatch())
{
qWarning() << "FITS: END missing";
return;
}
m_dataStart = ((endIdx + m_headerSize) / m_headerSize) * m_headerSize;
m_valid = true;
}
float FITS::value(int x, int y) const
{
int offset = m_dataStart + (m_height-1-y) * m_width * m_bytesPerPixel + x * m_bytesPerPixel;
const uchar *data = (const uchar *)m_data.data();
// Big-endian
int v = 0;
for (int i = m_bytesPerPixel - 1; i >= 0; i--)
v += data[offset++] << (i*8);
if (m_bitsPerPixel > 0)
{
// Sign-extend
switch (m_bytesPerPixel)
{
case 1:
v = (char)v;
break;
case 2:
v = (qint16)v;
break;
case 3:
v = (qint32)v;
break;
}
return v * m_bscale + m_bzero;
}
else
{
// Type-punning via unions apparently undefined behaviour in C++
uint32_t i = (uint32_t)v;
float f;
memcpy(&f, &i, sizeof(f));
return f;
}
}
float FITS::scaledValue(int x, int y) const
{
float v = value(x, y);
return v * m_uintScale;
}
int FITS::mod(int a, int b) const
{
return a - b * floor(a/(double)b);
}
float FITS::scaledWrappedValue(int x, int y) const
{
float v = value(mod(x, m_width), mod(y, m_height));
return v * m_uintScale;
}