mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-11-25 17:28:50 -05:00
189 lines
5.8 KiB
C++
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;
|
|
}
|