9 Commits

Author SHA1 Message Date
nou a675e97e0b Release 0.2.8 2023-06-13 12:25:13 +02:00
nou 6028354c61 Remove linking against zstd 2023-06-12 23:41:48 +02:00
nou 467af80cb3 Fix reading ICCprofile 2023-06-05 12:21:27 +02:00
nou 76ee3f361c Release 0.2.6 2023-06-02 08:50:37 +02:00
nou cde2fc9a17 Fix "string sub-command REPLACE requires at least four arguments" 2023-05-31 14:56:26 +02:00
nou 0ddff094ef Fix generating pkgconfig file 2023-05-29 23:16:53 +02:00
nou 5dcc383090 Fix generating pkgconfig file 2023-05-29 22:59:24 +02:00
nou 0893bfa048 Add method to for ICC profile 2023-05-24 16:28:11 +02:00
nou a536a271ab Add StreamBuffer to optimize save/load from memory 2023-05-21 19:33:36 +02:00
6 changed files with 223 additions and 22 deletions
+9 -1
View File
@@ -1,6 +1,6 @@
cmake_minimum_required(VERSION 3.14) cmake_minimum_required(VERSION 3.14)
project(libXISF VERSION 0.2.4 LANGUAGES CXX C project(libXISF VERSION 0.2.8 LANGUAGES CXX C
HOMEPAGE_URL https://gitea.nouspiro.space/nou/libXISF HOMEPAGE_URL https://gitea.nouspiro.space/nou/libXISF
DESCRIPTION "LibXISF is C++ library that can read and write XISF files produced by PixInsight.") DESCRIPTION "LibXISF is C++ library that can read and write XISF files produced by PixInsight.")
@@ -66,6 +66,8 @@ add_library(XISF
libXISF_global.h libXISF_global.h
libxisf.cpp libxisf.cpp
libxisf.h libxisf.h
streambuffer.cpp
streambuffer.h
utils.cpp utils.cpp
variant.cpp variant.cpp
${THIRD_PARTY_SRC} ${THIRD_PARTY_SRC}
@@ -106,6 +108,12 @@ install(FILES ${XISF_PUBLIC_HEADERS} DESTINATION ${CMAKE_INSTALL_INCLUDEDIR})
install(FILES ${CMAKE_CURRENT_BINARY_DIR}/libxisf.pc DESTINATION ${CMAKE_INSTALL_LIBDIR}/pkgconfig) install(FILES ${CMAKE_CURRENT_BINARY_DIR}/libxisf.pc DESTINATION ${CMAKE_INSTALL_LIBDIR}/pkgconfig)
install(TARGETS XISF LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}) install(TARGETS XISF LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR})
list(JOIN PC_LIBS_REQUIRE ", " PC_REQUIRE_STR2)
if(NOT PC_REQUIRE_STR2 STREQUAL "")
string(REPLACE "lz4" "liblz4" PC_REQUIRE_STR ${PC_REQUIRE_STR2})
endif()
list(TRANSFORM PC_LIBS_REQUIRE PREPEND "-l")
list(JOIN PC_LIBS_REQUIRE " " PC_LIBS_STR)
configure_file(libxisf.pc.in libxisf.pc @ONLY) configure_file(libxisf.pc.in libxisf.pc @ONLY)
#testing #testing
+33 -18
View File
@@ -27,6 +27,7 @@
#include <lz4hc.h> #include <lz4hc.h>
#include <pugixml.hpp> #include <pugixml.hpp>
#include <zlib.h> #include <zlib.h>
#include "streambuffer.h"
namespace LibXISF namespace LibXISF
{ {
@@ -357,6 +358,16 @@ bool Image::addFITSKeywordAsProperty(const String &name, const String &value)
return false; return false;
} }
const ByteArray &Image::iccProfile() const
{
return _iccProfile;
}
void Image::setIccProfile(const ByteArray &iccProfile)
{
_iccProfile = iccProfile;
}
void *Image::imageData() void *Image::imageData()
{ {
return _dataBlock.data.size() ? _dataBlock.data.data() : nullptr; return _dataBlock.data.size() ? _dataBlock.data.data() : nullptr;
@@ -530,8 +541,10 @@ private:
FITSKeyword parseFITSKeyword(const pugi::xml_node &node); FITSKeyword parseFITSKeyword(const pugi::xml_node &node);
ColorFilterArray parseCFA(const pugi::xml_node &node); ColorFilterArray parseCFA(const pugi::xml_node &node);
Image parseImage(const pugi::xml_node &node); Image parseImage(const pugi::xml_node &node);
void readAttachment(DataBlock &dataBlock);
std::unique_ptr<std::istream> _io; std::unique_ptr<std::istream> _io;
std::unique_ptr<StreamBuffer> _buffer;
std::vector<Image> _images; std::vector<Image> _images;
Image _thumbnail; Image _thumbnail;
std::vector<Property> _properties; std::vector<Property> _properties;
@@ -548,8 +561,8 @@ void XISFReaderPrivate::open(const String &name)
void XISFReaderPrivate::open(const ByteArray &data) void XISFReaderPrivate::open(const ByteArray &data)
{ {
close(); close();
std::string str((char*)data.data(), data.size()); _buffer = std::make_unique<StreamBuffer>(data);
_io = std::make_unique<std::istringstream>(str, std::ios_base::in | std::ios_base::binary); _io = std::make_unique<std::istream>(_buffer.get());
readSignature(); readSignature();
readXISFHeader(); readXISFHeader();
} }
@@ -565,6 +578,7 @@ void XISFReaderPrivate::open(std::istream *io)
void XISFReaderPrivate::close() void XISFReaderPrivate::close()
{ {
_io.reset(); _io.reset();
_buffer.reset();
_images.clear(); _images.clear();
_properties.clear(); _properties.clear();
} }
@@ -582,10 +596,7 @@ const Image& XISFReaderPrivate::getImage(uint32_t n, bool readPixels)
Image &img = _images[n]; Image &img = _images[n];
if(img._dataBlock.attachmentPos && readPixels) if(img._dataBlock.attachmentPos && readPixels)
{ {
_io->seekg(img._dataBlock.attachmentPos); readAttachment(img._dataBlock);
ByteArray data(img._dataBlock.attachmentSize);
_io->read(data.data(), data.size());
img._dataBlock.decompress(data);
} }
return img; return img;
} }
@@ -721,17 +732,10 @@ Property XISFReaderPrivate::parseProperty(const pugi::xml_node &node)
{ {
DataBlock dataBlock = parseDataBlock(node); DataBlock dataBlock = parseDataBlock(node);
if(dataBlock.attachmentPos) if(dataBlock.attachmentPos)
{ readAttachment(dataBlock);
data.resize(dataBlock.attachmentSize);
_io->seekg(dataBlock.attachmentPos);
_io->read(data.data(), dataBlock.attachmentSize);
dataBlock.decompress(data);
}
else
{
data = dataBlock.data; data = dataBlock.data;
} }
}
deserializeVariant(node, property.value, data); deserializeVariant(node, property.value, data);
@@ -799,6 +803,9 @@ Image XISFReaderPrivate::parseImage(const pugi::xml_node &node)
if(node.child("ICCProfile")) if(node.child("ICCProfile"))
{ {
DataBlock icc = parseDataBlock(node.child("ICCProfile")); DataBlock icc = parseDataBlock(node.child("ICCProfile"));
if(icc.attachmentPos)
readAttachment(icc);
image._iccProfile = icc.data; image._iccProfile = icc.data;
} }
@@ -808,6 +815,14 @@ Image XISFReaderPrivate::parseImage(const pugi::xml_node &node)
return image; return image;
} }
void XISFReaderPrivate::readAttachment(DataBlock &dataBlock)
{
ByteArray data(dataBlock.attachmentSize);
_io->seekg(dataBlock.attachmentPos);
_io->read(data.data(), dataBlock.attachmentSize);
dataBlock.decompress(data);
}
class XISFWriterPrivate class XISFWriterPrivate
{ {
public: public:
@@ -839,10 +854,10 @@ void XISFWriterPrivate::save(const String &name)
void XISFWriterPrivate::save(ByteArray &data) void XISFWriterPrivate::save(ByteArray &data)
{ {
std::ostringstream oss; StreamBuffer buffer;
std::ostream oss(&buffer);
save(oss); save(oss);
std::string str = oss.str(); data = buffer.byteArray();
data = ByteArray(str.data(), str.size());
} }
void XISFWriterPrivate::save(std::ostream &io) void XISFWriterPrivate::save(std::ostream &io)
+2
View File
@@ -301,6 +301,8 @@ public:
* For example OBSERVER => Observer:Name, SITELAT => Observation:Location:Latitude * For example OBSERVER => Observer:Name, SITELAT => Observation:Location:Latitude
*/ */
bool addFITSKeywordAsProperty(const String &name, const String &value); bool addFITSKeywordAsProperty(const String &name, const String &value);
const ByteArray& iccProfile() const;
void setIccProfile(const ByteArray &iccProfile);
void* imageData(); void* imageData();
const void* imageData() const; const void* imageData() const;
+1 -2
View File
@@ -7,7 +7,6 @@ Name: @PROJECT_NAME@
Description: @CMAKE_PROJECT_DESCRIPTION@ Description: @CMAKE_PROJECT_DESCRIPTION@
URL: @CMAKE_PROJECT_HOMEPAGE_URL@ URL: @CMAKE_PROJECT_HOMEPAGE_URL@
Version: @PROJECT_VERSION@ Version: @PROJECT_VERSION@
Requires.private: @PC_LIBS_REQUIRE@ Requires.private: @PC_REQUIRE_STR@
Cflags: -I"${includedir}" Cflags: -I"${includedir}"
Libs: -L"${libdir}" -lXISF Libs: -L"${libdir}" -lXISF
Libs.private: -L"${libdir}" @PC_LIBS_REQUIRE@
+144
View File
@@ -0,0 +1,144 @@
#include "streambuffer.h"
namespace LibXISF
{
StreamBuffer::StreamBuffer()
{
setg(nullptr, nullptr, nullptr);
setp(nullptr, nullptr);
}
StreamBuffer::StreamBuffer(const ByteArray &byteArray) :
_size(byteArray.size()),
_byteArray(byteArray)
{
if(_byteArray.size())
{
char *ptr = _byteArray.data();
setg(ptr, ptr, ptr + _size);
setp(ptr, ptr + _size);
}
else
{
setg(nullptr, nullptr, nullptr);
setp(nullptr, nullptr);
}
}
ByteArray StreamBuffer::byteArray()
{
return _byteArray;
}
StreamBuffer::pos_type StreamBuffer::seekoff(off_type off, std::ios_base::seekdir dir, std::ios_base::openmode mode)
{
pos_type ret = pos_type(off_type(-1));
off_type newoffi = off;
off_type newoffo = off;
if(dir == std::ios_base::cur)
{
newoffi += gptr() - eback();
newoffo += pptr() - pbase();
}
else if(dir == std::ios_base::end)
newoffo = newoffi = _byteArray.size() - off;
char *ptr = _byteArray.data();
if(mode & std::ios_base::in && newoffi >= 0 && newoffi <= _size)
{
setg(ptr, ptr + newoffi, ptr + _size);
ret = pos_type(newoffi);
}
if(mode & std::ios_base::out && newoffo >= 0 && newoffo <= _size)
{
setp(ptr, ptr + _size);
pbump(newoffo);
ret = pos_type(newoffo);
}
return ret;
}
StreamBuffer::pos_type StreamBuffer::seekpos(pos_type pos, std::ios_base::openmode mode)
{
pos_type ret = pos_type(off_type(-1));
off_type off = pos;
if(off >= 0 && off <= (off_type)_byteArray.size())
{
char *ptr = _byteArray.data();
if(mode & std::ios_base::in)
setg(ptr, ptr + pos, ptr + _size);
if(mode & std::ios_base::out)
{
setp(ptr, ptr + _size);
pbump(pos);
}
ret = pos;
}
return ret;
}
std::streamsize StreamBuffer::xsgetn(char_type *s, std::streamsize n)
{
std::streamsize ret = 0;
std::streamsize len = egptr() - gptr();
if(len > 0)
{
std::streamsize c = std::min(n, len);
std::memcpy(s, gptr(), c);
gbump(c);
ret = c;
}
return ret;
}
StreamBuffer::int_type StreamBuffer::underflow()
{
if(gptr() < egptr())
return traits_type::to_int_type(*gptr());
else
return traits_type::eof();
}
std::streamsize StreamBuffer::xsputn(const char_type *s, std::streamsize n)
{
off_type len = epptr() - pptr();
if(len < n)
{
_size += n - len;
_byteArray.resize(_size);
update_ptrs();
}
std::memcpy(pptr(), s, n);
pbump(n);
return n;
}
StreamBuffer::int_type StreamBuffer::overflow(int_type c)
{
if(traits_type::eq_int_type(traits_type::eof(), c))
return traits_type::eof();
_byteArray.append(c);
_size++;
pbump(1);
update_ptrs();
return c;
}
void StreamBuffer::update_ptrs()
{
off_type ipos = gptr() - eback();
off_type opos = pptr() - pbase();
char *ptr = _byteArray.data();
setg(ptr, ptr + ipos, ptr + _size);
setp(ptr, ptr + _size);
pbump(opos);
}
}
+33
View File
@@ -0,0 +1,33 @@
#ifndef STREAMBUFFER_H
#define STREAMBUFFER_H
#include <streambuf>
#include "libxisf.h"
namespace LibXISF
{
class StreamBuffer : public std::streambuf
{
public:
StreamBuffer();
StreamBuffer(const ByteArray &byteArray);
ByteArray byteArray();
protected:
pos_type seekoff(off_type off, std::ios_base::seekdir dir, std::ios_base::openmode mode = std::ios_base::in | std::ios_base::out) override;
pos_type seekpos(pos_type pos, std::ios_base::openmode mode = std::ios_base::in | std::ios_base::out) override;
std::streamsize xsgetn(char_type *s, std::streamsize n) override;
int_type underflow() override;
std::streamsize xsputn(const char_type *s, std::streamsize n) override;
int_type overflow(int_type c = traits_type::eof()) override;
private:
void update_ptrs();
off_type _size = 0;
ByteArray _byteArray;
};
}
#endif // STREAMBUFFER_H