Compare commits
36 Commits
v0.2.0
...
033a34e248
| Author | SHA1 | Date | |
|---|---|---|---|
| 033a34e248 | |||
| cf9c903a90 | |||
| aa356443d3 | |||
| 16676540b4 | |||
| bae3edd12a | |||
| c848ba75ff | |||
| 8a1f305cc7 | |||
| 8e05a58610 | |||
| 0b0c865df0 | |||
| a675e97e0b | |||
| 6028354c61 | |||
| 467af80cb3 | |||
| 76ee3f361c | |||
| cde2fc9a17 | |||
| 0ddff094ef | |||
| 5dcc383090 | |||
| 0893bfa048 | |||
| a536a271ab | |||
| 3ee8bd4027 | |||
| b7cd8cb72f | |||
| 1c0d6e4e05 | |||
| f2907d7e53 | |||
| dafc26984e | |||
| f9731614c8 | |||
| ae46f09edf | |||
| ad50c5e323 | |||
| 164183f348 | |||
| c74a67184c | |||
| 77b83e4832 | |||
| e2d3c878c4 | |||
| 3b3d472149 | |||
| 8f8cc94ddb | |||
| bf20f8eb9c | |||
| dd66850fdf | |||
| 83f12a434c | |||
| 43533aa363 |
+78
-30
@@ -1,6 +1,10 @@
|
|||||||
cmake_minimum_required(VERSION 3.14)
|
cmake_minimum_required(VERSION 3.14)
|
||||||
|
|
||||||
project(libXISF VERSION 0.2.0 LANGUAGES CXX C)
|
project(libXISF VERSION 0.2.11 LANGUAGES CXX C
|
||||||
|
HOMEPAGE_URL https://gitea.nouspiro.space/nou/libXISF
|
||||||
|
DESCRIPTION "LibXISF is C++ library that can read and write XISF files produced by PixInsight.")
|
||||||
|
|
||||||
|
include(CMakeDependentOption)
|
||||||
|
|
||||||
set(CMAKE_INCLUDE_CURRENT_DIR ON)
|
set(CMAKE_INCLUDE_CURRENT_DIR ON)
|
||||||
set(CMAKE_CXX_STANDARD 17)
|
set(CMAKE_CXX_STANDARD 17)
|
||||||
@@ -9,58 +13,93 @@ set(CMAKE_C_VISIBILITY_PRESET hidden)
|
|||||||
set(CMAKE_CXX_VISIBILITY_PRESET hidden)
|
set(CMAKE_CXX_VISIBILITY_PRESET hidden)
|
||||||
|
|
||||||
option(BUILD_SHARED_LIBS "Build using shared libraries" ON)
|
option(BUILD_SHARED_LIBS "Build using shared libraries" ON)
|
||||||
option(USE_BUNDLED_LIBS "Use bundled LZ4 PugiXML and Zlib" ON)
|
option(USE_BUNDLED_LIBS "Use bundled LZ4 PugiXML and Zlib. You can still exclude some" ON)
|
||||||
|
cmake_dependent_option(USE_BUNDLED_LZ4 "Use bundled LZ4" ON "USE_BUNDLED_LIBS" OFF)
|
||||||
|
cmake_dependent_option(USE_BUNDLED_PUGIXML "Use bundled PugiXML" ON "USE_BUNDLED_LIBS" OFF)
|
||||||
|
cmake_dependent_option(USE_BUNDLED_ZLIB "Use bundled Zlib" ON "USE_BUNDLED_LIBS" OFF)
|
||||||
|
|
||||||
if(USE_BUNDLED_LIBS)
|
find_package(PkgConfig)
|
||||||
set(THIRD_PARTY_SRC
|
|
||||||
lz4/lz4.c
|
if(USE_BUNDLED_LZ4)
|
||||||
lz4/lz4.h
|
list(APPEND THIRD_PARTY_SRC
|
||||||
lz4/lz4hc.c
|
lz4/lz4.c
|
||||||
lz4/lz4hc.h
|
lz4/lz4.h
|
||||||
pugixml/pugixml.cpp
|
lz4/lz4hc.c
|
||||||
zlib/adler32.c
|
lz4/lz4hc.h)
|
||||||
zlib/compress.c
|
list(APPEND THIRD_PARTY_INCLUDE lz4)
|
||||||
zlib/crc32.c
|
else(USE_BUNDLED_LZ4)
|
||||||
zlib/deflate.c
|
pkg_check_modules(LZ4 liblz4 IMPORTED_TARGET REQUIRED)
|
||||||
zlib/gzclose.c
|
endif(USE_BUNDLED_LZ4)
|
||||||
zlib/gzlib.c
|
if(USE_BUNDLED_PUGIXML)
|
||||||
zlib/gzread.c
|
list(APPEND THIRD_PARTY_SRC pugixml/pugixml.cpp)
|
||||||
zlib/gzwrite.c
|
list(APPEND THIRD_PARTY_INCLUDE pugixml)
|
||||||
zlib/inflate.c
|
else(USE_BUNDLED_PUGIXML)
|
||||||
zlib/infback.c
|
pkg_check_modules(PUGIXML pugixml IMPORTED_TARGET REQUIRED)
|
||||||
zlib/inftrees.c
|
endif(USE_BUNDLED_PUGIXML)
|
||||||
zlib/inffast.c
|
if(USE_BUNDLED_ZLIB)
|
||||||
zlib/trees.c
|
list(APPEND THIRD_PARTY_SRC
|
||||||
zlib/uncompr.c
|
zlib/adler32.c
|
||||||
zlib/zutil.c)
|
zlib/compress.c
|
||||||
|
zlib/crc32.c
|
||||||
|
zlib/deflate.c
|
||||||
|
zlib/gzclose.c
|
||||||
|
zlib/gzlib.c
|
||||||
|
zlib/gzread.c
|
||||||
|
zlib/gzwrite.c
|
||||||
|
zlib/inflate.c
|
||||||
|
zlib/infback.c
|
||||||
|
zlib/inftrees.c
|
||||||
|
zlib/inffast.c
|
||||||
|
zlib/trees.c
|
||||||
|
zlib/uncompr.c
|
||||||
|
zlib/zutil.c)
|
||||||
set(ZLIB_PC ${CMAKE_CURRENT_BINARY_DIR}/zlib.pc)
|
set(ZLIB_PC ${CMAKE_CURRENT_BINARY_DIR}/zlib.pc)
|
||||||
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/zlib/zlib.pc.cmakein ${ZLIB_PC} @ONLY)
|
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/zlib/zlib.pc.cmakein ${ZLIB_PC} @ONLY)
|
||||||
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/zlib/zconf.h.cmakein ${CMAKE_CURRENT_BINARY_DIR}/zconf.h @ONLY)
|
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/zlib/zconf.h.cmakein ${CMAKE_CURRENT_BINARY_DIR}/zconf.h @ONLY)
|
||||||
else(USE_BUNDLED_LIBS)
|
list(APPEND THIRD_PARTY_INCLUDE zlib)
|
||||||
find_package(PkgConfig REQUIRED)
|
else(USE_BUNDLED_ZLIB)
|
||||||
pkg_check_modules(LZ4 liblz4 IMPORTED_TARGET REQUIRED)
|
|
||||||
pkg_check_modules(PUGIXML pugixml IMPORTED_TARGET REQUIRED)
|
|
||||||
pkg_check_modules(ZLIB zlib IMPORTED_TARGET REQUIRED)
|
pkg_check_modules(ZLIB zlib IMPORTED_TARGET REQUIRED)
|
||||||
endif(USE_BUNDLED_LIBS)
|
endif(USE_BUNDLED_ZLIB)
|
||||||
|
|
||||||
add_library(XISF
|
add_library(XISF
|
||||||
bytearray.cpp
|
bytearray.cpp
|
||||||
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}
|
||||||
)
|
)
|
||||||
|
|
||||||
if(USE_BUNDLED_LIBS)
|
if(USE_BUNDLED_LIBS)
|
||||||
target_include_directories(XISF PRIVATE lz4 pugixml zlib)
|
target_include_directories(XISF PRIVATE ${THIRD_PARTY_INCLUDE})
|
||||||
|
if(NOT USE_BUNDLED_LZ4)
|
||||||
|
target_link_libraries(XISF PUBLIC PkgConfig::LZ4)
|
||||||
|
list(APPEND PC_LIBS_REQUIRE lz4)
|
||||||
|
endif(NOT USE_BUNDLED_LZ4)
|
||||||
|
if(NOT USE_BUNDLED_PUGIXML)
|
||||||
|
target_link_libraries(XISF PUBLIC PkgConfig::PUGIXML)
|
||||||
|
list(APPEND PC_LIBS_REQUIRE pugixml)
|
||||||
|
endif(NOT USE_BUNDLED_PUGIXML)
|
||||||
|
if(NOT USE_BUNDLED_ZLIB)
|
||||||
|
target_link_libraries(XISF PUBLIC PkgConfig::ZLIB)
|
||||||
|
list(APPEND PC_LIBS_REQUIRE zlib)
|
||||||
|
endif(NOT USE_BUNDLED_ZLIB)
|
||||||
else(USE_BUNDLED_LIBS)
|
else(USE_BUNDLED_LIBS)
|
||||||
target_link_libraries(XISF PUBLIC PkgConfig::LZ4 PkgConfig::PUGIXML PkgConfig::ZLIB)
|
target_link_libraries(XISF PUBLIC PkgConfig::LZ4 PkgConfig::PUGIXML PkgConfig::ZLIB)
|
||||||
|
list(APPEND PC_LIBS_REQUIRE lz4 pugixml zlib)
|
||||||
endif(USE_BUNDLED_LIBS)
|
endif(USE_BUNDLED_LIBS)
|
||||||
|
|
||||||
set_target_properties(XISF PROPERTIES VERSION ${PROJECT_VERSION} SOVERSION ${PROJECT_VERSION_MAJOR})
|
set_target_properties(XISF PROPERTIES VERSION ${PROJECT_VERSION} SOVERSION ${PROJECT_VERSION_MAJOR})
|
||||||
|
|
||||||
|
pkg_check_modules(ZSTD libzstd IMPORTED_TARGET)
|
||||||
|
if(ZSTD_FOUND)
|
||||||
|
target_compile_definitions(XISF PRIVATE HAVE_ZSTD)
|
||||||
|
target_link_libraries(XISF PUBLIC PkgConfig::ZSTD)
|
||||||
|
endif(ZSTD_FOUND)
|
||||||
|
|
||||||
if(BUILD_SHARED_LIBS)
|
if(BUILD_SHARED_LIBS)
|
||||||
target_compile_definitions(XISF PRIVATE LIBXISF_LIBRARY)
|
target_compile_definitions(XISF PRIVATE LIBXISF_LIBRARY)
|
||||||
else(BUILD_SHARED_LIBS)
|
else(BUILD_SHARED_LIBS)
|
||||||
@@ -72,8 +111,17 @@ set(XISF_PUBLIC_HEADERS libxisf.h libXISF_global.h)
|
|||||||
include(GNUInstallDirs)
|
include(GNUInstallDirs)
|
||||||
|
|
||||||
install(FILES ${XISF_PUBLIC_HEADERS} DESTINATION ${CMAKE_INSTALL_INCLUDEDIR})
|
install(FILES ${XISF_PUBLIC_HEADERS} DESTINATION ${CMAKE_INSTALL_INCLUDEDIR})
|
||||||
|
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)
|
||||||
|
|
||||||
#testing
|
#testing
|
||||||
|
|
||||||
enable_testing()
|
enable_testing()
|
||||||
|
|||||||
@@ -3,4 +3,15 @@ LibXISF
|
|||||||
|
|
||||||
LibXISF is C++ library that can read and write XISF files produced by [PixInsight](https://pixinsight.com/).
|
LibXISF is C++ library that can read and write XISF files produced by [PixInsight](https://pixinsight.com/).
|
||||||
It implement [XISF 1.0 specification](https://pixinsight.com/doc/docs/XISF-1.0-spec/XISF-1.0-spec.html).
|
It implement [XISF 1.0 specification](https://pixinsight.com/doc/docs/XISF-1.0-spec/XISF-1.0-spec.html).
|
||||||
It is licensed under GPLv3. To compile you will need C++17 compiler.
|
It is licensed under GPLv3 or later. To compile you will need C++17 compiler.
|
||||||
|
|
||||||
|
To compile simply run these commands
|
||||||
|
```
|
||||||
|
cmake -B build -S .
|
||||||
|
cmake --build build --parallel
|
||||||
|
cmake --install build
|
||||||
|
```
|
||||||
|
|
||||||
|
By default it use bundled libraries. If you wish to use external libraries you will may add
|
||||||
|
`-DUSE_BUNDLED_LIBS=Off` to first command. Then you will need *lz4 pkg-config pugixml zlib* installed.
|
||||||
|
You may also specify `-DBUILD_SHARED_LIBS=Off` if you want build static lib.
|
||||||
|
|||||||
@@ -1,3 +1,21 @@
|
|||||||
|
/************************************************************************
|
||||||
|
* LibXISF - library to load and save XISF files *
|
||||||
|
* Copyright (C) 2023 Dušan Poizl *
|
||||||
|
* *
|
||||||
|
* 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, either 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 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 "libxisf.h"
|
#include "libxisf.h"
|
||||||
|
|
||||||
namespace LibXISF
|
namespace LibXISF
|
||||||
|
|||||||
Vendored
-29
@@ -1,29 +0,0 @@
|
|||||||
libxisf (0.2.0-ubuntu1) focal; urgency=medium
|
|
||||||
|
|
||||||
* Rewrite to remove Qt as dependency
|
|
||||||
|
|
||||||
-- Dušan Poizl <nou@nouspiro.space> Fri, 10 Mar 2023 22:10:01 +0100
|
|
||||||
|
|
||||||
libxisf (0.1.3-ubuntu1) focal; urgency=medium
|
|
||||||
|
|
||||||
* Write missing pixelStorage attribute
|
|
||||||
|
|
||||||
-- Dušan Poizl <nou@nouspiro.space> Fri, 17 Feb 2023 16:57:30 +0100
|
|
||||||
|
|
||||||
libxisf (0.1.2-ubuntu1) focal; urgency=medium
|
|
||||||
|
|
||||||
* Add qtbase5-dev as dependency to libxisf-dev
|
|
||||||
|
|
||||||
-- Dušan Poizl <nou@nouspiro.space> Thu, 09 Feb 2023 21:28:45 +0100
|
|
||||||
|
|
||||||
libxisf (0.1.1-ubuntu1) focal; urgency=medium
|
|
||||||
|
|
||||||
* Fixed packaging
|
|
||||||
|
|
||||||
-- Dušan Poizl <nou@nouspiro.space> Wed, 08 Feb 2023 18:57:50 +0100
|
|
||||||
|
|
||||||
libxisf (0.1.0-ubuntu1) focal; urgency=low
|
|
||||||
|
|
||||||
* Initial release.
|
|
||||||
|
|
||||||
-- Dušan Poizl <nou@nouspiro.space> Tue, 07 Feb 2023 10:47:08 +0100
|
|
||||||
Vendored
-1
@@ -1 +0,0 @@
|
|||||||
10
|
|
||||||
Vendored
-25
@@ -1,25 +0,0 @@
|
|||||||
Source: libxisf
|
|
||||||
Priority: optional
|
|
||||||
Maintainer: Dušan Poizl <nou@nouspiro.space>
|
|
||||||
Build-Depends: debhelper (>= 9),
|
|
||||||
cdbs,
|
|
||||||
cmake,
|
|
||||||
Standards-Version: 4.5.0
|
|
||||||
Section: libs
|
|
||||||
Homepage: https://gitea.nouspiro.space/nou/libXISF
|
|
||||||
Vcs-Git: https://gitea.nouspiro.space/nou/libXISF.git
|
|
||||||
|
|
||||||
Package: libxisf-dev
|
|
||||||
Section: libdevel
|
|
||||||
Architecture: any
|
|
||||||
Multi-Arch: same
|
|
||||||
Depends: libxisf (= ${binary:Version}), ${misc:Depends}, qtbase5-dev
|
|
||||||
Description: Library to load and save XISF images
|
|
||||||
Native format of PixInsight astroprocessing suite
|
|
||||||
|
|
||||||
Package: libxisf
|
|
||||||
Architecture: any
|
|
||||||
Multi-Arch: same
|
|
||||||
Depends: ${shlibs:Depends}, ${misc:Depends}
|
|
||||||
Description: Library to load and save XISF images
|
|
||||||
Native format of PixInsight astroprocessing suite
|
|
||||||
Vendored
-52
@@ -1,52 +0,0 @@
|
|||||||
Format: https://www.debian.org/doc/packaging-manuals/copyright-format/1.0/
|
|
||||||
Upstream-Name: libxisf
|
|
||||||
Source: https://gitea.nouspiro.space/nou/libXISF
|
|
||||||
|
|
||||||
Files: * debian/*
|
|
||||||
Copyright: (c) 2023 <Dušan Poizl nou@nouspiro.space>
|
|
||||||
License: GPL-3+
|
|
||||||
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, either 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 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/>.
|
|
||||||
.
|
|
||||||
On Debian systems, the complete texts of the GNU General Public Licenses
|
|
||||||
version 2 and 3 can be found in '/usr/share/common-licenses/GPL-2' and
|
|
||||||
'/usr/share/common-licenses/GPL-3'.
|
|
||||||
|
|
||||||
Files: lz4/*
|
|
||||||
Copyright: (c) 2011-2020, Yann Collet
|
|
||||||
License: BSD-2-clause
|
|
||||||
LZ4 Library
|
|
||||||
Copyright (c) 2011-2020, Yann Collet
|
|
||||||
All rights reserved.
|
|
||||||
.
|
|
||||||
Redistribution and use in source and binary forms, with or without modification,
|
|
||||||
are permitted provided that the following conditions are met:
|
|
||||||
.
|
|
||||||
* Redistributions of source code must retain the above copyright notice, this
|
|
||||||
list of conditions and the following disclaimer.
|
|
||||||
.
|
|
||||||
* Redistributions in binary form must reproduce the above copyright notice, this
|
|
||||||
list of conditions and the following disclaimer in the documentation and/or
|
|
||||||
other materials provided with the distribution.
|
|
||||||
.
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
||||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
|
|
||||||
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
||||||
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
|
|
||||||
Vendored
-2
@@ -1,2 +0,0 @@
|
|||||||
usr/include
|
|
||||||
usr/lib/*/*.so
|
|
||||||
Vendored
-1
@@ -1 +0,0 @@
|
|||||||
usr/lib/*/*.so.*
|
|
||||||
Vendored
-10
@@ -1,10 +0,0 @@
|
|||||||
#!/usr/bin/make -f
|
|
||||||
|
|
||||||
#!/usr/bin/make -f
|
|
||||||
|
|
||||||
include /usr/share/cdbs/1/rules/debhelper.mk
|
|
||||||
include /usr/share/cdbs/1/class/cmake.mk
|
|
||||||
|
|
||||||
export CFLAGS := $(shell dpkg-buildflags --get CFLAGS)
|
|
||||||
export CXXFLAGS := $(shell dpkg-buildflags --get CXXFLAGS)
|
|
||||||
export LDFLAGS := $(shell dpkg-buildflags --get LDFLAGS)
|
|
||||||
Vendored
-1
@@ -1 +0,0 @@
|
|||||||
3.0 (quilt)
|
|
||||||
+118
-22
@@ -27,6 +27,10 @@
|
|||||||
#include <lz4hc.h>
|
#include <lz4hc.h>
|
||||||
#include <pugixml.hpp>
|
#include <pugixml.hpp>
|
||||||
#include <zlib.h>
|
#include <zlib.h>
|
||||||
|
#ifdef HAVE_ZSTD
|
||||||
|
#include <zstd.h>
|
||||||
|
#endif
|
||||||
|
#include "streambuffer.h"
|
||||||
|
|
||||||
namespace LibXISF
|
namespace LibXISF
|
||||||
{
|
{
|
||||||
@@ -82,12 +86,12 @@ static void byteShuffle(ByteArray &data, int itemSize)
|
|||||||
{
|
{
|
||||||
ByteArray &input = data;
|
ByteArray &input = data;
|
||||||
ByteArray output(input.size());
|
ByteArray output(input.size());
|
||||||
int num = input.size() / itemSize;
|
size_t num = input.size() / itemSize;
|
||||||
char *s = output.data();
|
char *s = output.data();
|
||||||
for(int i=0; i<itemSize; i++)
|
for(int i=0; i<itemSize; i++)
|
||||||
{
|
{
|
||||||
const char *u = input.constData() + i;
|
const char *u = input.constData() + i;
|
||||||
for(int o=0; o<num; o++, s++, u += itemSize)
|
for(size_t o=0; o<num; o++, s++, u += itemSize)
|
||||||
*s = *u;
|
*s = *u;
|
||||||
}
|
}
|
||||||
memcpy(s, input.constData() + num * itemSize, input.size() % itemSize);
|
memcpy(s, input.constData() + num * itemSize, input.size() % itemSize);
|
||||||
@@ -101,12 +105,12 @@ static void byteUnshuffle(ByteArray &data, int itemSize)
|
|||||||
{
|
{
|
||||||
ByteArray &input = data;
|
ByteArray &input = data;
|
||||||
ByteArray output(input.size());
|
ByteArray output(input.size());
|
||||||
int num = input.size() / itemSize;
|
size_t num = input.size() / itemSize;
|
||||||
const char *s = input.constData();
|
const char *s = input.constData();
|
||||||
for(int i=0; i<itemSize; i++)
|
for(int i=0; i<itemSize; i++)
|
||||||
{
|
{
|
||||||
char *u = output.data() + i;
|
char *u = output.data() + i;
|
||||||
for(int o=0; o<num; o++, s++, u += itemSize)
|
for(size_t o=0; o<num; o++, s++, u += itemSize)
|
||||||
*u = *s;
|
*u = *s;
|
||||||
}
|
}
|
||||||
memcpy(output.data() + num * itemSize, s, input.size() % itemSize);
|
memcpy(output.data() + num * itemSize, s, input.size() % itemSize);
|
||||||
@@ -141,6 +145,15 @@ void DataBlock::decompress(const ByteArray &input, const String &encoding)
|
|||||||
if(LZ4_decompress_safe(tmp.constData(), data.data(), tmp.size(), data.size()) < 0)
|
if(LZ4_decompress_safe(tmp.constData(), data.data(), tmp.size(), data.size()) < 0)
|
||||||
throw Error("LZ4 decompression failed");
|
throw Error("LZ4 decompression failed");
|
||||||
break;
|
break;
|
||||||
|
case ZSTD:
|
||||||
|
#ifdef HAVE_ZSTD
|
||||||
|
data.resize(uncompressedSize);
|
||||||
|
if(ZSTD_isError(ZSTD_decompress(data.data(), data.size(), tmp.constData(), tmp.size())))
|
||||||
|
throw Error("ZSTD decompression failed");
|
||||||
|
#else
|
||||||
|
throw Error("ZSTD support not compiled");
|
||||||
|
#endif
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
byteUnshuffle(data, byteShuffling);
|
byteUnshuffle(data, byteShuffling);
|
||||||
@@ -191,7 +204,32 @@ void DataBlock::compress(int sampleFormatSize)
|
|||||||
data.resize(compSize);
|
data.resize(compSize);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case ZSTD:
|
||||||
|
{
|
||||||
|
#ifdef HAVE_ZSTD
|
||||||
|
size_t compSize = 0;
|
||||||
|
data.resize(ZSTD_compressBound(uncompressedSize));
|
||||||
|
compSize = ZSTD_compress(data.data(), data.size(), tmp.data(), tmp.size(), compressLevel < 0 ? ZSTD_CLEVEL_DEFAULT : compressLevel);
|
||||||
|
if(ZSTD_isError(compSize))
|
||||||
|
throw Error("ZSTD compression failed");
|
||||||
|
|
||||||
|
data.resize(compSize);
|
||||||
|
#else
|
||||||
|
throw Error("ZSTD support not compiled");
|
||||||
|
#endif
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool DataBlock::CompressionCodecSupported(CompressionCodec codec)
|
||||||
|
{
|
||||||
|
(void)codec;
|
||||||
|
#ifndef HAVE_ZSTD
|
||||||
|
if(codec == ZSTD)
|
||||||
|
return false;
|
||||||
|
#endif
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Property::Property(const String &_id, const char *_value) :
|
Property::Property(const String &_id, const char *_value) :
|
||||||
@@ -351,12 +389,26 @@ bool Image::addFITSKeywordAsProperty(const String &name, const String &value)
|
|||||||
{
|
{
|
||||||
auto &c = fitsNameToPropertyIdTypeConvert.at(name);
|
auto &c = fitsNameToPropertyIdTypeConvert.at(name);
|
||||||
Property prop(c.first, variantFromString(c.second, value));
|
Property prop(c.first, variantFromString(c.second, value));
|
||||||
|
|
||||||
|
if(name == "APTDIA" || name == "FOCALLEN")
|
||||||
|
prop.value.value<LibXISF::Float32>() /= 1000.0f;
|
||||||
|
|
||||||
updateProperty(prop);
|
updateProperty(prop);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
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;
|
||||||
@@ -520,6 +572,7 @@ public:
|
|||||||
* @param readPixel when false it will read pixel data from file and imageData()
|
* @param readPixel when false it will read pixel data from file and imageData()
|
||||||
* will return nullptr */
|
* will return nullptr */
|
||||||
const Image& getImage(uint32_t n, bool readPixels = true);
|
const Image& getImage(uint32_t n, bool readPixels = true);
|
||||||
|
const Image& getThumbnail();
|
||||||
private:
|
private:
|
||||||
void readXISFHeader();
|
void readXISFHeader();
|
||||||
void readSignature();
|
void readSignature();
|
||||||
@@ -529,9 +582,12 @@ 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;
|
||||||
std::vector<Property> _properties;
|
std::vector<Property> _properties;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -546,8 +602,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();
|
||||||
}
|
}
|
||||||
@@ -563,6 +619,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();
|
||||||
}
|
}
|
||||||
@@ -579,13 +636,23 @@ 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)
|
||||||
|
{
|
||||||
|
readAttachment(img._dataBlock);
|
||||||
|
}
|
||||||
|
return img;
|
||||||
|
}
|
||||||
|
|
||||||
|
const Image &XISFReaderPrivate::getThumbnail()
|
||||||
|
{
|
||||||
|
Image &img = _thumbnail;
|
||||||
|
if(_thumbnail._dataBlock.attachmentPos)
|
||||||
{
|
{
|
||||||
_io->seekg(img._dataBlock.attachmentPos);
|
_io->seekg(img._dataBlock.attachmentPos);
|
||||||
ByteArray data(img._dataBlock.attachmentSize);
|
ByteArray data(img._dataBlock.attachmentSize);
|
||||||
_io->read(data.data(), data.size());
|
_io->read(data.data(), data.size());
|
||||||
img._dataBlock.decompress(data);
|
img._dataBlock.decompress(data);
|
||||||
}
|
}
|
||||||
return img;
|
return _thumbnail;
|
||||||
}
|
}
|
||||||
|
|
||||||
void XISFReaderPrivate::readXISFHeader()
|
void XISFReaderPrivate::readXISFHeader()
|
||||||
@@ -608,6 +675,9 @@ void XISFReaderPrivate::readXISFHeader()
|
|||||||
|
|
||||||
for(auto &property : root.children("Property"))
|
for(auto &property : root.children("Property"))
|
||||||
_properties.push_back(parseProperty(property));
|
_properties.push_back(parseProperty(property));
|
||||||
|
|
||||||
|
if(root.child("Thumbnail"))
|
||||||
|
_thumbnail = parseImage(root.child("Thumbnail"));
|
||||||
}
|
}
|
||||||
else throw Error("Unknown root XML element");
|
else throw Error("Unknown root XML element");
|
||||||
}
|
}
|
||||||
@@ -634,6 +704,10 @@ void XISFReaderPrivate::parseCompression(const pugi::xml_node &node, DataBlock &
|
|||||||
dataBlock.codec = DataBlock::LZ4HC;
|
dataBlock.codec = DataBlock::LZ4HC;
|
||||||
else if(compression[0].find("lz4") == 0)
|
else if(compression[0].find("lz4") == 0)
|
||||||
dataBlock.codec = DataBlock::LZ4;
|
dataBlock.codec = DataBlock::LZ4;
|
||||||
|
#ifdef HAVE_ZSTD
|
||||||
|
else if(compression[0].find("zstd") == 0)
|
||||||
|
dataBlock.codec = DataBlock::ZSTD;
|
||||||
|
#endif
|
||||||
else
|
else
|
||||||
throw Error("Unknown compression codec");
|
throw Error("Unknown compression codec");
|
||||||
|
|
||||||
@@ -703,16 +777,9 @@ 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);
|
data = dataBlock.data;
|
||||||
_io->read(data.data(), dataBlock.attachmentSize);
|
|
||||||
dataBlock.decompress(data);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
data = dataBlock.data;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
deserializeVariant(node, property.value, data);
|
deserializeVariant(node, property.value, data);
|
||||||
@@ -781,12 +848,26 @@ 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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(node.child("Thumbnail") && std::strcmp(node.name(), "Thumbnail"))
|
||||||
|
_thumbnail = parseImage(node.child("Thumbnail"));
|
||||||
|
|
||||||
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:
|
||||||
@@ -818,10 +899,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)
|
||||||
@@ -880,7 +961,7 @@ void XISFWriterPrivate::writeHeader()
|
|||||||
offset += image._dataBlock.data.size();
|
offset += image._dataBlock.data.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t headerSize = size - sizeof(signature);
|
uint32_t headerSize = header.size() - sizeof(signature);
|
||||||
header.resize(size, 0);
|
header.resize(size, 0);
|
||||||
header.replace(8, sizeof(uint32_t), (const char*)&headerSize, sizeof(uint32_t));
|
header.replace(8, sizeof(uint32_t), (const char*)&headerSize, sizeof(uint32_t));
|
||||||
|
|
||||||
@@ -912,7 +993,7 @@ void XISFWriterPrivate::writeImageElement(pugi::xml_node &node, const Image &ima
|
|||||||
|
|
||||||
if(image._cfa.width && image._cfa.height)
|
if(image._cfa.width && image._cfa.height)
|
||||||
{
|
{
|
||||||
pugi::xml_node cfa_node = node.append_child("ColorFilterArray");
|
pugi::xml_node cfa_node = image_node.append_child("ColorFilterArray");
|
||||||
cfa_node.append_attribute("pattern").set_value(image._cfa.pattern.c_str());
|
cfa_node.append_attribute("pattern").set_value(image._cfa.pattern.c_str());
|
||||||
cfa_node.append_attribute("width").set_value(image._cfa.width);
|
cfa_node.append_attribute("width").set_value(image._cfa.width);
|
||||||
cfa_node.append_attribute("height").set_value(image._cfa.height);
|
cfa_node.append_attribute("height").set_value(image._cfa.height);
|
||||||
@@ -952,6 +1033,12 @@ void XISFWriterPrivate::writeDataBlockAttributes(pugi::xml_node &image_node, con
|
|||||||
codec = "lz4";
|
codec = "lz4";
|
||||||
else if(dataBlock.codec == DataBlock::LZ4HC)
|
else if(dataBlock.codec == DataBlock::LZ4HC)
|
||||||
codec = "lz4hc";
|
codec = "lz4hc";
|
||||||
|
else if(dataBlock.codec == DataBlock::ZSTD)
|
||||||
|
#ifdef HAVE_ZSTD
|
||||||
|
codec = "zstd";
|
||||||
|
#else
|
||||||
|
throw Error("ZSTD support not compiled");
|
||||||
|
#endif
|
||||||
|
|
||||||
if(dataBlock.byteShuffling > 1)
|
if(dataBlock.byteShuffling > 1)
|
||||||
codec += "+sh";
|
codec += "+sh";
|
||||||
@@ -1034,6 +1121,11 @@ const Image &XISFReader::getImage(uint32_t n, bool readPixels)
|
|||||||
return p->getImage(n, readPixels);
|
return p->getImage(n, readPixels);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const Image &XISFReader::getThumbnail()
|
||||||
|
{
|
||||||
|
return p->getThumbnail();
|
||||||
|
}
|
||||||
|
|
||||||
XISFWriter::XISFWriter()
|
XISFWriter::XISFWriter()
|
||||||
{
|
{
|
||||||
p = new XISFWriterPrivate;
|
p = new XISFWriterPrivate;
|
||||||
@@ -1108,6 +1200,10 @@ struct Init
|
|||||||
compressionCodecOverride = DataBlock::LZ4HC;
|
compressionCodecOverride = DataBlock::LZ4HC;
|
||||||
else if(compression.find("lz4") == 0)
|
else if(compression.find("lz4") == 0)
|
||||||
compressionCodecOverride = DataBlock::LZ4;
|
compressionCodecOverride = DataBlock::LZ4;
|
||||||
|
#ifdef HAVE_ZSTD
|
||||||
|
else if(compression.find("zstd") == 0)
|
||||||
|
compressionCodecOverride = DataBlock::ZSTD;
|
||||||
|
#endif
|
||||||
|
|
||||||
if(compression.find("+sh") != std::string::npos)
|
if(compression.find("+sh") != std::string::npos)
|
||||||
byteShuffleOverride = true;
|
byteShuffleOverride = true;
|
||||||
|
|||||||
@@ -172,7 +172,8 @@ struct LIBXISF_EXPORT DataBlock
|
|||||||
None,
|
None,
|
||||||
Zlib,
|
Zlib,
|
||||||
LZ4,
|
LZ4,
|
||||||
LZ4HC
|
LZ4HC,
|
||||||
|
ZSTD
|
||||||
};
|
};
|
||||||
bool embedded = false;
|
bool embedded = false;
|
||||||
uint32_t byteShuffling = 0;
|
uint32_t byteShuffling = 0;
|
||||||
@@ -184,6 +185,8 @@ struct LIBXISF_EXPORT DataBlock
|
|||||||
ByteArray data;
|
ByteArray data;
|
||||||
void decompress(const ByteArray &input, const std::string &encoding = "");
|
void decompress(const ByteArray &input, const std::string &encoding = "");
|
||||||
void compress(int sampleFormatSize);
|
void compress(int sampleFormatSize);
|
||||||
|
/// ZSTD compression can be disabled at compile time
|
||||||
|
static bool CompressionCodecSupported(CompressionCodec codec);
|
||||||
};
|
};
|
||||||
|
|
||||||
struct LIBXISF_EXPORT Property
|
struct LIBXISF_EXPORT Property
|
||||||
@@ -301,6 +304,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;
|
||||||
@@ -364,9 +369,14 @@ public:
|
|||||||
int imagesCount() const;
|
int imagesCount() const;
|
||||||
/** Return reference to Image
|
/** Return reference to Image
|
||||||
* @param n index of image
|
* @param n index of image
|
||||||
* @param readPixel when false it will read pixel data from file and imageData()
|
* @param readPixel when false it will not read pixel data from file and imageData()
|
||||||
* will return nullptr */
|
* will return nullptr. Other properties like width, height, format etc will be returned correctly */
|
||||||
const Image& getImage(uint32_t n, bool readPixels = true);
|
const Image& getImage(uint32_t n, bool readPixels = true);
|
||||||
|
/**
|
||||||
|
* @brief getThumbnail
|
||||||
|
* @return image thumbnail
|
||||||
|
*/
|
||||||
|
const Image& getThumbnail();
|
||||||
private:
|
private:
|
||||||
XISFReaderPrivate *p;
|
XISFReaderPrivate *p;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,12 @@
|
|||||||
|
prefix="@CMAKE_INSTALL_PREFIX@"
|
||||||
|
exec_prefix="${prefix}"
|
||||||
|
libdir="${exec_prefix}/@CMAKE_INSTALL_LIBDIR@"
|
||||||
|
includedir="${prefix}/@CMAKE_INSTALL_INCLUDEDIR@"
|
||||||
|
|
||||||
|
Name: @PROJECT_NAME@
|
||||||
|
Description: @CMAKE_PROJECT_DESCRIPTION@
|
||||||
|
URL: @CMAKE_PROJECT_HOMEPAGE_URL@
|
||||||
|
Version: @PROJECT_VERSION@
|
||||||
|
Requires.private: @PC_REQUIRE_STR@
|
||||||
|
Cflags: -I"${includedir}"
|
||||||
|
Libs: -L"${libdir}" -lXISF
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2006-2022 Arseny Kapoulkine
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person
|
||||||
|
obtaining a copy of this software and associated documentation
|
||||||
|
files (the "Software"), to deal in the Software without
|
||||||
|
restriction, including without limitation the rights to use,
|
||||||
|
copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the
|
||||||
|
Software is furnished to do so, subject to the following
|
||||||
|
conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
||||||
|
OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
||||||
|
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||||
|
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||||
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||||||
|
OTHER DEALINGS IN THE SOFTWARE.
|
||||||
@@ -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 = n < len ? 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);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -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
|
||||||
@@ -78,6 +78,17 @@ void benchmarkType(float avg, float stdDev)
|
|||||||
std::cout << "LZ4HC compression \tElapsed time: " << timer.elapsed() << " " << "ms\tSpeed: "
|
std::cout << "LZ4HC compression \tElapsed time: " << timer.elapsed() << " " << "ms\tSpeed: "
|
||||||
<< size/1024.0/1.024/timer.elapsed() << "MiB/s\tRatio: " << baseSize/xisfImage.size() << std::endl;
|
<< size/1024.0/1.024/timer.elapsed() << "MiB/s\tRatio: " << baseSize/xisfImage.size() << std::endl;
|
||||||
}
|
}
|
||||||
|
if(DataBlock::CompressionCodecSupported(DataBlock::ZSTD))
|
||||||
|
{
|
||||||
|
image.setCompression(DataBlock::ZSTD);
|
||||||
|
timer.start();
|
||||||
|
XISFWriter writer;
|
||||||
|
writer.writeImage(image);
|
||||||
|
ByteArray xisfImage;
|
||||||
|
writer.save(xisfImage);
|
||||||
|
std::cout << "ZSTD compression \tElapsed time: " << timer.elapsed() << " " << "ms\tSpeed: "
|
||||||
|
<< size/1024.0/1.024/timer.elapsed() << "MiB/s\tRatio: " << baseSize/xisfImage.size() << std::endl;
|
||||||
|
}
|
||||||
image.setByteshuffling(true);
|
image.setByteshuffling(true);
|
||||||
{
|
{
|
||||||
image.setCompression(DataBlock::Zlib);
|
image.setCompression(DataBlock::Zlib);
|
||||||
@@ -109,6 +120,17 @@ void benchmarkType(float avg, float stdDev)
|
|||||||
std::cout << "LZ4HC compression SH\tElapsed time: " << timer.elapsed() << " " << "ms\tSpeed: "
|
std::cout << "LZ4HC compression SH\tElapsed time: " << timer.elapsed() << " " << "ms\tSpeed: "
|
||||||
<< size/1024.0/1.024/timer.elapsed() << "MiB/s\tRatio: " << baseSize/xisfImage.size() << std::endl;
|
<< size/1024.0/1.024/timer.elapsed() << "MiB/s\tRatio: " << baseSize/xisfImage.size() << std::endl;
|
||||||
}
|
}
|
||||||
|
if(DataBlock::CompressionCodecSupported(DataBlock::ZSTD))
|
||||||
|
{
|
||||||
|
image.setCompression(DataBlock::ZSTD);
|
||||||
|
timer.start();
|
||||||
|
XISFWriter writer;
|
||||||
|
writer.writeImage(image);
|
||||||
|
ByteArray xisfImage;
|
||||||
|
writer.save(xisfImage);
|
||||||
|
std::cout << "ZSTD compression SH\tElapsed time: " << timer.elapsed() << " " << "ms\tSpeed: "
|
||||||
|
<< size/1024.0/1.024/timer.elapsed() << "MiB/s\tRatio: " << baseSize/xisfImage.size() << std::endl;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void benchmark()
|
void benchmark()
|
||||||
|
|||||||
+9
-2
@@ -54,7 +54,7 @@ int main(int argc, char **argv)
|
|||||||
m(0, 2) = 2;
|
m(0, 2) = 2;
|
||||||
m(1, 0) = 10;
|
m(1, 0) = 10;
|
||||||
image.addProperty(Property("UI16Matrix", m));
|
image.addProperty(Property("UI16Matrix", m));
|
||||||
std::tm tm = {12, 22, 23, 1, 2, 2023, 0, 0, 0, 0, 0};
|
std::tm tm = {12, 22, 23, 1, 2, 2023, 0, 0, 0};
|
||||||
image.addProperty(Property("TimeObs", tm));
|
image.addProperty(Property("TimeObs", tm));
|
||||||
image.addFITSKeyword({"RA", "226.9751163116387", "Right ascension of the center of the image (deg)"});
|
image.addFITSKeyword({"RA", "226.9751163116387", "Right ascension of the center of the image (deg)"});
|
||||||
image.addFITSKeyword({"DEC", "62.02302376908295", "Declination of the center of the image (deg)"});
|
image.addFITSKeyword({"DEC", "62.02302376908295", "Declination of the center of the image (deg)"});
|
||||||
@@ -94,7 +94,14 @@ int main(int argc, char **argv)
|
|||||||
TEST(image.height() != 10, "Invalid height");
|
TEST(image.height() != 10, "Invalid height");
|
||||||
TEST(image.colorSpace() != LibXISF::Image::Gray, "Invalid color space");
|
TEST(image.colorSpace() != LibXISF::Image::Gray, "Invalid color space");
|
||||||
TEST(image.pixelStorage() != LibXISF::Image::Planar, "Invalid pixel storage");
|
TEST(image.pixelStorage() != LibXISF::Image::Planar, "Invalid pixel storage");
|
||||||
TEST(image.compression() != LibXISF::DataBlock::None, "Invalid compression codec");
|
if(std::strstr(argv[1], "lz4"))
|
||||||
|
{
|
||||||
|
TEST(image.compression() != LibXISF::DataBlock::LZ4, "Invalid compression codec");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
TEST(image.compression() != LibXISF::DataBlock::None, "Invalid compression codec");
|
||||||
|
}
|
||||||
//TEST(!image.dataBlock.embedded, "Not embedded");
|
//TEST(!image.dataBlock.embedded, "Not embedded");
|
||||||
TEST(image.imageDataSize() != 80*2, "Invalid data size");
|
TEST(image.imageDataSize() != 80*2, "Invalid data size");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,7 +52,7 @@ void sha1(uint8_t *data, size_t len, uint8_t *hash)
|
|||||||
nlen += 64 - nlen % 64;
|
nlen += 64 - nlen % 64;
|
||||||
tmp.resize(nlen, 0);
|
tmp.resize(nlen, 0);
|
||||||
|
|
||||||
size_t ml = len * 8;
|
uint64_t ml = len * 8;
|
||||||
tmp[nlen - 1] = ml & 0xff;
|
tmp[nlen - 1] = ml & 0xff;
|
||||||
tmp[nlen - 2] = ml >> 8 & 0xff;
|
tmp[nlen - 2] = ml >> 8 & 0xff;
|
||||||
tmp[nlen - 3] = ml >> 16 & 0xff;
|
tmp[nlen - 3] = ml >> 16 & 0xff;
|
||||||
|
|||||||
+34
-5
@@ -21,8 +21,9 @@
|
|||||||
#include <map>
|
#include <map>
|
||||||
#include <regex>
|
#include <regex>
|
||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
|
#include <sstream>
|
||||||
#include "libxisf.h"
|
#include "libxisf.h"
|
||||||
#include "pugixml/pugixml.hpp"
|
#include <pugixml.hpp>
|
||||||
|
|
||||||
namespace LibXISF
|
namespace LibXISF
|
||||||
{
|
{
|
||||||
@@ -107,8 +108,8 @@ static std::map<Variant::Type, const char*> idToType = {
|
|||||||
{Variant::Type::UI32Matrix, "UI32Matrix"},
|
{Variant::Type::UI32Matrix, "UI32Matrix"},
|
||||||
{Variant::Type::I64Matrix, "I64Matrix"},
|
{Variant::Type::I64Matrix, "I64Matrix"},
|
||||||
{Variant::Type::UI64Matrix, "UI64Matrix"},
|
{Variant::Type::UI64Matrix, "UI64Matrix"},
|
||||||
{Variant::Type::F32Matrix, "I8Matrix"},
|
{Variant::Type::F32Matrix, "F32Matrix"},
|
||||||
{Variant::Type::F64Matrix, "UI8Matrix"},
|
{Variant::Type::F64Matrix, "F64Matrix"},
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
@@ -195,9 +196,12 @@ void deserializeVariant(const pugi::xml_node &node, Variant &variant, const Byte
|
|||||||
std::string type = node.attribute("type").as_string();
|
std::string type = node.attribute("type").as_string();
|
||||||
Variant::Type typeId = typeToId[type];
|
Variant::Type typeId = typeToId[type];
|
||||||
|
|
||||||
if(typeId == Variant::Type::String && !node.attribute("location"))
|
if(typeId == Variant::Type::String)
|
||||||
{
|
{
|
||||||
variant.setValue(node.text().as_string());
|
if(!node.attribute("location"))
|
||||||
|
variant.setValue(node.text().as_string());
|
||||||
|
else
|
||||||
|
variant.setValue(String(data.constData(), data.size()));
|
||||||
}
|
}
|
||||||
else if(node.attribute("value"))
|
else if(node.attribute("value"))
|
||||||
{
|
{
|
||||||
@@ -399,6 +403,7 @@ Variant variantFromString(Variant::Type type, const String &str)
|
|||||||
|
|
||||||
Variant::Type Variant::type() const
|
Variant::Type Variant::type() const
|
||||||
{
|
{
|
||||||
|
int idx = _value.index();
|
||||||
return (Variant::Type)_value.index();
|
return (Variant::Type)_value.index();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -429,6 +434,28 @@ String Variant::toString() const
|
|||||||
return ss.str();
|
return ss.str();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
auto matrixString = [](auto matrix) {
|
||||||
|
std::stringstream ss;
|
||||||
|
ss << "{";
|
||||||
|
for(int i=0; i<matrix.rows(); i++)
|
||||||
|
{
|
||||||
|
ss << "{";
|
||||||
|
for(int o=0; o<matrix.cols(); o++)
|
||||||
|
{
|
||||||
|
char str[128] = {0};
|
||||||
|
char *end = str + sizeof(str);
|
||||||
|
std::to_chars(str, end, matrix(i, o));
|
||||||
|
ss << str;
|
||||||
|
if(o < matrix.cols() - 1)
|
||||||
|
ss << ",";
|
||||||
|
}
|
||||||
|
ss << "}";
|
||||||
|
if(i < matrix.rows() - 1)
|
||||||
|
ss << ",";
|
||||||
|
}
|
||||||
|
return ss.str();
|
||||||
|
};
|
||||||
|
|
||||||
switch(type())
|
switch(type())
|
||||||
{
|
{
|
||||||
case Variant::Type::Int8: toChars<Int8>(_value, str, end); break;
|
case Variant::Type::Int8: toChars<Int8>(_value, str, end); break;
|
||||||
@@ -453,6 +480,8 @@ String Variant::toString() const
|
|||||||
case Variant::Type::UI64Vector: string = vectorString(std::get<UI64Vector>(_value)); break;
|
case Variant::Type::UI64Vector: string = vectorString(std::get<UI64Vector>(_value)); break;
|
||||||
case Variant::Type::F32Vector: string = vectorString(std::get<F32Vector>(_value)); break;
|
case Variant::Type::F32Vector: string = vectorString(std::get<F32Vector>(_value)); break;
|
||||||
case Variant::Type::F64Vector: string = vectorString(std::get<F64Vector>(_value)); break;
|
case Variant::Type::F64Vector: string = vectorString(std::get<F64Vector>(_value)); break;
|
||||||
|
case Variant::Type::F32Matrix: string = matrixString(std::get<F32Matrix>(_value)); break;
|
||||||
|
case Variant::Type::F64Matrix: string = matrixString(std::get<F64Matrix>(_value)); break;
|
||||||
case Variant::Type::String: string = std::get<String>(_value); break;
|
case Variant::Type::String: string = std::get<String>(_value); break;
|
||||||
case Variant::Type::TimePoint:
|
case Variant::Type::TimePoint:
|
||||||
{
|
{
|
||||||
|
|||||||
Binary file not shown.
Reference in New Issue
Block a user