mirror of
https://github.com/opencv/opencv.git
synced 2025-01-12 07:42:32 +08:00
b14ea19466
core: persistence: output reals as human-friendly expression. #25351 Close #25073 Related https://github.com/opencv/opencv/pull/25087 This patch is need to merge same time with https://github.com/opencv/opencv_contrib/pull/3714 ### Pull Request Readiness Checklist See details at https://github.com/opencv/opencv/wiki/How_to_contribute#making-a-good-pull-request - [x] I agree to contribute to the project under Apache 2 License. - [x] To the best of my knowledge, the proposed patch is not based on a code under GPL or another license that is incompatible with OpenCV - [x] The PR is proposed to the proper branch - [x] There is a reference to the original bug report and related work - [x] There is accuracy test, performance test and test data in opencv_extra repository, if applicable Patch to opencv_extra has the same branch name. - [x] The feature is well documented and sample code can be built with the project CMake
2752 lines
82 KiB
C++
2752 lines
82 KiB
C++
// This file is part of OpenCV project.
|
|
// It is subject to the license terms in the LICENSE file found in the top-level directory
|
|
// of this distribution and at http://opencv.org/license.html
|
|
|
|
#include "precomp.hpp"
|
|
#include "persistence.hpp"
|
|
#include "persistence_impl.hpp"
|
|
#include "persistence_base64_encoding.hpp"
|
|
#include <unordered_map>
|
|
#include <iterator>
|
|
|
|
#include <opencv2/core/utils/logger.hpp>
|
|
|
|
namespace cv
|
|
{
|
|
|
|
namespace fs
|
|
{
|
|
|
|
int strcasecmp(const char* s1, const char* s2)
|
|
{
|
|
const char* dummy="";
|
|
if(!s1) s1=dummy;
|
|
if(!s2) s2=dummy;
|
|
|
|
size_t len1 = strlen(s1);
|
|
size_t len2 = strlen(s2);
|
|
size_t i, len = std::min(len1, len2);
|
|
for( i = 0; i < len; i++ )
|
|
{
|
|
int d = tolower((int)s1[i]) - tolower((int)s2[i]);
|
|
if( d != 0 )
|
|
return d;
|
|
}
|
|
return len1 < len2 ? -1 : len1 > len2 ? 1 : 0;
|
|
}
|
|
|
|
char* itoa( int _val, char* buffer, int /*radix*/ )
|
|
{
|
|
const int radix = 10;
|
|
char* ptr=buffer + 23 /* enough even for 64-bit integers */;
|
|
unsigned val = abs(_val);
|
|
|
|
*ptr = '\0';
|
|
do
|
|
{
|
|
unsigned r = val / radix;
|
|
*--ptr = (char)(val - (r*radix) + '0');
|
|
val = r;
|
|
}
|
|
while( val != 0 );
|
|
|
|
if( _val < 0 )
|
|
*--ptr = '-';
|
|
|
|
return ptr;
|
|
}
|
|
|
|
char* doubleToString( char* buf, size_t bufSize, double value, bool explicitZero )
|
|
{
|
|
Cv64suf val;
|
|
unsigned ieee754_hi;
|
|
|
|
val.f = value;
|
|
ieee754_hi = (unsigned)(val.u >> 32);
|
|
|
|
if( (ieee754_hi & 0x7ff00000) != 0x7ff00000 )
|
|
{
|
|
int ivalue = cvRound(value);
|
|
if( ivalue == value )
|
|
{
|
|
if( explicitZero )
|
|
snprintf( buf, bufSize, "%d.0", ivalue );
|
|
else
|
|
snprintf( buf, bufSize, "%d.", ivalue );
|
|
}
|
|
else
|
|
{
|
|
// binary64 has 52 bit fraction with hidden bit.
|
|
// 53 * log_10(2) is 15.955. So "%.16f" should be fine, but its test fails.
|
|
snprintf( buf, bufSize, "%.17g", value );
|
|
|
|
char* ptr = buf;
|
|
if( *ptr == '+' || *ptr == '-' )
|
|
ptr++;
|
|
for( ; cv_isdigit(*ptr); ptr++ )
|
|
;
|
|
if( *ptr == ',' )
|
|
*ptr = '.';
|
|
}
|
|
}
|
|
else
|
|
{
|
|
unsigned ieee754_lo = (unsigned)val.u;
|
|
if( (ieee754_hi & 0x7fffffff) + (ieee754_lo != 0) > 0x7ff00000 )
|
|
strcpy( buf, ".Nan" );
|
|
else
|
|
strcpy( buf, (int)ieee754_hi < 0 ? "-.Inf" : ".Inf" );
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
char* floatToString( char* buf, size_t bufSize, float value, bool halfprecision, bool explicitZero )
|
|
{
|
|
Cv32suf val;
|
|
unsigned ieee754;
|
|
val.f = value;
|
|
ieee754 = val.u;
|
|
|
|
if( (ieee754 & 0x7f800000) != 0x7f800000 )
|
|
{
|
|
int ivalue = cvRound(value);
|
|
if( ivalue == value )
|
|
{
|
|
if( explicitZero )
|
|
snprintf( buf, bufSize, "%d.0", ivalue );
|
|
else
|
|
snprintf( buf, bufSize, "%d.", ivalue );
|
|
}
|
|
else
|
|
{
|
|
if (halfprecision)
|
|
{
|
|
// bfloat16 has 7 bit fraction with hidden bit.
|
|
// binary16 has 10 bit fraction with hidden bit.
|
|
// 11 * log_10(2) is 3.311. So "%.4f" should be fine, but its test fails.
|
|
snprintf(buf, bufSize, "%.5g", value);
|
|
}
|
|
else
|
|
{
|
|
// binray32 has 23 bit fraction with hidden bit.
|
|
// 24 * log_10(2) is 7.225. So "%.8f" should be fine, but its test fails.
|
|
snprintf(buf, bufSize, "%.9g", value);
|
|
}
|
|
|
|
char* ptr = buf;
|
|
if( *ptr == '+' || *ptr == '-' )
|
|
ptr++;
|
|
for( ; cv_isdigit(*ptr); ptr++ )
|
|
;
|
|
if( *ptr == ',' )
|
|
*ptr = '.';
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if( (ieee754 & 0x7fffffff) != 0x7f800000 )
|
|
strcpy( buf, ".Nan" );
|
|
else
|
|
strcpy( buf, (int)ieee754 < 0 ? "-.Inf" : ".Inf" );
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
static const char symbols[9] = "ucwsifdh";
|
|
|
|
static char typeSymbol(int depth)
|
|
{
|
|
CV_StaticAssert(CV_64F == 6, "");
|
|
CV_CheckDepth(depth, depth >=0 && depth <= CV_16F, "");
|
|
return symbols[depth];
|
|
}
|
|
|
|
static int symbolToType(char c)
|
|
{
|
|
if (c == 'r')
|
|
return CV_SEQ_ELTYPE_PTR;
|
|
const char* pos = strchr( symbols, c );
|
|
if( !pos )
|
|
CV_Error( cv::Error::StsBadArg, "Invalid data type specification" );
|
|
return static_cast<int>(pos - symbols);
|
|
}
|
|
|
|
char* encodeFormat(int elem_type, char* dt, size_t dt_len)
|
|
{
|
|
int cn = (elem_type == CV_SEQ_ELTYPE_PTR/*CV_USRTYPE1*/) ? 1 : CV_MAT_CN(elem_type);
|
|
char symbol = (elem_type == CV_SEQ_ELTYPE_PTR/*CV_USRTYPE1*/) ? 'r' : typeSymbol(CV_MAT_DEPTH(elem_type));
|
|
snprintf(dt, dt_len, "%d%c", cn, symbol);
|
|
return dt + (cn == 1 ? 1 : 0);
|
|
}
|
|
|
|
// Deprecated due to size of dt buffer being unknowable.
|
|
char* encodeFormat(int elem_type, char* dt)
|
|
{
|
|
constexpr size_t max = 20+1+1; // UINT64_MAX + one char + nul termination.
|
|
return encodeFormat(elem_type, dt, max);
|
|
}
|
|
|
|
int decodeFormat( const char* dt, int* fmt_pairs, int max_len )
|
|
{
|
|
int fmt_pair_count = 0;
|
|
int i = 0, k = 0, len = dt ? (int)strlen(dt) : 0;
|
|
|
|
if( !dt || !len )
|
|
return 0;
|
|
|
|
CV_Assert( fmt_pairs != 0 && max_len > 0 );
|
|
fmt_pairs[0] = 0;
|
|
max_len *= 2;
|
|
|
|
for( ; k < len; k++ )
|
|
{
|
|
char c = dt[k];
|
|
|
|
if( cv_isdigit(c) )
|
|
{
|
|
int count = c - '0';
|
|
if( cv_isdigit(dt[k+1]) )
|
|
{
|
|
char* endptr = 0;
|
|
count = (int)strtol( dt+k, &endptr, 10 );
|
|
k = (int)(endptr - dt) - 1;
|
|
}
|
|
|
|
if( count <= 0 )
|
|
CV_Error( cv::Error::StsBadArg, "Invalid data type specification" );
|
|
|
|
fmt_pairs[i] = count;
|
|
}
|
|
else
|
|
{
|
|
int depth = symbolToType(c);
|
|
if( fmt_pairs[i] == 0 )
|
|
fmt_pairs[i] = 1;
|
|
fmt_pairs[i+1] = depth;
|
|
if( i > 0 && fmt_pairs[i+1] == fmt_pairs[i-1] )
|
|
fmt_pairs[i-2] += fmt_pairs[i];
|
|
else
|
|
{
|
|
i += 2;
|
|
if( i >= max_len )
|
|
CV_Error( cv::Error::StsBadArg, "Too long data type specification" );
|
|
}
|
|
fmt_pairs[i] = 0;
|
|
}
|
|
}
|
|
|
|
fmt_pair_count = i/2;
|
|
return fmt_pair_count;
|
|
}
|
|
|
|
int calcElemSize( const char* dt, int initial_size )
|
|
{
|
|
int size = 0;
|
|
int fmt_pairs[CV_FS_MAX_FMT_PAIRS], i, fmt_pair_count;
|
|
int comp_size;
|
|
|
|
fmt_pair_count = decodeFormat( dt, fmt_pairs, CV_FS_MAX_FMT_PAIRS );
|
|
fmt_pair_count *= 2;
|
|
for( i = 0, size = initial_size; i < fmt_pair_count; i += 2 )
|
|
{
|
|
comp_size = CV_ELEM_SIZE(fmt_pairs[i+1]);
|
|
size = cvAlign( size, comp_size );
|
|
size += comp_size * fmt_pairs[i];
|
|
}
|
|
if( initial_size == 0 )
|
|
{
|
|
comp_size = CV_ELEM_SIZE(fmt_pairs[1]);
|
|
size = cvAlign( size, comp_size );
|
|
}
|
|
return size;
|
|
}
|
|
|
|
|
|
int calcStructSize( const char* dt, int initial_size )
|
|
{
|
|
int size = calcElemSize( dt, initial_size );
|
|
size_t elem_max_size = 0;
|
|
for ( const char * type = dt; *type != '\0'; type++ )
|
|
{
|
|
char v = *type;
|
|
if (v >= '0' && v <= '9')
|
|
continue; // skip vector size
|
|
switch (v)
|
|
{
|
|
case 'u': { elem_max_size = std::max( elem_max_size, sizeof(uchar ) ); break; }
|
|
case 'c': { elem_max_size = std::max( elem_max_size, sizeof(schar ) ); break; }
|
|
case 'w': { elem_max_size = std::max( elem_max_size, sizeof(ushort) ); break; }
|
|
case 's': { elem_max_size = std::max( elem_max_size, sizeof(short ) ); break; }
|
|
case 'i': { elem_max_size = std::max( elem_max_size, sizeof(int ) ); break; }
|
|
case 'f': { elem_max_size = std::max( elem_max_size, sizeof(float ) ); break; }
|
|
case 'd': { elem_max_size = std::max( elem_max_size, sizeof(double) ); break; }
|
|
case 'h': { elem_max_size = std::max(elem_max_size, sizeof(hfloat)); break; }
|
|
default:
|
|
CV_Error_(Error::StsNotImplemented, ("Unknown type identifier: '%c' in '%s'", (char)(*type), dt));
|
|
}
|
|
}
|
|
size = cvAlign( size, static_cast<int>(elem_max_size) );
|
|
return size;
|
|
}
|
|
|
|
int decodeSimpleFormat( const char* dt )
|
|
{
|
|
int elem_type = -1;
|
|
int fmt_pairs[CV_FS_MAX_FMT_PAIRS], fmt_pair_count;
|
|
|
|
fmt_pair_count = decodeFormat( dt, fmt_pairs, CV_FS_MAX_FMT_PAIRS );
|
|
if( fmt_pair_count != 1 || fmt_pairs[0] >= CV_CN_MAX)
|
|
CV_Error( cv::Error::StsError, "Too complex format for the matrix" );
|
|
|
|
elem_type = CV_MAKETYPE( fmt_pairs[1], fmt_pairs[0] );
|
|
|
|
return elem_type;
|
|
}
|
|
|
|
}
|
|
|
|
#if defined __i386__ || defined(_M_IX86) || defined __x86_64__ || defined(_M_X64) || \
|
|
(defined (__LITTLE_ENDIAN__) && __LITTLE_ENDIAN__)
|
|
#define CV_LITTLE_ENDIAN_MEM_ACCESS 1
|
|
#else
|
|
#define CV_LITTLE_ENDIAN_MEM_ACCESS 0
|
|
#endif
|
|
|
|
static inline int readInt(const uchar* p)
|
|
{
|
|
// On little endian CPUs, both branches produce the same result. On big endian, only the else branch does.
|
|
#if CV_LITTLE_ENDIAN_MEM_ACCESS
|
|
int val;
|
|
memcpy(&val, p, sizeof(val));
|
|
return val;
|
|
#else
|
|
int val = (int)(p[0] | (p[1] << 8) | (p[2] << 16) | (p[3] << 24));
|
|
return val;
|
|
#endif
|
|
}
|
|
|
|
static inline double readReal(const uchar* p)
|
|
{
|
|
// On little endian CPUs, both branches produce the same result. On big endian, only the else branch does.
|
|
#if CV_LITTLE_ENDIAN_MEM_ACCESS
|
|
double val;
|
|
memcpy(&val, p, sizeof(val));
|
|
return val;
|
|
#else
|
|
unsigned val0 = (unsigned)(p[0] | (p[1] << 8) | (p[2] << 16) | (p[3] << 24));
|
|
unsigned val1 = (unsigned)(p[4] | (p[5] << 8) | (p[6] << 16) | (p[7] << 24));
|
|
Cv64suf val;
|
|
val.u = val0 | ((uint64)val1 << 32);
|
|
return val.f;
|
|
#endif
|
|
}
|
|
|
|
static inline void writeInt(uchar* p, int ival)
|
|
{
|
|
// On little endian CPUs, both branches produce the same result. On big endian, only the else branch does.
|
|
#if CV_LITTLE_ENDIAN_MEM_ACCESS
|
|
memcpy(p, &ival, sizeof(ival));
|
|
#else
|
|
p[0] = (uchar)ival;
|
|
p[1] = (uchar)(ival >> 8);
|
|
p[2] = (uchar)(ival >> 16);
|
|
p[3] = (uchar)(ival >> 24);
|
|
#endif
|
|
}
|
|
|
|
static inline void writeReal(uchar* p, double fval)
|
|
{
|
|
// On little endian CPUs, both branches produce the same result. On big endian, only the else branch does.
|
|
#if CV_LITTLE_ENDIAN_MEM_ACCESS
|
|
memcpy(p, &fval, sizeof(fval));
|
|
#else
|
|
Cv64suf v;
|
|
v.f = fval;
|
|
p[0] = (uchar)v.u;
|
|
p[1] = (uchar)(v.u >> 8);
|
|
p[2] = (uchar)(v.u >> 16);
|
|
p[3] = (uchar)(v.u >> 24);
|
|
p[4] = (uchar)(v.u >> 32);
|
|
p[5] = (uchar)(v.u >> 40);
|
|
p[6] = (uchar)(v.u >> 48);
|
|
p[7] = (uchar)(v.u >> 56);
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
void FileStorage::Impl::init() {
|
|
flags = 0;
|
|
buffer.clear();
|
|
bufofs = 0;
|
|
state = UNDEFINED;
|
|
is_using_base64 = false;
|
|
state_of_writing_base64 = FileStorage_API::Base64State::Uncertain;
|
|
is_write_struct_delayed = false;
|
|
delayed_struct_key = nullptr;
|
|
delayed_struct_flags = 0;
|
|
delayed_type_name = nullptr;
|
|
base64_writer = nullptr;
|
|
is_opened = false;
|
|
dummy_eof = false;
|
|
write_mode = false;
|
|
mem_mode = false;
|
|
space = 0;
|
|
wrap_margin = 71;
|
|
fmt = 0;
|
|
file = 0;
|
|
gzfile = 0;
|
|
empty_stream = true;
|
|
|
|
strbufv.clear();
|
|
strbuf = 0;
|
|
strbufsize = strbufpos = 0;
|
|
roots.clear();
|
|
|
|
fs_data.clear();
|
|
fs_data_ptrs.clear();
|
|
fs_data_blksz.clear();
|
|
freeSpaceOfs = 0;
|
|
|
|
str_hash.clear();
|
|
str_hash_data.clear();
|
|
str_hash_data.resize(1);
|
|
str_hash_data[0] = '\0';
|
|
|
|
filename.clear();
|
|
lineno = 0;
|
|
}
|
|
|
|
FileStorage::Impl::Impl(FileStorage *_fs) {
|
|
fs_ext = _fs;
|
|
init();
|
|
}
|
|
|
|
FileStorage::Impl::~Impl() {
|
|
release();
|
|
}
|
|
|
|
void FileStorage::Impl::release(String *out) {
|
|
if (is_opened) {
|
|
if (out)
|
|
out->clear();
|
|
if (write_mode) {
|
|
while (write_stack.size() > 1) {
|
|
endWriteStruct();
|
|
}
|
|
flush();
|
|
if (fmt == FileStorage::FORMAT_XML)
|
|
puts("</opencv_storage>\n");
|
|
else if (fmt == FileStorage::FORMAT_JSON)
|
|
puts("}\n");
|
|
}
|
|
if (mem_mode && out) {
|
|
*out = cv::String(outbuf.begin(), outbuf.end());
|
|
}
|
|
}
|
|
closeFile();
|
|
init();
|
|
}
|
|
|
|
void FileStorage::Impl::analyze_file_name(const std::string &file_name, std::vector<std::string> ¶ms) {
|
|
params.clear();
|
|
static const char not_file_name = '\n';
|
|
static const char parameter_begin = '?';
|
|
static const char parameter_separator = '&';
|
|
|
|
if (file_name.find(not_file_name, (size_t) 0) != std::string::npos)
|
|
return;
|
|
|
|
size_t beg = file_name.find_last_of(parameter_begin);
|
|
params.push_back(file_name.substr((size_t) 0, beg));
|
|
|
|
if (beg != std::string::npos) {
|
|
size_t end = file_name.size();
|
|
beg++;
|
|
for (size_t param_beg = beg, param_end = beg;
|
|
param_end < end;
|
|
param_beg = param_end + 1) {
|
|
param_end = file_name.find_first_of(parameter_separator, param_beg);
|
|
if ((param_end == std::string::npos || param_end != param_beg) && param_beg + 1 < end) {
|
|
params.push_back(file_name.substr(param_beg, param_end - param_beg));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool FileStorage::Impl::open(const char *filename_or_buf, int _flags, const char *encoding) {
|
|
bool ok = true;
|
|
release();
|
|
|
|
bool append = (_flags & 3) == FileStorage::APPEND;
|
|
mem_mode = (_flags & FileStorage::MEMORY) != 0;
|
|
|
|
write_mode = (_flags & 3) != 0;
|
|
bool write_base64 = (write_mode || append) && (_flags & FileStorage::BASE64) != 0;
|
|
|
|
bool isGZ = false;
|
|
size_t fnamelen = 0;
|
|
|
|
std::vector<std::string> params;
|
|
//if ( !mem_mode )
|
|
{
|
|
analyze_file_name(filename_or_buf, params);
|
|
if (!params.empty())
|
|
filename = params[0];
|
|
|
|
if (!write_base64 && params.size() >= 2 &&
|
|
std::find(params.begin() + 1, params.end(), std::string("base64")) != params.end())
|
|
write_base64 = (write_mode || append);
|
|
}
|
|
|
|
if (filename.size() == 0 && !mem_mode && !write_mode)
|
|
CV_Error(cv::Error::StsNullPtr, "NULL or empty filename");
|
|
|
|
if (mem_mode && append)
|
|
CV_Error(cv::Error::StsBadFlag, "FileStorage::APPEND and FileStorage::MEMORY are not currently compatible");
|
|
|
|
flags = _flags;
|
|
|
|
if (!mem_mode) {
|
|
char *dot_pos = strrchr((char *) filename.c_str(), '.');
|
|
char compression = '\0';
|
|
|
|
if (dot_pos && dot_pos[1] == 'g' && dot_pos[2] == 'z' &&
|
|
(dot_pos[3] == '\0' || (cv_isdigit(dot_pos[3]) && dot_pos[4] == '\0'))) {
|
|
if (append) {
|
|
CV_Error(cv::Error::StsNotImplemented, "Appending data to compressed file is not implemented");
|
|
}
|
|
isGZ = true;
|
|
compression = dot_pos[3];
|
|
if (compression)
|
|
dot_pos[3] = '\0', fnamelen--;
|
|
}
|
|
|
|
if (!isGZ) {
|
|
file = fopen(filename.c_str(), !write_mode ? "rt" : !append ? "wt" : "a+t");
|
|
if (!file)
|
|
{
|
|
CV_LOG_ERROR(NULL, "Can't open file: '" << filename << "' in " << (!write_mode ? "read" : !append ? "write" : "append") << " mode");
|
|
return false;
|
|
}
|
|
} else {
|
|
#if USE_ZLIB
|
|
char mode[] = {write_mode ? 'w' : 'r', 'b', compression ? compression : '3', '\0'};
|
|
gzfile = gzopen(filename.c_str(), mode);
|
|
if (!gzfile)
|
|
{
|
|
CV_LOG_ERROR(NULL, "Can't open archive: '" << filename << "' mode=" << mode);
|
|
return false;
|
|
}
|
|
#else
|
|
CV_Error(cv::Error::StsNotImplemented, "There is no compressed file storage support in this configuration");
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// FIXIT release() must do that, use CV_Assert() here instead
|
|
roots.clear();
|
|
fs_data.clear();
|
|
|
|
wrap_margin = 71;
|
|
fmt = FileStorage::FORMAT_AUTO;
|
|
|
|
if (write_mode) {
|
|
fmt = flags & FileStorage::FORMAT_MASK;
|
|
|
|
if (mem_mode)
|
|
outbuf.clear();
|
|
|
|
if (fmt == FileStorage::FORMAT_AUTO && !filename.empty()) {
|
|
const char *dot_pos = NULL;
|
|
const char *dot_pos2 = NULL;
|
|
// like strrchr() implementation, but save two last positions simultaneously
|
|
for (const char *pos = &filename[0]; pos[0] != 0; pos++) {
|
|
if (pos[0] == '.') {
|
|
dot_pos2 = dot_pos;
|
|
dot_pos = pos;
|
|
}
|
|
}
|
|
if (fs::strcasecmp(dot_pos, ".gz") == 0 && dot_pos2 != NULL) {
|
|
dot_pos = dot_pos2;
|
|
}
|
|
fmt = (fs::strcasecmp(dot_pos, ".xml") == 0 || fs::strcasecmp(dot_pos, ".xml.gz") == 0)
|
|
? FileStorage::FORMAT_XML
|
|
: (fs::strcasecmp(dot_pos, ".json") == 0 || fs::strcasecmp(dot_pos, ".json.gz") == 0)
|
|
? FileStorage::FORMAT_JSON
|
|
: FileStorage::FORMAT_YAML;
|
|
} else if (fmt == FileStorage::FORMAT_AUTO) {
|
|
fmt = FileStorage::FORMAT_XML;
|
|
}
|
|
|
|
// we use factor=6 for XML (the longest characters (' and ") are encoded with 6 bytes (' and ")
|
|
// and factor=4 for YAML ( as we use 4 bytes for non ASCII characters (e.g. \xAB))
|
|
int buf_size = CV_FS_MAX_LEN * (fmt == FileStorage::FORMAT_XML ? 6 : 4) + 1024;
|
|
|
|
if (append) {
|
|
fseek(file, 0, SEEK_END);
|
|
if (ftell(file) == 0)
|
|
append = false;
|
|
}
|
|
|
|
write_stack.clear();
|
|
empty_stream = true;
|
|
write_stack.push_back(FStructData("", FileNode::MAP | FileNode::EMPTY, 0));
|
|
buffer.reserve(buf_size + 1024);
|
|
buffer.resize(buf_size);
|
|
bufofs = 0;
|
|
is_using_base64 = write_base64;
|
|
state_of_writing_base64 = FileStorage_API::Base64State::Uncertain;
|
|
|
|
if (fmt == FileStorage::FORMAT_XML) {
|
|
size_t file_size = file ? (size_t) ftell(file) : (size_t) 0;
|
|
if (!append || file_size == 0) {
|
|
if (encoding && *encoding != '\0') {
|
|
if (fs::strcasecmp(encoding, "UTF-16") == 0) {
|
|
release();
|
|
CV_Error(cv::Error::StsBadArg, "UTF-16 XML encoding is not supported! Use 8-bit encoding\n");
|
|
}
|
|
|
|
CV_Assert(strlen(encoding) < 1000);
|
|
char buf[1100];
|
|
snprintf(buf, sizeof(buf), "<?xml version=\"1.0\" encoding=\"%s\"?>\n", encoding);
|
|
puts(buf);
|
|
} else
|
|
puts("<?xml version=\"1.0\"?>\n");
|
|
puts("<opencv_storage>\n");
|
|
} else {
|
|
int xml_buf_size = 1 << 10;
|
|
char substr[] = "</opencv_storage>";
|
|
int last_occurrence = -1;
|
|
xml_buf_size = MIN(xml_buf_size, int(file_size));
|
|
fseek(file, -xml_buf_size, SEEK_END);
|
|
// find the last occurrence of </opencv_storage>
|
|
for (;;) {
|
|
int line_offset = (int) ftell(file);
|
|
const char *ptr0 = this->gets(xml_buf_size);
|
|
const char *ptr = NULL;
|
|
if (!ptr0)
|
|
break;
|
|
ptr = ptr0;
|
|
for (;;) {
|
|
ptr = strstr(ptr, substr);
|
|
if (!ptr)
|
|
break;
|
|
last_occurrence = line_offset + (int) (ptr - ptr0);
|
|
ptr += strlen(substr);
|
|
}
|
|
}
|
|
if (last_occurrence < 0) {
|
|
release();
|
|
CV_Error(cv::Error::StsError, "Could not find </opencv_storage> in the end of file.\n");
|
|
}
|
|
closeFile();
|
|
file = fopen(filename.c_str(), "r+t");
|
|
CV_Assert(file != 0);
|
|
fseek(file, last_occurrence, SEEK_SET);
|
|
// replace the last "</opencv_storage>" with " <!-- resumed -->", which has the same length
|
|
puts(" <!-- resumed -->");
|
|
fseek(file, 0, SEEK_END);
|
|
puts("\n");
|
|
}
|
|
|
|
emitter_do_not_use_direct_dereference = createXMLEmitter(this);
|
|
} else if (fmt == FileStorage::FORMAT_YAML) {
|
|
if (!append)
|
|
puts("%YAML:1.0\n---\n");
|
|
else
|
|
puts("...\n---\n");
|
|
|
|
emitter_do_not_use_direct_dereference = createYAMLEmitter(this);
|
|
} else {
|
|
CV_Assert(fmt == FileStorage::FORMAT_JSON);
|
|
if (!append)
|
|
puts("{\n");
|
|
else {
|
|
bool valid = false;
|
|
long roffset = 0;
|
|
for (;
|
|
fseek(file, roffset, SEEK_END) == 0;
|
|
roffset -= 1) {
|
|
const char end_mark = '}';
|
|
if (fgetc(file) == end_mark) {
|
|
fseek(file, roffset, SEEK_END);
|
|
valid = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (valid) {
|
|
closeFile();
|
|
file = fopen(filename.c_str(), "r+t");
|
|
CV_Assert(file != 0);
|
|
fseek(file, roffset, SEEK_END);
|
|
fputs(",", file);
|
|
} else {
|
|
CV_Error(cv::Error::StsError, "Could not find '}' in the end of file.\n");
|
|
}
|
|
}
|
|
write_stack.back().indent = 4;
|
|
emitter_do_not_use_direct_dereference = createJSONEmitter(this);
|
|
}
|
|
is_opened = true;
|
|
} else {
|
|
const size_t buf_size0 = 40;
|
|
buffer.resize(buf_size0);
|
|
if (mem_mode) {
|
|
strbuf = (char *) filename_or_buf;
|
|
strbufsize = strlen(strbuf);
|
|
}
|
|
|
|
const char *yaml_signature = "%YAML";
|
|
const char *json_signature = "{";
|
|
const char *xml_signature = "<?xml";
|
|
char *buf = this->gets(16);
|
|
CV_Assert(buf);
|
|
char *bufPtr = cv_skip_BOM(buf);
|
|
size_t bufOffset = bufPtr - buf;
|
|
|
|
if (strncmp(bufPtr, yaml_signature, strlen(yaml_signature)) == 0)
|
|
fmt = FileStorage::FORMAT_YAML;
|
|
else if (strncmp(bufPtr, json_signature, strlen(json_signature)) == 0)
|
|
fmt = FileStorage::FORMAT_JSON;
|
|
else if (strncmp(bufPtr, xml_signature, strlen(xml_signature)) == 0)
|
|
fmt = FileStorage::FORMAT_XML;
|
|
else if (strbufsize == bufOffset)
|
|
CV_Error(cv::Error::StsBadArg, "Input file is invalid");
|
|
else
|
|
CV_Error(cv::Error::StsBadArg, "Unsupported file storage format");
|
|
|
|
rewind();
|
|
strbufpos = bufOffset;
|
|
bufofs = 0;
|
|
|
|
try {
|
|
char *ptr = bufferStart();
|
|
ptr[0] = ptr[1] = ptr[2] = '\0';
|
|
FileNode root_nodes(fs_ext, 0, 0);
|
|
|
|
uchar *rptr = reserveNodeSpace(root_nodes, 9);
|
|
*rptr = FileNode::SEQ;
|
|
writeInt(rptr + 1, 4);
|
|
writeInt(rptr + 5, 0);
|
|
|
|
roots.clear();
|
|
|
|
switch (fmt) {
|
|
case FileStorage::FORMAT_XML:
|
|
parser_do_not_use_direct_dereference = createXMLParser(this);
|
|
break;
|
|
case FileStorage::FORMAT_YAML:
|
|
parser_do_not_use_direct_dereference = createYAMLParser(this);
|
|
break;
|
|
case FileStorage::FORMAT_JSON:
|
|
parser_do_not_use_direct_dereference = createJSONParser(this);
|
|
break;
|
|
default:
|
|
parser_do_not_use_direct_dereference = Ptr<FileStorageParser>();
|
|
}
|
|
|
|
if (!parser_do_not_use_direct_dereference.empty()) {
|
|
ok = getParser().parse(ptr);
|
|
if (ok) {
|
|
finalizeCollection(root_nodes);
|
|
|
|
CV_Assert(!fs_data_ptrs.empty());
|
|
FileNode roots_node(fs_ext, 0, 0);
|
|
size_t i, nroots = roots_node.size();
|
|
FileNodeIterator it = roots_node.begin();
|
|
|
|
for (i = 0; i < nroots; i++, ++it)
|
|
roots.push_back(*it);
|
|
}
|
|
}
|
|
}
|
|
catch (...)
|
|
{
|
|
// FIXIT log error message
|
|
is_opened = true;
|
|
release();
|
|
throw;
|
|
}
|
|
|
|
// release resources that we do not need anymore
|
|
closeFile();
|
|
is_opened = true;
|
|
std::vector<char> tmpbuf;
|
|
std::swap(buffer, tmpbuf);
|
|
bufofs = 0;
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
void FileStorage::Impl::puts(const char *str) {
|
|
CV_Assert(write_mode);
|
|
if (mem_mode)
|
|
std::copy(str, str + strlen(str), std::back_inserter(outbuf));
|
|
else if (file)
|
|
fputs(str, file);
|
|
#if USE_ZLIB
|
|
else if (gzfile)
|
|
gzputs(gzfile, str);
|
|
#endif
|
|
else
|
|
CV_Error(cv::Error::StsError, "The storage is not opened");
|
|
}
|
|
|
|
char *FileStorage::Impl::getsFromFile(char *buf, int count) {
|
|
if (file)
|
|
return fgets(buf, count, file);
|
|
#if USE_ZLIB
|
|
if (gzfile)
|
|
return gzgets(gzfile, buf, count);
|
|
#endif
|
|
CV_Error(cv::Error::StsError, "The storage is not opened");
|
|
}
|
|
|
|
char *FileStorage::Impl::gets(size_t maxCount) {
|
|
if (strbuf) {
|
|
size_t i = strbufpos, len = strbufsize;
|
|
const char *instr = strbuf;
|
|
for (; i < len; i++) {
|
|
char c = instr[i];
|
|
if (c == '\0' || c == '\n') {
|
|
if (c == '\n')
|
|
i++;
|
|
break;
|
|
}
|
|
}
|
|
size_t count = i - strbufpos;
|
|
if (maxCount == 0 || maxCount > count)
|
|
maxCount = count;
|
|
buffer.resize(std::max(buffer.size(), maxCount + 8));
|
|
memcpy(&buffer[0], instr + strbufpos, maxCount);
|
|
buffer[maxCount] = '\0';
|
|
strbufpos = i;
|
|
return maxCount > 0 ? &buffer[0] : 0;
|
|
}
|
|
|
|
const size_t MAX_BLOCK_SIZE = INT_MAX / 2; // hopefully, that will be enough
|
|
if (maxCount == 0)
|
|
maxCount = MAX_BLOCK_SIZE;
|
|
else
|
|
CV_Assert(maxCount < MAX_BLOCK_SIZE);
|
|
size_t ofs = 0;
|
|
|
|
for (;;) {
|
|
int count = (int) std::min(buffer.size() - ofs - 16, maxCount);
|
|
char *ptr = getsFromFile(&buffer[ofs], count + 1);
|
|
if (!ptr)
|
|
break;
|
|
int delta = (int) strlen(ptr);
|
|
ofs += delta;
|
|
maxCount -= delta;
|
|
if (delta == 0 || ptr[delta - 1] == '\n' || maxCount == 0)
|
|
break;
|
|
if (delta == count)
|
|
buffer.resize((size_t) (buffer.size() * 1.5));
|
|
}
|
|
return ofs > 0 ? &buffer[0] : 0;
|
|
}
|
|
|
|
char *FileStorage::Impl::gets() {
|
|
char *ptr = this->gets(0);
|
|
if (!ptr) {
|
|
ptr = bufferStart(); // FIXIT Why do we need this hack? What is about other parsers JSON/YAML?
|
|
*ptr = '\0';
|
|
setEof();
|
|
return 0;
|
|
} else {
|
|
size_t l = strlen(ptr);
|
|
if (l > 0 && ptr[l - 1] != '\n' && ptr[l - 1] != '\r' && !eof()) {
|
|
ptr[l] = '\n';
|
|
ptr[l + 1] = '\0';
|
|
}
|
|
}
|
|
lineno++;
|
|
return ptr;
|
|
}
|
|
|
|
bool FileStorage::Impl::eof() {
|
|
if (dummy_eof)
|
|
return true;
|
|
if (strbuf)
|
|
return strbufpos >= strbufsize;
|
|
if (file)
|
|
return feof(file) != 0;
|
|
#if USE_ZLIB
|
|
if (gzfile)
|
|
return gzeof(gzfile) != 0;
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
void FileStorage::Impl::setEof() {
|
|
dummy_eof = true;
|
|
}
|
|
|
|
void FileStorage::Impl::closeFile() {
|
|
if (file)
|
|
fclose(file);
|
|
#if USE_ZLIB
|
|
else if (gzfile)
|
|
gzclose(gzfile);
|
|
#endif
|
|
file = 0;
|
|
gzfile = 0;
|
|
strbuf = 0;
|
|
strbufpos = 0;
|
|
is_opened = false;
|
|
}
|
|
|
|
void FileStorage::Impl::rewind() {
|
|
if (file)
|
|
::rewind(file);
|
|
#if USE_ZLIB
|
|
else if (gzfile)
|
|
gzrewind(gzfile);
|
|
#endif
|
|
strbufpos = 0;
|
|
}
|
|
|
|
char *FileStorage::Impl::resizeWriteBuffer(char *ptr, int len) {
|
|
const char *buffer_end = &buffer[0] + buffer.size();
|
|
if (ptr + len < buffer_end)
|
|
return ptr;
|
|
|
|
const char *buffer_start = &buffer[0];
|
|
int written_len = (int) (ptr - buffer_start);
|
|
|
|
CV_Assert(written_len <= (int) buffer.size());
|
|
int new_size = (int) ((buffer_end - buffer_start) * 3 / 2);
|
|
new_size = MAX(written_len + len, new_size);
|
|
buffer.reserve(new_size + 256);
|
|
buffer.resize(new_size);
|
|
bufofs = written_len;
|
|
return &buffer[0] + bufofs;
|
|
}
|
|
|
|
char *FileStorage::Impl::flush() {
|
|
char *buffer_start = &buffer[0];
|
|
char *ptr = buffer_start + bufofs;
|
|
|
|
if (ptr > buffer_start + space) {
|
|
ptr[0] = '\n';
|
|
ptr[1] = '\0';
|
|
puts(buffer_start);
|
|
bufofs = 0;
|
|
}
|
|
|
|
int indent = write_stack.back().indent;
|
|
|
|
if (space != indent) {
|
|
memset(buffer_start, ' ', indent);
|
|
space = indent;
|
|
}
|
|
bufofs = space;
|
|
ptr = buffer_start + bufofs;
|
|
|
|
return ptr;
|
|
}
|
|
|
|
void FileStorage::Impl::endWriteStruct() {
|
|
CV_Assert(write_mode);
|
|
|
|
check_if_write_struct_is_delayed(false);
|
|
if (state_of_writing_base64 != FileStorage_API::Uncertain)
|
|
switch_to_Base64_state(FileStorage_API::Uncertain);
|
|
|
|
CV_Assert(!write_stack.empty());
|
|
|
|
FStructData ¤t_struct = write_stack.back();
|
|
if (fmt == FileStorage::FORMAT_JSON && !FileNode::isFlow(current_struct.flags) && write_stack.size() > 1)
|
|
current_struct.indent = write_stack[write_stack.size() - 2].indent;
|
|
|
|
getEmitter().endWriteStruct(current_struct);
|
|
|
|
write_stack.pop_back();
|
|
if (!write_stack.empty())
|
|
write_stack.back().flags &= ~FileNode::EMPTY;
|
|
}
|
|
|
|
void FileStorage::Impl::startWriteStruct_helper(const char *key, int struct_flags,
|
|
const char *type_name) {
|
|
CV_Assert(write_mode);
|
|
|
|
struct_flags = (struct_flags & (FileNode::TYPE_MASK | FileNode::FLOW)) | FileNode::EMPTY;
|
|
if (!FileNode::isCollection(struct_flags))
|
|
CV_Error(cv::Error::StsBadArg,
|
|
"Some collection type: FileNode::SEQ or FileNode::MAP must be specified");
|
|
|
|
if (type_name && type_name[0] == '\0')
|
|
type_name = 0;
|
|
|
|
FStructData s = getEmitter().startWriteStruct(write_stack.back(), key, struct_flags, type_name);
|
|
|
|
write_stack.push_back(s);
|
|
size_t write_stack_size = write_stack.size();
|
|
if (write_stack_size > 1)
|
|
write_stack[write_stack_size - 2].flags &= ~FileNode::EMPTY;
|
|
|
|
if (fmt != FileStorage::FORMAT_JSON && !FileNode::isFlow(s.flags))
|
|
flush();
|
|
|
|
if (fmt == FileStorage::FORMAT_JSON && type_name && type_name[0] && FileNode::isMap(struct_flags)) {
|
|
getEmitter().write("type_id", type_name, false);
|
|
}
|
|
}
|
|
|
|
void FileStorage::Impl::startWriteStruct(const char *key, int struct_flags,
|
|
const char *type_name) {
|
|
check_if_write_struct_is_delayed(false);
|
|
if (state_of_writing_base64 == FileStorage_API::NotUse)
|
|
switch_to_Base64_state(FileStorage_API::Uncertain);
|
|
|
|
if (state_of_writing_base64 == FileStorage_API::Uncertain && FileNode::isSeq(struct_flags)
|
|
&& is_using_base64 && type_name == 0) {
|
|
/* Uncertain whether output Base64 data */
|
|
make_write_struct_delayed(key, struct_flags, type_name);
|
|
} else if (type_name && memcmp(type_name, "binary", 6) == 0) {
|
|
/* Must output Base64 data */
|
|
if ((FileNode::TYPE_MASK & struct_flags) != FileNode::SEQ)
|
|
CV_Error(cv::Error::StsBadArg, "must set 'struct_flags |= CV_NODE_SEQ' if using Base64.");
|
|
else if (state_of_writing_base64 != FileStorage_API::Uncertain)
|
|
CV_Error(cv::Error::StsError, "function \'cvStartWriteStruct\' calls cannot be nested if using Base64.");
|
|
|
|
startWriteStruct_helper(key, struct_flags, "binary");
|
|
|
|
if (state_of_writing_base64 != FileStorage_API::Uncertain)
|
|
switch_to_Base64_state(FileStorage_API::Uncertain);
|
|
switch_to_Base64_state(FileStorage_API::InUse);
|
|
} else {
|
|
/* Won't output Base64 data */
|
|
if (state_of_writing_base64 == FileStorage_API::InUse)
|
|
CV_Error(cv::Error::StsError, "At the end of the output Base64, `cvEndWriteStruct` is needed.");
|
|
|
|
startWriteStruct_helper(key, struct_flags, type_name);
|
|
|
|
if (state_of_writing_base64 != FileStorage_API::Uncertain)
|
|
switch_to_Base64_state(FileStorage_API::Uncertain);
|
|
switch_to_Base64_state(FileStorage_API::NotUse);
|
|
}
|
|
}
|
|
|
|
void FileStorage::Impl::writeComment(const char *comment, bool eol_comment) {
|
|
CV_Assert(write_mode);
|
|
getEmitter().writeComment(comment, eol_comment);
|
|
}
|
|
|
|
void FileStorage::Impl::startNextStream() {
|
|
CV_Assert(write_mode);
|
|
if (!empty_stream) {
|
|
while (!write_stack.empty())
|
|
endWriteStruct();
|
|
flush();
|
|
getEmitter().startNextStream();
|
|
empty_stream = true;
|
|
write_stack.push_back(FStructData("", FileNode::EMPTY, 0));
|
|
bufofs = 0;
|
|
}
|
|
}
|
|
|
|
void FileStorage::Impl::write(const String &key, int value) {
|
|
CV_Assert(write_mode);
|
|
getEmitter().write(key.c_str(), value);
|
|
}
|
|
|
|
void FileStorage::Impl::write(const String &key, double value) {
|
|
CV_Assert(write_mode);
|
|
getEmitter().write(key.c_str(), value);
|
|
}
|
|
|
|
void FileStorage::Impl::write(const String &key, const String &value) {
|
|
CV_Assert(write_mode);
|
|
getEmitter().write(key.c_str(), value.c_str(), false);
|
|
}
|
|
|
|
void FileStorage::Impl::writeRawData(const std::string &dt, const void *_data, size_t len) {
|
|
CV_Assert(write_mode);
|
|
|
|
if (is_using_base64 || state_of_writing_base64 == FileStorage_API::Base64State::InUse) {
|
|
writeRawDataBase64(_data, len, dt.c_str());
|
|
return;
|
|
} else if (state_of_writing_base64 == FileStorage_API::Base64State::Uncertain) {
|
|
switch_to_Base64_state(FileStorage_API::Base64State::NotUse);
|
|
}
|
|
|
|
size_t elemSize = fs::calcStructSize(dt.c_str(), 0);
|
|
CV_Assert(elemSize);
|
|
CV_Assert(len % elemSize == 0);
|
|
len /= elemSize;
|
|
|
|
bool explicitZero = fmt == FileStorage::FORMAT_JSON;
|
|
const uchar *data0 = (const uchar *) _data;
|
|
int fmt_pairs[CV_FS_MAX_FMT_PAIRS * 2], k, fmt_pair_count;
|
|
char buf[256] = "";
|
|
|
|
fmt_pair_count = fs::decodeFormat(dt.c_str(), fmt_pairs, CV_FS_MAX_FMT_PAIRS);
|
|
|
|
if (!len)
|
|
return;
|
|
|
|
if (!data0)
|
|
CV_Error(cv::Error::StsNullPtr, "Null data pointer");
|
|
|
|
if (fmt_pair_count == 1) {
|
|
fmt_pairs[0] *= (int) len;
|
|
len = 1;
|
|
}
|
|
|
|
for (; len--; data0 += elemSize) {
|
|
int offset = 0;
|
|
for (k = 0; k < fmt_pair_count; k++) {
|
|
int i, count = fmt_pairs[k * 2];
|
|
int elem_type = fmt_pairs[k * 2 + 1];
|
|
int elem_size = CV_ELEM_SIZE(elem_type);
|
|
const char *ptr;
|
|
|
|
offset = cvAlign(offset, elem_size);
|
|
const uchar *data = data0 + offset;
|
|
|
|
for (i = 0; i < count; i++) {
|
|
switch (elem_type) {
|
|
case CV_8U:
|
|
ptr = fs::itoa(*(uchar *) data, buf, 10);
|
|
data++;
|
|
break;
|
|
case CV_8S:
|
|
ptr = fs::itoa(*(char *) data, buf, 10);
|
|
data++;
|
|
break;
|
|
case CV_16U:
|
|
ptr = fs::itoa(*(ushort *) data, buf, 10);
|
|
data += sizeof(ushort);
|
|
break;
|
|
case CV_16S:
|
|
ptr = fs::itoa(*(short *) data, buf, 10);
|
|
data += sizeof(short);
|
|
break;
|
|
case CV_32S:
|
|
ptr = fs::itoa(*(int *) data, buf, 10);
|
|
data += sizeof(int);
|
|
break;
|
|
case CV_32F:
|
|
ptr = fs::floatToString(buf, sizeof(buf), *(float *) data, false, explicitZero);
|
|
data += sizeof(float);
|
|
break;
|
|
case CV_64F:
|
|
ptr = fs::doubleToString(buf, sizeof(buf), *(double *) data, explicitZero);
|
|
data += sizeof(double);
|
|
break;
|
|
case CV_16F: /* reference */
|
|
ptr = fs::floatToString(buf, sizeof(buf), (float) *(hfloat *) data, true, explicitZero);
|
|
data += sizeof(hfloat);
|
|
break;
|
|
default:
|
|
CV_Error(cv::Error::StsUnsupportedFormat, "Unsupported type");
|
|
return;
|
|
}
|
|
|
|
getEmitter().writeScalar(0, ptr);
|
|
}
|
|
|
|
offset = (int) (data - data0);
|
|
}
|
|
}
|
|
}
|
|
|
|
void FileStorage::Impl::workaround() {
|
|
check_if_write_struct_is_delayed(false);
|
|
|
|
if (state_of_writing_base64 != FileStorage_API::Base64State::Uncertain)
|
|
switch_to_Base64_state(FileStorage_API::Base64State::Uncertain);
|
|
}
|
|
|
|
void FileStorage::Impl::switch_to_Base64_state(FileStorage_API::Base64State new_state) {
|
|
const char *err_unkonwn_state = "Unexpected error, unable to determine the Base64 state.";
|
|
const char *err_unable_to_switch = "Unexpected error, unable to switch to this state.";
|
|
|
|
/* like a finite state machine */
|
|
switch (state_of_writing_base64) {
|
|
case FileStorage_API::Base64State::Uncertain:
|
|
switch (new_state) {
|
|
case FileStorage_API::Base64State::InUse:
|
|
{
|
|
CV_DbgAssert(base64_writer == 0);
|
|
bool can_indent = (fmt != cv::FileStorage::Mode::FORMAT_JSON);
|
|
base64_writer = new base64::Base64Writer(*this, can_indent);
|
|
if (!can_indent) {
|
|
char *ptr = bufferPtr();
|
|
*ptr++ = '\0';
|
|
puts(bufferStart());
|
|
setBufferPtr(bufferStart());
|
|
memset(bufferStart(), 0, static_cast<int>(space));
|
|
puts("\"$base64$");
|
|
}
|
|
break;
|
|
}
|
|
case FileStorage_API::Base64State::Uncertain:
|
|
break;
|
|
case FileStorage_API::Base64State::NotUse:
|
|
break;
|
|
default:
|
|
CV_Error(cv::Error::StsError, err_unkonwn_state);
|
|
break;
|
|
}
|
|
break;
|
|
case FileStorage_API::Base64State::InUse:
|
|
switch (new_state) {
|
|
case FileStorage_API::Base64State::InUse:
|
|
case FileStorage_API::Base64State::NotUse:
|
|
CV_Error(cv::Error::StsError, err_unable_to_switch);
|
|
break;
|
|
case FileStorage_API::Base64State::Uncertain:
|
|
delete base64_writer;
|
|
base64_writer = 0;
|
|
if ( fmt == cv::FileStorage::FORMAT_JSON )
|
|
{
|
|
puts("\"");
|
|
setBufferPtr(bufferStart());
|
|
flush();
|
|
memset(bufferStart(), 0, static_cast<int>(space) );
|
|
setBufferPtr(bufferStart());
|
|
}
|
|
break;
|
|
default:
|
|
CV_Error(cv::Error::StsError, err_unkonwn_state);
|
|
break;
|
|
}
|
|
break;
|
|
case FileStorage_API::Base64State::NotUse:
|
|
switch (new_state) {
|
|
case FileStorage_API::Base64State::InUse:
|
|
case FileStorage_API::Base64State::NotUse:
|
|
CV_Error(cv::Error::StsError, err_unable_to_switch);
|
|
break;
|
|
case FileStorage_API::Base64State::Uncertain:
|
|
break;
|
|
default:
|
|
CV_Error(cv::Error::StsError, err_unkonwn_state);
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
CV_Error(cv::Error::StsError, err_unkonwn_state);
|
|
break;
|
|
}
|
|
|
|
state_of_writing_base64 = new_state;
|
|
}
|
|
|
|
void FileStorage::Impl::make_write_struct_delayed(const char *key, int struct_flags, const char *type_name) {
|
|
CV_Assert(is_write_struct_delayed == false);
|
|
CV_DbgAssert(delayed_struct_key == nullptr);
|
|
CV_DbgAssert(delayed_struct_flags == 0);
|
|
CV_DbgAssert(delayed_type_name == nullptr);
|
|
|
|
delayed_struct_flags = struct_flags;
|
|
|
|
if (key != nullptr) {
|
|
delayed_struct_key = new char[strlen(key) + 1U];
|
|
strcpy(delayed_struct_key, key);
|
|
}
|
|
|
|
if (type_name != nullptr) {
|
|
delayed_type_name = new char[strlen(type_name) + 1U];
|
|
strcpy(delayed_type_name, type_name);
|
|
}
|
|
|
|
is_write_struct_delayed = true;
|
|
}
|
|
|
|
void FileStorage::Impl::check_if_write_struct_is_delayed(bool change_type_to_base64) {
|
|
if (is_write_struct_delayed) {
|
|
/* save data to prevent recursive call errors */
|
|
std::string struct_key;
|
|
std::string type_name;
|
|
int struct_flags = delayed_struct_flags;
|
|
|
|
if (delayed_struct_key != nullptr && *delayed_struct_key != '\0') {
|
|
struct_key.assign(delayed_struct_key);
|
|
}
|
|
if (delayed_type_name != nullptr && *delayed_type_name != '\0') {
|
|
type_name.assign(delayed_type_name);
|
|
}
|
|
|
|
/* reset */
|
|
delete[] delayed_struct_key;
|
|
delete[] delayed_type_name;
|
|
delayed_struct_key = nullptr;
|
|
delayed_struct_flags = 0;
|
|
delayed_type_name = nullptr;
|
|
|
|
is_write_struct_delayed = false;
|
|
|
|
/* call */
|
|
if (change_type_to_base64) {
|
|
startWriteStruct_helper(struct_key.c_str(), struct_flags, "binary");
|
|
if (state_of_writing_base64 != FileStorage_API::Uncertain)
|
|
switch_to_Base64_state(FileStorage_API::Uncertain);
|
|
switch_to_Base64_state(FileStorage_API::InUse);
|
|
} else {
|
|
startWriteStruct_helper(struct_key.c_str(), struct_flags, type_name.c_str());
|
|
if (state_of_writing_base64 != FileStorage_API::Uncertain)
|
|
switch_to_Base64_state(FileStorage_API::Uncertain);
|
|
switch_to_Base64_state(FileStorage_API::NotUse);
|
|
}
|
|
}
|
|
}
|
|
|
|
void FileStorage::Impl::writeRawDataBase64(const void *_data, size_t len, const char *dt) {
|
|
CV_Assert(write_mode);
|
|
|
|
check_if_write_struct_is_delayed(true);
|
|
|
|
if (state_of_writing_base64 == FileStorage_API::Base64State::Uncertain) {
|
|
switch_to_Base64_state(FileStorage_API::Base64State::InUse);
|
|
} else if (state_of_writing_base64 != FileStorage_API::Base64State::InUse) {
|
|
CV_Error(cv::Error::StsError, "Base64 should not be used at present.");
|
|
}
|
|
|
|
base64_writer->write(_data, len, dt);
|
|
}
|
|
|
|
FileNode FileStorage::Impl::getFirstTopLevelNode() const {
|
|
return roots.empty() ? FileNode() : roots[0];
|
|
}
|
|
|
|
FileNode FileStorage::Impl::root(int streamIdx) const {
|
|
return streamIdx >= 0 && streamIdx < (int) roots.size() ? roots[streamIdx] : FileNode();
|
|
}
|
|
|
|
FileNode FileStorage::Impl::operator[](const String &nodename) const {
|
|
return this->operator[](nodename.c_str());
|
|
}
|
|
|
|
FileNode FileStorage::Impl::operator[](const char * /*nodename*/) const {
|
|
return FileNode();
|
|
}
|
|
|
|
int FileStorage::Impl::getFormat() const { return fmt; }
|
|
|
|
char *FileStorage::Impl::bufferPtr() const { return (char *) (&buffer[0] + bufofs); }
|
|
|
|
char *FileStorage::Impl::bufferStart() const { return (char *) &buffer[0]; }
|
|
|
|
char *FileStorage::Impl::bufferEnd() const { return (char *) (&buffer[0] + buffer.size()); }
|
|
|
|
void FileStorage::Impl::setBufferPtr(char *ptr) {
|
|
char *bufferstart = bufferStart();
|
|
CV_Assert(ptr >= bufferstart && ptr <= bufferEnd());
|
|
bufofs = ptr - bufferstart;
|
|
}
|
|
|
|
int FileStorage::Impl::wrapMargin() const { return wrap_margin; }
|
|
|
|
FStructData &FileStorage::Impl::getCurrentStruct() {
|
|
CV_Assert(!write_stack.empty());
|
|
return write_stack.back();
|
|
}
|
|
|
|
void FileStorage::Impl::setNonEmpty() {
|
|
empty_stream = false;
|
|
}
|
|
|
|
void FileStorage::Impl::processSpecialDouble(char *buf, double *value, char **endptr) {
|
|
FileStorage_API *fs = this;
|
|
char c = buf[0];
|
|
int inf_hi = 0x7ff00000;
|
|
|
|
if (c == '-' || c == '+') {
|
|
inf_hi = c == '-' ? 0xfff00000 : 0x7ff00000;
|
|
c = *++buf;
|
|
}
|
|
|
|
if (c != '.')
|
|
CV_PARSE_ERROR_CPP("Bad format of floating-point constant");
|
|
|
|
Cv64suf v;
|
|
v.f = 0.;
|
|
if (toupper(buf[1]) == 'I' && toupper(buf[2]) == 'N' && toupper(buf[3]) == 'F')
|
|
v.u = (uint64) inf_hi << 32;
|
|
else if (toupper(buf[1]) == 'N' && toupper(buf[2]) == 'A' && toupper(buf[3]) == 'N')
|
|
v.u = (uint64) -1;
|
|
else
|
|
CV_PARSE_ERROR_CPP("Bad format of floating-point constant");
|
|
*value = v.f;
|
|
*endptr = buf + 4;
|
|
}
|
|
|
|
double FileStorage::Impl::strtod(char *ptr, char **endptr) {
|
|
double fval = ::strtod(ptr, endptr);
|
|
if (**endptr == '.') {
|
|
char *dot_pos = *endptr;
|
|
*dot_pos = ',';
|
|
double fval2 = ::strtod(ptr, endptr);
|
|
*dot_pos = '.';
|
|
if (*endptr > dot_pos)
|
|
fval = fval2;
|
|
else
|
|
*endptr = dot_pos;
|
|
}
|
|
|
|
if (*endptr == ptr || cv_isalpha(**endptr))
|
|
processSpecialDouble(ptr, &fval, endptr);
|
|
|
|
return fval;
|
|
}
|
|
|
|
void FileStorage::Impl::convertToCollection(int type, FileNode &node) {
|
|
CV_Assert(type == FileNode::SEQ || type == FileNode::MAP);
|
|
|
|
int node_type = node.type();
|
|
if (node_type == type)
|
|
return;
|
|
|
|
bool named = node.isNamed();
|
|
uchar *ptr = node.ptr() + 1 + (named ? 4 : 0);
|
|
|
|
int ival = 0;
|
|
double fval = 0;
|
|
std::string sval;
|
|
bool add_first_scalar = false;
|
|
|
|
if (node_type != FileNode::NONE) {
|
|
// scalar nodes can only be converted to sequences, e.g. in XML:
|
|
// <a>5[parser_position]... => create 5 with name "a"
|
|
// <a>5 6[parser_position]... => 5 is converted to [5] and then 6 is added to it
|
|
//
|
|
// otherwise we don't know where to get the element names from
|
|
CV_Assert(type == FileNode::SEQ);
|
|
if (node_type == FileNode::INT) {
|
|
ival = readInt(ptr);
|
|
add_first_scalar = true;
|
|
} else if (node_type == FileNode::REAL) {
|
|
fval = readReal(ptr);
|
|
add_first_scalar = true;
|
|
} else if (node_type == FileNode::STRING) {
|
|
sval = std::string(node);
|
|
add_first_scalar = true;
|
|
} else
|
|
CV_Error_(Error::StsError, ("The node of type %d cannot be converted to collection", node_type));
|
|
}
|
|
|
|
ptr = reserveNodeSpace(node, 1 + (named ? 4 : 0) + 4 + 4);
|
|
*ptr++ = (uchar) (type | (named ? FileNode::NAMED : 0));
|
|
// name has been copied automatically
|
|
if (named)
|
|
ptr += 4;
|
|
// set raw_size(collection)==4, nelems(collection)==1
|
|
writeInt(ptr, 4);
|
|
writeInt(ptr + 4, 0);
|
|
|
|
if (add_first_scalar)
|
|
addNode(node, std::string(), node_type,
|
|
node_type == FileNode::INT ? (const void *) &ival :
|
|
node_type == FileNode::REAL ? (const void *) &fval :
|
|
node_type == FileNode::STRING ? (const void *) sval.c_str() : 0,
|
|
-1);
|
|
}
|
|
|
|
// a) allocates new FileNode (for that just set blockIdx to the last block and ofs to freeSpaceOfs) or
|
|
// b) reallocates just created new node (blockIdx and ofs must be taken from FileNode).
|
|
// If there is no enough space in the current block (it should be the last block added so far),
|
|
// the last block is shrunk so that it ends immediately before the reallocated node. Then,
|
|
// a new block of sufficient size is allocated and the FileNode is placed in the beginning of it.
|
|
// The case (a) can be used to allocate the very first node by setting blockIdx == ofs == 0.
|
|
// In the case (b) the existing tag and the name are copied automatically.
|
|
uchar *FileStorage::Impl::reserveNodeSpace(FileNode &node, size_t sz) {
|
|
bool shrinkBlock = false;
|
|
size_t shrinkBlockIdx = 0, shrinkSize = 0;
|
|
|
|
uchar *ptr = 0, *blockEnd = 0;
|
|
|
|
if (!fs_data_ptrs.empty()) {
|
|
size_t blockIdx = node.blockIdx;
|
|
size_t ofs = node.ofs;
|
|
CV_Assert(blockIdx == fs_data_ptrs.size() - 1);
|
|
CV_Assert(ofs <= fs_data_blksz[blockIdx]);
|
|
CV_Assert(freeSpaceOfs <= fs_data_blksz[blockIdx]);
|
|
//CV_Assert( freeSpaceOfs <= ofs + sz );
|
|
|
|
ptr = fs_data_ptrs[blockIdx] + ofs;
|
|
blockEnd = fs_data_ptrs[blockIdx] + fs_data_blksz[blockIdx];
|
|
|
|
CV_Assert(ptr >= fs_data_ptrs[blockIdx] && ptr <= blockEnd);
|
|
if (ptr + sz <= blockEnd) {
|
|
freeSpaceOfs = ofs + sz;
|
|
return ptr;
|
|
}
|
|
|
|
if (ofs ==
|
|
0) // FileNode is a first component of this block. Resize current block instead of allocation of new one.
|
|
{
|
|
fs_data[blockIdx]->resize(sz);
|
|
ptr = &fs_data[blockIdx]->at(0);
|
|
fs_data_ptrs[blockIdx] = ptr;
|
|
fs_data_blksz[blockIdx] = sz;
|
|
freeSpaceOfs = sz;
|
|
return ptr;
|
|
}
|
|
|
|
shrinkBlock = true;
|
|
shrinkBlockIdx = blockIdx;
|
|
shrinkSize = ofs;
|
|
}
|
|
|
|
size_t blockSize = std::max((size_t) CV_FS_MAX_LEN * 4 - 256, sz) + 256;
|
|
Ptr<std::vector<uchar> > pv = makePtr<std::vector<uchar> >(blockSize);
|
|
fs_data.push_back(pv);
|
|
uchar *new_ptr = &pv->at(0);
|
|
fs_data_ptrs.push_back(new_ptr);
|
|
fs_data_blksz.push_back(blockSize);
|
|
node.blockIdx = fs_data_ptrs.size() - 1;
|
|
node.ofs = 0;
|
|
freeSpaceOfs = sz;
|
|
|
|
if (ptr && ptr + 5 <= blockEnd) {
|
|
new_ptr[0] = ptr[0];
|
|
if (ptr[0] & FileNode::NAMED) {
|
|
new_ptr[1] = ptr[1];
|
|
new_ptr[2] = ptr[2];
|
|
new_ptr[3] = ptr[3];
|
|
new_ptr[4] = ptr[4];
|
|
}
|
|
}
|
|
|
|
if (shrinkBlock) {
|
|
fs_data[shrinkBlockIdx]->resize(shrinkSize);
|
|
fs_data_blksz[shrinkBlockIdx] = shrinkSize;
|
|
}
|
|
|
|
return new_ptr;
|
|
}
|
|
|
|
unsigned FileStorage::Impl::getStringOfs(const std::string &key) const {
|
|
str_hash_t::const_iterator it = str_hash.find(key);
|
|
return it != str_hash.end() ? it->second : 0;
|
|
}
|
|
|
|
FileNode FileStorage::Impl::addNode(FileNode &collection, const std::string &key,
|
|
int elem_type, const void *value, int len) {
|
|
FileStorage_API *fs = this;
|
|
bool noname = key.empty() || (fmt == FileStorage::FORMAT_XML && strcmp(key.c_str(), "_") == 0);
|
|
convertToCollection(noname ? FileNode::SEQ : FileNode::MAP, collection);
|
|
|
|
bool isseq = collection.empty() ? false : collection.isSeq();
|
|
if (noname != isseq)
|
|
CV_PARSE_ERROR_CPP(noname ? "Map element should have a name" :
|
|
"Sequence element should not have name (use <_></_>)");
|
|
unsigned strofs = 0;
|
|
if (!noname) {
|
|
strofs = getStringOfs(key);
|
|
if (!strofs) {
|
|
strofs = (unsigned) str_hash_data.size();
|
|
size_t keysize = key.size() + 1;
|
|
str_hash_data.resize(strofs + keysize);
|
|
memcpy(&str_hash_data[0] + strofs, &key[0], keysize);
|
|
str_hash.insert(std::make_pair(key, strofs));
|
|
}
|
|
}
|
|
|
|
uchar *cp = collection.ptr();
|
|
|
|
size_t blockIdx = fs_data_ptrs.size() - 1;
|
|
size_t ofs = freeSpaceOfs;
|
|
FileNode node(fs_ext, blockIdx, ofs);
|
|
|
|
size_t sz0 = 1 + (noname ? 0 : 4) + 8;
|
|
uchar *ptr = reserveNodeSpace(node, sz0);
|
|
|
|
*ptr++ = (uchar) (elem_type | (noname ? 0 : FileNode::NAMED));
|
|
if (elem_type == FileNode::NONE)
|
|
freeSpaceOfs -= 8;
|
|
|
|
if (!noname) {
|
|
writeInt(ptr, (int) strofs);
|
|
ptr += 4;
|
|
}
|
|
|
|
if (elem_type == FileNode::SEQ || elem_type == FileNode::MAP) {
|
|
writeInt(ptr, 4);
|
|
writeInt(ptr, 0);
|
|
}
|
|
|
|
if (value)
|
|
node.setValue(elem_type, value, len);
|
|
|
|
if (collection.isNamed())
|
|
cp += 4;
|
|
int nelems = readInt(cp + 5);
|
|
writeInt(cp + 5, nelems + 1);
|
|
|
|
return node;
|
|
}
|
|
|
|
void FileStorage::Impl::finalizeCollection(FileNode &collection) {
|
|
if (!collection.isSeq() && !collection.isMap())
|
|
return;
|
|
uchar *ptr0 = collection.ptr(), *ptr = ptr0 + 1;
|
|
if (*ptr0 & FileNode::NAMED)
|
|
ptr += 4;
|
|
size_t blockIdx = collection.blockIdx;
|
|
size_t ofs = collection.ofs + (size_t) (ptr + 8 - ptr0);
|
|
size_t rawSize = 4;
|
|
unsigned sz = (unsigned) readInt(ptr + 4);
|
|
if (sz > 0) {
|
|
size_t lastBlockIdx = fs_data_ptrs.size() - 1;
|
|
|
|
for (; blockIdx < lastBlockIdx; blockIdx++) {
|
|
rawSize += fs_data_blksz[blockIdx] - ofs;
|
|
ofs = 0;
|
|
}
|
|
}
|
|
rawSize += freeSpaceOfs - ofs;
|
|
writeInt(ptr, (int) rawSize);
|
|
}
|
|
|
|
void FileStorage::Impl::normalizeNodeOfs(size_t &blockIdx, size_t &ofs) const {
|
|
while (ofs >= fs_data_blksz[blockIdx]) {
|
|
if (blockIdx == fs_data_blksz.size() - 1) {
|
|
CV_Assert(ofs == fs_data_blksz[blockIdx]);
|
|
break;
|
|
}
|
|
ofs -= fs_data_blksz[blockIdx];
|
|
blockIdx++;
|
|
}
|
|
}
|
|
|
|
FileStorage::Impl::Base64State FileStorage::Impl::get_state_of_writing_base64() {
|
|
return state_of_writing_base64;
|
|
}
|
|
|
|
int FileStorage::Impl::get_space() {
|
|
return space;
|
|
}
|
|
|
|
|
|
FileStorage::Impl::Base64Decoder::Base64Decoder() {
|
|
ofs = 0;
|
|
ptr = 0;
|
|
indent = 0;
|
|
totalchars = 0;
|
|
eos = true;
|
|
}
|
|
|
|
void FileStorage::Impl::Base64Decoder::init(const Ptr<FileStorageParser> &_parser, char *_ptr, int _indent) {
|
|
parser_do_not_use_direct_dereference = _parser;
|
|
ptr = _ptr;
|
|
indent = _indent;
|
|
encoded.clear();
|
|
decoded.clear();
|
|
ofs = 0;
|
|
totalchars = 0;
|
|
eos = false;
|
|
}
|
|
|
|
bool FileStorage::Impl::Base64Decoder::readMore(int needed) {
|
|
static const uchar base64tab[] =
|
|
{
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 0, 0, 63,
|
|
52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
|
|
15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 0, 0, 0, 0, 0,
|
|
0, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40,
|
|
41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
|
|
};
|
|
|
|
if (eos)
|
|
return false;
|
|
|
|
size_t sz = decoded.size();
|
|
CV_Assert(ofs <= sz);
|
|
sz -= ofs;
|
|
for (size_t i = 0; i < sz; i++)
|
|
decoded[i] = decoded[ofs + i];
|
|
|
|
decoded.resize(sz);
|
|
ofs = 0;
|
|
|
|
CV_Assert(ptr);
|
|
char *beg = 0, *end = 0;
|
|
bool ok = getParser().getBase64Row(ptr, indent, beg, end);
|
|
ptr = end;
|
|
std::copy(beg, end, std::back_inserter(encoded));
|
|
totalchars += end - beg;
|
|
|
|
if (!ok || beg == end) {
|
|
// in the end of base64 sequence pad it with '=' characters so that
|
|
// its total length is multiple of
|
|
eos = true;
|
|
size_t tc = totalchars;
|
|
for (; tc % 4 != 0; tc++)
|
|
encoded.push_back('=');
|
|
}
|
|
|
|
int i = 0, j, n = (int) encoded.size();
|
|
if (n > 0) {
|
|
const uchar *tab = base64tab;
|
|
char *src = &encoded[0];
|
|
|
|
for (; i <= n - 4; i += 4) {
|
|
// dddddd cccccc bbbbbb aaaaaa => ddddddcc ccccbbbb bbaaaaaa
|
|
uchar d = tab[(int) (uchar) src[i]], c = tab[(int) (uchar) src[i + 1]];
|
|
uchar b = tab[(int) (uchar) src[i + 2]], a = tab[(int) (uchar) src[i + 3]];
|
|
|
|
decoded.push_back((uchar) ((d << 2) | (c >> 4)));
|
|
decoded.push_back((uchar) ((c << 4) | (b >> 2)));
|
|
decoded.push_back((uchar) ((b << 6) | a));
|
|
}
|
|
}
|
|
|
|
if (i > 0 && encoded[i - 1] == '=') {
|
|
if (i > 1 && encoded[i - 2] == '=' && !decoded.empty())
|
|
decoded.pop_back();
|
|
if (!decoded.empty())
|
|
decoded.pop_back();
|
|
}
|
|
|
|
n -= i;
|
|
for (j = 0; j < n; j++)
|
|
encoded[j] = encoded[i + j];
|
|
encoded.resize(n);
|
|
|
|
return (int) decoded.size() >= needed;
|
|
}
|
|
|
|
uchar FileStorage::Impl::Base64Decoder::getUInt8() {
|
|
size_t sz = decoded.size();
|
|
if (ofs >= sz && !readMore(1))
|
|
return (uchar) 0;
|
|
return decoded[ofs++];
|
|
}
|
|
|
|
ushort FileStorage::Impl::Base64Decoder::getUInt16() {
|
|
size_t sz = decoded.size();
|
|
if (ofs + 2 > sz && !readMore(2))
|
|
return (ushort) 0;
|
|
ushort val = (decoded[ofs] + (decoded[ofs + 1] << 8));
|
|
ofs += 2;
|
|
return val;
|
|
}
|
|
|
|
int FileStorage::Impl::Base64Decoder::getInt32() {
|
|
size_t sz = decoded.size();
|
|
if (ofs + 4 > sz && !readMore(4))
|
|
return 0;
|
|
int ival = readInt(&decoded[ofs]);
|
|
ofs += 4;
|
|
return ival;
|
|
}
|
|
|
|
double FileStorage::Impl::Base64Decoder::getFloat64() {
|
|
size_t sz = decoded.size();
|
|
if (ofs + 8 > sz && !readMore(8))
|
|
return 0;
|
|
double fval = readReal(&decoded[ofs]);
|
|
ofs += 8;
|
|
return fval;
|
|
}
|
|
|
|
bool FileStorage::Impl::Base64Decoder::endOfStream() const { return eos; }
|
|
|
|
char *FileStorage::Impl::Base64Decoder::getPtr() const { return ptr; }
|
|
|
|
|
|
char *FileStorage::Impl::parseBase64(char *ptr, int indent, FileNode &collection) {
|
|
const int BASE64_HDR_SIZE = 24;
|
|
char dt[BASE64_HDR_SIZE + 1] = {0};
|
|
base64decoder.init(parser_do_not_use_direct_dereference, ptr, indent);
|
|
|
|
int i, k;
|
|
|
|
for (i = 0; i < BASE64_HDR_SIZE; i++)
|
|
dt[i] = (char) base64decoder.getUInt8();
|
|
for (i = 0; i < BASE64_HDR_SIZE; i++)
|
|
if (isspace(dt[i]))
|
|
break;
|
|
dt[i] = '\0';
|
|
|
|
CV_Assert(!base64decoder.endOfStream());
|
|
|
|
int fmt_pairs[CV_FS_MAX_FMT_PAIRS * 2];
|
|
int fmt_pair_count = fs::decodeFormat(dt, fmt_pairs, CV_FS_MAX_FMT_PAIRS);
|
|
int ival = 0;
|
|
double fval = 0;
|
|
|
|
for (;;) {
|
|
for (k = 0; k < fmt_pair_count; k++) {
|
|
int elem_type = fmt_pairs[k * 2 + 1];
|
|
int count = fmt_pairs[k * 2];
|
|
|
|
for (i = 0; i < count; i++) {
|
|
int node_type = FileNode::INT;
|
|
switch (elem_type) {
|
|
case CV_8U:
|
|
ival = base64decoder.getUInt8();
|
|
break;
|
|
case CV_8S:
|
|
ival = (char) base64decoder.getUInt8();
|
|
break;
|
|
case CV_16U:
|
|
ival = base64decoder.getUInt16();
|
|
break;
|
|
case CV_16S:
|
|
ival = (short) base64decoder.getUInt16();
|
|
break;
|
|
case CV_32S:
|
|
ival = base64decoder.getInt32();
|
|
break;
|
|
case CV_32F: {
|
|
Cv32suf v;
|
|
v.i = base64decoder.getInt32();
|
|
fval = v.f;
|
|
node_type = FileNode::REAL;
|
|
}
|
|
break;
|
|
case CV_64F:
|
|
fval = base64decoder.getFloat64();
|
|
node_type = FileNode::REAL;
|
|
break;
|
|
case CV_16F:
|
|
fval = float(hfloatFromBits(base64decoder.getUInt16()));
|
|
node_type = FileNode::REAL;
|
|
break;
|
|
default:
|
|
CV_Error(Error::StsUnsupportedFormat, "Unsupported type");
|
|
}
|
|
|
|
if (base64decoder.endOfStream())
|
|
break;
|
|
addNode(collection, std::string(), node_type,
|
|
node_type == FileNode::INT ? (void *) &ival : (void *) &fval, -1);
|
|
}
|
|
}
|
|
if (base64decoder.endOfStream())
|
|
break;
|
|
}
|
|
|
|
finalizeCollection(collection);
|
|
return base64decoder.getPtr();
|
|
}
|
|
|
|
void FileStorage::Impl::parseError(const char *func_name, const std::string &err_msg, const char *source_file,
|
|
int source_line) {
|
|
std::string msg = format("%s(%d): %s", filename.c_str(), lineno, err_msg.c_str());
|
|
error(Error::StsParseError, func_name, msg.c_str(), source_file, source_line);
|
|
}
|
|
|
|
const uchar *FileStorage::Impl::getNodePtr(size_t blockIdx, size_t ofs) const {
|
|
CV_Assert(blockIdx < fs_data_ptrs.size());
|
|
CV_Assert(ofs < fs_data_blksz[blockIdx]);
|
|
|
|
return fs_data_ptrs[blockIdx] + ofs;
|
|
}
|
|
|
|
std::string FileStorage::Impl::getName(size_t nameofs) const {
|
|
CV_Assert(nameofs < str_hash_data.size());
|
|
return std::string(&str_hash_data[nameofs]);
|
|
}
|
|
|
|
FileStorage *FileStorage::Impl::getFS() { return fs_ext; }
|
|
|
|
|
|
FileStorage::FileStorage()
|
|
: state(0)
|
|
{
|
|
p = makePtr<FileStorage::Impl>(this);
|
|
}
|
|
|
|
FileStorage::FileStorage(const String& filename, int flags, const String& encoding)
|
|
: state(0)
|
|
{
|
|
p = makePtr<FileStorage::Impl>(this);
|
|
bool ok = p->open(filename.c_str(), flags, encoding.c_str());
|
|
if(ok)
|
|
state = FileStorage::NAME_EXPECTED + FileStorage::INSIDE_MAP;
|
|
}
|
|
|
|
void FileStorage::startWriteStruct(const String& name, int struct_flags, const String& typeName)
|
|
{
|
|
p->startWriteStruct(name.size() ? name.c_str() : 0, struct_flags, typeName.size() ? typeName.c_str() : 0);
|
|
elname = String();
|
|
if ((struct_flags & FileNode::TYPE_MASK) == FileNode::SEQ)
|
|
state = FileStorage::VALUE_EXPECTED;
|
|
else
|
|
state = FileStorage::NAME_EXPECTED + FileStorage::INSIDE_MAP;
|
|
}
|
|
|
|
void FileStorage::endWriteStruct()
|
|
{
|
|
p->endWriteStruct();
|
|
state = p->write_stack.empty() || FileNode::isMap(p->write_stack.back().flags) ?
|
|
FileStorage::NAME_EXPECTED + FileStorage::INSIDE_MAP :
|
|
FileStorage::VALUE_EXPECTED;
|
|
elname = String();
|
|
}
|
|
|
|
FileStorage::~FileStorage()
|
|
{
|
|
}
|
|
|
|
bool FileStorage::open(const String& filename, int flags, const String& encoding)
|
|
{
|
|
try
|
|
{
|
|
bool ok = p->open(filename.c_str(), flags, encoding.c_str());
|
|
if(ok)
|
|
state = FileStorage::NAME_EXPECTED + FileStorage::INSIDE_MAP;
|
|
return ok;
|
|
}
|
|
catch (...)
|
|
{
|
|
release();
|
|
throw; // re-throw
|
|
}
|
|
}
|
|
|
|
bool FileStorage::isOpened() const { return p->is_opened; }
|
|
|
|
void FileStorage::release()
|
|
{
|
|
p->release();
|
|
}
|
|
|
|
FileNode FileStorage::root(int i) const
|
|
{
|
|
if( p.empty() || p->roots.empty() || i < 0 || i >= (int)p->roots.size() )
|
|
return FileNode();
|
|
|
|
return p->roots[i];
|
|
}
|
|
|
|
FileNode FileStorage::getFirstTopLevelNode() const
|
|
{
|
|
FileNode r = root();
|
|
FileNodeIterator it = r.begin();
|
|
return it != r.end() ? *it : FileNode();
|
|
}
|
|
|
|
std::string FileStorage::getDefaultObjectName(const std::string& _filename)
|
|
{
|
|
static const char* stubname = "unnamed";
|
|
const char* filename = _filename.c_str();
|
|
const char* ptr2 = filename + _filename.size();
|
|
const char* ptr = ptr2 - 1;
|
|
cv::AutoBuffer<char> name_buf(_filename.size()+1);
|
|
|
|
while( ptr >= filename && *ptr != '\\' && *ptr != '/' && *ptr != ':' )
|
|
{
|
|
if( *ptr == '.' && (!*ptr2 || strncmp(ptr2, ".gz", 3) == 0) )
|
|
ptr2 = ptr;
|
|
ptr--;
|
|
}
|
|
ptr++;
|
|
if( ptr == ptr2 )
|
|
CV_Error( cv::Error::StsBadArg, "Invalid filename" );
|
|
|
|
char* name = name_buf.data();
|
|
|
|
// name must start with letter or '_'
|
|
if( !cv_isalpha(*ptr) && *ptr!= '_' ){
|
|
*name++ = '_';
|
|
}
|
|
|
|
while( ptr < ptr2 )
|
|
{
|
|
char c = *ptr++;
|
|
if( !cv_isalnum(c) && c != '-' && c != '_' )
|
|
c = '_';
|
|
*name++ = c;
|
|
}
|
|
*name = '\0';
|
|
name = name_buf.data();
|
|
if( strcmp( name, "_" ) == 0 )
|
|
strcpy( name, stubname );
|
|
return name;
|
|
}
|
|
|
|
|
|
int FileStorage::getFormat() const
|
|
{
|
|
return p->fmt;
|
|
}
|
|
|
|
FileNode FileStorage::operator [](const char* key) const
|
|
{
|
|
return this->operator[](std::string(key));
|
|
}
|
|
|
|
FileNode FileStorage::operator [](const std::string& key) const
|
|
{
|
|
FileNode res;
|
|
for (size_t i = 0; i < p->roots.size(); i++)
|
|
{
|
|
res = p->roots[i][key];
|
|
if (!res.empty())
|
|
break;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
String FileStorage::releaseAndGetString()
|
|
{
|
|
String buf;
|
|
p->release(&buf);
|
|
return buf;
|
|
}
|
|
|
|
void FileStorage::writeRaw( const String& fmt, const void* vec, size_t len )
|
|
{
|
|
p->writeRawData(fmt, (const uchar*)vec, len);
|
|
}
|
|
|
|
void FileStorage::writeComment( const String& comment, bool eol_comment )
|
|
{
|
|
p->writeComment(comment.c_str(), eol_comment);
|
|
}
|
|
|
|
void writeScalar( FileStorage& fs, int value )
|
|
{
|
|
fs.p->write(String(), value);
|
|
}
|
|
|
|
void writeScalar( FileStorage& fs, float value )
|
|
{
|
|
fs.p->write(String(), (double)value);
|
|
}
|
|
|
|
void writeScalar( FileStorage& fs, double value )
|
|
{
|
|
fs.p->write(String(), value);
|
|
}
|
|
|
|
void writeScalar( FileStorage& fs, const String& value )
|
|
{
|
|
fs.p->write(String(), value);
|
|
}
|
|
|
|
void write( FileStorage& fs, const String& name, int value )
|
|
{
|
|
fs.p->write(name, value);
|
|
}
|
|
|
|
void write( FileStorage& fs, const String& name, float value )
|
|
{
|
|
fs.p->write(name, (double)value);
|
|
}
|
|
|
|
void write( FileStorage& fs, const String& name, double value )
|
|
{
|
|
fs.p->write(name, value);
|
|
}
|
|
|
|
void write( FileStorage& fs, const String& name, const String& value )
|
|
{
|
|
fs.p->write(name, value);
|
|
}
|
|
|
|
void FileStorage::write(const String& name, int val) { p->write(name, val); }
|
|
void FileStorage::write(const String& name, double val) { p->write(name, val); }
|
|
void FileStorage::write(const String& name, const String& val) { p->write(name, val); }
|
|
void FileStorage::write(const String& name, const Mat& val) { cv::write(*this, name, val); }
|
|
void FileStorage::write(const String& name, const std::vector<String>& val) { cv::write(*this, name, val); }
|
|
|
|
FileStorage& operator << (FileStorage& fs, const String& str)
|
|
{
|
|
enum { NAME_EXPECTED = FileStorage::NAME_EXPECTED,
|
|
VALUE_EXPECTED = FileStorage::VALUE_EXPECTED,
|
|
INSIDE_MAP = FileStorage::INSIDE_MAP };
|
|
const char* _str = str.c_str();
|
|
if( !fs.isOpened() || !_str )
|
|
return fs;
|
|
Ptr<FileStorage::Impl>& fs_impl = fs.p;
|
|
char c = *_str;
|
|
|
|
if( c == '}' || c == ']' )
|
|
{
|
|
if( fs_impl->write_stack.empty() )
|
|
CV_Error_( cv::Error::StsError, ("Extra closing '%c'", *_str) );
|
|
|
|
fs_impl->workaround();
|
|
|
|
int struct_flags = fs_impl->write_stack.back().flags;
|
|
char expected_bracket = FileNode::isMap(struct_flags) ? '}' : ']';
|
|
if( c != expected_bracket )
|
|
CV_Error_( cv::Error::StsError, ("The closing '%c' does not match the opening '%c'", c, expected_bracket));
|
|
fs_impl->endWriteStruct();
|
|
CV_Assert(!fs_impl->write_stack.empty());
|
|
struct_flags = fs_impl->write_stack.back().flags;
|
|
fs.state = FileNode::isMap(struct_flags) ? INSIDE_MAP + NAME_EXPECTED : VALUE_EXPECTED;
|
|
fs.elname = String();
|
|
}
|
|
else if( fs.state == NAME_EXPECTED + INSIDE_MAP )
|
|
{
|
|
if (!cv_isalpha(c) && c != '_')
|
|
CV_Error_( cv::Error::StsError, ("Incorrect element name %s; should start with a letter or '_'", _str) );
|
|
fs.elname = str;
|
|
fs.state = VALUE_EXPECTED + INSIDE_MAP;
|
|
}
|
|
else if( (fs.state & 3) == VALUE_EXPECTED )
|
|
{
|
|
if( c == '{' || c == '[' )
|
|
{
|
|
int struct_flags = c == '{' ? FileNode::MAP : FileNode::SEQ;
|
|
fs.state = struct_flags == FileNode::MAP ? INSIDE_MAP + NAME_EXPECTED : VALUE_EXPECTED;
|
|
_str++;
|
|
if( *_str == ':' )
|
|
{
|
|
_str++;
|
|
if( !*_str )
|
|
struct_flags |= FileNode::FLOW;
|
|
}
|
|
fs_impl->startWriteStruct(!fs.elname.empty() ? fs.elname.c_str() : 0, struct_flags, *_str ? _str : 0 );
|
|
fs.elname = String();
|
|
}
|
|
else
|
|
{
|
|
write( fs, fs.elname, (c == '\\' && (_str[1] == '{' || _str[1] == '}' ||
|
|
_str[1] == '[' || _str[1] == ']')) ? String(_str+1) : str );
|
|
if( fs.state == INSIDE_MAP + VALUE_EXPECTED )
|
|
fs.state = INSIDE_MAP + NAME_EXPECTED;
|
|
}
|
|
}
|
|
else
|
|
CV_Error( cv::Error::StsError, "Invalid fs.state" );
|
|
return fs;
|
|
}
|
|
|
|
|
|
FileNode::FileNode()
|
|
: fs(NULL)
|
|
{
|
|
blockIdx = ofs = 0;
|
|
}
|
|
|
|
FileNode::FileNode(FileStorage::Impl* _fs, size_t _blockIdx, size_t _ofs)
|
|
: fs(_fs)
|
|
{
|
|
blockIdx = _blockIdx;
|
|
ofs = _ofs;
|
|
}
|
|
|
|
FileNode::FileNode(const FileStorage* _fs, size_t _blockIdx, size_t _ofs)
|
|
: FileNode(_fs->p.get(), _blockIdx, _ofs)
|
|
{
|
|
// nothing
|
|
}
|
|
|
|
FileNode::FileNode(const FileNode& node)
|
|
{
|
|
fs = node.fs;
|
|
blockIdx = node.blockIdx;
|
|
ofs = node.ofs;
|
|
}
|
|
|
|
FileNode& FileNode::operator=(const FileNode& node)
|
|
{
|
|
fs = node.fs;
|
|
blockIdx = node.blockIdx;
|
|
ofs = node.ofs;
|
|
return *this;
|
|
}
|
|
|
|
FileNode FileNode::operator[](const std::string& nodename) const
|
|
{
|
|
if(!fs)
|
|
return FileNode();
|
|
|
|
CV_Assert( isMap() );
|
|
|
|
unsigned key = fs->getStringOfs(nodename);
|
|
size_t i, sz = size();
|
|
FileNodeIterator it = begin();
|
|
|
|
for( i = 0; i < sz; i++, ++it )
|
|
{
|
|
FileNode n = *it;
|
|
const uchar* p = n.ptr();
|
|
unsigned key2 = (unsigned)readInt(p + 1);
|
|
CV_Assert( key2 < fs->str_hash_data.size() );
|
|
if( key == key2 )
|
|
return n;
|
|
}
|
|
return FileNode();
|
|
}
|
|
|
|
FileNode FileNode::operator[](const char* nodename) const
|
|
{
|
|
return this->operator[](std::string(nodename));
|
|
}
|
|
|
|
FileNode FileNode::operator[](int i) const
|
|
{
|
|
if(!fs)
|
|
return FileNode();
|
|
|
|
CV_Assert( isSeq() );
|
|
|
|
int sz = (int)size();
|
|
CV_Assert( 0 <= i && i < sz );
|
|
|
|
FileNodeIterator it = begin();
|
|
it += i;
|
|
|
|
return *it;
|
|
}
|
|
|
|
std::vector<String> FileNode::keys() const
|
|
{
|
|
CV_Assert(isMap());
|
|
|
|
std::vector<String> res;
|
|
res.reserve(size());
|
|
for (FileNodeIterator it = begin(); it != end(); ++it)
|
|
{
|
|
res.push_back((*it).name());
|
|
}
|
|
return res;
|
|
}
|
|
|
|
int FileNode::type() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if(!p)
|
|
return NONE;
|
|
return (*p & TYPE_MASK);
|
|
}
|
|
|
|
bool FileNode::isMap(int flags) { return (flags & TYPE_MASK) == MAP; }
|
|
bool FileNode::isSeq(int flags) { return (flags & TYPE_MASK) == SEQ; }
|
|
bool FileNode::isCollection(int flags) { return isMap(flags) || isSeq(flags); }
|
|
bool FileNode::isFlow(int flags) { return (flags & FLOW) != 0; }
|
|
bool FileNode::isEmptyCollection(int flags) { return (flags & EMPTY) != 0; }
|
|
|
|
bool FileNode::empty() const { return fs == 0; }
|
|
bool FileNode::isNone() const { return type() == NONE; }
|
|
bool FileNode::isSeq() const { return type() == SEQ; }
|
|
bool FileNode::isMap() const { return type() == MAP; }
|
|
bool FileNode::isInt() const { return type() == INT; }
|
|
bool FileNode::isReal() const { return type() == REAL; }
|
|
bool FileNode::isString() const { return type() == STRING; }
|
|
bool FileNode::isNamed() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if(!p)
|
|
return false;
|
|
return (*p & NAMED) != 0;
|
|
}
|
|
|
|
std::string FileNode::name() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if(!p)
|
|
return std::string();
|
|
size_t nameofs = p[1] | (p[2]<<8) | (p[3]<<16) | (p[4]<<24);
|
|
return fs->getName(nameofs);
|
|
}
|
|
|
|
FileNode::operator int() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if(!p)
|
|
return 0;
|
|
int tag = *p;
|
|
int type = (tag & TYPE_MASK);
|
|
p += (tag & NAMED) ? 5 : 1;
|
|
|
|
if( type == INT )
|
|
{
|
|
return readInt(p);
|
|
}
|
|
else if( type == REAL )
|
|
{
|
|
return cvRound(readReal(p));
|
|
}
|
|
else
|
|
return 0x7fffffff;
|
|
}
|
|
|
|
FileNode::operator float() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if(!p)
|
|
return 0.f;
|
|
int tag = *p;
|
|
int type = (tag & TYPE_MASK);
|
|
p += (tag & NAMED) ? 5 : 1;
|
|
|
|
if( type == INT )
|
|
{
|
|
return (float)readInt(p);
|
|
}
|
|
else if( type == REAL )
|
|
{
|
|
return (float)readReal(p);
|
|
}
|
|
else
|
|
return FLT_MAX;
|
|
}
|
|
|
|
FileNode::operator double() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if(!p)
|
|
return 0.f;
|
|
int tag = *p;
|
|
int type = (tag & TYPE_MASK);
|
|
p += (tag & NAMED) ? 5 : 1;
|
|
|
|
if( type == INT )
|
|
{
|
|
return (double)readInt(p);
|
|
}
|
|
else if( type == REAL )
|
|
{
|
|
return readReal(p);
|
|
}
|
|
else
|
|
return DBL_MAX;
|
|
}
|
|
|
|
double FileNode::real() const { return double(*this); }
|
|
std::string FileNode::string() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if( !p || (*p & TYPE_MASK) != STRING )
|
|
return std::string();
|
|
p += (*p & NAMED) ? 5 : 1;
|
|
size_t sz = (size_t)(unsigned)readInt(p);
|
|
return std::string((const char*)(p + 4), sz - 1);
|
|
}
|
|
Mat FileNode::mat() const { Mat value; read(*this, value, Mat()); return value; }
|
|
|
|
FileNodeIterator FileNode::begin() const { return FileNodeIterator(*this, false); }
|
|
FileNodeIterator FileNode::end() const { return FileNodeIterator(*this, true); }
|
|
|
|
void FileNode::readRaw( const std::string& fmt, void* vec, size_t len ) const
|
|
{
|
|
FileNodeIterator it = begin();
|
|
it.readRaw( fmt, vec, len );
|
|
}
|
|
|
|
size_t FileNode::size() const
|
|
{
|
|
const uchar* p = ptr();
|
|
if( !p )
|
|
return 0;
|
|
int tag = *p;
|
|
int tp = tag & TYPE_MASK;
|
|
if( tp == MAP || tp == SEQ )
|
|
{
|
|
if( tag & NAMED )
|
|
p += 4;
|
|
return (size_t)(unsigned)readInt(p + 5);
|
|
}
|
|
return tp != NONE;
|
|
}
|
|
|
|
size_t FileNode::rawSize() const
|
|
{
|
|
const uchar* p0 = ptr(), *p = p0;
|
|
if( !p )
|
|
return 0;
|
|
int tag = *p++;
|
|
int tp = tag & TYPE_MASK;
|
|
if( tag & NAMED )
|
|
p += 4;
|
|
size_t sz0 = (size_t)(p - p0);
|
|
if( tp == INT )
|
|
return sz0 + 4;
|
|
if( tp == REAL )
|
|
return sz0 + 8;
|
|
if( tp == NONE )
|
|
return sz0;
|
|
CV_Assert( tp == STRING || tp == SEQ || tp == MAP );
|
|
return sz0 + 4 + readInt(p);
|
|
}
|
|
|
|
uchar* FileNode::ptr()
|
|
{
|
|
return !fs ? 0 : (uchar*)fs->getNodePtr(blockIdx, ofs);
|
|
}
|
|
|
|
const uchar* FileNode::ptr() const
|
|
{
|
|
return !fs ? 0 : fs->getNodePtr(blockIdx, ofs);
|
|
}
|
|
|
|
void FileNode::setValue( int type, const void* value, int len )
|
|
{
|
|
uchar *p = ptr();
|
|
CV_Assert(p != 0);
|
|
|
|
int tag = *p;
|
|
int current_type = tag & TYPE_MASK;
|
|
CV_Assert( current_type == NONE || current_type == type );
|
|
|
|
int sz = 1;
|
|
|
|
if( tag & NAMED )
|
|
sz += 4;
|
|
|
|
if( type == INT )
|
|
sz += 4;
|
|
else if( type == REAL )
|
|
sz += 8;
|
|
else if( type == STRING )
|
|
{
|
|
if( len < 0 )
|
|
len = (int)strlen((const char*)value);
|
|
sz += 4 + len + 1; // besides the string content,
|
|
// take the size (4 bytes) and the final '\0' into account
|
|
}
|
|
else
|
|
CV_Error(Error::StsNotImplemented, "Only scalar types can be dynamically assigned to a file node");
|
|
|
|
p = fs->reserveNodeSpace(*this, sz);
|
|
*p++ = (uchar)(type | (tag & NAMED));
|
|
if( tag & NAMED )
|
|
p += 4;
|
|
|
|
if( type == INT )
|
|
{
|
|
int ival = *(const int*)value;
|
|
writeInt(p, ival);
|
|
}
|
|
else if( type == REAL )
|
|
{
|
|
double dbval = *(const double*)value;
|
|
writeReal(p, dbval);
|
|
}
|
|
else if( type == STRING )
|
|
{
|
|
const char* str = (const char*)value;
|
|
writeInt(p, len + 1);
|
|
memcpy(p + 4, str, len);
|
|
p[4 + len] = (uchar)'\0';
|
|
}
|
|
}
|
|
|
|
FileNodeIterator::FileNodeIterator()
|
|
{
|
|
fs = 0;
|
|
blockIdx = 0;
|
|
ofs = 0;
|
|
blockSize = 0;
|
|
nodeNElems = 0;
|
|
idx = 0;
|
|
}
|
|
|
|
FileNodeIterator::FileNodeIterator( const FileNode& node, bool seekEnd )
|
|
{
|
|
fs = node.fs;
|
|
idx = 0;
|
|
if( !fs )
|
|
blockIdx = ofs = blockSize = nodeNElems = 0;
|
|
else
|
|
{
|
|
blockIdx = node.blockIdx;
|
|
ofs = node.ofs;
|
|
|
|
bool collection = node.isSeq() || node.isMap();
|
|
if( node.isNone() )
|
|
{
|
|
nodeNElems = 0;
|
|
}
|
|
else if( !collection )
|
|
{
|
|
nodeNElems = 1;
|
|
if( seekEnd )
|
|
{
|
|
idx = 1;
|
|
ofs += node.rawSize();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
nodeNElems = node.size();
|
|
const uchar* p0 = node.ptr(), *p = p0 + 1;
|
|
if(*p0 & FileNode::NAMED )
|
|
p += 4;
|
|
if( !seekEnd )
|
|
ofs += (p - p0) + 8;
|
|
else
|
|
{
|
|
size_t rawsz = (size_t)(unsigned)readInt(p);
|
|
ofs += (p - p0) + 4 + rawsz;
|
|
idx = nodeNElems;
|
|
}
|
|
}
|
|
fs->normalizeNodeOfs(blockIdx, ofs);
|
|
blockSize = fs->fs_data_blksz[blockIdx];
|
|
}
|
|
}
|
|
|
|
FileNodeIterator::FileNodeIterator(const FileNodeIterator& it)
|
|
{
|
|
fs = it.fs;
|
|
blockIdx = it.blockIdx;
|
|
ofs = it.ofs;
|
|
blockSize = it.blockSize;
|
|
nodeNElems = it.nodeNElems;
|
|
idx = it.idx;
|
|
}
|
|
|
|
FileNodeIterator& FileNodeIterator::operator=(const FileNodeIterator& it)
|
|
{
|
|
fs = it.fs;
|
|
blockIdx = it.blockIdx;
|
|
ofs = it.ofs;
|
|
blockSize = it.blockSize;
|
|
nodeNElems = it.nodeNElems;
|
|
idx = it.idx;
|
|
return *this;
|
|
}
|
|
|
|
FileNode FileNodeIterator::operator *() const
|
|
{
|
|
return FileNode(idx < nodeNElems ? fs : NULL, blockIdx, ofs);
|
|
}
|
|
|
|
FileNodeIterator& FileNodeIterator::operator ++ ()
|
|
{
|
|
if( idx == nodeNElems || !fs )
|
|
return *this;
|
|
idx++;
|
|
FileNode n(fs, blockIdx, ofs);
|
|
ofs += n.rawSize();
|
|
if( ofs >= blockSize )
|
|
{
|
|
fs->normalizeNodeOfs(blockIdx, ofs);
|
|
blockSize = fs->fs_data_blksz[blockIdx];
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
FileNodeIterator FileNodeIterator::operator ++ (int)
|
|
{
|
|
FileNodeIterator it = *this;
|
|
++(*this);
|
|
return it;
|
|
}
|
|
|
|
FileNodeIterator& FileNodeIterator::operator += (int _ofs)
|
|
{
|
|
CV_Assert( _ofs >= 0 );
|
|
for( ; _ofs > 0; _ofs-- )
|
|
this->operator ++();
|
|
return *this;
|
|
}
|
|
|
|
FileNodeIterator& FileNodeIterator::readRaw( const String& fmt, void* _data0, size_t maxsz)
|
|
{
|
|
if( fs && idx < nodeNElems )
|
|
{
|
|
uchar* data0 = (uchar*)_data0;
|
|
int fmt_pairs[CV_FS_MAX_FMT_PAIRS*2];
|
|
int fmt_pair_count = fs::decodeFormat( fmt.c_str(), fmt_pairs, CV_FS_MAX_FMT_PAIRS );
|
|
size_t esz = fs::calcStructSize( fmt.c_str(), 0 );
|
|
|
|
CV_Assert( maxsz % esz == 0 );
|
|
maxsz /= esz;
|
|
|
|
for( ; maxsz > 0; maxsz--, data0 += esz )
|
|
{
|
|
size_t offset = 0;
|
|
for( int k = 0; k < fmt_pair_count; k++ )
|
|
{
|
|
int elem_type = fmt_pairs[k*2+1];
|
|
int elem_size = CV_ELEM_SIZE(elem_type);
|
|
|
|
int count = fmt_pairs[k*2];
|
|
offset = alignSize( offset, elem_size );
|
|
uchar* data = data0 + offset;
|
|
|
|
for( int i = 0; i < count; i++, ++(*this) )
|
|
{
|
|
FileNode node = *(*this);
|
|
if( node.isInt() )
|
|
{
|
|
int ival = (int)node;
|
|
switch( elem_type )
|
|
{
|
|
case CV_8U:
|
|
*(uchar*)data = saturate_cast<uchar>(ival);
|
|
data++;
|
|
break;
|
|
case CV_8S:
|
|
*(char*)data = saturate_cast<schar>(ival);
|
|
data++;
|
|
break;
|
|
case CV_16U:
|
|
*(ushort*)data = saturate_cast<ushort>(ival);
|
|
data += sizeof(ushort);
|
|
break;
|
|
case CV_16S:
|
|
*(short*)data = saturate_cast<short>(ival);
|
|
data += sizeof(short);
|
|
break;
|
|
case CV_32S:
|
|
*(int*)data = ival;
|
|
data += sizeof(int);
|
|
break;
|
|
case CV_32F:
|
|
*(float*)data = (float)ival;
|
|
data += sizeof(float);
|
|
break;
|
|
case CV_64F:
|
|
*(double*)data = (double)ival;
|
|
data += sizeof(double);
|
|
break;
|
|
case CV_16F:
|
|
*(hfloat*)data = hfloat((float)ival);
|
|
data += sizeof(hfloat);
|
|
break;
|
|
default:
|
|
CV_Error( Error::StsUnsupportedFormat, "Unsupported type" );
|
|
}
|
|
}
|
|
else if( node.isReal() )
|
|
{
|
|
double fval = (double)node;
|
|
|
|
switch( elem_type )
|
|
{
|
|
case CV_8U:
|
|
*(uchar*)data = saturate_cast<uchar>(fval);
|
|
data++;
|
|
break;
|
|
case CV_8S:
|
|
*(char*)data = saturate_cast<schar>(fval);
|
|
data++;
|
|
break;
|
|
case CV_16U:
|
|
*(ushort*)data = saturate_cast<ushort>(fval);
|
|
data += sizeof(ushort);
|
|
break;
|
|
case CV_16S:
|
|
*(short*)data = saturate_cast<short>(fval);
|
|
data += sizeof(short);
|
|
break;
|
|
case CV_32S:
|
|
*(int*)data = saturate_cast<int>(fval);
|
|
data += sizeof(int);
|
|
break;
|
|
case CV_32F:
|
|
*(float*)data = (float)fval;
|
|
data += sizeof(float);
|
|
break;
|
|
case CV_64F:
|
|
*(double*)data = fval;
|
|
data += sizeof(double);
|
|
break;
|
|
case CV_16F:
|
|
*(hfloat*)data = hfloat((float)fval);
|
|
data += sizeof(hfloat);
|
|
break;
|
|
default:
|
|
CV_Error( Error::StsUnsupportedFormat, "Unsupported type" );
|
|
}
|
|
}
|
|
else
|
|
CV_Error( Error::StsError, "readRawData can only be used to read plain sequences of numbers" );
|
|
}
|
|
offset = (int)(data - data0);
|
|
}
|
|
}
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
bool FileNodeIterator::equalTo(const FileNodeIterator& it) const
|
|
{
|
|
return fs == it.fs && blockIdx == it.blockIdx && ofs == it.ofs &&
|
|
idx == it.idx && nodeNElems == it.nodeNElems;
|
|
}
|
|
|
|
size_t FileNodeIterator::remaining() const
|
|
{
|
|
return nodeNElems - idx;
|
|
}
|
|
|
|
bool operator == ( const FileNodeIterator& it1, const FileNodeIterator& it2 )
|
|
{
|
|
return it1.equalTo(it2);
|
|
}
|
|
|
|
bool operator != ( const FileNodeIterator& it1, const FileNodeIterator& it2 )
|
|
{
|
|
return !it1.equalTo(it2);
|
|
}
|
|
|
|
void read(const FileNode& node, int& val, int default_val)
|
|
{
|
|
val = default_val;
|
|
if( !node.empty() )
|
|
{
|
|
val = (int)node;
|
|
}
|
|
}
|
|
|
|
void read(const FileNode& node, double& val, double default_val)
|
|
{
|
|
val = default_val;
|
|
if( !node.empty() )
|
|
{
|
|
val = (double)node;
|
|
}
|
|
}
|
|
|
|
void read(const FileNode& node, float& val, float default_val)
|
|
{
|
|
val = default_val;
|
|
if( !node.empty() )
|
|
{
|
|
val = (float)node;
|
|
}
|
|
}
|
|
|
|
void read(const FileNode& node, std::string& val, const std::string& default_val)
|
|
{
|
|
val = default_val;
|
|
if( !node.empty() )
|
|
{
|
|
val = (std::string)node;
|
|
}
|
|
}
|
|
|
|
FileStorage_API::~FileStorage_API() {}
|
|
|
|
namespace internal
|
|
{
|
|
|
|
WriteStructContext::WriteStructContext(FileStorage& _fs, const std::string& name,
|
|
int flags, const std::string& typeName)
|
|
{
|
|
fs = &_fs;
|
|
fs->startWriteStruct(name, flags, typeName);
|
|
}
|
|
|
|
WriteStructContext::~WriteStructContext()
|
|
{
|
|
fs->endWriteStruct();
|
|
}
|
|
|
|
}
|
|
|
|
}
|