0f5271f14c
PUT /config/user_mask.tiff only accepted 32-bit unsigned TIFF, so masks exported by tools like PyFAI (8-bit) failed. Route the upload through the universal ReadTIFF reader and let PixelMask take a CompressedImage directly: it validates the 2D shape against the detector's converted/raw layouts, binarizes any 8/16/32-bit integer image (non-zero == masked), and rejects float/multi-channel images. Also dedupe the TIFF readers: ReadTIFFFromString16 is now a thin wrapper over ReadTIFF, and the now-unused ReadTIFFFromString32 is removed. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
166 lines
6.8 KiB
C++
166 lines
6.8 KiB
C++
// SPDX-FileCopyrightText: 2024 Filip Leonarski, Paul Scherrer Institute <filip.leonarski@psi.ch>
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
#include <tiffio.h>
|
|
#include <tiffio.hxx>
|
|
#include <cstring>
|
|
#include <sstream>
|
|
|
|
#include "JFJochTIFF.h"
|
|
#include "../common/JFJochException.h"
|
|
|
|
|
|
void WriteTIFF(TIFF *tiff, const CompressedImage& image) {
|
|
|
|
if (tiff == nullptr)
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError, "TIFFStreamOpen error");
|
|
|
|
std::vector<uint8_t> buffer;
|
|
image.GetUncompressed(buffer);
|
|
|
|
TIFFSetField(tiff, TIFFTAG_IMAGEWIDTH, image.GetWidth()); // set the width of the image
|
|
TIFFSetField(tiff, TIFFTAG_IMAGELENGTH, image.GetHeight()); // set the height of the image
|
|
|
|
switch (image.GetMode()) {
|
|
case CompressedImageMode::RGB:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 8);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 3);
|
|
TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_RGB);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_UINT);
|
|
break;
|
|
case CompressedImageMode::Int8:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 8);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_INT);
|
|
break;
|
|
case CompressedImageMode::Int16:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 16);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_INT);
|
|
break;
|
|
case CompressedImageMode::Int32:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 32);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_INT);
|
|
break;
|
|
case CompressedImageMode::Uint8:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 8);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_UINT);
|
|
break;
|
|
case CompressedImageMode::Uint16:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 16);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_UINT);
|
|
break;
|
|
case CompressedImageMode::Uint32:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 32);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_UINT);
|
|
break;
|
|
case CompressedImageMode::Float16:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 16);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_IEEEFP); // Float format
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
break;
|
|
case CompressedImageMode::Float32:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 32);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_IEEEFP); // Float format
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
break;
|
|
case CompressedImageMode::Float64:
|
|
TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 64);
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLEFORMAT, SAMPLEFORMAT_IEEEFP); // Float format
|
|
TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 1);
|
|
break;
|
|
}
|
|
TIFFSetField(tiff, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
|
|
TIFFSetField(tiff, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
|
|
TIFFSetField(tiff, TIFFTAG_COMPRESSION, COMPRESSION_NONE); // set compression to LZW
|
|
TIFFSetField(tiff, TIFFTAG_ROWSPERSTRIP, image.GetHeight());
|
|
|
|
if (TIFFWriteEncodedStrip(tiff, 0, buffer.data(), image.GetUncompressedSize()) < 0)
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError, "TIFFWriteEncodedStrip error");
|
|
}
|
|
|
|
std::string WriteTIFFToString(const CompressedImage& image) {
|
|
std::stringstream os;
|
|
|
|
TIFF *tiff = TIFFStreamOpen("x", (std::ostream *) &os);
|
|
WriteTIFF(tiff, image);
|
|
TIFFClose(tiff);
|
|
|
|
return os.str();
|
|
}
|
|
|
|
void WriteTIFFToFile(const std::string &filename, const CompressedImage& image) {
|
|
TIFF *tiff = TIFFOpen(filename.c_str(), "w");
|
|
|
|
WriteTIFF(tiff, image);
|
|
|
|
TIFFClose(tiff);
|
|
}
|
|
|
|
CompressedImage ReadTIFF(const std::string &s, std::vector<uint8_t> &buffer) {
|
|
if (s.empty())
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError, "No TIFF file provided");
|
|
uint32_t rows_per_string = 0;
|
|
|
|
uint16_t elem_size = 0;
|
|
uint16_t format = 0;
|
|
uint32_t cols = 0;
|
|
uint32_t lines = 0;
|
|
|
|
std::istringstream input_TIFF_stream(s);
|
|
TIFF* tiff = TIFFStreamOpen("MemTIFF", &input_TIFF_stream);
|
|
if (tiff == nullptr)
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError,"Not a proper TIFF file");
|
|
|
|
TIFFGetField(tiff, TIFFTAG_IMAGEWIDTH, &cols); // get the width of the image
|
|
TIFFGetField(tiff, TIFFTAG_IMAGELENGTH, &lines); // get the height of the image
|
|
TIFFGetField(tiff, TIFFTAG_BITSPERSAMPLE, &elem_size); // get the size of the channels
|
|
TIFFGetField(tiff, TIFFTAG_ROWSPERSTRIP, &rows_per_string);
|
|
TIFFGetField(tiff, TIFFTAG_SAMPLEFORMAT, &format);
|
|
|
|
if (elem_size % 8 != 0)
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError, "Only byte-aligned TIFF samples are supported");
|
|
|
|
const size_t elem_size_bytes = elem_size / 8;
|
|
if (cols == 0 || lines == 0 || elem_size_bytes == 0)
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError,"Size wrong");
|
|
|
|
if (cols * elem_size_bytes != TIFFScanlineSize(tiff))
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError, "TIFFScanlineSize mismatch");
|
|
|
|
buffer.resize(static_cast<size_t>(cols) * static_cast<size_t>(lines) * elem_size_bytes);
|
|
|
|
for (uint32_t i = 0; i < lines; i++) {
|
|
if (TIFFReadScanline(tiff, buffer.data() + static_cast<size_t>(i) * cols * elem_size_bytes, i, 0) < 0)
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError, "TIFFReadScanline error");
|
|
}
|
|
|
|
TIFFClose(tiff);
|
|
|
|
CompressedImageMode mode = CalcImageMode(elem_size_bytes, (format == SAMPLEFORMAT_IEEEFP), (format == SAMPLEFORMAT_INT));
|
|
return CompressedImage(buffer, cols, lines, mode);
|
|
}
|
|
|
|
std::vector<uint16_t> ReadTIFFFromString16(const std::string &s, uint32_t &cols, uint32_t &lines) {
|
|
std::vector<uint8_t> buffer;
|
|
CompressedImage image = ReadTIFF(s, buffer);
|
|
|
|
if (image.GetByteDepth() != sizeof(uint16_t) || image.GetNumChannels() != 1)
|
|
throw JFJochException(JFJochExceptionCategory::TIFFGeneratorError, "Only 16-bit format supported");
|
|
|
|
cols = image.GetWidth();
|
|
lines = image.GetHeight();
|
|
|
|
std::vector<uint16_t> ret(static_cast<size_t>(cols) * lines);
|
|
memcpy(ret.data(), buffer.data(), buffer.size());
|
|
return ret;
|
|
}
|
|
|
|
void SuppressTIFFErrors() {
|
|
TIFFSetErrorHandler(nullptr);
|
|
}
|