Reorganized the directory structure
This commit is contained in:
20
lib/xli/Makefile.am
Normal file
20
lib/xli/Makefile.am
Normal file
@ -0,0 +1,20 @@
|
||||
AUTOMAKE_OPTIONS = nostdinc
|
||||
|
||||
AM_CPPFLAGS = \
|
||||
-I$(top_builddir)/include \
|
||||
-I$(top_srcdir)/include
|
||||
|
||||
lib_LTLIBRARIES = libqsexli.la
|
||||
libqsexli_la_SOURCES = xli-prv.h xli.c err.c \
|
||||
read.c read-ini.c read-json.c \
|
||||
write.c write-ini.c write-json.c \
|
||||
std.c json.c
|
||||
libqsexli_la_LDFLAGS = -L../cmn -L../si -version-info 1:0:0 -no-undefined
|
||||
libqsexli_la_LIBADD = -lqsesi -lqsecmn
|
||||
libqsexli_la_DEPENDENCIES = ../../lib/si/libqsesi.la ../../lib/cmn/libqsecmn.la
|
||||
|
||||
|
||||
if ENABLE_CXX
|
||||
libqsexli_la_SOURCES += SkvEnv.cpp
|
||||
endif
|
||||
|
785
lib/xli/Makefile.in
Normal file
785
lib/xli/Makefile.in
Normal file
@ -0,0 +1,785 @@
|
||||
# Makefile.in generated by automake 1.16.2 from Makefile.am.
|
||||
# @configure_input@
|
||||
|
||||
# Copyright (C) 1994-2020 Free Software Foundation, Inc.
|
||||
|
||||
# This Makefile.in is free software; the Free Software Foundation
|
||||
# gives unlimited permission to copy and/or distribute it,
|
||||
# with or without modifications, as long as this notice is preserved.
|
||||
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY, to the extent permitted by law; without
|
||||
# even the implied warranty of MERCHANTABILITY or FITNESS FOR A
|
||||
# PARTICULAR PURPOSE.
|
||||
|
||||
@SET_MAKE@
|
||||
|
||||
VPATH = @srcdir@
|
||||
am__is_gnu_make = { \
|
||||
if test -z '$(MAKELEVEL)'; then \
|
||||
false; \
|
||||
elif test -n '$(MAKE_HOST)'; then \
|
||||
true; \
|
||||
elif test -n '$(MAKE_VERSION)' && test -n '$(CURDIR)'; then \
|
||||
true; \
|
||||
else \
|
||||
false; \
|
||||
fi; \
|
||||
}
|
||||
am__make_running_with_option = \
|
||||
case $${target_option-} in \
|
||||
?) ;; \
|
||||
*) echo "am__make_running_with_option: internal error: invalid" \
|
||||
"target option '$${target_option-}' specified" >&2; \
|
||||
exit 1;; \
|
||||
esac; \
|
||||
has_opt=no; \
|
||||
sane_makeflags=$$MAKEFLAGS; \
|
||||
if $(am__is_gnu_make); then \
|
||||
sane_makeflags=$$MFLAGS; \
|
||||
else \
|
||||
case $$MAKEFLAGS in \
|
||||
*\\[\ \ ]*) \
|
||||
bs=\\; \
|
||||
sane_makeflags=`printf '%s\n' "$$MAKEFLAGS" \
|
||||
| sed "s/$$bs$$bs[$$bs $$bs ]*//g"`;; \
|
||||
esac; \
|
||||
fi; \
|
||||
skip_next=no; \
|
||||
strip_trailopt () \
|
||||
{ \
|
||||
flg=`printf '%s\n' "$$flg" | sed "s/$$1.*$$//"`; \
|
||||
}; \
|
||||
for flg in $$sane_makeflags; do \
|
||||
test $$skip_next = yes && { skip_next=no; continue; }; \
|
||||
case $$flg in \
|
||||
*=*|--*) continue;; \
|
||||
-*I) strip_trailopt 'I'; skip_next=yes;; \
|
||||
-*I?*) strip_trailopt 'I';; \
|
||||
-*O) strip_trailopt 'O'; skip_next=yes;; \
|
||||
-*O?*) strip_trailopt 'O';; \
|
||||
-*l) strip_trailopt 'l'; skip_next=yes;; \
|
||||
-*l?*) strip_trailopt 'l';; \
|
||||
-[dEDm]) skip_next=yes;; \
|
||||
-[JT]) skip_next=yes;; \
|
||||
esac; \
|
||||
case $$flg in \
|
||||
*$$target_option*) has_opt=yes; break;; \
|
||||
esac; \
|
||||
done; \
|
||||
test $$has_opt = yes
|
||||
am__make_dryrun = (target_option=n; $(am__make_running_with_option))
|
||||
am__make_keepgoing = (target_option=k; $(am__make_running_with_option))
|
||||
pkgdatadir = $(datadir)/@PACKAGE@
|
||||
pkgincludedir = $(includedir)/@PACKAGE@
|
||||
pkglibdir = $(libdir)/@PACKAGE@
|
||||
pkglibexecdir = $(libexecdir)/@PACKAGE@
|
||||
am__cd = CDPATH="$${ZSH_VERSION+.}$(PATH_SEPARATOR)" && cd
|
||||
install_sh_DATA = $(install_sh) -c -m 644
|
||||
install_sh_PROGRAM = $(install_sh) -c
|
||||
install_sh_SCRIPT = $(install_sh) -c
|
||||
INSTALL_HEADER = $(INSTALL_DATA)
|
||||
transform = $(program_transform_name)
|
||||
NORMAL_INSTALL = :
|
||||
PRE_INSTALL = :
|
||||
POST_INSTALL = :
|
||||
NORMAL_UNINSTALL = :
|
||||
PRE_UNINSTALL = :
|
||||
POST_UNINSTALL = :
|
||||
build_triplet = @build@
|
||||
host_triplet = @host@
|
||||
@ENABLE_CXX_TRUE@am__append_1 = SkvEnv.cpp
|
||||
subdir = lib/xli
|
||||
ACLOCAL_M4 = $(top_srcdir)/aclocal.m4
|
||||
am__aclocal_m4_deps = $(top_srcdir)/m4/ax_check_sign.m4 \
|
||||
$(top_srcdir)/m4/ax_cxx_compile_stdcxx.m4 \
|
||||
$(top_srcdir)/m4/ax_cxx_namespace.m4 \
|
||||
$(top_srcdir)/m4/ax_lib_mysql.m4 $(top_srcdir)/m4/ax_numval.m4 \
|
||||
$(top_srcdir)/m4/ax_pthread.m4 $(top_srcdir)/m4/libtool.m4 \
|
||||
$(top_srcdir)/m4/ltoptions.m4 $(top_srcdir)/m4/ltsugar.m4 \
|
||||
$(top_srcdir)/m4/ltversion.m4 $(top_srcdir)/m4/lt~obsolete.m4 \
|
||||
$(top_srcdir)/m4/lx_find_mpi.m4 \
|
||||
$(top_srcdir)/m4/qse_try_cflags.m4 $(top_srcdir)/configure.ac
|
||||
am__configure_deps = $(am__aclocal_m4_deps) $(CONFIGURE_DEPENDENCIES) \
|
||||
$(ACLOCAL_M4)
|
||||
DIST_COMMON = $(srcdir)/Makefile.am $(am__DIST_COMMON)
|
||||
mkinstalldirs = $(install_sh) -d
|
||||
CONFIG_HEADER = $(top_builddir)/include/qse/config.h
|
||||
CONFIG_CLEAN_FILES =
|
||||
CONFIG_CLEAN_VPATH_FILES =
|
||||
am__vpath_adj_setup = srcdirstrip=`echo "$(srcdir)" | sed 's|.|.|g'`;
|
||||
am__vpath_adj = case $$p in \
|
||||
$(srcdir)/*) f=`echo "$$p" | sed "s|^$$srcdirstrip/||"`;; \
|
||||
*) f=$$p;; \
|
||||
esac;
|
||||
am__strip_dir = f=`echo $$p | sed -e 's|^.*/||'`;
|
||||
am__install_max = 40
|
||||
am__nobase_strip_setup = \
|
||||
srcdirstrip=`echo "$(srcdir)" | sed 's/[].[^$$\\*|]/\\\\&/g'`
|
||||
am__nobase_strip = \
|
||||
for p in $$list; do echo "$$p"; done | sed -e "s|$$srcdirstrip/||"
|
||||
am__nobase_list = $(am__nobase_strip_setup); \
|
||||
for p in $$list; do echo "$$p $$p"; done | \
|
||||
sed "s| $$srcdirstrip/| |;"' / .*\//!s/ .*/ ./; s,\( .*\)/[^/]*$$,\1,' | \
|
||||
$(AWK) 'BEGIN { files["."] = "" } { files[$$2] = files[$$2] " " $$1; \
|
||||
if (++n[$$2] == $(am__install_max)) \
|
||||
{ print $$2, files[$$2]; n[$$2] = 0; files[$$2] = "" } } \
|
||||
END { for (dir in files) print dir, files[dir] }'
|
||||
am__base_list = \
|
||||
sed '$$!N;$$!N;$$!N;$$!N;$$!N;$$!N;$$!N;s/\n/ /g' | \
|
||||
sed '$$!N;$$!N;$$!N;$$!N;s/\n/ /g'
|
||||
am__uninstall_files_from_dir = { \
|
||||
test -z "$$files" \
|
||||
|| { test ! -d "$$dir" && test ! -f "$$dir" && test ! -r "$$dir"; } \
|
||||
|| { echo " ( cd '$$dir' && rm -f" $$files ")"; \
|
||||
$(am__cd) "$$dir" && rm -f $$files; }; \
|
||||
}
|
||||
am__installdirs = "$(DESTDIR)$(libdir)"
|
||||
LTLIBRARIES = $(lib_LTLIBRARIES)
|
||||
am__libqsexli_la_SOURCES_DIST = xli-prv.h xli.c err.c read.c \
|
||||
read-ini.c read-json.c write.c write-ini.c write-json.c std.c \
|
||||
json.c SkvEnv.cpp
|
||||
@ENABLE_CXX_TRUE@am__objects_1 = SkvEnv.lo
|
||||
am_libqsexli_la_OBJECTS = xli.lo err.lo read.lo read-ini.lo \
|
||||
read-json.lo write.lo write-ini.lo write-json.lo std.lo \
|
||||
json.lo $(am__objects_1)
|
||||
libqsexli_la_OBJECTS = $(am_libqsexli_la_OBJECTS)
|
||||
AM_V_lt = $(am__v_lt_@AM_V@)
|
||||
am__v_lt_ = $(am__v_lt_@AM_DEFAULT_V@)
|
||||
am__v_lt_0 = --silent
|
||||
am__v_lt_1 =
|
||||
libqsexli_la_LINK = $(LIBTOOL) $(AM_V_lt) --tag=CXX $(AM_LIBTOOLFLAGS) \
|
||||
$(LIBTOOLFLAGS) --mode=link $(CXXLD) $(AM_CXXFLAGS) \
|
||||
$(CXXFLAGS) $(libqsexli_la_LDFLAGS) $(LDFLAGS) -o $@
|
||||
AM_V_P = $(am__v_P_@AM_V@)
|
||||
am__v_P_ = $(am__v_P_@AM_DEFAULT_V@)
|
||||
am__v_P_0 = false
|
||||
am__v_P_1 = :
|
||||
AM_V_GEN = $(am__v_GEN_@AM_V@)
|
||||
am__v_GEN_ = $(am__v_GEN_@AM_DEFAULT_V@)
|
||||
am__v_GEN_0 = @echo " GEN " $@;
|
||||
am__v_GEN_1 =
|
||||
AM_V_at = $(am__v_at_@AM_V@)
|
||||
am__v_at_ = $(am__v_at_@AM_DEFAULT_V@)
|
||||
am__v_at_0 = @
|
||||
am__v_at_1 =
|
||||
DEFAULT_INCLUDES =
|
||||
depcomp = $(SHELL) $(top_srcdir)/ac/depcomp
|
||||
am__maybe_remake_depfiles = depfiles
|
||||
am__depfiles_remade = ./$(DEPDIR)/SkvEnv.Plo ./$(DEPDIR)/err.Plo \
|
||||
./$(DEPDIR)/json.Plo ./$(DEPDIR)/read-ini.Plo \
|
||||
./$(DEPDIR)/read-json.Plo ./$(DEPDIR)/read.Plo \
|
||||
./$(DEPDIR)/std.Plo ./$(DEPDIR)/write-ini.Plo \
|
||||
./$(DEPDIR)/write-json.Plo ./$(DEPDIR)/write.Plo \
|
||||
./$(DEPDIR)/xli.Plo
|
||||
am__mv = mv -f
|
||||
COMPILE = $(CC) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) \
|
||||
$(CPPFLAGS) $(AM_CFLAGS) $(CFLAGS)
|
||||
LTCOMPILE = $(LIBTOOL) $(AM_V_lt) --tag=CC $(AM_LIBTOOLFLAGS) \
|
||||
$(LIBTOOLFLAGS) --mode=compile $(CC) $(DEFS) \
|
||||
$(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) \
|
||||
$(AM_CFLAGS) $(CFLAGS)
|
||||
AM_V_CC = $(am__v_CC_@AM_V@)
|
||||
am__v_CC_ = $(am__v_CC_@AM_DEFAULT_V@)
|
||||
am__v_CC_0 = @echo " CC " $@;
|
||||
am__v_CC_1 =
|
||||
CCLD = $(CC)
|
||||
LINK = $(LIBTOOL) $(AM_V_lt) --tag=CC $(AM_LIBTOOLFLAGS) \
|
||||
$(LIBTOOLFLAGS) --mode=link $(CCLD) $(AM_CFLAGS) $(CFLAGS) \
|
||||
$(AM_LDFLAGS) $(LDFLAGS) -o $@
|
||||
AM_V_CCLD = $(am__v_CCLD_@AM_V@)
|
||||
am__v_CCLD_ = $(am__v_CCLD_@AM_DEFAULT_V@)
|
||||
am__v_CCLD_0 = @echo " CCLD " $@;
|
||||
am__v_CCLD_1 =
|
||||
CXXCOMPILE = $(CXX) $(DEFS) $(DEFAULT_INCLUDES) $(INCLUDES) \
|
||||
$(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CXXFLAGS) $(CXXFLAGS)
|
||||
LTCXXCOMPILE = $(LIBTOOL) $(AM_V_lt) --tag=CXX $(AM_LIBTOOLFLAGS) \
|
||||
$(LIBTOOLFLAGS) --mode=compile $(CXX) $(DEFS) \
|
||||
$(DEFAULT_INCLUDES) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) \
|
||||
$(AM_CXXFLAGS) $(CXXFLAGS)
|
||||
AM_V_CXX = $(am__v_CXX_@AM_V@)
|
||||
am__v_CXX_ = $(am__v_CXX_@AM_DEFAULT_V@)
|
||||
am__v_CXX_0 = @echo " CXX " $@;
|
||||
am__v_CXX_1 =
|
||||
CXXLD = $(CXX)
|
||||
CXXLINK = $(LIBTOOL) $(AM_V_lt) --tag=CXX $(AM_LIBTOOLFLAGS) \
|
||||
$(LIBTOOLFLAGS) --mode=link $(CXXLD) $(AM_CXXFLAGS) \
|
||||
$(CXXFLAGS) $(AM_LDFLAGS) $(LDFLAGS) -o $@
|
||||
AM_V_CXXLD = $(am__v_CXXLD_@AM_V@)
|
||||
am__v_CXXLD_ = $(am__v_CXXLD_@AM_DEFAULT_V@)
|
||||
am__v_CXXLD_0 = @echo " CXXLD " $@;
|
||||
am__v_CXXLD_1 =
|
||||
SOURCES = $(libqsexli_la_SOURCES)
|
||||
DIST_SOURCES = $(am__libqsexli_la_SOURCES_DIST)
|
||||
am__can_run_installinfo = \
|
||||
case $$AM_UPDATE_INFO_DIR in \
|
||||
n|no|NO) false;; \
|
||||
*) (install-info --version) >/dev/null 2>&1;; \
|
||||
esac
|
||||
am__tagged_files = $(HEADERS) $(SOURCES) $(TAGS_FILES) $(LISP)
|
||||
# Read a list of newline-separated strings from the standard input,
|
||||
# and print each of them once, without duplicates. Input order is
|
||||
# *not* preserved.
|
||||
am__uniquify_input = $(AWK) '\
|
||||
BEGIN { nonempty = 0; } \
|
||||
{ items[$$0] = 1; nonempty = 1; } \
|
||||
END { if (nonempty) { for (i in items) print i; }; } \
|
||||
'
|
||||
# Make sure the list of sources is unique. This is necessary because,
|
||||
# e.g., the same source file might be shared among _SOURCES variables
|
||||
# for different programs/libraries.
|
||||
am__define_uniq_tagged_files = \
|
||||
list='$(am__tagged_files)'; \
|
||||
unique=`for i in $$list; do \
|
||||
if test -f "$$i"; then echo $$i; else echo $(srcdir)/$$i; fi; \
|
||||
done | $(am__uniquify_input)`
|
||||
ETAGS = etags
|
||||
CTAGS = ctags
|
||||
am__DIST_COMMON = $(srcdir)/Makefile.in $(top_srcdir)/ac/depcomp
|
||||
DISTFILES = $(DIST_COMMON) $(DIST_SOURCES) $(TEXINFOS) $(EXTRA_DIST)
|
||||
ACLOCAL = @ACLOCAL@
|
||||
AMTAR = @AMTAR@
|
||||
AM_DEFAULT_VERBOSITY = @AM_DEFAULT_VERBOSITY@
|
||||
AR = @AR@
|
||||
AUTOCONF = @AUTOCONF@
|
||||
AUTOHEADER = @AUTOHEADER@
|
||||
AUTOMAKE = @AUTOMAKE@
|
||||
AWK = @AWK@
|
||||
BUILD_MODE = @BUILD_MODE@
|
||||
CC = @CC@
|
||||
CCDEPMODE = @CCDEPMODE@
|
||||
CFLAGS = @CFLAGS@
|
||||
CPP = @CPP@
|
||||
CPPFLAGS = @CPPFLAGS@
|
||||
CXX = @CXX@
|
||||
CXXCPP = @CXXCPP@
|
||||
CXXDEPMODE = @CXXDEPMODE@
|
||||
CXXFLAGS = @CXXFLAGS@
|
||||
CYGPATH_W = @CYGPATH_W@
|
||||
DEFS = @DEFS@
|
||||
DEPDIR = @DEPDIR@
|
||||
DLLTOOL = @DLLTOOL@
|
||||
DL_LIBS = @DL_LIBS@
|
||||
DSYMUTIL = @DSYMUTIL@
|
||||
DUMPBIN = @DUMPBIN@
|
||||
ECHO = @ECHO@
|
||||
ECHO_C = @ECHO_C@
|
||||
ECHO_N = @ECHO_N@
|
||||
ECHO_T = @ECHO_T@
|
||||
EGREP = @EGREP@
|
||||
EXEEXT = @EXEEXT@
|
||||
FGREP = @FGREP@
|
||||
GREP = @GREP@
|
||||
HAVE_CXX = @HAVE_CXX@
|
||||
HAVE_CXX11 = @HAVE_CXX11@
|
||||
INSTALL = @INSTALL@
|
||||
INSTALL_DATA = @INSTALL_DATA@
|
||||
INSTALL_PROGRAM = @INSTALL_PROGRAM@
|
||||
INSTALL_SCRIPT = @INSTALL_SCRIPT@
|
||||
INSTALL_STRIP_PROGRAM = @INSTALL_STRIP_PROGRAM@
|
||||
LD = @LD@
|
||||
LDFLAGS = @LDFLAGS@
|
||||
LIBM = @LIBM@
|
||||
LIBOBJS = @LIBOBJS@
|
||||
LIBS = @LIBS@
|
||||
LIBTOOL = @LIBTOOL@
|
||||
LIBTOOL_DEPS = @LIBTOOL_DEPS@
|
||||
LIPO = @LIPO@
|
||||
LN_S = @LN_S@
|
||||
LTDL_LIBS = @LTDL_LIBS@
|
||||
LTLIBOBJS = @LTLIBOBJS@
|
||||
LT_SYS_LIBRARY_PATH = @LT_SYS_LIBRARY_PATH@
|
||||
MAKEINFO = @MAKEINFO@
|
||||
MANIFEST_TOOL = @MANIFEST_TOOL@
|
||||
MKDIR_P = @MKDIR_P@
|
||||
MPICC = @MPICC@
|
||||
MPI_CFLAGS = @MPI_CFLAGS@
|
||||
MPI_CLDFLAGS = @MPI_CLDFLAGS@
|
||||
MYSQL_CFLAGS = @MYSQL_CFLAGS@
|
||||
MYSQL_CONFIG = @MYSQL_CONFIG@
|
||||
MYSQL_LDFLAGS = @MYSQL_LDFLAGS@
|
||||
MYSQL_LIBS = @MYSQL_LIBS@
|
||||
MYSQL_VERSION = @MYSQL_VERSION@
|
||||
NM = @NM@
|
||||
NMEDIT = @NMEDIT@
|
||||
OBJDUMP = @OBJDUMP@
|
||||
OBJEXT = @OBJEXT@
|
||||
OTOOL = @OTOOL@
|
||||
OTOOL64 = @OTOOL64@
|
||||
PACKAGE = @PACKAGE@
|
||||
PACKAGE_BUGREPORT = @PACKAGE_BUGREPORT@
|
||||
PACKAGE_NAME = @PACKAGE_NAME@
|
||||
PACKAGE_STRING = @PACKAGE_STRING@
|
||||
PACKAGE_TARNAME = @PACKAGE_TARNAME@
|
||||
PACKAGE_URL = @PACKAGE_URL@
|
||||
PACKAGE_VERSION = @PACKAGE_VERSION@
|
||||
PACKAGE_VERSION_MAJOR = @PACKAGE_VERSION_MAJOR@
|
||||
PACKAGE_VERSION_MINOR = @PACKAGE_VERSION_MINOR@
|
||||
PACKAGE_VERSION_PATCH = @PACKAGE_VERSION_PATCH@
|
||||
PATH_SEPARATOR = @PATH_SEPARATOR@
|
||||
PTHREAD_CC = @PTHREAD_CC@
|
||||
PTHREAD_CFLAGS = @PTHREAD_CFLAGS@
|
||||
PTHREAD_LIBS = @PTHREAD_LIBS@
|
||||
QSE_PROJECT_AUTHOR = @QSE_PROJECT_AUTHOR@
|
||||
QSE_PROJECT_URL = @QSE_PROJECT_URL@
|
||||
QUADMATH_LIBS = @QUADMATH_LIBS@
|
||||
RANLIB = @RANLIB@
|
||||
RM = @RM@
|
||||
RMDIR = @RMDIR@
|
||||
SED = @SED@
|
||||
SENDFILE_LIBS = @SENDFILE_LIBS@
|
||||
SET_MAKE = @SET_MAKE@
|
||||
SHELL = @SHELL@
|
||||
SOCKET_LIBS = @SOCKET_LIBS@
|
||||
SSL_LIBS = @SSL_LIBS@
|
||||
STRIP = @STRIP@
|
||||
TRUE = @TRUE@
|
||||
UCI_LIBS = @UCI_LIBS@
|
||||
UNICOWS_LIBS = @UNICOWS_LIBS@
|
||||
VERSION = @VERSION@
|
||||
abs_builddir = @abs_builddir@
|
||||
abs_srcdir = @abs_srcdir@
|
||||
abs_top_builddir = @abs_top_builddir@
|
||||
abs_top_srcdir = @abs_top_srcdir@
|
||||
ac_ct_AR = @ac_ct_AR@
|
||||
ac_ct_CC = @ac_ct_CC@
|
||||
ac_ct_CXX = @ac_ct_CXX@
|
||||
ac_ct_DUMPBIN = @ac_ct_DUMPBIN@
|
||||
am__include = @am__include@
|
||||
am__leading_dot = @am__leading_dot@
|
||||
am__quote = @am__quote@
|
||||
am__tar = @am__tar@
|
||||
am__untar = @am__untar@
|
||||
ax_pthread_config = @ax_pthread_config@
|
||||
bindir = @bindir@
|
||||
build = @build@
|
||||
build_alias = @build_alias@
|
||||
build_cpu = @build_cpu@
|
||||
build_os = @build_os@
|
||||
build_vendor = @build_vendor@
|
||||
builddir = @builddir@
|
||||
datadir = @datadir@
|
||||
datarootdir = @datarootdir@
|
||||
docdir = @docdir@
|
||||
dvidir = @dvidir@
|
||||
exec_prefix = @exec_prefix@
|
||||
host = @host@
|
||||
host_alias = @host_alias@
|
||||
host_cpu = @host_cpu@
|
||||
host_os = @host_os@
|
||||
host_vendor = @host_vendor@
|
||||
htmldir = @htmldir@
|
||||
includedir = @includedir@
|
||||
infodir = @infodir@
|
||||
install_sh = @install_sh@
|
||||
libdir = @libdir@
|
||||
libexecdir = @libexecdir@
|
||||
localedir = @localedir@
|
||||
localstatedir = @localstatedir@
|
||||
mandir = @mandir@
|
||||
mkdir_p = @mkdir_p@
|
||||
oldincludedir = @oldincludedir@
|
||||
pdfdir = @pdfdir@
|
||||
prefix = @prefix@
|
||||
program_transform_name = @program_transform_name@
|
||||
psdir = @psdir@
|
||||
runstatedir = @runstatedir@
|
||||
sbindir = @sbindir@
|
||||
sharedstatedir = @sharedstatedir@
|
||||
srcdir = @srcdir@
|
||||
sysconfdir = @sysconfdir@
|
||||
target_alias = @target_alias@
|
||||
top_build_prefix = @top_build_prefix@
|
||||
top_builddir = @top_builddir@
|
||||
top_srcdir = @top_srcdir@
|
||||
AUTOMAKE_OPTIONS = nostdinc
|
||||
AM_CPPFLAGS = \
|
||||
-I$(top_builddir)/include \
|
||||
-I$(top_srcdir)/include
|
||||
|
||||
lib_LTLIBRARIES = libqsexli.la
|
||||
libqsexli_la_SOURCES = xli-prv.h xli.c err.c read.c read-ini.c \
|
||||
read-json.c write.c write-ini.c write-json.c std.c json.c \
|
||||
$(am__append_1)
|
||||
libqsexli_la_LDFLAGS = -L../cmn -L../si -version-info 1:0:0 -no-undefined
|
||||
libqsexli_la_LIBADD = -lqsesi -lqsecmn
|
||||
libqsexli_la_DEPENDENCIES = ../../lib/si/libqsesi.la ../../lib/cmn/libqsecmn.la
|
||||
all: all-am
|
||||
|
||||
.SUFFIXES:
|
||||
.SUFFIXES: .c .cpp .lo .o .obj
|
||||
$(srcdir)/Makefile.in: $(srcdir)/Makefile.am $(am__configure_deps)
|
||||
@for dep in $?; do \
|
||||
case '$(am__configure_deps)' in \
|
||||
*$$dep*) \
|
||||
( cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh ) \
|
||||
&& { if test -f $@; then exit 0; else break; fi; }; \
|
||||
exit 1;; \
|
||||
esac; \
|
||||
done; \
|
||||
echo ' cd $(top_srcdir) && $(AUTOMAKE) --foreign lib/xli/Makefile'; \
|
||||
$(am__cd) $(top_srcdir) && \
|
||||
$(AUTOMAKE) --foreign lib/xli/Makefile
|
||||
Makefile: $(srcdir)/Makefile.in $(top_builddir)/config.status
|
||||
@case '$?' in \
|
||||
*config.status*) \
|
||||
cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh;; \
|
||||
*) \
|
||||
echo ' cd $(top_builddir) && $(SHELL) ./config.status $(subdir)/$@ $(am__maybe_remake_depfiles)'; \
|
||||
cd $(top_builddir) && $(SHELL) ./config.status $(subdir)/$@ $(am__maybe_remake_depfiles);; \
|
||||
esac;
|
||||
|
||||
$(top_builddir)/config.status: $(top_srcdir)/configure $(CONFIG_STATUS_DEPENDENCIES)
|
||||
cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh
|
||||
|
||||
$(top_srcdir)/configure: $(am__configure_deps)
|
||||
cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh
|
||||
$(ACLOCAL_M4): $(am__aclocal_m4_deps)
|
||||
cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh
|
||||
$(am__aclocal_m4_deps):
|
||||
|
||||
install-libLTLIBRARIES: $(lib_LTLIBRARIES)
|
||||
@$(NORMAL_INSTALL)
|
||||
@list='$(lib_LTLIBRARIES)'; test -n "$(libdir)" || list=; \
|
||||
list2=; for p in $$list; do \
|
||||
if test -f $$p; then \
|
||||
list2="$$list2 $$p"; \
|
||||
else :; fi; \
|
||||
done; \
|
||||
test -z "$$list2" || { \
|
||||
echo " $(MKDIR_P) '$(DESTDIR)$(libdir)'"; \
|
||||
$(MKDIR_P) "$(DESTDIR)$(libdir)" || exit 1; \
|
||||
echo " $(LIBTOOL) $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=install $(INSTALL) $(INSTALL_STRIP_FLAG) $$list2 '$(DESTDIR)$(libdir)'"; \
|
||||
$(LIBTOOL) $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=install $(INSTALL) $(INSTALL_STRIP_FLAG) $$list2 "$(DESTDIR)$(libdir)"; \
|
||||
}
|
||||
|
||||
uninstall-libLTLIBRARIES:
|
||||
@$(NORMAL_UNINSTALL)
|
||||
@list='$(lib_LTLIBRARIES)'; test -n "$(libdir)" || list=; \
|
||||
for p in $$list; do \
|
||||
$(am__strip_dir) \
|
||||
echo " $(LIBTOOL) $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=uninstall rm -f '$(DESTDIR)$(libdir)/$$f'"; \
|
||||
$(LIBTOOL) $(AM_LIBTOOLFLAGS) $(LIBTOOLFLAGS) --mode=uninstall rm -f "$(DESTDIR)$(libdir)/$$f"; \
|
||||
done
|
||||
|
||||
clean-libLTLIBRARIES:
|
||||
-test -z "$(lib_LTLIBRARIES)" || rm -f $(lib_LTLIBRARIES)
|
||||
@list='$(lib_LTLIBRARIES)'; \
|
||||
locs=`for p in $$list; do echo $$p; done | \
|
||||
sed 's|^[^/]*$$|.|; s|/[^/]*$$||; s|$$|/so_locations|' | \
|
||||
sort -u`; \
|
||||
test -z "$$locs" || { \
|
||||
echo rm -f $${locs}; \
|
||||
rm -f $${locs}; \
|
||||
}
|
||||
|
||||
libqsexli.la: $(libqsexli_la_OBJECTS) $(libqsexli_la_DEPENDENCIES) $(EXTRA_libqsexli_la_DEPENDENCIES)
|
||||
$(AM_V_CXXLD)$(libqsexli_la_LINK) -rpath $(libdir) $(libqsexli_la_OBJECTS) $(libqsexli_la_LIBADD) $(LIBS)
|
||||
|
||||
mostlyclean-compile:
|
||||
-rm -f *.$(OBJEXT)
|
||||
|
||||
distclean-compile:
|
||||
-rm -f *.tab.c
|
||||
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/SkvEnv.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/err.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/json.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/read-ini.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/read-json.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/read.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/std.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/write-ini.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/write-json.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/write.Plo@am__quote@ # am--include-marker
|
||||
@AMDEP_TRUE@@am__include@ @am__quote@./$(DEPDIR)/xli.Plo@am__quote@ # am--include-marker
|
||||
|
||||
$(am__depfiles_remade):
|
||||
@$(MKDIR_P) $(@D)
|
||||
@echo '# dummy' >$@-t && $(am__mv) $@-t $@
|
||||
|
||||
am--depfiles: $(am__depfiles_remade)
|
||||
|
||||
.c.o:
|
||||
@am__fastdepCC_TRUE@ $(AM_V_CC)$(COMPILE) -MT $@ -MD -MP -MF $(DEPDIR)/$*.Tpo -c -o $@ $<
|
||||
@am__fastdepCC_TRUE@ $(AM_V_at)$(am__mv) $(DEPDIR)/$*.Tpo $(DEPDIR)/$*.Po
|
||||
@AMDEP_TRUE@@am__fastdepCC_FALSE@ $(AM_V_CC)source='$<' object='$@' libtool=no @AMDEPBACKSLASH@
|
||||
@AMDEP_TRUE@@am__fastdepCC_FALSE@ DEPDIR=$(DEPDIR) $(CCDEPMODE) $(depcomp) @AMDEPBACKSLASH@
|
||||
@am__fastdepCC_FALSE@ $(AM_V_CC@am__nodep@)$(COMPILE) -c -o $@ $<
|
||||
|
||||
.c.obj:
|
||||
@am__fastdepCC_TRUE@ $(AM_V_CC)$(COMPILE) -MT $@ -MD -MP -MF $(DEPDIR)/$*.Tpo -c -o $@ `$(CYGPATH_W) '$<'`
|
||||
@am__fastdepCC_TRUE@ $(AM_V_at)$(am__mv) $(DEPDIR)/$*.Tpo $(DEPDIR)/$*.Po
|
||||
@AMDEP_TRUE@@am__fastdepCC_FALSE@ $(AM_V_CC)source='$<' object='$@' libtool=no @AMDEPBACKSLASH@
|
||||
@AMDEP_TRUE@@am__fastdepCC_FALSE@ DEPDIR=$(DEPDIR) $(CCDEPMODE) $(depcomp) @AMDEPBACKSLASH@
|
||||
@am__fastdepCC_FALSE@ $(AM_V_CC@am__nodep@)$(COMPILE) -c -o $@ `$(CYGPATH_W) '$<'`
|
||||
|
||||
.c.lo:
|
||||
@am__fastdepCC_TRUE@ $(AM_V_CC)$(LTCOMPILE) -MT $@ -MD -MP -MF $(DEPDIR)/$*.Tpo -c -o $@ $<
|
||||
@am__fastdepCC_TRUE@ $(AM_V_at)$(am__mv) $(DEPDIR)/$*.Tpo $(DEPDIR)/$*.Plo
|
||||
@AMDEP_TRUE@@am__fastdepCC_FALSE@ $(AM_V_CC)source='$<' object='$@' libtool=yes @AMDEPBACKSLASH@
|
||||
@AMDEP_TRUE@@am__fastdepCC_FALSE@ DEPDIR=$(DEPDIR) $(CCDEPMODE) $(depcomp) @AMDEPBACKSLASH@
|
||||
@am__fastdepCC_FALSE@ $(AM_V_CC@am__nodep@)$(LTCOMPILE) -c -o $@ $<
|
||||
|
||||
.cpp.o:
|
||||
@am__fastdepCXX_TRUE@ $(AM_V_CXX)$(CXXCOMPILE) -MT $@ -MD -MP -MF $(DEPDIR)/$*.Tpo -c -o $@ $<
|
||||
@am__fastdepCXX_TRUE@ $(AM_V_at)$(am__mv) $(DEPDIR)/$*.Tpo $(DEPDIR)/$*.Po
|
||||
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ $(AM_V_CXX)source='$<' object='$@' libtool=no @AMDEPBACKSLASH@
|
||||
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
|
||||
@am__fastdepCXX_FALSE@ $(AM_V_CXX@am__nodep@)$(CXXCOMPILE) -c -o $@ $<
|
||||
|
||||
.cpp.obj:
|
||||
@am__fastdepCXX_TRUE@ $(AM_V_CXX)$(CXXCOMPILE) -MT $@ -MD -MP -MF $(DEPDIR)/$*.Tpo -c -o $@ `$(CYGPATH_W) '$<'`
|
||||
@am__fastdepCXX_TRUE@ $(AM_V_at)$(am__mv) $(DEPDIR)/$*.Tpo $(DEPDIR)/$*.Po
|
||||
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ $(AM_V_CXX)source='$<' object='$@' libtool=no @AMDEPBACKSLASH@
|
||||
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
|
||||
@am__fastdepCXX_FALSE@ $(AM_V_CXX@am__nodep@)$(CXXCOMPILE) -c -o $@ `$(CYGPATH_W) '$<'`
|
||||
|
||||
.cpp.lo:
|
||||
@am__fastdepCXX_TRUE@ $(AM_V_CXX)$(LTCXXCOMPILE) -MT $@ -MD -MP -MF $(DEPDIR)/$*.Tpo -c -o $@ $<
|
||||
@am__fastdepCXX_TRUE@ $(AM_V_at)$(am__mv) $(DEPDIR)/$*.Tpo $(DEPDIR)/$*.Plo
|
||||
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ $(AM_V_CXX)source='$<' object='$@' libtool=yes @AMDEPBACKSLASH@
|
||||
@AMDEP_TRUE@@am__fastdepCXX_FALSE@ DEPDIR=$(DEPDIR) $(CXXDEPMODE) $(depcomp) @AMDEPBACKSLASH@
|
||||
@am__fastdepCXX_FALSE@ $(AM_V_CXX@am__nodep@)$(LTCXXCOMPILE) -c -o $@ $<
|
||||
|
||||
mostlyclean-libtool:
|
||||
-rm -f *.lo
|
||||
|
||||
clean-libtool:
|
||||
-rm -rf .libs _libs
|
||||
|
||||
ID: $(am__tagged_files)
|
||||
$(am__define_uniq_tagged_files); mkid -fID $$unique
|
||||
tags: tags-am
|
||||
TAGS: tags
|
||||
|
||||
tags-am: $(TAGS_DEPENDENCIES) $(am__tagged_files)
|
||||
set x; \
|
||||
here=`pwd`; \
|
||||
$(am__define_uniq_tagged_files); \
|
||||
shift; \
|
||||
if test -z "$(ETAGS_ARGS)$$*$$unique"; then :; else \
|
||||
test -n "$$unique" || unique=$$empty_fix; \
|
||||
if test $$# -gt 0; then \
|
||||
$(ETAGS) $(ETAGSFLAGS) $(AM_ETAGSFLAGS) $(ETAGS_ARGS) \
|
||||
"$$@" $$unique; \
|
||||
else \
|
||||
$(ETAGS) $(ETAGSFLAGS) $(AM_ETAGSFLAGS) $(ETAGS_ARGS) \
|
||||
$$unique; \
|
||||
fi; \
|
||||
fi
|
||||
ctags: ctags-am
|
||||
|
||||
CTAGS: ctags
|
||||
ctags-am: $(TAGS_DEPENDENCIES) $(am__tagged_files)
|
||||
$(am__define_uniq_tagged_files); \
|
||||
test -z "$(CTAGS_ARGS)$$unique" \
|
||||
|| $(CTAGS) $(CTAGSFLAGS) $(AM_CTAGSFLAGS) $(CTAGS_ARGS) \
|
||||
$$unique
|
||||
|
||||
GTAGS:
|
||||
here=`$(am__cd) $(top_builddir) && pwd` \
|
||||
&& $(am__cd) $(top_srcdir) \
|
||||
&& gtags -i $(GTAGS_ARGS) "$$here"
|
||||
cscopelist: cscopelist-am
|
||||
|
||||
cscopelist-am: $(am__tagged_files)
|
||||
list='$(am__tagged_files)'; \
|
||||
case "$(srcdir)" in \
|
||||
[\\/]* | ?:[\\/]*) sdir="$(srcdir)" ;; \
|
||||
*) sdir=$(subdir)/$(srcdir) ;; \
|
||||
esac; \
|
||||
for i in $$list; do \
|
||||
if test -f "$$i"; then \
|
||||
echo "$(subdir)/$$i"; \
|
||||
else \
|
||||
echo "$$sdir/$$i"; \
|
||||
fi; \
|
||||
done >> $(top_builddir)/cscope.files
|
||||
|
||||
distclean-tags:
|
||||
-rm -f TAGS ID GTAGS GRTAGS GSYMS GPATH tags
|
||||
|
||||
distdir: $(BUILT_SOURCES)
|
||||
$(MAKE) $(AM_MAKEFLAGS) distdir-am
|
||||
|
||||
distdir-am: $(DISTFILES)
|
||||
@srcdirstrip=`echo "$(srcdir)" | sed 's/[].[^$$\\*]/\\\\&/g'`; \
|
||||
topsrcdirstrip=`echo "$(top_srcdir)" | sed 's/[].[^$$\\*]/\\\\&/g'`; \
|
||||
list='$(DISTFILES)'; \
|
||||
dist_files=`for file in $$list; do echo $$file; done | \
|
||||
sed -e "s|^$$srcdirstrip/||;t" \
|
||||
-e "s|^$$topsrcdirstrip/|$(top_builddir)/|;t"`; \
|
||||
case $$dist_files in \
|
||||
*/*) $(MKDIR_P) `echo "$$dist_files" | \
|
||||
sed '/\//!d;s|^|$(distdir)/|;s,/[^/]*$$,,' | \
|
||||
sort -u` ;; \
|
||||
esac; \
|
||||
for file in $$dist_files; do \
|
||||
if test -f $$file || test -d $$file; then d=.; else d=$(srcdir); fi; \
|
||||
if test -d $$d/$$file; then \
|
||||
dir=`echo "/$$file" | sed -e 's,/[^/]*$$,,'`; \
|
||||
if test -d "$(distdir)/$$file"; then \
|
||||
find "$(distdir)/$$file" -type d ! -perm -700 -exec chmod u+rwx {} \;; \
|
||||
fi; \
|
||||
if test -d $(srcdir)/$$file && test $$d != $(srcdir); then \
|
||||
cp -fpR $(srcdir)/$$file "$(distdir)$$dir" || exit 1; \
|
||||
find "$(distdir)/$$file" -type d ! -perm -700 -exec chmod u+rwx {} \;; \
|
||||
fi; \
|
||||
cp -fpR $$d/$$file "$(distdir)$$dir" || exit 1; \
|
||||
else \
|
||||
test -f "$(distdir)/$$file" \
|
||||
|| cp -p $$d/$$file "$(distdir)/$$file" \
|
||||
|| exit 1; \
|
||||
fi; \
|
||||
done
|
||||
check-am: all-am
|
||||
check: check-am
|
||||
all-am: Makefile $(LTLIBRARIES)
|
||||
installdirs:
|
||||
for dir in "$(DESTDIR)$(libdir)"; do \
|
||||
test -z "$$dir" || $(MKDIR_P) "$$dir"; \
|
||||
done
|
||||
install: install-am
|
||||
install-exec: install-exec-am
|
||||
install-data: install-data-am
|
||||
uninstall: uninstall-am
|
||||
|
||||
install-am: all-am
|
||||
@$(MAKE) $(AM_MAKEFLAGS) install-exec-am install-data-am
|
||||
|
||||
installcheck: installcheck-am
|
||||
install-strip:
|
||||
if test -z '$(STRIP)'; then \
|
||||
$(MAKE) $(AM_MAKEFLAGS) INSTALL_PROGRAM="$(INSTALL_STRIP_PROGRAM)" \
|
||||
install_sh_PROGRAM="$(INSTALL_STRIP_PROGRAM)" INSTALL_STRIP_FLAG=-s \
|
||||
install; \
|
||||
else \
|
||||
$(MAKE) $(AM_MAKEFLAGS) INSTALL_PROGRAM="$(INSTALL_STRIP_PROGRAM)" \
|
||||
install_sh_PROGRAM="$(INSTALL_STRIP_PROGRAM)" INSTALL_STRIP_FLAG=-s \
|
||||
"INSTALL_PROGRAM_ENV=STRIPPROG='$(STRIP)'" install; \
|
||||
fi
|
||||
mostlyclean-generic:
|
||||
|
||||
clean-generic:
|
||||
|
||||
distclean-generic:
|
||||
-test -z "$(CONFIG_CLEAN_FILES)" || rm -f $(CONFIG_CLEAN_FILES)
|
||||
-test . = "$(srcdir)" || test -z "$(CONFIG_CLEAN_VPATH_FILES)" || rm -f $(CONFIG_CLEAN_VPATH_FILES)
|
||||
|
||||
maintainer-clean-generic:
|
||||
@echo "This command is intended for maintainers to use"
|
||||
@echo "it deletes files that may require special tools to rebuild."
|
||||
clean: clean-am
|
||||
|
||||
clean-am: clean-generic clean-libLTLIBRARIES clean-libtool \
|
||||
mostlyclean-am
|
||||
|
||||
distclean: distclean-am
|
||||
-rm -f ./$(DEPDIR)/SkvEnv.Plo
|
||||
-rm -f ./$(DEPDIR)/err.Plo
|
||||
-rm -f ./$(DEPDIR)/json.Plo
|
||||
-rm -f ./$(DEPDIR)/read-ini.Plo
|
||||
-rm -f ./$(DEPDIR)/read-json.Plo
|
||||
-rm -f ./$(DEPDIR)/read.Plo
|
||||
-rm -f ./$(DEPDIR)/std.Plo
|
||||
-rm -f ./$(DEPDIR)/write-ini.Plo
|
||||
-rm -f ./$(DEPDIR)/write-json.Plo
|
||||
-rm -f ./$(DEPDIR)/write.Plo
|
||||
-rm -f ./$(DEPDIR)/xli.Plo
|
||||
-rm -f Makefile
|
||||
distclean-am: clean-am distclean-compile distclean-generic \
|
||||
distclean-tags
|
||||
|
||||
dvi: dvi-am
|
||||
|
||||
dvi-am:
|
||||
|
||||
html: html-am
|
||||
|
||||
html-am:
|
||||
|
||||
info: info-am
|
||||
|
||||
info-am:
|
||||
|
||||
install-data-am:
|
||||
|
||||
install-dvi: install-dvi-am
|
||||
|
||||
install-dvi-am:
|
||||
|
||||
install-exec-am: install-libLTLIBRARIES
|
||||
|
||||
install-html: install-html-am
|
||||
|
||||
install-html-am:
|
||||
|
||||
install-info: install-info-am
|
||||
|
||||
install-info-am:
|
||||
|
||||
install-man:
|
||||
|
||||
install-pdf: install-pdf-am
|
||||
|
||||
install-pdf-am:
|
||||
|
||||
install-ps: install-ps-am
|
||||
|
||||
install-ps-am:
|
||||
|
||||
installcheck-am:
|
||||
|
||||
maintainer-clean: maintainer-clean-am
|
||||
-rm -f ./$(DEPDIR)/SkvEnv.Plo
|
||||
-rm -f ./$(DEPDIR)/err.Plo
|
||||
-rm -f ./$(DEPDIR)/json.Plo
|
||||
-rm -f ./$(DEPDIR)/read-ini.Plo
|
||||
-rm -f ./$(DEPDIR)/read-json.Plo
|
||||
-rm -f ./$(DEPDIR)/read.Plo
|
||||
-rm -f ./$(DEPDIR)/std.Plo
|
||||
-rm -f ./$(DEPDIR)/write-ini.Plo
|
||||
-rm -f ./$(DEPDIR)/write-json.Plo
|
||||
-rm -f ./$(DEPDIR)/write.Plo
|
||||
-rm -f ./$(DEPDIR)/xli.Plo
|
||||
-rm -f Makefile
|
||||
maintainer-clean-am: distclean-am maintainer-clean-generic
|
||||
|
||||
mostlyclean: mostlyclean-am
|
||||
|
||||
mostlyclean-am: mostlyclean-compile mostlyclean-generic \
|
||||
mostlyclean-libtool
|
||||
|
||||
pdf: pdf-am
|
||||
|
||||
pdf-am:
|
||||
|
||||
ps: ps-am
|
||||
|
||||
ps-am:
|
||||
|
||||
uninstall-am: uninstall-libLTLIBRARIES
|
||||
|
||||
.MAKE: install-am install-strip
|
||||
|
||||
.PHONY: CTAGS GTAGS TAGS all all-am am--depfiles check check-am clean \
|
||||
clean-generic clean-libLTLIBRARIES clean-libtool cscopelist-am \
|
||||
ctags ctags-am distclean distclean-compile distclean-generic \
|
||||
distclean-libtool distclean-tags distdir dvi dvi-am html \
|
||||
html-am info info-am install install-am install-data \
|
||||
install-data-am install-dvi install-dvi-am install-exec \
|
||||
install-exec-am install-html install-html-am install-info \
|
||||
install-info-am install-libLTLIBRARIES install-man install-pdf \
|
||||
install-pdf-am install-ps install-ps-am install-strip \
|
||||
installcheck installcheck-am installdirs maintainer-clean \
|
||||
maintainer-clean-generic mostlyclean mostlyclean-compile \
|
||||
mostlyclean-generic mostlyclean-libtool pdf pdf-am ps ps-am \
|
||||
tags tags-am uninstall uninstall-am uninstall-libLTLIBRARIES
|
||||
|
||||
.PRECIOUS: Makefile
|
||||
|
||||
|
||||
# Tell versions [3.59,3.63) of GNU make to not export all variables.
|
||||
# Otherwise a system limit (for SysV at least) may be exceeded.
|
||||
.NOEXPORT:
|
326
lib/xli/SkvEnv.cpp
Normal file
326
lib/xli/SkvEnv.cpp
Normal file
@ -0,0 +1,326 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
|
||||
|
||||
#include <qse/xli/SkvEnv.hpp>
|
||||
#include <qse/xli/stdxli.h>
|
||||
|
||||
#include <qse/cmn/str.h>
|
||||
#include "../cmn/mem-prv.h"
|
||||
#include "xli-prv.h"
|
||||
|
||||
#define SCTN_KEY_SPLITTER QSE_T('*')
|
||||
|
||||
QSE_BEGIN_NAMESPACE(QSE)
|
||||
|
||||
static qse_char_t splitter[2] = { SCTN_KEY_SPLITTER, QSE_T('\0') };
|
||||
|
||||
SkvEnv::SkvEnv (Mmgr* mmgr): Mmged(mmgr), xli(QSE_NULL)
|
||||
{
|
||||
}
|
||||
|
||||
SkvEnv::~SkvEnv ()
|
||||
{
|
||||
if (this->xli) qse_xli_close (this->xli);
|
||||
}
|
||||
|
||||
int SkvEnv::addItem (const qse_char_t* name, const qse_char_t* dval, ProbeProc probe)
|
||||
{
|
||||
qse_char_t* sctn, * key;
|
||||
qse_size_t sctn_len, key_len;
|
||||
ItemList::Node* np;
|
||||
|
||||
if (this->split_name(name, &sctn, &sctn_len, &key, &key_len) == -1) return -1;
|
||||
|
||||
for (np = this->item_list.getHeadNode(); np; np = np->getNextNode())
|
||||
{
|
||||
Item& item = np->getValue();
|
||||
if (qse_strxcmp(sctn, sctn_len, item.sctn) == 0 &&
|
||||
qse_strxcmp(key, key_len, item.key) == 0) return -1;
|
||||
}
|
||||
|
||||
try { np = this->item_list.append (Item()); }
|
||||
catch (...) { return -1; }
|
||||
|
||||
Item& item = np->getValue();
|
||||
qse_strxncpy (item.sctn, QSE_COUNTOF(item.sctn), sctn, sctn_len);
|
||||
qse_strxncpy (item.key, QSE_COUNTOF(item.key), key, key_len);
|
||||
|
||||
qse_strxjoin (
|
||||
item.name, QSE_COUNTOF(item.name),
|
||||
item.sctn, splitter, item.key, QSE_NULL);
|
||||
qse_strxcpy (item.dval, QSE_COUNTOF(item.dval), dval);
|
||||
item.probe = probe;
|
||||
|
||||
// set its default value
|
||||
if (this->setValue(name, item.dval) <= -1)
|
||||
{
|
||||
this->item_list.remove (np);
|
||||
return -1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
int SkvEnv::removeItem (const qse_char_t* name)
|
||||
{
|
||||
qse_char_t* sctn, * key;
|
||||
qse_size_t sctn_len, key_len;
|
||||
|
||||
if (this->split_name (name, &sctn, &sctn_len, &key, &key_len) == -1) return -1;
|
||||
|
||||
for (ItemList::Node* np = this->item_list.getHeadNode(); np; np = np->getNextNode())
|
||||
{
|
||||
Item& item = np->value;
|
||||
if (qse_strxcmp(sctn, sctn_len, item.sctn) == 0 &&
|
||||
qse_strxcmp(key, key_len, item.key) == 0)
|
||||
{
|
||||
this->item_list.remove (np);
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
const qse_char_t* SkvEnv::getValue (const qse_char_t* name) const
|
||||
{
|
||||
if (!this->xli) return QSE_NULL;
|
||||
|
||||
qse_xli_pair_t* pair = qse_xli_findpair(this->xli, QSE_NULL, name);
|
||||
if (!pair) return QSE_NULL;
|
||||
|
||||
QSE_ASSERT (pair->val != QSE_NULL);
|
||||
QSE_ASSERT (pair->val->type == QSE_XLI_STR);
|
||||
|
||||
return (((qse_xli_str_t*)pair->val))->ptr;
|
||||
}
|
||||
|
||||
int SkvEnv::setValue (const qse_char_t* name, const qse_char_t* value)
|
||||
{
|
||||
qse_char_t* sctn, * key;
|
||||
qse_size_t sctn_len, key_len;
|
||||
|
||||
if (this->split_name(name, &sctn, &sctn_len, &key, &key_len) <= -1) return -1;
|
||||
|
||||
if (!this->xli)
|
||||
{
|
||||
this->xli = qse_xli_openstdwithmmgr(this->getMmgr(), 0, 0, QSE_NULL);
|
||||
if (!this->xli) return -1;
|
||||
qse_xli_setopt (this->xli, QSE_XLI_KEYSPLITTER, splitter);
|
||||
}
|
||||
|
||||
// find if the name is a registered item name.
|
||||
for (ItemList::Node* np = this->item_list.getHeadNode(); np; np = np->getNextNode())
|
||||
{
|
||||
Item& item = np->value;
|
||||
if (qse_strxcmp(sctn, sctn_len, item.sctn) == 0 &&
|
||||
qse_strxcmp(key, key_len, item.key) == 0)
|
||||
{
|
||||
// if it's the registered item name, change the value.
|
||||
return this->set_value_with_item (item, value);
|
||||
}
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
int SkvEnv::split_name (const qse_char_t* name, qse_char_t** sctn, qse_size_t* sctn_len, qse_char_t** key, qse_size_t* key_len)
|
||||
{
|
||||
QSE_ASSERT (name != QSE_NULL);
|
||||
QSE_ASSERT (sctn != QSE_NULL);
|
||||
QSE_ASSERT (sctn_len != QSE_NULL);
|
||||
QSE_ASSERT (key != QSE_NULL);
|
||||
QSE_ASSERT (key_len != QSE_NULL);
|
||||
|
||||
qse_char_t* p;
|
||||
qse_cstr_t s, k;
|
||||
|
||||
p = qse_strtok(name, splitter, &s);
|
||||
if (!p || s.len == 0) return -1;
|
||||
qse_strtok(p, QSE_NULL, &k);
|
||||
if (k.len == 0) return -1;
|
||||
|
||||
*sctn = s.ptr;
|
||||
*sctn_len = s.len;
|
||||
*key = k.ptr;
|
||||
*key_len = k.len;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int SkvEnv::set_value_with_item (Item& item, const qse_char_t* value)
|
||||
{
|
||||
#if 0
|
||||
if (this->probe_item_value(item, value) <= -1) return -1;
|
||||
|
||||
qse_cstr_t v = { (qse_char_t*)value, qse_strlen(value) };
|
||||
|
||||
qse_xli_pair_t* pair;
|
||||
pair = qse_xli_setpairwithstr(this->xli, QSE_NULL, item.name, &v, QSE_NULL);
|
||||
if (!pair)
|
||||
{
|
||||
if (qse_xli_geterrnum(this->xli) != QSE_XLI_ENOENT) return -1;
|
||||
|
||||
pair = qse_xli_findpair(this->xli, QSE_NULL, item.sctn);
|
||||
if (!pair)
|
||||
{
|
||||
pair = qse_xli_insertpairwithemptylist(this->xli, QSE_NULL, QSE_NULL, item.sctn, QSE_NULL, QSE_NULL);
|
||||
if (!pair) return -1;
|
||||
}
|
||||
|
||||
QSE_ASSERT (pair->val != QSE_NULL);
|
||||
QSE_ASSERT (pair->val->type == QSE_XLI_LIST);
|
||||
|
||||
if (!qse_xli_insertpairwithstr(this->xli, (qse_xli_list_t*)pair->val, QSE_NULL, item.key, QSE_NULL, QSE_NULL, &v, QSE_NULL)) return -1;
|
||||
}
|
||||
|
||||
this->accept_item_value (item, value);
|
||||
return 0;
|
||||
#else
|
||||
|
||||
// this function adds/updates the pair with the given value
|
||||
// before it verifies the value. as long as verification is
|
||||
// ok, the value is guaranteed to be set.
|
||||
|
||||
qse_cstr_t v = { (qse_char_t*)value, qse_strlen(value) };
|
||||
qse_xli_pair_t* sctn_pair = QSE_NULL, * kv_pair = QSE_NULL;
|
||||
|
||||
sctn_pair = qse_xli_findpair(this->xli, QSE_NULL, item.sctn);
|
||||
if (sctn_pair)
|
||||
{
|
||||
// section exists.
|
||||
QSE_ASSERT (sctn_pair->val && sctn_pair->val->type == QSE_XLI_LIST);
|
||||
kv_pair = qse_xli_findpair(this->xli, (qse_xli_list_t*)sctn_pair->val, item.key);
|
||||
if (kv_pair)
|
||||
{
|
||||
// key also exists
|
||||
qse_xli_pair_t* existing_kv_pair = kv_pair;
|
||||
|
||||
kv_pair = QSE_NULL;
|
||||
sctn_pair = QSE_NULL;
|
||||
|
||||
qse_xli_str_t* newstrv = qse_xli_makestrval(this->xli, &v, QSE_NULL);
|
||||
if (!newstrv) goto rollback;
|
||||
|
||||
if (this->call_probe_item_value(item, value) <= -1)
|
||||
{
|
||||
qse_xli_freeval (this->xli, (qse_xli_val_t*)newstrv);
|
||||
goto rollback;
|
||||
}
|
||||
|
||||
qse_xli_freeval (this->xli, existing_kv_pair->val);
|
||||
existing_kv_pair->val = (qse_xli_val_t*)newstrv;
|
||||
}
|
||||
else
|
||||
{
|
||||
kv_pair = qse_xli_insertpairwithstr(this->xli, (qse_xli_list_t*)sctn_pair->val, QSE_NULL, item.key, QSE_NULL, QSE_NULL, &v, QSE_NULL);
|
||||
sctn_pair = QSE_NULL;
|
||||
if (!kv_pair) goto rollback;
|
||||
|
||||
if (this->call_probe_item_value(item, value) <= -1) goto rollback;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
sctn_pair = qse_xli_insertpairwithemptylist(this->xli, QSE_NULL, QSE_NULL, item.sctn, QSE_NULL, QSE_NULL);
|
||||
if (!sctn_pair) return -1;
|
||||
|
||||
kv_pair = qse_xli_insertpairwithstr(this->xli, (qse_xli_list_t*)sctn_pair->val, QSE_NULL, item.key, QSE_NULL, QSE_NULL, &v, QSE_NULL);
|
||||
if (!kv_pair) goto rollback;
|
||||
|
||||
if (this->call_probe_item_value(item, value) <= -1) goto rollback;
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
rollback:
|
||||
if (kv_pair) qse_xli_deletepair (this->xli, kv_pair);
|
||||
if (sctn_pair) qse_xli_deletepair (this->xli, sctn_pair);
|
||||
return -1;
|
||||
#endif
|
||||
}
|
||||
|
||||
int SkvEnv::load (const qse_char_t* path)
|
||||
{
|
||||
qse_xli_t* xli;
|
||||
|
||||
if (!this->xli)
|
||||
{
|
||||
// this means that no items have been registered with
|
||||
// this->addItem(). so loading is meaningless.
|
||||
return -1;
|
||||
}
|
||||
|
||||
xli = qse_xli_openstdwithmmgr (this->getMmgr(), 0, 0, QSE_NULL);
|
||||
if (!xli) return -1;
|
||||
|
||||
qse_xli_setopt (xli, QSE_XLI_KEYSPLITTER, splitter);
|
||||
|
||||
qse_xli_iostd_t in;
|
||||
QSE_MEMSET (&in, 0, QSE_SIZEOF(in));
|
||||
in.type = QSE_XLI_IOSTD_FILE;
|
||||
in.u.file.path = path;
|
||||
|
||||
if (qse_xli_readinistd(xli, &in) <= -1)
|
||||
{
|
||||
qse_xli_close (xli);
|
||||
return -1;
|
||||
}
|
||||
|
||||
for (ItemList::Node* np = this->item_list.getHeadNode(); np; np = np->getNextNode())
|
||||
{
|
||||
Item& item = np->value;
|
||||
qse_xli_pair_t* pair;
|
||||
|
||||
pair = qse_xli_findpair(xli, QSE_NULL, item.name);
|
||||
if (pair)
|
||||
{
|
||||
qse_xli_str_t* strv = (qse_xli_str_t*)pair->val;
|
||||
this->set_value_with_item (item, strv->ptr);
|
||||
// ignore failure.
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: check if there are any unknown names. flag it via callbacks???
|
||||
qse_xli_close(xli);
|
||||
return 0;
|
||||
}
|
||||
|
||||
int SkvEnv::store (const qse_char_t* path)
|
||||
{
|
||||
if (!this->xli) return -1;
|
||||
|
||||
qse_xli_iostd_t out;
|
||||
|
||||
QSE_MEMSET (&out, 0, QSE_SIZEOF(out));
|
||||
out.type = QSE_XLI_IOSTD_FILE;
|
||||
out.u.file.path = path;
|
||||
|
||||
return qse_xli_writeinistd(this->xli, QSE_NULL, &out);
|
||||
}
|
||||
|
||||
QSE_END_NAMESPACE(QSE)
|
151
lib/xli/err.c
Normal file
151
lib/xli/err.c
Normal file
@ -0,0 +1,151 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#include "xli-prv.h"
|
||||
#include "../cmn/mem-prv.h"
|
||||
|
||||
const qse_char_t* qse_xli_dflerrstr (const qse_xli_t* xli, qse_xli_errnum_t errnum)
|
||||
{
|
||||
static const qse_char_t* errstr[] =
|
||||
{
|
||||
QSE_T("no error"),
|
||||
QSE_T("other error"),
|
||||
QSE_T("not implemented"),
|
||||
QSE_T("subsystem error"),
|
||||
QSE_T("internal error that should never have happened"),
|
||||
|
||||
QSE_T("insufficient memory"),
|
||||
QSE_T("invalid parameter or data"),
|
||||
QSE_T("'${0}' not found"),
|
||||
QSE_T("'${0}' already exists"),
|
||||
QSE_T("I/O error with file '${0}'"),
|
||||
QSE_T("error returned by user I/O handler"),
|
||||
|
||||
QSE_T("syntax error"),
|
||||
QSE_T("colon expected in place of '${0}'"),
|
||||
QSE_T("semicolon expected in place of '${0}'"),
|
||||
QSE_T("equal-sign expected in place of '${0}'"),
|
||||
QSE_T("left-brace or equal-sign expected in place of '${0}'"),
|
||||
QSE_T("left-brace or left-bracket expected in place of '${0}'"),
|
||||
QSE_T("right-brace expected in place of '${0}'"),
|
||||
QSE_T("right-bracket expected in place of '${0}'"),
|
||||
QSE_T("comma expected in place of '${0}'"),
|
||||
QSE_T("value expected in place of '${0}'"),
|
||||
QSE_T("string not closed"),
|
||||
QSE_T("string tag not closed"),
|
||||
QSE_T("'@include' not followed by a string"),
|
||||
QSE_T("invalid character '${0}'"),
|
||||
QSE_T("invalid tag character '${0}'"),
|
||||
QSE_T("'${0}' not recognized"),
|
||||
QSE_T("@ not followed by a valid word"),
|
||||
QSE_T("invalid identifier '${0}'"),
|
||||
QSE_T("key expected in place of '${0}'"),
|
||||
QSE_T("missing key after key tag"),
|
||||
QSE_T("undefined key '${0}'"),
|
||||
QSE_T("no alias for '${0}'"),
|
||||
QSE_T("value expected in place of '${0}'"),
|
||||
QSE_T("illegal value for '${0}'"),
|
||||
QSE_T("no value for '${0}'"),
|
||||
QSE_T("uncomplying number of string segments for '${0}'"),
|
||||
QSE_T("section tag expected in place of '${0}'")
|
||||
};
|
||||
|
||||
return (errnum >= 0 && errnum < QSE_COUNTOF(errstr))?
|
||||
errstr[errnum]: QSE_T("unknown error");
|
||||
}
|
||||
|
||||
qse_xli_errstr_t qse_xli_geterrstr (const qse_xli_t* xli)
|
||||
{
|
||||
return xli->errstr;
|
||||
}
|
||||
|
||||
void qse_xli_seterrstr (qse_xli_t* xli, qse_xli_errstr_t errstr)
|
||||
{
|
||||
xli->errstr = errstr;
|
||||
}
|
||||
|
||||
qse_xli_errnum_t qse_xli_geterrnum (const qse_xli_t* xli)
|
||||
{
|
||||
return xli->errnum;
|
||||
}
|
||||
|
||||
const qse_xli_loc_t* qse_xli_geterrloc (const qse_xli_t* xli)
|
||||
{
|
||||
return &xli->errloc;
|
||||
}
|
||||
|
||||
const qse_char_t* qse_xli_geterrmsg (const qse_xli_t* xli)
|
||||
{
|
||||
return (xli->errmsg[0] == QSE_T('\0'))?
|
||||
qse_xli_geterrstr(xli)(xli,xli->errnum): xli->errmsg;
|
||||
}
|
||||
|
||||
void qse_xli_geterror (
|
||||
const qse_xli_t* xli, qse_xli_errnum_t* errnum,
|
||||
const qse_char_t** errmsg, qse_xli_loc_t* errloc)
|
||||
{
|
||||
if (errnum != QSE_NULL) *errnum = xli->errnum;
|
||||
if (errmsg != QSE_NULL)
|
||||
{
|
||||
*errmsg = (xli->errmsg[0] == QSE_T('\0'))?
|
||||
qse_xli_geterrstr(xli)(xli,xli->errnum):
|
||||
xli->errmsg;
|
||||
}
|
||||
if (errloc != QSE_NULL) *errloc = xli->errloc;
|
||||
}
|
||||
|
||||
void qse_xli_seterrnum (
|
||||
qse_xli_t* xli, qse_xli_errnum_t errnum, const qse_cstr_t* errarg)
|
||||
{
|
||||
qse_xli_seterror (xli, errnum, errarg, QSE_NULL);
|
||||
}
|
||||
|
||||
void qse_xli_seterrmsg (
|
||||
qse_xli_t* xli, qse_xli_errnum_t errnum,
|
||||
const qse_char_t* errmsg, const qse_xli_loc_t* errloc)
|
||||
{
|
||||
xli->errnum = errnum;
|
||||
qse_strxcpy (xli->errmsg, QSE_COUNTOF(xli->errmsg), errmsg);
|
||||
if (errloc != QSE_NULL) xli->errloc = *errloc;
|
||||
else QSE_MEMSET (&xli->errloc, 0, QSE_SIZEOF(xli->errloc));
|
||||
}
|
||||
|
||||
void qse_xli_seterror (
|
||||
qse_xli_t* xli, qse_xli_errnum_t errnum,
|
||||
const qse_cstr_t* errarg, const qse_xli_loc_t* errloc)
|
||||
{
|
||||
const qse_char_t* errfmt;
|
||||
|
||||
xli->errnum = errnum;
|
||||
|
||||
errfmt = qse_xli_geterrstr(xli)(xli,xli->errnum);
|
||||
QSE_ASSERT (errfmt != QSE_NULL);
|
||||
qse_strxfncpy (xli->errmsg, QSE_COUNTOF(xli->errmsg), errfmt, errarg);
|
||||
|
||||
if (errloc != QSE_NULL) xli->errloc = *errloc;
|
||||
else QSE_MEMSET (&xli->errloc, 0, QSE_SIZEOF(xli->errloc));
|
||||
}
|
||||
|
995
lib/xli/json.c
Normal file
995
lib/xli/json.c
Normal file
@ -0,0 +1,995 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#include <qse/xli/json.h>
|
||||
#include <qse/cmn/chr.h>
|
||||
#include <qse/cmn/str.h>
|
||||
#include <qse/cmn/mbwc.h>
|
||||
#include "../cmn/mem-prv.h"
|
||||
|
||||
#define QSE_JSON_TOKEN_NAME_ALIGN 64
|
||||
|
||||
typedef struct qse_json_state_node_t qse_json_state_node_t;
|
||||
struct qse_json_state_node_t
|
||||
{
|
||||
qse_json_state_t state;
|
||||
union
|
||||
{
|
||||
struct
|
||||
{
|
||||
int got_value;
|
||||
} ia; /* in array */
|
||||
|
||||
struct
|
||||
{
|
||||
/* 0: ready to get key (at the beginning or got comma),
|
||||
* 1: got key, 2: got colon, 3: got value */
|
||||
int state;
|
||||
} id; /* in dictionary */
|
||||
struct
|
||||
{
|
||||
int escaped;
|
||||
int digit_count;
|
||||
/* acc is always of unicode type to handle \u and \U.
|
||||
* in the bch mode, it will get converted to a utf8 stream. */
|
||||
qse_wchar_t acc;
|
||||
} sv;
|
||||
struct
|
||||
{
|
||||
int escaped;
|
||||
int digit_count;
|
||||
/* for a character, no way to support the unicode character
|
||||
* in the bch mode */
|
||||
qse_char_t acc;
|
||||
} cv;
|
||||
struct
|
||||
{
|
||||
int dotted;
|
||||
} nv;
|
||||
} u;
|
||||
qse_json_state_node_t* next;
|
||||
};
|
||||
|
||||
struct qse_json_t
|
||||
{
|
||||
QSE_JSON_HDR;
|
||||
|
||||
qse_json_errnum_t errnum;
|
||||
struct
|
||||
{
|
||||
qse_char_t backup[256];
|
||||
qse_char_t buf[256];
|
||||
/*qse_size_t len;*/
|
||||
} errmsg;
|
||||
|
||||
qse_json_prim_t prim;
|
||||
|
||||
struct
|
||||
{
|
||||
int trait;
|
||||
} cfg;
|
||||
|
||||
qse_json_state_node_t state_top;
|
||||
qse_json_state_node_t* state_stack;
|
||||
|
||||
qse_cstr_t tok;
|
||||
qse_size_t tok_capa;
|
||||
};
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static void clear_token (qse_json_t* json)
|
||||
{
|
||||
json->tok.len = 0;
|
||||
if (json->tok_capa > 0) json->tok.ptr[json->tok.len] = QSE_T('\0');
|
||||
}
|
||||
|
||||
static int add_char_to_token (qse_json_t* json, qse_char_t ch)
|
||||
{
|
||||
if (json->tok.len >= json->tok_capa)
|
||||
{
|
||||
qse_char_t* tmp;
|
||||
qse_size_t newcapa;
|
||||
|
||||
newcapa = QSE_ALIGNTO_POW2(json->tok.len + 2, QSE_JSON_TOKEN_NAME_ALIGN); /* +2 here because of -1 when setting newcapa */
|
||||
tmp = (qse_char_t*)qse_json_reallocmem(json, json->tok.ptr, newcapa * QSE_SIZEOF(*tmp));
|
||||
if (!tmp) return -1;
|
||||
|
||||
json->tok_capa = newcapa - 1; /* -1 to secure space for terminating null */
|
||||
json->tok.ptr = tmp;
|
||||
}
|
||||
|
||||
json->tok.ptr[json->tok.len++] = ch;
|
||||
json->tok.ptr[json->tok.len] = QSE_T('\0');
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int add_chars_to_token (qse_json_t* json, const qse_char_t* ptr, qse_size_t len)
|
||||
{
|
||||
qse_size_t i;
|
||||
|
||||
if (json->tok_capa - json->tok.len > len)
|
||||
{
|
||||
qse_char_t* tmp;
|
||||
qse_size_t newcapa;
|
||||
|
||||
newcapa = QSE_ALIGNTO_POW2(json->tok.len + len + 1, QSE_JSON_TOKEN_NAME_ALIGN);
|
||||
tmp = (qse_char_t*)qse_json_reallocmem(json, json->tok.ptr, newcapa * QSE_SIZEOF(*tmp));
|
||||
if (!tmp) return -1;
|
||||
|
||||
json->tok_capa = newcapa - 1;
|
||||
json->tok.ptr = tmp;
|
||||
}
|
||||
|
||||
for (i = 0; i < len; i++)
|
||||
json->tok.ptr[json->tok.len++] = ptr[i];
|
||||
json->tok.ptr[json->tok.len] = QSE_T('\0');
|
||||
return 0;
|
||||
}
|
||||
|
||||
static QSE_INLINE qse_char_t unescape (qse_char_t c)
|
||||
{
|
||||
switch (c)
|
||||
{
|
||||
case QSE_T('a'): return QSE_T('\a');
|
||||
case QSE_T('b'): return QSE_T('\b');
|
||||
case QSE_T('f'): return QSE_T('\f');
|
||||
case QSE_T('n'): return QSE_T('\n');
|
||||
case QSE_T('r'): return QSE_T('\r');
|
||||
case QSE_T('t'): return QSE_T('\t');
|
||||
case QSE_T('v'): return QSE_T('\v');
|
||||
default: return c;
|
||||
}
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static int push_state (qse_json_t* json, qse_json_state_t state)
|
||||
{
|
||||
qse_json_state_node_t* ss;
|
||||
|
||||
ss = (qse_json_state_node_t*)qse_json_callocmem(json, QSE_SIZEOF(*ss));
|
||||
if (!ss) return -1;
|
||||
|
||||
ss->state = state;
|
||||
ss->next = json->state_stack;
|
||||
|
||||
json->state_stack = ss;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void pop_state (qse_json_t* json)
|
||||
{
|
||||
qse_json_state_node_t* ss;
|
||||
|
||||
ss = json->state_stack;
|
||||
QSE_ASSERT (ss != QSE_NULL && ss != &json->state_top);
|
||||
json->state_stack = ss->next;
|
||||
|
||||
if (json->state_stack->state == QSE_JSON_STATE_IN_ARRAY)
|
||||
{
|
||||
json->state_stack->u.ia.got_value = 1;
|
||||
}
|
||||
else if (json->state_stack->state == QSE_JSON_STATE_IN_DIC)
|
||||
{
|
||||
json->state_stack->u.id.state++;
|
||||
}
|
||||
|
||||
/* TODO: don't free this. move it to the free list? */
|
||||
qse_json_freemem (json, ss);
|
||||
}
|
||||
|
||||
static void pop_all_states (qse_json_t* json)
|
||||
{
|
||||
while (json->state_stack != &json->state_top) pop_state (json);
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static int invoke_data_inst (qse_json_t* json, qse_json_inst_t inst)
|
||||
{
|
||||
if (json->state_stack->state == QSE_JSON_STATE_IN_DIC && json->state_stack->u.id.state == 1)
|
||||
{
|
||||
if (inst != QSE_JSON_INST_STRING)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("dictionary key not a string - %.*js"), (int)json->tok.len, json->tok.ptr);
|
||||
return -1;
|
||||
}
|
||||
|
||||
inst = QSE_JSON_INST_KEY;
|
||||
}
|
||||
|
||||
if (json->prim.instcb(json, inst, &json->tok) <= -1) return -1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int handle_string_value_char (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
int ret = 1;
|
||||
|
||||
if (json->state_stack->u.sv.escaped == 3)
|
||||
{
|
||||
if (c >= '0' && c <= '7')
|
||||
{
|
||||
json->state_stack->u.sv.acc = json->state_stack->u.sv.acc * 8 + c - '0';
|
||||
json->state_stack->u.sv.digit_count++;
|
||||
if (json->state_stack->u.sv.digit_count >= json->state_stack->u.sv.escaped) goto add_sv_acc;
|
||||
}
|
||||
else
|
||||
{
|
||||
ret = 0;
|
||||
goto add_sv_acc;
|
||||
}
|
||||
}
|
||||
else if (json->state_stack->u.sv.escaped >= 2)
|
||||
{
|
||||
if (c >= '0' && c <= '9')
|
||||
{
|
||||
json->state_stack->u.sv.acc = json->state_stack->u.sv.acc * 16 + c - '0';
|
||||
json->state_stack->u.sv.digit_count++;
|
||||
if (json->state_stack->u.sv.digit_count >= json->state_stack->u.sv.escaped) goto add_sv_acc;
|
||||
}
|
||||
else if (c >= 'a' && c <= 'f')
|
||||
{
|
||||
json->state_stack->u.sv.acc = json->state_stack->u.sv.acc * 16 + c - 'a' + 10;
|
||||
json->state_stack->u.sv.digit_count++;
|
||||
if (json->state_stack->u.sv.digit_count >= json->state_stack->u.sv.escaped) goto add_sv_acc;
|
||||
}
|
||||
else if (c >= 'A' && c <= 'F')
|
||||
{
|
||||
json->state_stack->u.sv.acc = json->state_stack->u.sv.acc * 16 + c - 'A' + 10;
|
||||
json->state_stack->u.sv.digit_count++;
|
||||
if (json->state_stack->u.sv.digit_count >= json->state_stack->u.sv.escaped) goto add_sv_acc;
|
||||
}
|
||||
else
|
||||
{
|
||||
ret = 0;
|
||||
add_sv_acc:
|
||||
#if defined(QSE_CHAR_IS_WCHAR)
|
||||
if (add_char_to_token(json, json->state_stack->u.sv.acc) <= -1) return -1;
|
||||
#else
|
||||
/* convert the character to utf8 */
|
||||
{
|
||||
qse_mchar_t bcsbuf[QSE_MBLEN_MAX];
|
||||
qse_size_t n;
|
||||
|
||||
n = json->_cmgr->wctomb(json->state_stack->u.sv.acc, bcsbuf, QSE_COUNTOF(bcsbuf));
|
||||
if (n == 0 || n > QSE_COUNTOF(bcsbuf))
|
||||
{
|
||||
/* illegal character or buffer to small */
|
||||
qse_json_seterrfmt (json, QSE_JSON_EECERR, QSE_T("unable to convert %jc"), json->state_stack->u.sv.acc);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (add_chars_to_token(json, bcsbuf, n) <= -1) return -1;
|
||||
}
|
||||
#endif
|
||||
json->state_stack->u.sv.escaped = 0;
|
||||
}
|
||||
}
|
||||
else if (json->state_stack->u.sv.escaped == 1)
|
||||
{
|
||||
if (c >= '0' && c <= '8')
|
||||
{
|
||||
json->state_stack->u.sv.escaped = 3;
|
||||
json->state_stack->u.sv.digit_count = 0;
|
||||
json->state_stack->u.sv.acc = c - '0';
|
||||
}
|
||||
else if (c == 'x')
|
||||
{
|
||||
json->state_stack->u.sv.escaped = 2;
|
||||
json->state_stack->u.sv.digit_count = 0;
|
||||
json->state_stack->u.sv.acc = 0;
|
||||
}
|
||||
else if (c == 'u')
|
||||
{
|
||||
json->state_stack->u.sv.escaped = 4;
|
||||
json->state_stack->u.sv.digit_count = 0;
|
||||
json->state_stack->u.sv.acc = 0;
|
||||
}
|
||||
else if (c == 'U')
|
||||
{
|
||||
json->state_stack->u.sv.escaped = 8;
|
||||
json->state_stack->u.sv.digit_count = 0;
|
||||
json->state_stack->u.sv.acc = 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
json->state_stack->u.sv.escaped = 0;
|
||||
if (add_char_to_token(json, unescape(c)) <= -1) return -1;
|
||||
}
|
||||
}
|
||||
else if (c == '\\')
|
||||
{
|
||||
json->state_stack->u.sv.escaped = 1;
|
||||
}
|
||||
else if (c == '\"')
|
||||
{
|
||||
pop_state (json);
|
||||
if (invoke_data_inst(json, QSE_JSON_INST_STRING) <= -1) return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int handle_character_value_char (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
/* The real JSON dones't support character literal. this is HCL's own extension. */
|
||||
int ret = 1;
|
||||
|
||||
if (json->state_stack->u.cv.escaped == 3)
|
||||
{
|
||||
if (c >= '0' && c <= '7')
|
||||
{
|
||||
json->state_stack->u.cv.acc = json->state_stack->u.cv.acc * 8 + c - '0';
|
||||
json->state_stack->u.cv.digit_count++;
|
||||
if (json->state_stack->u.cv.digit_count >= json->state_stack->u.cv.escaped) goto add_cv_acc;
|
||||
}
|
||||
else
|
||||
{
|
||||
ret = 0;
|
||||
goto add_cv_acc;
|
||||
}
|
||||
}
|
||||
if (json->state_stack->u.cv.escaped >= 2)
|
||||
{
|
||||
if (c >= '0' && c <= '9')
|
||||
{
|
||||
json->state_stack->u.cv.acc = json->state_stack->u.cv.acc * 16 + c - '0';
|
||||
json->state_stack->u.cv.digit_count++;
|
||||
if (json->state_stack->u.cv.digit_count >= json->state_stack->u.cv.escaped) goto add_cv_acc;
|
||||
}
|
||||
else if (c >= 'a' && c <= 'f')
|
||||
{
|
||||
json->state_stack->u.cv.acc = json->state_stack->u.cv.acc * 16 + c - 'a' + 10;
|
||||
json->state_stack->u.cv.digit_count++;
|
||||
if (json->state_stack->u.cv.digit_count >= json->state_stack->u.cv.escaped) goto add_cv_acc;
|
||||
}
|
||||
else if (c >= 'A' && c <= 'F')
|
||||
{
|
||||
json->state_stack->u.cv.acc = json->state_stack->u.cv.acc * 16 + c - 'A' + 10;
|
||||
json->state_stack->u.cv.digit_count++;
|
||||
if (json->state_stack->u.cv.digit_count >= json->state_stack->u.cv.escaped) goto add_cv_acc;
|
||||
}
|
||||
else
|
||||
{
|
||||
ret = 0;
|
||||
add_cv_acc:
|
||||
if (add_char_to_token(json, json->state_stack->u.cv.acc) <= -1) return -1;
|
||||
json->state_stack->u.cv.escaped = 0;
|
||||
}
|
||||
}
|
||||
else if (json->state_stack->u.cv.escaped == 1)
|
||||
{
|
||||
if (c >= '0' && c <= '8')
|
||||
{
|
||||
json->state_stack->u.cv.escaped = 3;
|
||||
json->state_stack->u.cv.digit_count = 0;
|
||||
json->state_stack->u.cv.acc = c - '0';
|
||||
}
|
||||
else if (c == 'x')
|
||||
{
|
||||
json->state_stack->u.cv.escaped = 2;
|
||||
json->state_stack->u.cv.digit_count = 0;
|
||||
json->state_stack->u.cv.acc = 0;
|
||||
}
|
||||
else if (c == 'u')
|
||||
{
|
||||
json->state_stack->u.cv.escaped = 4;
|
||||
json->state_stack->u.cv.digit_count = 0;
|
||||
json->state_stack->u.cv.acc = 0;
|
||||
}
|
||||
else if (c == 'U')
|
||||
{
|
||||
json->state_stack->u.cv.escaped = 8;
|
||||
json->state_stack->u.cv.digit_count = 0;
|
||||
json->state_stack->u.cv.acc = 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
json->state_stack->u.cv.escaped = 0;
|
||||
if (add_char_to_token(json, unescape(c)) <= -1) return -1;
|
||||
}
|
||||
}
|
||||
else if (c == '\\')
|
||||
{
|
||||
json->state_stack->u.cv.escaped = 1;
|
||||
}
|
||||
else if (c == '\'')
|
||||
{
|
||||
pop_state (json);
|
||||
|
||||
if (json->tok.len < 1)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("no character in a character literal"));
|
||||
return -1;
|
||||
}
|
||||
if (invoke_data_inst(json, QSE_JSON_INST_CHARACTER) <= -1) return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
}
|
||||
|
||||
if (json->tok.len > 1)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("too many characters in a character literal - %.*js"), (int)json->tok.len, json->tok.ptr);
|
||||
return -1;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int handle_numeric_value_char (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
if (QSE_ISMDIGIT(c) || (json->tok.len == 0 && (c == '+' || c == '-')))
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else if (!json->state_stack->u.nv.dotted && c == '.' &&
|
||||
json->tok.len > 0 && QSE_ISMDIGIT(json->tok.ptr[json->tok.len - 1]))
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
json->state_stack->u.nv.dotted = 1;
|
||||
return 1;
|
||||
}
|
||||
|
||||
pop_state (json);
|
||||
|
||||
QSE_ASSERT (json->tok.len > 0);
|
||||
if (!QSE_ISMDIGIT(json->tok.ptr[json->tok.len - 1]))
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("invalid numeric value - %.*js"), (int)json->tok.len, json->tok.ptr);
|
||||
return -1;
|
||||
}
|
||||
if (invoke_data_inst(json, QSE_JSON_INST_NUMBER) <= -1) return -1;
|
||||
return 0; /* start over */
|
||||
}
|
||||
|
||||
static int handle_word_value_char (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
qse_json_inst_t inst;
|
||||
|
||||
if (QSE_ISMALPHA(c))
|
||||
{
|
||||
if (add_char_to_token(json, c) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
|
||||
pop_state (json);
|
||||
|
||||
if (qse_strxcmp(json->tok.ptr, json->tok.len, QSE_T("null")) == 0) inst = QSE_JSON_INST_NIL;
|
||||
else if (qse_strxcmp(json->tok.ptr, json->tok.len, QSE_T("true")) == 0) inst = QSE_JSON_INST_TRUE;
|
||||
else if (qse_strxcmp(json->tok.ptr, json->tok.len, QSE_T("false")) == 0) inst = QSE_JSON_INST_FALSE;
|
||||
else
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("invalid word value - %.*js"), (int)json->tok.len, json->tok.ptr);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (invoke_data_inst(json, inst) <= -1) return -1;
|
||||
return 0; /* start over */
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static int handle_start_char (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
if (c == '[')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_ARRAY) <= -1) return -1;
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_START_ARRAY, QSE_NULL) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else if (c == '{')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_DIC) <= -1) return -1;
|
||||
json->state_stack->u.id.state = 0;
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_START_DIC, QSE_NULL) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else if (QSE_ISMSPACE(c))
|
||||
{
|
||||
/* do nothing */
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("not starting with [ or { - %jc"), (qse_char_t)c);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
static int handle_char_in_array (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
if (c == ']')
|
||||
{
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_END_ARRAY, QSE_NULL) <= -1) return -1;
|
||||
pop_state (json);
|
||||
return 1;
|
||||
}
|
||||
else if (c == ',')
|
||||
{
|
||||
if (!json->state_stack->u.ia.got_value)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("redundant comma in array - %jc"), (qse_char_t)c);
|
||||
return -1;
|
||||
}
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
return 1;
|
||||
}
|
||||
else if (QSE_ISMSPACE(c))
|
||||
{
|
||||
/* do nothing */
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (json->state_stack->u.ia.got_value)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("comma required in array - %jc"), (qse_char_t)c);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (c == '\"')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_STRING_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 1;
|
||||
}
|
||||
else if (c == '\'')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_CHARACTER_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 1;
|
||||
}
|
||||
/* TOOD: else if (c == '#') HCL radixed number
|
||||
*/
|
||||
else if (QSE_ISMDIGIT(c) || c == '+' || c == '-')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_NUMERIC_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
json->state_stack->u.nv.dotted = 0;
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (QSE_ISMALPHA(c))
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_WORD_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (c == '[')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_ARRAY) <= -1) return -1;
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_START_ARRAY, QSE_NULL) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else if (c == '{')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_DIC) <= -1) return -1;
|
||||
json->state_stack->u.id.state = 0;
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_START_DIC, QSE_NULL) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("wrong character inside array - %jc[%d]"), (qse_char_t)c, (int)c);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int handle_char_in_dic (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
if (c == '}')
|
||||
{
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_END_DIC, QSE_NULL) <= -1) return -1;
|
||||
pop_state (json);
|
||||
return 1;
|
||||
}
|
||||
else if (c == ':')
|
||||
{
|
||||
if (json->state_stack->u.id.state != 1)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("redundant colon in dictionary - %jc"), (qse_char_t)c);
|
||||
return -1;
|
||||
}
|
||||
json->state_stack->u.id.state++;
|
||||
return 1;
|
||||
}
|
||||
else if (c == ',')
|
||||
{
|
||||
if (json->state_stack->u.id.state != 3)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("redundant comma in dicitonary - %jc"), (qse_char_t)c);
|
||||
return -1;
|
||||
}
|
||||
json->state_stack->u.id.state = 0;
|
||||
return 1;
|
||||
}
|
||||
else if (QSE_ISMSPACE(c))
|
||||
{
|
||||
/* do nothing */
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (json->state_stack->u.id.state == 1)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("colon required in dicitonary - %jc"), (qse_char_t)c);
|
||||
return -1;
|
||||
}
|
||||
else if (json->state_stack->u.id.state == 3)
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("comma required in dicitonary - %jc"), (qse_char_t)c);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (c == '\"')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_STRING_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 1;
|
||||
}
|
||||
else if (c == '\'')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_CHARACTER_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 1;
|
||||
}
|
||||
/* TOOD: else if (c == '#') HCL radixed number
|
||||
*/
|
||||
else if (QSE_ISMDIGIT(c) || c == '+' || c == '-')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_NUMERIC_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
json->state_stack->u.nv.dotted = 0;
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (QSE_ISMALPHA(c))
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_WORD_VALUE) <= -1) return -1;
|
||||
clear_token (json);
|
||||
return 0; /* start over */
|
||||
}
|
||||
else if (c == '[')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_ARRAY) <= -1) return -1;
|
||||
json->state_stack->u.ia.got_value = 0;
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_START_ARRAY, QSE_NULL) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else if (c == '{')
|
||||
{
|
||||
if (push_state(json, QSE_JSON_STATE_IN_DIC) <= -1) return -1;
|
||||
json->state_stack->u.id.state = 0;
|
||||
if (json->prim.instcb(json, QSE_JSON_INST_START_DIC, QSE_NULL) <= -1) return -1;
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINVAL, QSE_T("wrong character inside dictionary - %jc[%d]"), (qse_char_t)c, (int)c);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static int handle_char (qse_json_t* json, qse_cint_t c)
|
||||
{
|
||||
int x;
|
||||
|
||||
start_over:
|
||||
if (c == QSE_CHAR_EOF)
|
||||
{
|
||||
if (json->state_stack->state == QSE_JSON_STATE_START)
|
||||
{
|
||||
/* no input data */
|
||||
return 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_json_seterrnum (json, QSE_JSON_EFINIS);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
switch (json->state_stack->state)
|
||||
{
|
||||
case QSE_JSON_STATE_START:
|
||||
x = handle_start_char(json, c);
|
||||
break;
|
||||
|
||||
case QSE_JSON_STATE_IN_ARRAY:
|
||||
x = handle_char_in_array(json, c);
|
||||
break;
|
||||
|
||||
case QSE_JSON_STATE_IN_DIC:
|
||||
x = handle_char_in_dic(json, c);
|
||||
break;
|
||||
|
||||
case QSE_JSON_STATE_IN_WORD_VALUE:
|
||||
x = handle_word_value_char(json, c);
|
||||
break;
|
||||
|
||||
case QSE_JSON_STATE_IN_STRING_VALUE:
|
||||
x = handle_string_value_char(json, c);
|
||||
break;
|
||||
|
||||
case QSE_JSON_STATE_IN_CHARACTER_VALUE:
|
||||
x = handle_character_value_char(json, c);
|
||||
break;
|
||||
|
||||
case QSE_JSON_STATE_IN_NUMERIC_VALUE:
|
||||
x = handle_numeric_value_char(json, c);
|
||||
break;
|
||||
|
||||
default:
|
||||
qse_json_seterrfmt (json, QSE_JSON_EINTERN, QSE_T("internal error - must not be called for state %d"), (int)json->state_stack->state);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (x <= -1) return -1;
|
||||
if (x == 0) goto start_over;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
static int feed_json_data (qse_json_t* json, const qse_mchar_t* data, qse_size_t len, qse_size_t* xlen)
|
||||
{
|
||||
const qse_mchar_t* ptr;
|
||||
const qse_mchar_t* end;
|
||||
|
||||
ptr = data;
|
||||
end = ptr + len;
|
||||
|
||||
while (ptr < end)
|
||||
{
|
||||
qse_cint_t c;
|
||||
|
||||
#if defined(QSE_CHAR_IS_WCHAR)
|
||||
qse_char_t uc;
|
||||
qse_size_t bcslen;
|
||||
qse_size_t n;
|
||||
|
||||
bcslen = end - ptr;
|
||||
n = json->_cmgr->mbtowc(ptr, bcslen, &uc);
|
||||
if (n == 0)
|
||||
{
|
||||
/* invalid sequence */
|
||||
uc = *ptr;
|
||||
n = 1;
|
||||
}
|
||||
else if (n > bcslen)
|
||||
{
|
||||
/* incomplete sequence */
|
||||
*xlen = ptr - data; /* didn't manage to process in full */
|
||||
return 0; /* feed more for incomplete sequence */
|
||||
}
|
||||
|
||||
ptr += n;
|
||||
c = uc;
|
||||
#else
|
||||
c = *ptr++;
|
||||
#endif
|
||||
|
||||
/* handle a single character */
|
||||
if (handle_char(json, c) <= -1) goto oops;
|
||||
}
|
||||
|
||||
*xlen = ptr - data;
|
||||
return 1;
|
||||
|
||||
oops:
|
||||
/* TODO: compute the number of processed bytes so far and return it via a parameter??? */
|
||||
/*printf ("feed oops....\n");*/
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
qse_json_t* qse_json_open (qse_mmgr_t* mmgr, qse_size_t xtnsize, qse_json_prim_t* prim, qse_json_errnum_t* errnum)
|
||||
{
|
||||
qse_json_t* json;
|
||||
|
||||
json = (qse_json_t*)QSE_MMGR_ALLOC(mmgr, QSE_SIZEOF(*json) + xtnsize);
|
||||
if (!json)
|
||||
{
|
||||
if (errnum) *errnum = QSE_JSON_ENOMEM;
|
||||
return QSE_NULL;
|
||||
}
|
||||
|
||||
QSE_MEMSET (json, 0, QSE_SIZEOF(*json) + xtnsize);
|
||||
json->_instsize = QSE_SIZEOF(*json);
|
||||
json->_mmgr = mmgr;
|
||||
json->_cmgr = qse_getdflcmgr();
|
||||
json->prim = *prim;
|
||||
|
||||
json->state_top.state = QSE_JSON_STATE_START;
|
||||
json->state_top.next = QSE_NULL;
|
||||
json->state_stack = &json->state_top;
|
||||
|
||||
return json;
|
||||
}
|
||||
|
||||
void qse_json_close (qse_json_t* json)
|
||||
{
|
||||
pop_all_states (json);
|
||||
if (json->tok.ptr) qse_json_freemem (json, json->tok.ptr);
|
||||
QSE_MMGR_FREE (json->_mmgr, json);
|
||||
}
|
||||
|
||||
int qse_json_setoption (qse_json_t* json, qse_json_option_t id, const void* value)
|
||||
{
|
||||
switch (id)
|
||||
{
|
||||
case QSE_JSON_TRAIT:
|
||||
json->cfg.trait = *(const int*)value;
|
||||
return 0;
|
||||
}
|
||||
|
||||
qse_json_seterrnum (json, QSE_JSON_EINVAL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
int qse_json_getoption (qse_json_t* json, qse_json_option_t id, void* value)
|
||||
{
|
||||
switch (id)
|
||||
{
|
||||
case QSE_JSON_TRAIT:
|
||||
*(int*)value = json->cfg.trait;
|
||||
return 0;
|
||||
};
|
||||
|
||||
qse_json_seterrnum (json, QSE_JSON_EINVAL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
qse_json_errnum_t qse_json_geterrnum (qse_json_t* json)
|
||||
{
|
||||
return json->errnum;
|
||||
}
|
||||
|
||||
const qse_char_t* qse_json_geterrmsg (qse_json_t* json)
|
||||
{
|
||||
const qse_char_t* __errstr[] =
|
||||
{
|
||||
QSE_T("no error"),
|
||||
QSE_T("other error"),
|
||||
QSE_T("not implemented"),
|
||||
QSE_T("subsystem error"),
|
||||
QSE_T("internal error"),
|
||||
|
||||
QSE_T("insufficient memory"),
|
||||
QSE_T("invalid parameter or data"),
|
||||
QSE_T("unexpected end of data")
|
||||
};
|
||||
return (json->errmsg.buf[0] == QSE_T('\0'))? __errstr[json->errnum]: json->errmsg.buf;
|
||||
|
||||
}
|
||||
|
||||
const qse_char_t* qse_json_backuperrmsg (qse_json_t* json)
|
||||
{
|
||||
qse_strxcpy (json->errmsg.backup, QSE_COUNTOF(json->errmsg.backup), qse_json_geterrmsg(json));
|
||||
return json->errmsg.backup;
|
||||
}
|
||||
|
||||
void qse_json_seterrnum (qse_json_t* json, qse_json_errnum_t errnum)
|
||||
{
|
||||
/*if (json->shuterr) return; */
|
||||
json->errnum = errnum;
|
||||
json->errmsg.buf[0] = QSE_T('\0');
|
||||
}
|
||||
|
||||
void qse_json_seterrfmt (qse_json_t* json, qse_json_errnum_t errnum, const qse_char_t* errfmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
json->errnum = errnum;
|
||||
va_start (ap, errfmt);
|
||||
qse_strxvfmt(json->errmsg.buf, QSE_COUNTOF(json->errmsg.buf), errfmt, ap);
|
||||
va_end (ap);
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
void* qse_json_allocmem (qse_json_t* json, qse_size_t size)
|
||||
{
|
||||
void* ptr;
|
||||
|
||||
ptr = QSE_MMGR_ALLOC(qse_json_getmmgr(json), size);
|
||||
if (!ptr) qse_json_seterrnum (json, QSE_JSON_ENOMEM);
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void* qse_json_callocmem (qse_json_t* json, qse_size_t size)
|
||||
{
|
||||
void* ptr;
|
||||
|
||||
ptr = QSE_MMGR_ALLOC(qse_json_getmmgr(json), size);
|
||||
if (!ptr) qse_json_seterrnum (json, QSE_JSON_ENOMEM);
|
||||
else QSE_MEMSET (ptr, 0, size);
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void* qse_json_reallocmem (qse_json_t* json, void* ptr, qse_size_t size)
|
||||
{
|
||||
ptr = QSE_MMGR_REALLOC(qse_json_getmmgr(json), ptr, size);
|
||||
if (!ptr) qse_json_seterrnum (json, QSE_JSON_ENOMEM);
|
||||
return ptr;
|
||||
}
|
||||
|
||||
void qse_json_freemem (qse_json_t* json, void* ptr)
|
||||
{
|
||||
QSE_MMGR_FREE (qse_json_getmmgr(json), ptr);
|
||||
}
|
||||
|
||||
/* ========================================================================= */
|
||||
|
||||
qse_json_state_t qse_json_getstate (qse_json_t* json)
|
||||
{
|
||||
return json->state_stack->state;
|
||||
}
|
||||
|
||||
void qse_json_reset (qse_json_t* json)
|
||||
{
|
||||
/* TODO: reset XXXXXXXXXXXXXXXXXXXXXXXXXXXxxxxx */
|
||||
pop_all_states (json);
|
||||
QSE_ASSERT (json->state_stack == &json->state_top);
|
||||
json->state_stack->state = QSE_JSON_STATE_START;
|
||||
}
|
||||
|
||||
int qse_json_feed (qse_json_t* json, const void* ptr, qse_size_t len, qse_size_t* xlen)
|
||||
{
|
||||
int x;
|
||||
qse_size_t total, ylen;
|
||||
const qse_mchar_t* buf;
|
||||
|
||||
buf = (const qse_mchar_t*)ptr;
|
||||
total = 0;
|
||||
while (total < len)
|
||||
{
|
||||
x = feed_json_data(json, &buf[total], len - total, &ylen);
|
||||
if (x <= -1) return -1;
|
||||
|
||||
total += ylen;
|
||||
if (x == 0) break; /* incomplete sequence encountered */
|
||||
}
|
||||
|
||||
*xlen = total;
|
||||
return 0;
|
||||
}
|
515
lib/xli/read-ini.c
Normal file
515
lib/xli/read-ini.c
Normal file
@ -0,0 +1,515 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#include "xli-prv.h"
|
||||
#include <qse/cmn/chr.h>
|
||||
|
||||
/*
|
||||
* It reads key-value pairs under sections as shown below:
|
||||
*
|
||||
* [SECTION1]
|
||||
* key1 = value1
|
||||
* key2 = value2
|
||||
* [SECTION2]
|
||||
* key1 = value1
|
||||
*
|
||||
* The above can get translated to the native XLI format shown below:
|
||||
*
|
||||
* SECTION1 {
|
||||
* key1 = "value1";
|
||||
* key2 = "value2";
|
||||
* }
|
||||
* SECTION2 {
|
||||
* key1 = "value1";
|
||||
* }
|
||||
*
|
||||
* The ini-format reader doesn't support file inclusion via @include.
|
||||
*/
|
||||
|
||||
enum
|
||||
{
|
||||
TOK_STATUS_SAME_LINE = (1 << 0),
|
||||
TOK_STATUS_UPTO_EOL = (1 << 1)
|
||||
};
|
||||
|
||||
#define GET_CHAR(xli) \
|
||||
do { if (qse_xli_getchar(xli) <= -1) return -1; } while(0)
|
||||
|
||||
#define GET_CHAR_TO(xli,c) \
|
||||
do { \
|
||||
if (qse_xli_getchar(xli) <= -1) return -1; \
|
||||
c = (xli)->rio.last.c; \
|
||||
} while(0)
|
||||
|
||||
#define ADD_TOKEN_CHAR(xli,tok,c) \
|
||||
do { \
|
||||
if (qse_str_ccat((tok)->name,(c)) == (qse_size_t)-1) \
|
||||
{ \
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOMEM, QSE_NULL); \
|
||||
return -1; \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#define ADD_TOKEN_STR(xli,tok,s,l) \
|
||||
do { \
|
||||
if (qse_str_ncat((tok)->name,(s),(l)) == (qse_size_t)-1) \
|
||||
{ \
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOMEM, QSE_NULL); \
|
||||
return -1; \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#define SET_TOKEN_TYPE(xli,tok,code) \
|
||||
do { (tok)->type = (code); } while (0)
|
||||
|
||||
#define MATCH(xli,tok_type) ((xli)->tok.type == (tok_type))
|
||||
#define MATCH(xli,tok_type) ((xli)->tok.type == (tok_type))
|
||||
|
||||
static int skip_spaces (qse_xli_t* xli)
|
||||
{
|
||||
qse_cint_t c = xli->rio.last.c;
|
||||
if (xli->tok_status & TOK_STATUS_SAME_LINE)
|
||||
{
|
||||
while (QSE_ISSPACE(c) && c != QSE_T('\n')) GET_CHAR_TO (xli, c);
|
||||
}
|
||||
else
|
||||
{
|
||||
while (QSE_ISSPACE(c)) GET_CHAR_TO (xli, c);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int skip_comment (qse_xli_t* xli, qse_xli_tok_t* tok)
|
||||
{
|
||||
qse_cint_t c = xli->rio.last.c;
|
||||
|
||||
if (c == QSE_T(';'))
|
||||
{
|
||||
/* skip up to \n */
|
||||
qse_str_clear (tok->name);
|
||||
|
||||
do
|
||||
{
|
||||
GET_CHAR_TO (xli, c);
|
||||
if (c == QSE_T('\n') || c == QSE_CHAR_EOF) break;
|
||||
|
||||
if (xli->opt.trait & QSE_XLI_KEEPTEXT) ADD_TOKEN_CHAR (xli, tok, c);
|
||||
}
|
||||
while (1);
|
||||
|
||||
if ((xli->opt.trait & QSE_XLI_KEEPTEXT) &&
|
||||
qse_xli_inserttext (xli, xli->parlink->list, QSE_NULL, QSE_STR_PTR(tok->name)) == QSE_NULL) return -1;
|
||||
|
||||
GET_CHAR (xli); /* eat the new line letter */
|
||||
return 1; /* comment by ; */
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int get_token_into (qse_xli_t* xli, qse_xli_tok_t* tok)
|
||||
{
|
||||
qse_cint_t c;
|
||||
int n;
|
||||
|
||||
/*retry:*/
|
||||
do
|
||||
{
|
||||
if (skip_spaces (xli) <= -1) return -1;
|
||||
if (!(xli->tok_status & TOK_STATUS_UPTO_EOL))
|
||||
{
|
||||
if ((n = skip_comment (xli, tok)) <= -1) return -1;
|
||||
}
|
||||
else n = 0;
|
||||
}
|
||||
while (n >= 1);
|
||||
|
||||
qse_str_clear (tok->name);
|
||||
tok->loc.file = xli->rio.last.file;
|
||||
tok->loc.line = xli->rio.last.line;
|
||||
tok->loc.colm = xli->rio.last.colm;
|
||||
|
||||
c = xli->rio.last.c;
|
||||
|
||||
if (c == QSE_CHAR_EOF)
|
||||
{
|
||||
ADD_TOKEN_STR (xli, tok, QSE_T("<EOF>"), 5);
|
||||
SET_TOKEN_TYPE (xli, tok, QSE_XLI_TOK_EOF);
|
||||
}
|
||||
else if (xli->tok_status & TOK_STATUS_UPTO_EOL)
|
||||
{
|
||||
qse_size_t xlen = 0;
|
||||
|
||||
SET_TOKEN_TYPE (xli, tok, QSE_XLI_TOK_SQSTR);
|
||||
|
||||
do
|
||||
{
|
||||
if (c == QSE_CHAR_EOF || c == QSE_T(';')) break;
|
||||
if (c == QSE_T('\n'))
|
||||
{
|
||||
GET_CHAR (xli);
|
||||
break;
|
||||
}
|
||||
|
||||
ADD_TOKEN_CHAR (xli, tok, c);
|
||||
if (!QSE_ISSPACE(c)) xlen = QSE_STR_LEN(tok->name);
|
||||
|
||||
GET_CHAR_TO (xli, c);
|
||||
}
|
||||
while (1);
|
||||
|
||||
/* trim away trailing spaces */
|
||||
qse_str_setlen (tok->name, xlen);
|
||||
}
|
||||
else if (c == QSE_T('['))
|
||||
{
|
||||
/* in the ini-styled format, a tag is used as a section name.
|
||||
* but the kinds of allowed charaters are more limited than
|
||||
* a normal tag in the xli format. */
|
||||
SET_TOKEN_TYPE (xli, tok, QSE_XLI_TOK_TAG);
|
||||
|
||||
while (1)
|
||||
{
|
||||
GET_CHAR_TO (xli, c);
|
||||
|
||||
if (c == QSE_CHAR_EOF || c == QSE_T('\n'))
|
||||
{
|
||||
/* the string tag is not closed */
|
||||
qse_xli_seterror (xli, QSE_XLI_ETAGNC, QSE_NULL, &xli->tok.loc);
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (c == QSE_T(']'))
|
||||
{
|
||||
/* terminating quote */
|
||||
GET_CHAR (xli);
|
||||
break;
|
||||
}
|
||||
|
||||
if (!QSE_ISALNUM(c) && c != QSE_T('-') && c != QSE_T('_') && c != QSE_T(':'))
|
||||
{
|
||||
qse_char_t cc = (qse_char_t)c;
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = &cc;
|
||||
ea.len = 1;
|
||||
qse_xli_seterror (xli, QSE_XLI_ETAGCHR, &ea, &tok->loc);
|
||||
return -1;
|
||||
}
|
||||
|
||||
ADD_TOKEN_CHAR (xli, tok, c);
|
||||
}
|
||||
}
|
||||
else if (c == QSE_T('_') || QSE_ISALPHA(c) ||
|
||||
((xli->opt.trait & QSE_XLI_LEADDIGIT) && QSE_ISDIGIT(c)))
|
||||
{
|
||||
int lead_digit = QSE_ISDIGIT(c);
|
||||
int all_digits = 1;
|
||||
|
||||
/* a normal identifier can be composed of wider varieties of
|
||||
* characters than a keyword/directive */
|
||||
while (1)
|
||||
{
|
||||
ADD_TOKEN_CHAR (xli, tok, c);
|
||||
GET_CHAR_TO (xli, c);
|
||||
|
||||
if (c == QSE_T('_') || c == QSE_T('-') ||
|
||||
c == QSE_T(':') || c == QSE_T('*') ||
|
||||
c == QSE_T('/') || QSE_ISALPHA(c))
|
||||
{
|
||||
all_digits = 0;
|
||||
}
|
||||
else if (QSE_ISDIGIT(c))
|
||||
{
|
||||
/* nothing to do */
|
||||
}
|
||||
else break;
|
||||
}
|
||||
|
||||
if (lead_digit && all_digits)
|
||||
{
|
||||
/* if an identifier begins with a digit, it must contain a non-digits character */
|
||||
qse_xli_seterror (xli, QSE_XLI_EIDENT, QSE_STR_XSTR(tok->name), &tok->loc);
|
||||
return -1;
|
||||
}
|
||||
|
||||
SET_TOKEN_TYPE (xli, tok, QSE_XLI_TOK_IDENT);
|
||||
}
|
||||
else if (c == QSE_T('='))
|
||||
{
|
||||
SET_TOKEN_TYPE (xli, tok, QSE_XLI_TOK_EQ);
|
||||
ADD_TOKEN_CHAR (xli, tok, c);
|
||||
GET_CHAR (xli);
|
||||
}
|
||||
else
|
||||
{
|
||||
if ((xli->tok_status & TOK_STATUS_SAME_LINE) && c == QSE_T('\n'))
|
||||
{
|
||||
SET_TOKEN_TYPE (xli, tok, QSE_XLI_TOK_NL);
|
||||
ADD_TOKEN_STR (xli, tok, QSE_T("<NL>"), 4);
|
||||
GET_CHAR (xli);
|
||||
}
|
||||
/* not handled yet */
|
||||
else if (c == QSE_T('\0'))
|
||||
{
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = QSE_T("<NUL>");
|
||||
ea.len = 5;
|
||||
qse_xli_seterror (xli, QSE_XLI_ELXCHR, &ea, &tok->loc);
|
||||
return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_char_t cc = (qse_char_t)c;
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = &cc;
|
||||
ea.len = 1;
|
||||
qse_xli_seterror (xli, QSE_XLI_ELXCHR, &ea, &tok->loc);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int get_token (qse_xli_t* xli)
|
||||
{
|
||||
return get_token_into (xli, &xli->tok);
|
||||
}
|
||||
|
||||
static int read_list (qse_xli_t* xli)
|
||||
{
|
||||
qse_xli_pair_t* pair;
|
||||
qse_cstr_t key;
|
||||
qse_xli_list_t* curlist;
|
||||
|
||||
key.ptr = QSE_NULL;
|
||||
key.len = 0;
|
||||
|
||||
while (1)
|
||||
{
|
||||
if (MATCH(xli, QSE_XLI_TOK_EOF)) break;
|
||||
|
||||
if (MATCH(xli, QSE_XLI_TOK_TAG))
|
||||
{
|
||||
/* insert a pair with an empty list */
|
||||
pair = qse_xli_insertpairwithemptylist (xli, &xli->root->list, QSE_NULL, QSE_STR_PTR(xli->tok.name), QSE_NULL, QSE_NULL);
|
||||
if (pair == QSE_NULL) goto oops;
|
||||
|
||||
curlist = (qse_xli_list_t*)pair->val;
|
||||
xli->parlink->list = curlist;
|
||||
|
||||
while (1)
|
||||
{
|
||||
if (get_token (xli) <= -1) goto oops;
|
||||
|
||||
if (MATCH(xli, QSE_XLI_TOK_EOF)) break;
|
||||
if (MATCH(xli, QSE_XLI_TOK_TAG))
|
||||
{
|
||||
/* switch to a new tag */
|
||||
pair = qse_xli_insertpairwithemptylist (xli, &xli->root->list, QSE_NULL, QSE_STR_PTR(xli->tok.name), QSE_NULL, QSE_NULL);
|
||||
if (pair == QSE_NULL) goto oops;
|
||||
|
||||
curlist = (qse_xli_list_t*)pair->val;
|
||||
xli->parlink->list = curlist;
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!MATCH(xli, QSE_XLI_TOK_IDENT))
|
||||
{
|
||||
qse_xli_seterror (xli, QSE_XLI_EKEY, QSE_STR_XSTR(xli->tok.name), &xli->tok.loc);
|
||||
goto oops;
|
||||
}
|
||||
|
||||
if (xli->opt.trait & QSE_XLI_KEYNODUP)
|
||||
{
|
||||
qse_xli_atom_t* atom;
|
||||
|
||||
/* find any key conflicts in the current scope */
|
||||
/* TODO: optimization. no sequential search */
|
||||
atom = curlist->tail;
|
||||
while (atom)
|
||||
{
|
||||
if (atom->type == QSE_XLI_PAIR &&
|
||||
xli->opt.strcmp(((qse_xli_pair_t*)atom)->key, QSE_STR_PTR(xli->tok.name)) == 0)
|
||||
{
|
||||
qse_xli_seterror (xli, QSE_XLI_EEXIST, QSE_STR_XSTR(xli->tok.name), &xli->tok.loc);
|
||||
goto oops;
|
||||
}
|
||||
|
||||
atom = atom->prev;
|
||||
}
|
||||
}
|
||||
|
||||
QSE_ASSERT (key.ptr == QSE_NULL);
|
||||
key.len = QSE_STR_LEN(xli->tok.name);
|
||||
key.ptr = qse_strdup(QSE_STR_PTR(xli->tok.name), qse_xli_getmmgr(xli));
|
||||
if (key.ptr == QSE_NULL)
|
||||
{
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOMEM, QSE_NULL);
|
||||
goto oops;
|
||||
}
|
||||
|
||||
xli->tok_status |= TOK_STATUS_SAME_LINE;
|
||||
if (get_token (xli) <= -1) goto oops;
|
||||
|
||||
if (!MATCH(xli, QSE_XLI_TOK_EQ))
|
||||
{
|
||||
qse_xli_seterror (xli, QSE_XLI_EEQ, QSE_STR_XSTR(xli->tok.name), &xli->tok.loc);
|
||||
goto oops;
|
||||
}
|
||||
|
||||
/* read the value */
|
||||
xli->tok_status |= TOK_STATUS_UPTO_EOL;
|
||||
if (get_token (xli) <= -1) goto oops;
|
||||
|
||||
xli->tok_status &= ~(TOK_STATUS_SAME_LINE | TOK_STATUS_UPTO_EOL);
|
||||
|
||||
if (MATCH(xli, QSE_XLI_TOK_EOF))
|
||||
{
|
||||
/* empty value */
|
||||
qse_cstr_t empty;
|
||||
|
||||
empty.ptr = QSE_T("");
|
||||
empty.len = 0;
|
||||
|
||||
pair = qse_xli_insertpairwithstr (xli, curlist, QSE_NULL, key.ptr, QSE_NULL, QSE_NULL, &empty, QSE_NULL);
|
||||
|
||||
qse_xli_freemem (xli, key.ptr);
|
||||
key.ptr = QSE_NULL;
|
||||
|
||||
if (pair == QSE_NULL) goto oops;
|
||||
break;
|
||||
}
|
||||
|
||||
if (MATCH(xli, QSE_XLI_TOK_SQSTR))
|
||||
{
|
||||
/* add a new pair with the initial string segment */
|
||||
pair = qse_xli_insertpairwithstr (xli, curlist, QSE_NULL, key.ptr, QSE_NULL, QSE_NULL, QSE_STR_XSTR(xli->tok.name), QSE_NULL);
|
||||
|
||||
qse_xli_freemem (xli, key.ptr);
|
||||
key.ptr = QSE_NULL;
|
||||
|
||||
if (pair == QSE_NULL) goto oops;
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_xli_seterror (xli, QSE_XLI_EVAL, QSE_STR_XSTR(xli->tok.name), &xli->tok.loc);
|
||||
goto oops;
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_xli_seterror (xli, QSE_XLI_ESECTAG, QSE_STR_XSTR(xli->tok.name), &xli->tok.loc);
|
||||
goto oops;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
||||
oops:
|
||||
if (key.ptr) qse_xli_freemem (xli, key.ptr);
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int read_root_list (qse_xli_t* xli)
|
||||
{
|
||||
qse_xli_list_link_t* link;
|
||||
|
||||
link = qse_xli_makelistlink (xli, &xli->root->list);
|
||||
if (!link) return -1;
|
||||
|
||||
if (qse_xli_getchar (xli) <= -1 || get_token (xli) <= -1 || read_list (xli) <= -1)
|
||||
{
|
||||
qse_xli_freelistlink (xli, link);
|
||||
return -1;
|
||||
}
|
||||
|
||||
QSE_ASSERT (link == xli->parlink);
|
||||
qse_xli_freelistlink (xli, link);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qse_xli_readini (qse_xli_t* xli, qse_xli_io_impl_t io)
|
||||
{
|
||||
if (!io)
|
||||
{
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
QSE_MEMSET (&xli->rio, 0, QSE_SIZEOF(xli->rio));
|
||||
xli->rio.impl = io;
|
||||
xli->rio.top.line = 1;
|
||||
xli->rio.top.colm = 1;
|
||||
xli->rio.inp = &xli->rio.top;
|
||||
|
||||
xli->tok_status = 0;
|
||||
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOERR, QSE_NULL);
|
||||
qse_xli_clearrionames (xli);
|
||||
|
||||
QSE_ASSERT (QSE_STR_LEN(xli->dotted_curkey) == 0);
|
||||
|
||||
if (qse_xli_openrstream(xli, xli->rio.inp) <= -1) return -1;
|
||||
/* the input stream is open now */
|
||||
|
||||
if (read_root_list(xli) <= -1) goto oops;
|
||||
|
||||
if (!MATCH(xli, QSE_XLI_TOK_EOF))
|
||||
{
|
||||
qse_xli_seterror (xli, QSE_XLI_ESYNTAX, QSE_NULL, &xli->tok.loc);
|
||||
goto oops;
|
||||
}
|
||||
|
||||
QSE_ASSERT (xli->rio.inp == &xli->rio.top);
|
||||
qse_xli_closeactiverstream (xli);
|
||||
qse_str_clear (xli->tok.name);
|
||||
return 0;
|
||||
|
||||
oops:
|
||||
/* an error occurred and control has reached here
|
||||
* probably, some included files might not have been
|
||||
* closed. close them */
|
||||
while (xli->rio.inp != &xli->rio.top)
|
||||
{
|
||||
qse_xli_io_arg_t* prev;
|
||||
|
||||
/* nothing much to do about a close error */
|
||||
qse_xli_closeactiverstream (xli);
|
||||
|
||||
prev = xli->rio.inp->prev;
|
||||
QSE_ASSERT (xli->rio.inp->name != QSE_NULL);
|
||||
qse_xli_freemem (xli, xli->rio.inp);
|
||||
xli->rio.inp = prev;
|
||||
}
|
||||
|
||||
qse_xli_closeactiverstream (xli);
|
||||
qse_str_clear (xli->tok.name);
|
||||
return -1;
|
||||
}
|
1094
lib/xli/read-json.c
Normal file
1094
lib/xli/read-json.c
Normal file
File diff suppressed because it is too large
Load Diff
1287
lib/xli/read.c
Normal file
1287
lib/xli/read.c
Normal file
File diff suppressed because it is too large
Load Diff
745
lib/xli/std.c
Normal file
745
lib/xli/std.c
Normal file
@ -0,0 +1,745 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#include "xli-prv.h"
|
||||
#include <qse/xli/stdxli.h>
|
||||
#include <qse/cmn/path.h>
|
||||
#include "../cmn/mem-prv.h"
|
||||
|
||||
typedef struct xtn_t
|
||||
{
|
||||
struct
|
||||
{
|
||||
struct
|
||||
{
|
||||
qse_xli_iostd_t* x;
|
||||
|
||||
union
|
||||
{
|
||||
/*
|
||||
nothing to maintain for file here
|
||||
*/
|
||||
|
||||
struct
|
||||
{
|
||||
const qse_char_t* ptr;
|
||||
const qse_char_t* end;
|
||||
} str;
|
||||
} u;
|
||||
} in;
|
||||
|
||||
struct
|
||||
{
|
||||
qse_xli_iostd_t* x;
|
||||
union
|
||||
{
|
||||
struct
|
||||
{
|
||||
qse_sio_t* sio;
|
||||
} file;
|
||||
struct
|
||||
{
|
||||
qse_str_t* buf;
|
||||
} str;
|
||||
} u;
|
||||
} out;
|
||||
} s;
|
||||
|
||||
qse_xli_ecb_t ecb;
|
||||
} xtn_t;
|
||||
|
||||
#if defined(QSE_HAVE_INLINE)
|
||||
static QSE_INLINE xtn_t* GET_XTN(qse_xli_t* xli) { return (xtn_t*)((qse_uint8_t*)qse_xli_getxtn(xli) - QSE_SIZEOF(xtn_t)); }
|
||||
#else
|
||||
#define GET_XTN(xli) ((xtn_t*)((qse_uint8_t*)qse_xli_getxtn(xli) - QSE_SIZEOF(xtn_t)))
|
||||
#endif
|
||||
|
||||
qse_xli_t* qse_xli_openstd (qse_size_t xtnsize, qse_size_t rootxtnsize, qse_xli_errnum_t* errnum)
|
||||
{
|
||||
return qse_xli_openstdwithmmgr (QSE_MMGR_GETDFL(), xtnsize, rootxtnsize, errnum);
|
||||
}
|
||||
|
||||
static void fini_xtn (qse_xli_t* xli)
|
||||
{
|
||||
/* nothing to do */
|
||||
}
|
||||
|
||||
static void clear_xtn (qse_xli_t* xli)
|
||||
{
|
||||
/* nothing to do */
|
||||
}
|
||||
|
||||
qse_xli_t* qse_xli_openstdwithmmgr (qse_mmgr_t* mmgr, qse_size_t xtnsize, qse_size_t rootxtnsize, qse_xli_errnum_t* errnum)
|
||||
{
|
||||
qse_xli_t* xli;
|
||||
xtn_t* xtn;
|
||||
|
||||
/* create an object */
|
||||
xli = qse_xli_open(mmgr, QSE_SIZEOF(xtn_t) + xtnsize, rootxtnsize, errnum);
|
||||
if (xli == QSE_NULL) goto oops;
|
||||
|
||||
xli->_instsize += QSE_SIZEOF(xtn_t);
|
||||
|
||||
/* initialize extension */
|
||||
xtn = GET_XTN(xli);
|
||||
/* the extension area has been cleared in qse_httpd_open().
|
||||
* QSE_MEMSET (xtn, 0, QSE_SIZEOF(*xtn));*/
|
||||
|
||||
xtn->ecb.close = fini_xtn;
|
||||
xtn->ecb.clear = clear_xtn;
|
||||
qse_xli_pushecb (xli, &xtn->ecb);
|
||||
|
||||
return xli;
|
||||
|
||||
oops:
|
||||
if (xli) qse_xli_close (xli);
|
||||
return QSE_NULL;
|
||||
}
|
||||
|
||||
static qse_sio_t* open_sio (qse_xli_t* xli, const qse_char_t* file, int flags)
|
||||
{
|
||||
qse_sio_t* sio;
|
||||
sio = qse_sio_open(qse_xli_getmmgr(xli), 0, file, flags);
|
||||
if (sio == QSE_NULL)
|
||||
{
|
||||
qse_cstr_t errarg;
|
||||
errarg.ptr = (qse_char_t*)file;
|
||||
errarg.len = qse_strlen(file);
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &errarg);
|
||||
}
|
||||
return sio;
|
||||
}
|
||||
|
||||
static qse_cstr_t sio_std_names[] =
|
||||
{
|
||||
{ QSE_T("stdin"), 5 },
|
||||
{ QSE_T("stdout"), 6 },
|
||||
{ QSE_T("stderr"), 6 }
|
||||
};
|
||||
|
||||
static qse_sio_t* open_sio_std (qse_xli_t* xli, qse_sio_std_t std, int flags)
|
||||
{
|
||||
qse_sio_t* sio;
|
||||
sio = qse_sio_openstd (qse_xli_getmmgr(xli), 0, std, flags);
|
||||
if (sio == QSE_NULL) qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &sio_std_names[std]);
|
||||
return sio;
|
||||
}
|
||||
|
||||
static qse_ssize_t sf_in_open (qse_xli_t* xli, qse_xli_io_arg_t* arg, xtn_t* xtn)
|
||||
{
|
||||
if (arg->prev == QSE_NULL)
|
||||
{
|
||||
qse_xli_iostd_t* psin = xtn->s.in.x;
|
||||
|
||||
QSE_ASSERT (arg == &xli->rio.top);
|
||||
|
||||
switch (psin->type)
|
||||
{
|
||||
/* normal source files */
|
||||
case QSE_XLI_IOSTD_FILE:
|
||||
if (psin->u.file.path == QSE_NULL ||
|
||||
(psin->u.file.path[0] == QSE_T('-') &&
|
||||
psin->u.file.path[1] == QSE_T('\0')))
|
||||
{
|
||||
/* no path name or - -> stdin */
|
||||
arg->handle = open_sio_std (xli, QSE_SIO_STDIN, QSE_SIO_READ | QSE_SIO_IGNOREMBWCERR | QSE_SIO_LINEBREAK);
|
||||
}
|
||||
else
|
||||
{
|
||||
arg->handle = open_sio (xli, psin->u.file.path, QSE_SIO_READ | QSE_SIO_IGNOREMBWCERR | QSE_SIO_KEEPPATH);
|
||||
}
|
||||
if (arg->handle == QSE_NULL) return -1;
|
||||
if (psin->u.file.cmgr) qse_sio_setcmgr (arg->handle, psin->u.file.cmgr);
|
||||
|
||||
/* update the object name to something more specific */
|
||||
arg->name = psin->u.file.path;
|
||||
if (arg->name == QSE_NULL) arg->name = sio_std_names[QSE_SIO_STDIN].ptr;
|
||||
|
||||
return 0;
|
||||
|
||||
case QSE_XLI_IOSTD_STR:
|
||||
xtn->s.in.u.str.ptr = psin->u.str.ptr;
|
||||
xtn->s.in.u.str.end = psin->u.str.ptr + psin->u.str.len;
|
||||
return 0;
|
||||
|
||||
default:
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINTERN, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
/* handle the included file - @include */
|
||||
const qse_char_t* path;
|
||||
qse_char_t fbuf[64];
|
||||
qse_char_t* dbuf = QSE_NULL;
|
||||
|
||||
QSE_ASSERT (arg->name != QSE_NULL);
|
||||
|
||||
path = arg->name;
|
||||
if (arg->prev->handle)
|
||||
{
|
||||
const qse_char_t* outer;
|
||||
|
||||
outer = qse_sio_getpath (arg->prev->handle);
|
||||
|
||||
if (outer)
|
||||
{
|
||||
const qse_char_t* base;
|
||||
|
||||
/* i'm being included from another file */
|
||||
base = qse_basename(outer);
|
||||
if (base != outer && arg->name[0] != QSE_T('/'))
|
||||
{
|
||||
qse_size_t tmplen, totlen, dirlen;
|
||||
|
||||
dirlen = base - outer;
|
||||
|
||||
totlen = qse_strlen(arg->name) + dirlen;
|
||||
if (totlen >= QSE_COUNTOF(fbuf))
|
||||
{
|
||||
dbuf = qse_xli_allocmem (
|
||||
xli, QSE_SIZEOF(qse_char_t) * (totlen + 1)
|
||||
);
|
||||
if (dbuf == QSE_NULL) return -1;
|
||||
|
||||
path = dbuf;
|
||||
}
|
||||
else path = fbuf;
|
||||
|
||||
tmplen = qse_strncpy ((qse_char_t*)path, outer, dirlen);
|
||||
qse_strcpy ((qse_char_t*)path + tmplen, arg->name);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
arg->handle = qse_sio_open (
|
||||
qse_xli_getmmgr(xli), 0, path, QSE_SIO_READ | QSE_SIO_IGNOREMBWCERR | QSE_SIO_KEEPPATH
|
||||
);
|
||||
|
||||
if (dbuf) qse_xli_freemem (xli, dbuf);
|
||||
if (arg->handle == QSE_NULL)
|
||||
{
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = (qse_char_t*)arg->name;
|
||||
ea.len = qse_strlen(ea.ptr);
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &ea);
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static qse_ssize_t sf_in_close (
|
||||
qse_xli_t* xli, qse_xli_io_arg_t* arg, xtn_t* xtn)
|
||||
{
|
||||
if (arg->prev == QSE_NULL)
|
||||
{
|
||||
switch (xtn->s.in.x->type)
|
||||
{
|
||||
case QSE_XLI_IOSTD_FILE:
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
qse_sio_close (arg->handle);
|
||||
break;
|
||||
|
||||
case QSE_XLI_IOSTD_STR:
|
||||
/* nothing to close */
|
||||
break;
|
||||
|
||||
default:
|
||||
/* nothing to close */
|
||||
break;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
/* handle the included source file - @include */
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
qse_sio_close (arg->handle);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static qse_ssize_t sf_in_read (
|
||||
qse_xli_t* xli, qse_xli_io_arg_t* arg,
|
||||
qse_char_t* data, qse_size_t size, xtn_t* xtn)
|
||||
{
|
||||
if (arg->prev == QSE_NULL)
|
||||
{
|
||||
qse_ssize_t n;
|
||||
|
||||
switch (xtn->s.in.x->type)
|
||||
{
|
||||
case QSE_XLI_IOSTD_FILE:
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
n = qse_sio_getstrn (arg->handle, data, size);
|
||||
if (n <= -1)
|
||||
{
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = (qse_char_t*)xtn->s.in.x->u.file.path;
|
||||
if (ea.ptr == QSE_NULL) ea.ptr = sio_std_names[QSE_SIO_STDIN].ptr;
|
||||
ea.len = qse_strlen(ea.ptr);
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &ea);
|
||||
}
|
||||
break;
|
||||
|
||||
case QSE_XLI_IOSTD_STR:
|
||||
n = 0;
|
||||
while (n < size && xtn->s.in.u.str.ptr < xtn->s.in.u.str.end)
|
||||
{
|
||||
data[n++] = *xtn->s.in.u.str.ptr++;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
/* this should never happen */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINTERN, QSE_NULL);
|
||||
n = -1;
|
||||
break;
|
||||
}
|
||||
|
||||
return n;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* handle the included source file - @include */
|
||||
qse_ssize_t n;
|
||||
|
||||
QSE_ASSERT (arg->name != QSE_NULL);
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
|
||||
n = qse_sio_getstrn (arg->handle, data, size);
|
||||
if (n <= -1)
|
||||
{
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = (qse_char_t*)arg->name;
|
||||
ea.len = qse_strlen(ea.ptr);
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &ea);
|
||||
}
|
||||
return n;
|
||||
}
|
||||
}
|
||||
|
||||
static qse_ssize_t sf_in (
|
||||
qse_xli_t* xli, qse_xli_io_cmd_t cmd,
|
||||
qse_xli_io_arg_t* arg, qse_char_t* data, qse_size_t size)
|
||||
{
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
QSE_ASSERT (arg != QSE_NULL);
|
||||
|
||||
switch (cmd)
|
||||
{
|
||||
case QSE_XLI_IO_OPEN:
|
||||
return sf_in_open (xli, arg, xtn);
|
||||
|
||||
case QSE_XLI_IO_CLOSE:
|
||||
return sf_in_close (xli, arg, xtn);
|
||||
|
||||
case QSE_XLI_IO_READ:
|
||||
return sf_in_read (xli, arg, data, size, xtn);
|
||||
|
||||
default:
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINTERN, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
static qse_ssize_t sf_out_open (qse_xli_t* xli, qse_xli_io_arg_t* arg, xtn_t* xtn)
|
||||
{
|
||||
if (arg->prev == QSE_NULL)
|
||||
{
|
||||
qse_xli_iostd_t* psout = xtn->s.out.x;
|
||||
|
||||
QSE_ASSERT (arg == &xli->wio.top);
|
||||
|
||||
switch (psout->type)
|
||||
{
|
||||
/* normal source files */
|
||||
case QSE_XLI_IOSTD_FILE:
|
||||
if (psout->u.file.path == QSE_NULL ||
|
||||
(psout->u.file.path[0] == QSE_T('-') &&
|
||||
psout->u.file.path[1] == QSE_T('\0')))
|
||||
{
|
||||
/* no path name or - -> stdout */
|
||||
arg->handle = open_sio_std (xli, QSE_SIO_STDOUT, QSE_SIO_WRITE | QSE_SIO_CREATE | QSE_SIO_TRUNCATE | QSE_SIO_IGNOREMBWCERR | QSE_SIO_LINEBREAK);
|
||||
}
|
||||
else
|
||||
{
|
||||
arg->handle = open_sio (xli, psout->u.file.path, QSE_SIO_WRITE | QSE_SIO_CREATE | QSE_SIO_TRUNCATE | QSE_SIO_IGNOREMBWCERR | QSE_SIO_KEEPPATH);
|
||||
}
|
||||
if (arg->handle == QSE_NULL) return -1;
|
||||
if (psout->u.file.cmgr) qse_sio_setcmgr (arg->handle, psout->u.file.cmgr);
|
||||
|
||||
/* update the object name to something more specific */
|
||||
arg->name = psout->u.file.path;
|
||||
if (arg->name == QSE_NULL) arg->name = sio_std_names[QSE_SIO_STDOUT].ptr;
|
||||
|
||||
return 0;
|
||||
|
||||
case QSE_XLI_IOSTD_STR:
|
||||
xtn->s.out.u.str.buf = qse_str_open(qse_xli_getmmgr(xli), 0, 512);
|
||||
if (xtn->s.out.u.str.buf == QSE_NULL)
|
||||
{
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOMEM, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
default:
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINTERN, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
/* handle the included file - @include */
|
||||
const qse_char_t* path;
|
||||
qse_char_t fbuf[64];
|
||||
qse_char_t* dbuf = QSE_NULL;
|
||||
|
||||
QSE_ASSERT (arg->name != QSE_NULL);
|
||||
|
||||
path = arg->name;
|
||||
if (arg->prev->handle)
|
||||
{
|
||||
const qse_char_t* outer;
|
||||
|
||||
outer = qse_sio_getpath (arg->prev->handle);
|
||||
|
||||
if (outer)
|
||||
{
|
||||
const qse_char_t* base;
|
||||
|
||||
/* i'm being included from another file */
|
||||
base = qse_basename(outer);
|
||||
if (base != outer && arg->name[0] != QSE_T('/'))
|
||||
{
|
||||
qse_size_t tmplen, totlen, dirlen;
|
||||
|
||||
dirlen = base - outer;
|
||||
|
||||
totlen = qse_strlen(arg->name) + dirlen;
|
||||
if (totlen >= QSE_COUNTOF(fbuf))
|
||||
{
|
||||
dbuf = qse_xli_allocmem (
|
||||
xli, QSE_SIZEOF(qse_char_t) * (totlen + 1)
|
||||
);
|
||||
if (dbuf == QSE_NULL) return -1;
|
||||
|
||||
path = dbuf;
|
||||
}
|
||||
else path = fbuf;
|
||||
|
||||
tmplen = qse_strncpy ((qse_char_t*)path, outer, dirlen);
|
||||
qse_strcpy ((qse_char_t*)path + tmplen, arg->name);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
arg->handle = qse_sio_open (
|
||||
qse_xli_getmmgr(xli), 0, path, QSE_SIO_WRITE | QSE_SIO_CREATE | QSE_SIO_TRUNCATE | QSE_SIO_IGNOREMBWCERR | QSE_SIO_KEEPPATH
|
||||
);
|
||||
|
||||
if (dbuf) qse_xli_freemem (xli, dbuf);
|
||||
if (arg->handle == QSE_NULL)
|
||||
{
|
||||
qse_cstr_t ea;
|
||||
/*ea.ptr = (qse_char_t*)arg->name;
|
||||
ea.len = qse_strlen(ea.ptr);*/
|
||||
ea.ptr = (qse_char_t*)path;
|
||||
ea.len = qse_strlen(path);
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &ea);
|
||||
return -1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static qse_ssize_t sf_out_close (
|
||||
qse_xli_t* xli, qse_xli_io_arg_t* arg, xtn_t* xtn)
|
||||
{
|
||||
if (arg->prev == QSE_NULL)
|
||||
{
|
||||
switch (xtn->s.out.x->type)
|
||||
{
|
||||
case QSE_XLI_IOSTD_FILE:
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
qse_sio_close (arg->handle);
|
||||
break;
|
||||
|
||||
case QSE_XLI_IOSTD_STR:
|
||||
/* nothing to close */
|
||||
break;
|
||||
|
||||
default:
|
||||
/* nothing to close */
|
||||
break;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
/* handle the included source file - @include */
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
qse_sio_close (arg->handle);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static qse_ssize_t sf_out_write (
|
||||
qse_xli_t* xli, qse_xli_io_arg_t* arg,
|
||||
qse_char_t* data, qse_size_t size, xtn_t* xtn)
|
||||
{
|
||||
if (arg->prev == QSE_NULL)
|
||||
{
|
||||
qse_ssize_t n;
|
||||
|
||||
switch (xtn->s.out.x->type)
|
||||
{
|
||||
case QSE_XLI_IOSTD_FILE:
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
n = qse_sio_putstrn (arg->handle, data, size);
|
||||
if (n <= -1)
|
||||
{
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = (qse_char_t*)xtn->s.out.x->u.file.path;
|
||||
if (ea.ptr == QSE_NULL) ea.ptr = sio_std_names[QSE_SIO_STDOUT].ptr;
|
||||
ea.len = qse_strlen(ea.ptr);
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &ea);
|
||||
}
|
||||
break;
|
||||
|
||||
case QSE_XLI_IOSTD_STR:
|
||||
if (size > QSE_TYPE_MAX(qse_ssize_t)) size = QSE_TYPE_MAX(qse_ssize_t);
|
||||
if (qse_str_ncat (xtn->s.out.u.str.buf, data, size) == (qse_size_t)-1)
|
||||
{
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOMEM, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
n = size;
|
||||
break;
|
||||
|
||||
default:
|
||||
/* this should never happen */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINTERN, QSE_NULL);
|
||||
n = -1;
|
||||
break;
|
||||
}
|
||||
|
||||
return n;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* handle the included source file - @include */
|
||||
qse_ssize_t n;
|
||||
|
||||
QSE_ASSERT (arg->name != QSE_NULL);
|
||||
QSE_ASSERT (arg->handle != QSE_NULL);
|
||||
|
||||
n = qse_sio_putstrn (arg->handle, data, size);
|
||||
if (n <= -1)
|
||||
{
|
||||
qse_cstr_t ea;
|
||||
ea.ptr = (qse_char_t*)arg->name;
|
||||
ea.len = qse_strlen(ea.ptr);
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOFIL, &ea);
|
||||
}
|
||||
return n;
|
||||
}
|
||||
}
|
||||
|
||||
static qse_ssize_t sf_out (
|
||||
qse_xli_t* xli, qse_xli_io_cmd_t cmd,
|
||||
qse_xli_io_arg_t* arg, qse_char_t* data, qse_size_t size)
|
||||
{
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
QSE_ASSERT (arg != QSE_NULL);
|
||||
|
||||
switch (cmd)
|
||||
{
|
||||
case QSE_XLI_IO_OPEN:
|
||||
return sf_out_open (xli, arg, xtn);
|
||||
|
||||
case QSE_XLI_IO_CLOSE:
|
||||
return sf_out_close (xli, arg, xtn);
|
||||
|
||||
case QSE_XLI_IO_WRITE:
|
||||
return sf_out_write (xli, arg, data, size, xtn);
|
||||
|
||||
default:
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINTERN, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
int qse_xli_readstd (qse_xli_t* xli, qse_xli_iostd_t* in)
|
||||
{
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
if (in == QSE_NULL || (in->type != QSE_XLI_IOSTD_FILE &&
|
||||
in->type != QSE_XLI_IOSTD_STR))
|
||||
{
|
||||
/* the input is a must. at least 1 file or 1 string
|
||||
* must be specified */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
xtn->s.in.x = in;
|
||||
return qse_xli_read(xli, sf_in);
|
||||
}
|
||||
|
||||
|
||||
int qse_xli_readinistd (qse_xli_t* xli, qse_xli_iostd_t* in)
|
||||
{
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
if (in == QSE_NULL || (in->type != QSE_XLI_IOSTD_FILE &&
|
||||
in->type != QSE_XLI_IOSTD_STR))
|
||||
{
|
||||
/* the input is a must. at least 1 file or 1 string
|
||||
* must be specified */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
xtn->s.in.x = in;
|
||||
return qse_xli_readini(xli, sf_in);
|
||||
}
|
||||
|
||||
int qse_xli_readjsonstd (qse_xli_t* xli, qse_xli_iostd_t* in)
|
||||
{
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
if (in == QSE_NULL || (in->type != QSE_XLI_IOSTD_FILE &&
|
||||
in->type != QSE_XLI_IOSTD_STR))
|
||||
{
|
||||
/* the input is a must. at least 1 file or 1 string
|
||||
* must be specified */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
xtn->s.in.x = in;
|
||||
return qse_xli_readjson(xli, sf_in);
|
||||
}
|
||||
|
||||
int qse_xli_writestd (qse_xli_t* xli, qse_xli_list_t* root_list, qse_xli_iostd_t* out)
|
||||
{
|
||||
int n;
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
if (out == QSE_NULL || (out->type != QSE_XLI_IOSTD_FILE &&
|
||||
out->type != QSE_XLI_IOSTD_STR))
|
||||
{
|
||||
/* the input is a must. at least 1 file or 1 string
|
||||
* must be specified */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
xtn->s.out.x = out;
|
||||
n = qse_xli_write (xli, root_list, sf_out);
|
||||
|
||||
if (out->type == QSE_XLI_IOSTD_STR)
|
||||
{
|
||||
if (n >= 0)
|
||||
{
|
||||
QSE_ASSERT (xtn->s.out.u.str.buf != QSE_NULL);
|
||||
qse_str_yield (xtn->s.out.u.str.buf, &out->u.str, 0);
|
||||
}
|
||||
if (xtn->s.out.u.str.buf) qse_str_close (xtn->s.out.u.str.buf);
|
||||
}
|
||||
|
||||
return n;
|
||||
}
|
||||
|
||||
int qse_xli_writeinistd (qse_xli_t* xli, qse_xli_list_t* root_list, qse_xli_iostd_t* out)
|
||||
{
|
||||
int n;
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
if (out == QSE_NULL || (out->type != QSE_XLI_IOSTD_FILE &&
|
||||
out->type != QSE_XLI_IOSTD_STR))
|
||||
{
|
||||
/* the input is a must. at least 1 file or 1 string
|
||||
* must be specified */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
xtn->s.out.x = out;
|
||||
n = qse_xli_writeini (xli, root_list, sf_out);
|
||||
|
||||
if (out->type == QSE_XLI_IOSTD_STR)
|
||||
{
|
||||
if (n >= 0)
|
||||
{
|
||||
QSE_ASSERT (xtn->s.out.u.str.buf != QSE_NULL);
|
||||
qse_str_yield (xtn->s.out.u.str.buf, &out->u.str, 0);
|
||||
}
|
||||
if (xtn->s.out.u.str.buf) qse_str_close (xtn->s.out.u.str.buf);
|
||||
}
|
||||
|
||||
return n;
|
||||
}
|
||||
|
||||
int qse_xli_writejsonstd (qse_xli_t* xli, qse_xli_list_t* root_list, qse_xli_iostd_t* out)
|
||||
{
|
||||
int n;
|
||||
xtn_t* xtn = GET_XTN(xli);
|
||||
|
||||
if (out == QSE_NULL || (out->type != QSE_XLI_IOSTD_FILE &&
|
||||
out->type != QSE_XLI_IOSTD_STR))
|
||||
{
|
||||
/* the input is a must. at least 1 file or 1 string
|
||||
* must be specified */
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
xtn->s.out.x = out;
|
||||
n = qse_xli_writejson (xli, root_list, sf_out);
|
||||
|
||||
if (out->type == QSE_XLI_IOSTD_STR)
|
||||
{
|
||||
if (n >= 0)
|
||||
{
|
||||
QSE_ASSERT (xtn->s.out.u.str.buf != QSE_NULL);
|
||||
qse_str_yield (xtn->s.out.u.str.buf, &out->u.str, 0);
|
||||
}
|
||||
if (xtn->s.out.u.str.buf) qse_str_close (xtn->s.out.u.str.buf);
|
||||
}
|
||||
|
||||
return n;
|
||||
}
|
199
lib/xli/write-ini.c
Normal file
199
lib/xli/write-ini.c
Normal file
@ -0,0 +1,199 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#include "xli-prv.h"
|
||||
|
||||
static int write_to_current_stream (qse_xli_t* xli, const qse_char_t* ptr, qse_size_t len)
|
||||
{
|
||||
qse_xli_io_arg_t* arg;
|
||||
qse_size_t i;
|
||||
|
||||
arg = xli->wio.inp;
|
||||
|
||||
for (i = 0; i < len; i++)
|
||||
{
|
||||
if (arg->b.len + 1 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream (xli, arg) <= -1) return -1;
|
||||
arg->b.buf[arg->b.len++] = ptr[i];
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int write_list (qse_xli_t* xli, qse_xli_list_t* list, int depth)
|
||||
{
|
||||
qse_xli_atom_t* curatom;
|
||||
|
||||
for (curatom = list->head; curatom; curatom = curatom->next)
|
||||
{
|
||||
switch (curatom->type)
|
||||
{
|
||||
case QSE_XLI_PAIR:
|
||||
{
|
||||
qse_xli_pair_t* pair = (qse_xli_pair_t*)curatom;
|
||||
|
||||
if (xli->opt.cbs.pair_writable && !xli->opt.cbs.pair_writable(xli, pair)) continue;
|
||||
|
||||
if (pair->tag)
|
||||
{
|
||||
/* the tag can't be written. so ignore it */
|
||||
/* do nothing */
|
||||
}
|
||||
|
||||
if (depth <= 0)
|
||||
{
|
||||
if (write_to_current_stream (xli, QSE_T("["), 1) <= -1 ||
|
||||
write_to_current_stream (xli, pair->key, qse_strlen(pair->key)) <= -1 ||
|
||||
write_to_current_stream (xli, QSE_T("]\n"), 2) <= -1) return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (write_to_current_stream (xli, pair->key, qse_strlen(pair->key)) <= -1) return -1;
|
||||
}
|
||||
|
||||
if (pair->alias)
|
||||
{
|
||||
/* no alias is supported. so ignore it */
|
||||
/* do nothing */
|
||||
}
|
||||
|
||||
switch (pair->val->type)
|
||||
{
|
||||
case QSE_XLI_NIL:
|
||||
if (write_to_current_stream (xli, QSE_T("\n"), 1) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_TRUE:
|
||||
if (write_to_current_stream (xli, QSE_T("true\n"), 5) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_FALSE:
|
||||
if (write_to_current_stream (xli, QSE_T("false\n"), 6) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_STR:
|
||||
{
|
||||
qse_xli_str_t* str = (qse_xli_str_t*)pair->val;
|
||||
|
||||
if (depth > 0)
|
||||
{
|
||||
/* key = value is not supported at the top level */
|
||||
|
||||
if (write_to_current_stream (xli, QSE_T("="), 1) <= -1) return -1;
|
||||
while (1)
|
||||
{
|
||||
if (str->tag)
|
||||
{
|
||||
/* no string tag is supported. ignore it */
|
||||
/* do nothing */
|
||||
}
|
||||
|
||||
if (write_to_current_stream (xli, str->ptr, str->len) <= -1) return -1;
|
||||
|
||||
#if 0
|
||||
if (!str->next) break;
|
||||
|
||||
if (write_to_current_stream (xli, QSE_T(", "), 2) <= -1) return -1;
|
||||
str = str->next;
|
||||
#else
|
||||
/* no multi-segment string is supported. ignore it */
|
||||
break;
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
if (write_to_current_stream (xli, QSE_T("\n"), 1) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_LIST:
|
||||
if (depth < 1)
|
||||
{
|
||||
if (write_list(xli, (qse_xli_list_t*)pair->val, depth + 1) <= -1 ||
|
||||
write_to_current_stream (xli, QSE_T("\n"), 1) <= -1) return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* the ini format doesn't support deep nesting */
|
||||
if (write_to_current_stream (xli, QSE_T("={}\n"), 4) <= -1) return -1;
|
||||
}
|
||||
break;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_TEXT:
|
||||
{
|
||||
const qse_char_t* str = ((qse_xli_text_t*)curatom)->ptr;
|
||||
|
||||
/* don't honor VERBATIM and DEINDENT flags */
|
||||
if (write_to_current_stream(xli, QSE_T(";"), 1) <= -1 ||
|
||||
write_to_current_stream(xli, str, qse_strlen(str)) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("\n"), 1) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_FILE:
|
||||
/* no file inclusion is supported by the ini-format. ignore it */
|
||||
break;
|
||||
|
||||
case QSE_XLI_EOF:
|
||||
/* no file inclusion is supported by the ini-format. so no EOF. ignore it */
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qse_xli_writeini (qse_xli_t* xli, qse_xli_list_t* root_list, qse_xli_io_impl_t io)
|
||||
{
|
||||
int n;
|
||||
|
||||
if (io == QSE_NULL)
|
||||
{
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
QSE_MEMSET (&xli->wio, 0, QSE_SIZEOF(xli->wio));
|
||||
xli->wio.impl = io;
|
||||
xli->wio.inp = &xli->wio.top;
|
||||
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOERR, QSE_NULL);
|
||||
qse_xli_clearwionames (xli);
|
||||
|
||||
/* open the top level stream */
|
||||
if (qse_xli_openwstream (xli, QSE_NULL, 0) <= -1) return -1;
|
||||
|
||||
/* begin writing the root list */
|
||||
n = write_list (xli, (root_list? root_list: &xli->root->list), 0);
|
||||
|
||||
/* close all open streams. there should be only the
|
||||
* top-level stream here if there occurred no errors */
|
||||
while (xli->wio.inp) qse_xli_closeactivewstream (xli, QSE_NULL);
|
||||
|
||||
return n;
|
||||
}
|
351
lib/xli/write-json.c
Normal file
351
lib/xli/write-json.c
Normal file
@ -0,0 +1,351 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#include "xli-prv.h"
|
||||
#include <qse/cmn/chr.h>
|
||||
|
||||
typedef struct arg_data_t arg_data_t;
|
||||
struct arg_data_t
|
||||
{
|
||||
int org_depth;
|
||||
};
|
||||
|
||||
static int write_to_current_stream(qse_xli_t* xli, const qse_char_t* ptr, qse_size_t len, int escape)
|
||||
{
|
||||
qse_xli_io_arg_t* arg;
|
||||
qse_size_t i;
|
||||
|
||||
arg = xli->wio.inp;
|
||||
|
||||
for (i = 0; i < len; i++)
|
||||
{
|
||||
|
||||
if (escape)
|
||||
{
|
||||
if (ptr[i] == QSE_T('\\') || ptr[i] == QSE_T('\"') ||
|
||||
ptr[i] == QSE_T('\b') || ptr[i] == QSE_T('\f') ||
|
||||
ptr[i] == QSE_T('\n') || ptr[i] == QSE_T('\r') || ptr[i] == QSE_T('\t'))
|
||||
{
|
||||
qse_char_t ac;
|
||||
if (arg->b.len + 2 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream(xli, arg) <= -1) return -1;
|
||||
arg->b.buf[arg->b.len++] = QSE_T('\\');
|
||||
switch (ptr[i])
|
||||
{
|
||||
case QSE_T('\b'): ac = 'b'; break;
|
||||
case QSE_T('\f'): ac = 'f'; break;
|
||||
case QSE_T('\n'): ac = 'n'; break;
|
||||
case QSE_T('\r'): ac = 'r'; break;
|
||||
case QSE_T('\t'): ac = 't'; break;
|
||||
default: ac = ptr[i]; break;
|
||||
}
|
||||
arg->b.buf[arg->b.len++] = ac;
|
||||
}
|
||||
else if (ptr[i] >= 0x00 && ptr[i] <= 0x1F)
|
||||
{
|
||||
qse_char_t tmp[5];
|
||||
if (arg->b.len + 6 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream(xli, arg) <= -1) return -1;
|
||||
arg->b.buf[arg->b.len++] = QSE_T('\\');
|
||||
arg->b.buf[arg->b.len++] = QSE_T('u');
|
||||
/* use 'tmp' to avoid null termination beyond the buffer end */
|
||||
qse_strxfmt(tmp, QSE_COUNTOF(tmp), QSE_T("%04X"), ptr[i]);
|
||||
arg->b.len += qse_strcpy(&arg->b.buf[arg->b.len], tmp);
|
||||
}
|
||||
else
|
||||
{
|
||||
goto unescaped;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
unescaped:
|
||||
if (arg->b.len + 1 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream(xli, arg) <= -1) return -1;
|
||||
arg->b.buf[arg->b.len++] = ptr[i];
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
if (escape && (ptr[i] == QSE_T('\\') || ptr[i] == QSE_T('\"')))
|
||||
{
|
||||
if (arg->b.len + 2 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream(xli, arg) <= -1) return -1;
|
||||
arg->b.buf[arg->b.len++] = QSE_T('\\');
|
||||
}
|
||||
else
|
||||
{
|
||||
if (arg->b.len + 1 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream(xli, arg) <= -1) return -1;
|
||||
}
|
||||
arg->b.buf[arg->b.len++] = ptr[i];
|
||||
*/
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int write_indentation (qse_xli_t* xli, int depth)
|
||||
{
|
||||
static const qse_char_t tabs[16] =
|
||||
{
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t'),
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t'),
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t'),
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t')
|
||||
};
|
||||
|
||||
if (depth <= QSE_COUNTOF(tabs))
|
||||
{
|
||||
if (write_to_current_stream(xli, tabs, depth, 0) <= -1) return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
int i;
|
||||
if (write_to_current_stream(xli, tabs, QSE_COUNTOF(tabs), 0) <= -1) return -1;
|
||||
for (i = QSE_COUNTOF(tabs); i < depth; i++)
|
||||
{
|
||||
if (write_to_current_stream(xli, QSE_T("\t"), 1, 0) <= -1) return -1;
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int need_comma (qse_xli_t* xli, qse_xli_atom_t* start)
|
||||
{
|
||||
qse_xli_atom_t* cur;
|
||||
for (cur = start; cur; cur = cur->next)
|
||||
{
|
||||
if (cur->type == QSE_XLI_PAIR)
|
||||
{
|
||||
if (xli->opt.cbs.pair_writable && !xli->opt.cbs.pair_writable(xli, (qse_xli_pair_t*)cur)) continue;
|
||||
return 1;
|
||||
}
|
||||
if (cur->type == QSE_XLI_EOF) return 0; /* if EOF encountered in the included file, i don't want a comma at the end of the file */
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int write_list (qse_xli_t* xli, qse_xli_list_t* list, int depth)
|
||||
{
|
||||
qse_xli_atom_t* curatom;
|
||||
|
||||
for (curatom = list->head; curatom; curatom = curatom->next)
|
||||
{
|
||||
switch (curatom->type)
|
||||
{
|
||||
case QSE_XLI_PAIR:
|
||||
{
|
||||
qse_xli_pair_t* pair = (qse_xli_pair_t*)curatom;
|
||||
|
||||
if (xli->opt.cbs.pair_writable && !xli->opt.cbs.pair_writable(xli, pair)) continue;
|
||||
|
||||
if (write_indentation(xli, depth) <= -1) return -1;
|
||||
|
||||
if (!(list->flags & QSE_XLI_LIST_ARRAYED))
|
||||
{
|
||||
if (pair->tag)
|
||||
{
|
||||
/* ignore tag in the json format */
|
||||
}
|
||||
|
||||
if (write_to_current_stream(xli, QSE_T("\""), 1, 0) <= -1 ||
|
||||
write_to_current_stream(xli, pair->key, qse_strlen(pair->key), 1) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("\""), 1, 0) <= -1) return -1;
|
||||
|
||||
if (pair->alias)
|
||||
{
|
||||
/* ignore alias in the json format */
|
||||
}
|
||||
|
||||
if (write_to_current_stream(xli, QSE_T(": "), 2, 0) <= -1) return -1;
|
||||
}
|
||||
|
||||
switch (pair->val->type)
|
||||
{
|
||||
case QSE_XLI_NIL:
|
||||
if (write_to_current_stream(xli, QSE_T("nil"), 3, 0) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_TRUE:
|
||||
if (write_to_current_stream(xli, QSE_T("true"), 4, 0) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_FALSE:
|
||||
if (write_to_current_stream(xli, QSE_T("false"), 5, 0) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_STR:
|
||||
{
|
||||
qse_xli_str_t* str = (qse_xli_str_t*)pair->val;
|
||||
|
||||
/* ignore the string tag(str->tag) in the json format.
|
||||
* concatenate multi-segmented string into 1 seperated by a comma */
|
||||
int quote_needed;
|
||||
|
||||
/* if the string value is a non-numeric string or
|
||||
* it is multi-segmented, quoting is needed */
|
||||
quote_needed = !(str->flags & QSE_XLI_STR_NSTR) || str->next;
|
||||
|
||||
if (quote_needed && write_to_current_stream(xli, QSE_T("\""), 1, 0) <= -1) return -1;
|
||||
while (1)
|
||||
{
|
||||
if (write_to_current_stream(xli, str->ptr, str->len, quote_needed) <= -1) return -1;
|
||||
if (!str->next) break;
|
||||
if (write_to_current_stream(xli, QSE_T(","), 1, 0) <= -1) return -1;
|
||||
str = str->next;
|
||||
}
|
||||
|
||||
if (quote_needed && write_to_current_stream(xli, QSE_T("\""), 1, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_LIST:
|
||||
{
|
||||
static qse_char_t* obrac[] =
|
||||
{
|
||||
QSE_T("{\n"),
|
||||
QSE_T("[\n")
|
||||
};
|
||||
static qse_char_t* cbrac[] =
|
||||
{
|
||||
QSE_T("}"),
|
||||
QSE_T("]")
|
||||
};
|
||||
qse_xli_list_t* lv = (qse_xli_list_t*)pair->val;
|
||||
if (write_to_current_stream(xli, obrac[lv->flags & QSE_XLI_LIST_ARRAYED], 2, 0) <= -1 ||
|
||||
write_list (xli, (qse_xli_list_t*)pair->val, depth + 1) <= -1 ||
|
||||
write_indentation (xli, depth) <= -1 ||
|
||||
write_to_current_stream(xli, cbrac[lv->flags & QSE_XLI_LIST_ARRAYED], 1, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (need_comma(xli, curatom->next) && write_to_current_stream(xli, QSE_T(","), 1, 0) <= -1) return -1;
|
||||
if (write_to_current_stream(xli, QSE_T("\n"), 1, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_TEXT:
|
||||
{
|
||||
qse_xli_text_t* ta;
|
||||
const qse_char_t* str;
|
||||
|
||||
ta = (qse_xli_text_t*)curatom;
|
||||
str = ta->ptr;
|
||||
|
||||
if (write_indentation(xli, depth - !!(ta->flags & QSE_XLI_TEXT_DEINDENT)) <= -1) return -1;
|
||||
|
||||
if ((!(ta->flags & QSE_XLI_TEXT_VERBATIM) && write_to_current_stream(xli, QSE_T("#"), 1, 0) <= -1) ||
|
||||
write_to_current_stream(xli, str, qse_strlen(str), 0) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("\n"), 1, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_FILE:
|
||||
{
|
||||
const qse_char_t* path = ((qse_xli_file_t*)curatom)->path;
|
||||
|
||||
if (write_indentation(xli, depth) <= -1) return -1;
|
||||
|
||||
if (write_to_current_stream(xli, QSE_T("@include \""), 10, 0) <= -1 ||
|
||||
write_to_current_stream(xli, path, qse_strlen(path), 1) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("\""), 1, 0) <= -1) return -1;
|
||||
|
||||
if (qse_xli_openwstream (xli, ((qse_xli_file_t*)curatom)->path, depth) <= -1) return -1;
|
||||
depth = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_EOF:
|
||||
if (qse_xli_closeactivewstream(xli, &depth) <= -1) return -1;
|
||||
|
||||
if (need_comma(xli, curatom->next) && write_to_current_stream(xli, QSE_T(","), 1, 0) <= -1) return -1;
|
||||
if (write_to_current_stream(xli, QSE_T("\n"), 1, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static int have_opening_marker (qse_xli_t* xli, qse_xli_list_t* list)
|
||||
{
|
||||
qse_xli_atom_t* curatom;
|
||||
|
||||
for (curatom = list->head; curatom; curatom = curatom->next)
|
||||
{
|
||||
qse_xli_text_t* ta;
|
||||
if (curatom->type != QSE_XLI_TEXT) break;
|
||||
ta = (qse_xli_text_t*)curatom;
|
||||
if (ta->flags & (QSE_XLI_TEXT_ARRAYED_LIST_OPENER | QSE_XLI_TEXT_LIST_OPENER)) return 1;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qse_xli_writejson (qse_xli_t* xli, qse_xli_list_t* root_list, qse_xli_io_impl_t io)
|
||||
{
|
||||
int n, marker;
|
||||
|
||||
if (io == QSE_NULL)
|
||||
{
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
QSE_MEMSET (&xli->wio, 0, QSE_SIZEOF(xli->wio));
|
||||
xli->wio.impl = io;
|
||||
xli->wio.inp = &xli->wio.top;
|
||||
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOERR, QSE_NULL);
|
||||
qse_xli_clearwionames (xli);
|
||||
|
||||
/* open the top level stream */
|
||||
if (qse_xli_openwstream (xli, QSE_NULL, 0) <= -1) return -1;
|
||||
|
||||
if (!root_list) root_list = &xli->root->list;
|
||||
|
||||
marker = have_opening_marker(xli, root_list);
|
||||
if (!marker)
|
||||
{
|
||||
/* if the data has been loaded from a different format like xli or ini,
|
||||
* there are no opening and closing markers. so emit them manually */
|
||||
if (write_to_current_stream(xli, ((root_list->flags & QSE_XLI_LIST_ARRAYED)? QSE_T("[\n"): QSE_T("{\n")), 2, 0) <= -1) return -1;
|
||||
}
|
||||
|
||||
/* begin writing the root list */
|
||||
n = write_list (xli, root_list, 1);
|
||||
|
||||
if (!marker)
|
||||
{
|
||||
if (write_to_current_stream(xli, ((root_list->flags & QSE_XLI_LIST_ARRAYED)? QSE_T("]\n"): QSE_T("}\n")), 2, 0) <= -1) return -1;
|
||||
}
|
||||
|
||||
/* close all open streams. there should be only the
|
||||
* top-level stream here if there occurred no errors */
|
||||
while (xli->wio.inp) qse_xli_closeactivewstream (xli, QSE_NULL);
|
||||
|
||||
return n;
|
||||
}
|
||||
|
||||
|
424
lib/xli/write.c
Normal file
424
lib/xli/write.c
Normal file
@ -0,0 +1,424 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#include "xli-prv.h"
|
||||
#include <qse/cmn/chr.h>
|
||||
|
||||
typedef struct arg_data_t arg_data_t;
|
||||
struct arg_data_t
|
||||
{
|
||||
int org_depth;
|
||||
};
|
||||
|
||||
int qse_xli_flushwstream (qse_xli_t* xli, qse_xli_io_arg_t* arg)
|
||||
{
|
||||
qse_ssize_t n;
|
||||
|
||||
while (arg->b.pos < arg->b.len)
|
||||
{
|
||||
n = xli->wio.impl (xli, QSE_XLI_IO_WRITE, xli->wio.inp, &arg->b.buf[arg->b.pos], arg->b.len - arg->b.pos);
|
||||
if (n <= -1)
|
||||
{
|
||||
if (xli->errnum == QSE_XLI_ENOERR)
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOUSR, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
arg->b.pos += n;
|
||||
}
|
||||
|
||||
arg->b.pos = 0;
|
||||
arg->b.len = 0;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qse_xli_openwstream (qse_xli_t* xli, const qse_char_t* path, int old_depth)
|
||||
{
|
||||
qse_ssize_t n;
|
||||
qse_xli_io_arg_t* arg;
|
||||
|
||||
if (path == QSE_NULL)
|
||||
{
|
||||
/* top-level */
|
||||
arg = &xli->wio.top;
|
||||
}
|
||||
else
|
||||
{
|
||||
qse_link_t* link;
|
||||
qse_size_t plen;
|
||||
arg_data_t* arg_data;
|
||||
|
||||
plen = qse_strlen (path);
|
||||
|
||||
link = (qse_link_t*) qse_xli_callocmem (xli,
|
||||
QSE_SIZEOF(*link) + QSE_SIZEOF(qse_char_t) * (plen + 1));
|
||||
if (link == QSE_NULL) return -1;
|
||||
|
||||
qse_strcpy ((qse_char_t*)(link + 1), path);
|
||||
link->link = xli->wio_names;
|
||||
xli->wio_names = link;
|
||||
|
||||
arg = (qse_xli_io_arg_t*) qse_xli_callocmem (xli, QSE_SIZEOF(*arg) + QSE_SIZEOF(*arg_data));
|
||||
if (arg == QSE_NULL) return -1;
|
||||
|
||||
arg_data = (arg_data_t*)(arg + 1);
|
||||
arg_data->org_depth = old_depth;
|
||||
|
||||
arg->name = (const qse_char_t*)(link + 1);
|
||||
arg->prev = xli->wio.inp;
|
||||
}
|
||||
|
||||
n = xli->wio.impl (xli, QSE_XLI_IO_OPEN, arg, QSE_NULL, 0);
|
||||
if (n <= -1)
|
||||
{
|
||||
if (xli->errnum == QSE_XLI_ENOERR)
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOUSR, QSE_NULL);
|
||||
if (arg != &xli->wio.top)
|
||||
{
|
||||
qse_xli_freemem (xli, arg);
|
||||
/* don't clean up 'link' since it's linked to xli->wio_names */
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
xli->wio.inp = arg;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int qse_xli_closeactivewstream (qse_xli_t* xli, int* org_depth)
|
||||
{
|
||||
qse_ssize_t n;
|
||||
qse_xli_io_arg_t* arg;
|
||||
|
||||
arg = xli->wio.inp;
|
||||
|
||||
qse_xli_flushwstream (xli, arg); /* TODO: do i have to care about the result? */
|
||||
|
||||
n = xli->wio.impl (xli, QSE_XLI_IO_CLOSE, arg, QSE_NULL, 0);
|
||||
if (n <= -1)
|
||||
{
|
||||
if (xli->errnum == QSE_XLI_ENOERR)
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EIOUSR, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
xli->wio.inp = arg->prev;
|
||||
if (arg == &xli->wio.top)
|
||||
{
|
||||
if (org_depth) *org_depth = 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (org_depth) *org_depth = ((arg_data_t*)(arg + 1))->org_depth;
|
||||
qse_xli_freemem (xli, arg);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int write_to_current_stream (qse_xli_t* xli, const qse_char_t* ptr, qse_size_t len, int escape)
|
||||
{
|
||||
qse_xli_io_arg_t* arg;
|
||||
qse_size_t i;
|
||||
|
||||
arg = xli->wio.inp;
|
||||
|
||||
for (i = 0; i < len; i++)
|
||||
{
|
||||
if (escape && (ptr[i] == QSE_T('\\') || ptr[i] == QSE_T('\"')))
|
||||
{
|
||||
if (arg->b.len + 2 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream (xli, arg) <= -1) return -1;
|
||||
arg->b.buf[arg->b.len++] = QSE_T('\\');
|
||||
}
|
||||
else
|
||||
{
|
||||
if (arg->b.len + 1 > QSE_COUNTOF(arg->b.buf) && qse_xli_flushwstream (xli, arg) <= -1) return -1;
|
||||
}
|
||||
arg->b.buf[arg->b.len++] = ptr[i];
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int write_indentation (qse_xli_t* xli, int depth)
|
||||
{
|
||||
static const qse_char_t tabs[16] =
|
||||
{
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t'),
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t'),
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t'),
|
||||
QSE_T('\t'), QSE_T('\t'), QSE_T('\t'), QSE_T('\t')
|
||||
};
|
||||
|
||||
if (depth <= QSE_COUNTOF(tabs))
|
||||
{
|
||||
if (write_to_current_stream (xli, tabs, depth, 0) <= -1) return -1;
|
||||
}
|
||||
else
|
||||
{
|
||||
int i;
|
||||
if (write_to_current_stream (xli, tabs, QSE_COUNTOF(tabs), 0) <= -1) return -1;
|
||||
for (i = QSE_COUNTOF(tabs); i < depth; i++)
|
||||
{
|
||||
if (write_to_current_stream (xli, QSE_T("\t"), 1, 0) <= -1) return -1;
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int key_needs_quoting (qse_xli_t* xli, const qse_char_t* str, int nstr)
|
||||
{
|
||||
/* this determines if a key or an alias requires quoting for output.
|
||||
* NSTR is not taken into account because it's only allowed as a value */
|
||||
|
||||
/* refer to the tokenization rule in get_token_into() in read.c */
|
||||
qse_char_t c;
|
||||
|
||||
c = *str++;
|
||||
if (c == QSE_T('\0')) return 1; /* an empty string requires a quoting */
|
||||
|
||||
if (c == QSE_T('_') || QSE_ISALPHA(c) || (!nstr && (xli->opt.trait & QSE_XLI_LEADDIGIT) && QSE_ISDIGIT(c)))
|
||||
{
|
||||
int lead_digit = QSE_ISDIGIT(c);
|
||||
int all_digits = 1;
|
||||
|
||||
/* a normal identifier can be composed of wider varieties of
|
||||
* characters than a keyword/directive */
|
||||
while (1)
|
||||
{
|
||||
c = *str++;
|
||||
if (c == QSE_T('\0')) break;
|
||||
|
||||
if (c == QSE_T('_') || c == QSE_T('-') || c == QSE_T(':') ||
|
||||
c == QSE_T('*') || c == QSE_T('/') || QSE_ISALPHA(c))
|
||||
{
|
||||
all_digits = 0;
|
||||
}
|
||||
else if (QSE_ISDIGIT(c))
|
||||
{
|
||||
/* nothing to do */
|
||||
}
|
||||
else
|
||||
{
|
||||
/* a disallowed character for an identifier */
|
||||
return 1; /* quote it */
|
||||
}
|
||||
}
|
||||
|
||||
if (lead_digit && all_digits)
|
||||
{
|
||||
/* if an identifier begins with a digit, it must contain a non-digit character */
|
||||
/* in fact, it is not a valid identifer. so quote it */
|
||||
return 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* this must be a normal identifer */
|
||||
return 0; /* no quoting needed */
|
||||
}
|
||||
}
|
||||
else if (nstr && QSE_ISDIGIT(c))
|
||||
{
|
||||
do
|
||||
{
|
||||
c = *str++;
|
||||
if (c == QSE_T('\0')) return 0; /* it's a numeric string */
|
||||
}
|
||||
while (QSE_ISDIGIT(c));
|
||||
}
|
||||
|
||||
/* quote all the rest */
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int write_list (qse_xli_t* xli, qse_xli_list_t* list, int depth)
|
||||
{
|
||||
qse_xli_atom_t* curatom;
|
||||
|
||||
for (curatom = list->head; curatom; curatom = curatom->next)
|
||||
{
|
||||
int quote_key = 0;
|
||||
|
||||
switch (curatom->type)
|
||||
{
|
||||
case QSE_XLI_PAIR:
|
||||
{
|
||||
qse_xli_pair_t* pair = (qse_xli_pair_t*)curatom;
|
||||
|
||||
if (xli->opt.cbs.pair_writable && !xli->opt.cbs.pair_writable(xli, pair)) continue;
|
||||
|
||||
if (write_indentation (xli, depth) <= -1) return -1;
|
||||
|
||||
if (pair->tag)
|
||||
{
|
||||
if (write_to_current_stream (xli, QSE_T("["), 1, 0) <= -1 ||
|
||||
write_to_current_stream (xli, pair->tag, qse_strlen(pair->tag), 0) <= -1 ||
|
||||
write_to_current_stream (xli, QSE_T("]"), 1, 0) <= -1) return -1;
|
||||
}
|
||||
|
||||
if (key_needs_quoting(xli, pair->key, 0)) quote_key = 1;
|
||||
|
||||
if ((quote_key && write_to_current_stream (xli, QSE_T("\""), 1, 0) <= -1) ||
|
||||
write_to_current_stream (xli, pair->key, qse_strlen(pair->key), 0) <= -1 ||
|
||||
(quote_key && write_to_current_stream (xli, QSE_T("\""), 1, 0) <= -1)) return -1;
|
||||
|
||||
if (pair->alias)
|
||||
{
|
||||
if (write_to_current_stream (xli, QSE_T(" \""), 2, 0) <= -1 ||
|
||||
write_to_current_stream (xli, pair->alias, qse_strlen(pair->alias), 1) <= -1 ||
|
||||
write_to_current_stream (xli, QSE_T("\""), 1, 0) <= -1) return -1;
|
||||
}
|
||||
|
||||
switch (pair->val->type)
|
||||
{
|
||||
case QSE_XLI_NIL:
|
||||
if (write_to_current_stream (xli, QSE_T(";\n"), 2, 0) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_TRUE:
|
||||
if (write_to_current_stream (xli, QSE_T(" = true;\n"), 9, 0) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_FALSE:
|
||||
if (write_to_current_stream (xli, QSE_T(" = false;\n"), 10, 0) <= -1) return -1;
|
||||
break;
|
||||
|
||||
case QSE_XLI_STR:
|
||||
{
|
||||
qse_xli_str_t* str = (qse_xli_str_t*)pair->val;
|
||||
|
||||
if (write_to_current_stream (xli, QSE_T(" = "), 3, 0) <= -1) return -1;
|
||||
while (1)
|
||||
{
|
||||
if (str->tag)
|
||||
{
|
||||
if (write_to_current_stream (xli, QSE_T("["), 1, 0) <= -1 ||
|
||||
write_to_current_stream (xli, str->tag, qse_strlen(str->tag), 0) <= -1 ||
|
||||
write_to_current_stream (xli, QSE_T("]"), 1, 0) <= -1) return -1;
|
||||
}
|
||||
|
||||
if ((!(str->flags & QSE_XLI_STR_NSTR) &&
|
||||
write_to_current_stream (xli, QSE_T("\""), 1, 0) <= -1) ||
|
||||
write_to_current_stream (xli, str->ptr, str->len, 1) <= -1 ||
|
||||
(!(str->flags & QSE_XLI_STR_NSTR) &&
|
||||
write_to_current_stream (xli, QSE_T("\""), 1, 0) <= -1)) return -1;
|
||||
if (!str->next) break;
|
||||
|
||||
if (write_to_current_stream (xli, QSE_T(", "), 2, 0) <= -1) return -1;
|
||||
str = str->next;
|
||||
}
|
||||
if (write_to_current_stream (xli, QSE_T(";\n"), 2, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_LIST:
|
||||
{
|
||||
if (write_to_current_stream (xli, QSE_T(" {\n"), 3, 0) <= -1 ||
|
||||
write_list (xli, (qse_xli_list_t*)pair->val, depth + 1) <= -1 ||
|
||||
write_indentation (xli, depth) <= -1 ||
|
||||
write_to_current_stream (xli, QSE_T("}\n"), 2, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_TEXT:
|
||||
{
|
||||
const qse_char_t* str = ((qse_xli_text_t*)curatom)->ptr;
|
||||
|
||||
/* don't honor VERBATIM and DEINDENT flags */
|
||||
if (write_indentation(xli, depth) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("#"), 1, 0) <= -1 ||
|
||||
write_to_current_stream(xli, str, qse_strlen(str), 0) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("\n"), 1, 0) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_FILE:
|
||||
{
|
||||
const qse_char_t* path = ((qse_xli_file_t*)curatom)->path;
|
||||
|
||||
if (write_indentation(xli, depth) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("@include \""), 10, 0) <= -1 ||
|
||||
write_to_current_stream(xli, path, qse_strlen(path), 1) <= -1 ||
|
||||
write_to_current_stream(xli, QSE_T("\";\n"), 3, 0) <= -1) return -1;
|
||||
|
||||
if (qse_xli_openwstream(xli, ((qse_xli_file_t*)curatom)->path, depth) <= -1) return -1;
|
||||
depth = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
case QSE_XLI_EOF:
|
||||
if (qse_xli_closeactivewstream (xli, &depth) <= -1) return -1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void qse_xli_clearwionames (qse_xli_t* xli)
|
||||
{
|
||||
qse_link_t* cur;
|
||||
while (xli->wio_names)
|
||||
{
|
||||
cur = xli->wio_names;
|
||||
xli->wio_names = cur->link;
|
||||
qse_xli_freemem (xli, cur);
|
||||
}
|
||||
}
|
||||
|
||||
int qse_xli_write (qse_xli_t* xli, qse_xli_list_t* root_list, qse_xli_io_impl_t io)
|
||||
{
|
||||
int n;
|
||||
|
||||
if (io == QSE_NULL)
|
||||
{
|
||||
qse_xli_seterrnum (xli, QSE_XLI_EINVAL, QSE_NULL);
|
||||
return -1;
|
||||
}
|
||||
|
||||
QSE_MEMSET (&xli->wio, 0, QSE_SIZEOF(xli->wio));
|
||||
xli->wio.impl = io;
|
||||
xli->wio.inp = &xli->wio.top;
|
||||
|
||||
qse_xli_seterrnum (xli, QSE_XLI_ENOERR, QSE_NULL);
|
||||
qse_xli_clearwionames (xli);
|
||||
|
||||
/* open the top level stream */
|
||||
if (qse_xli_openwstream (xli, QSE_NULL, 0) <= -1) return -1;
|
||||
|
||||
/* begin writing the root list */
|
||||
n = write_list (xli, (root_list? root_list: &xli->root->list), 0);
|
||||
|
||||
/* close all open streams. there should be only the
|
||||
* top-level stream here if there occurred no errors */
|
||||
while (xli->wio.inp) qse_xli_closeactivewstream (xli, QSE_NULL);
|
||||
|
||||
return n;
|
||||
}
|
||||
|
177
lib/xli/xli-prv.h
Normal file
177
lib/xli/xli-prv.h
Normal file
@ -0,0 +1,177 @@
|
||||
/*
|
||||
* $Id$
|
||||
*
|
||||
Copyright (c) 2006-2019 Chung, Hyung-Hwan. All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions
|
||||
are met:
|
||||
1. Redistributions of source code must retain the above copyright
|
||||
notice, this list of conditions and the following disclaimer.
|
||||
2. 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 AUTHOR "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 AUTHOR 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.
|
||||
*/
|
||||
|
||||
#ifndef _QSE_LIB_XLI_XLI_H_
|
||||
#define _QSE_LIB_XLI_XLI_H_
|
||||
|
||||
#include <qse/xli/xli.h>
|
||||
#include <qse/cmn/str.h>
|
||||
#include <qse/cmn/rbt.h>
|
||||
#include "../cmn/mem-prv.h"
|
||||
|
||||
enum qse_xli_tok_type_t
|
||||
{
|
||||
QSE_XLI_TOK_EOF,
|
||||
QSE_XLI_TOK_XINCLUDE,
|
||||
QSE_XLI_TOK_TRUE,
|
||||
QSE_XLI_TOK_FALSE,
|
||||
QSE_XLI_TOK_NIL,
|
||||
QSE_XLI_TOK_COLON,
|
||||
QSE_XLI_TOK_SEMICOLON,
|
||||
QSE_XLI_TOK_LBRACE,
|
||||
QSE_XLI_TOK_RBRACE,
|
||||
QSE_XLI_TOK_LBRACK,
|
||||
QSE_XLI_TOK_RBRACK,
|
||||
QSE_XLI_TOK_LPAREN,
|
||||
QSE_XLI_TOK_RPAREN,
|
||||
QSE_XLI_TOK_EQ,
|
||||
QSE_XLI_TOK_COMMA,
|
||||
QSE_XLI_TOK_DQSTR,
|
||||
QSE_XLI_TOK_SQSTR,
|
||||
QSE_XLI_TOK_NSTR,
|
||||
QSE_XLI_TOK_IDENT,
|
||||
QSE_XLI_TOK_TEXT,
|
||||
QSE_XLI_TOK_TAG,
|
||||
QSE_XLI_TOK_NL
|
||||
};
|
||||
|
||||
typedef enum qse_xli_tok_type_t qse_xli_tok_type_t;
|
||||
|
||||
typedef struct qse_xli_tok_t qse_xli_tok_t;
|
||||
|
||||
struct qse_xli_tok_t
|
||||
{
|
||||
qse_xli_tok_type_t type;
|
||||
qse_str_t* name;
|
||||
qse_xli_loc_t loc;
|
||||
};
|
||||
|
||||
typedef struct qse_xli_list_link_t qse_xli_list_link_t;
|
||||
|
||||
struct qse_xli_list_link_t
|
||||
{
|
||||
qse_xli_list_link_t* next;
|
||||
qse_xli_list_t* list;
|
||||
};
|
||||
|
||||
typedef struct qse_xli_root_list_t qse_xli_root_list_t;
|
||||
|
||||
struct qse_xli_root_list_t
|
||||
{
|
||||
qse_xli_list_t list;
|
||||
qse_xli_nil_t xnil;
|
||||
qse_xli_true_t xtrue;
|
||||
qse_xli_false_t xfalse;
|
||||
qse_mmgr_t* mmgr;
|
||||
};
|
||||
|
||||
struct qse_xli_t
|
||||
{
|
||||
QSE_XLI_HDR;
|
||||
|
||||
qse_xli_errstr_t errstr; /**< error string getter */
|
||||
qse_xli_errnum_t errnum; /**< stores an error number */
|
||||
qse_char_t errmsg[128]; /**< error message holder */
|
||||
qse_xli_loc_t errloc; /**< location of the last error */
|
||||
|
||||
struct
|
||||
{
|
||||
int trait;
|
||||
qse_size_t pair_xtnsize;
|
||||
qse_size_t root_xtnsize;
|
||||
qse_char_t key_splitter; /**< character to use to split a key in the fqpn format */
|
||||
qse_xli_cbs_t cbs;
|
||||
|
||||
int (*strcmp) (const qse_char_t* s1, const qse_char_t* s2);
|
||||
int (*strxcmp) (const qse_char_t* s1, qse_size_t len1, const qse_char_t* s2);
|
||||
} opt;
|
||||
|
||||
qse_xli_ecb_t* ecb;
|
||||
|
||||
qse_xli_root_list_t* root;
|
||||
qse_xli_list_link_t* parlink; /* internal use only - link that points to the list being read currently */
|
||||
|
||||
qse_str_t* dotted_curkey;
|
||||
qse_rbt_t* schema;
|
||||
|
||||
qse_xli_tok_t tok;
|
||||
int tok_status;
|
||||
struct
|
||||
{
|
||||
qse_xli_io_impl_t impl; /* input handler */
|
||||
qse_xli_io_lxc_t last;
|
||||
qse_xli_io_arg_t top; /* for top level */
|
||||
qse_xli_io_arg_t* inp; /* current */
|
||||
} rio;
|
||||
qse_link_t* rio_names;
|
||||
|
||||
struct
|
||||
{
|
||||
qse_xli_io_impl_t impl; /* output handler */
|
||||
qse_xli_io_arg_t top; /* for top level */
|
||||
qse_xli_io_arg_t* inp; /* current */
|
||||
} wio;
|
||||
qse_link_t* wio_names;
|
||||
};
|
||||
|
||||
|
||||
#if defined(__cplusplus)
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
int qse_xli_init (qse_xli_t* xli, qse_mmgr_t* mmgr, qse_size_t rootxtnsize);
|
||||
|
||||
void qse_xli_fini (qse_xli_t* xli);
|
||||
|
||||
const qse_char_t* qse_xli_dflerrstr (
|
||||
const qse_xli_t* xli, qse_xli_errnum_t errnum);
|
||||
|
||||
void qse_xli_clearrionames (qse_xli_t* xli);
|
||||
void qse_xli_clearwionames (qse_xli_t* xli);
|
||||
|
||||
int qse_xli_getchar (qse_xli_t* xli);
|
||||
int qse_xli_openrstream (qse_xli_t* xli, qse_xli_io_arg_t* arg);
|
||||
int qse_xli_closeactiverstream (qse_xli_t* xli);
|
||||
|
||||
int qse_xli_openwstream (qse_xli_t* xli, const qse_char_t* path, int old_depth);
|
||||
int qse_xli_closeactivewstream (qse_xli_t* xli, int* org_depth);
|
||||
int qse_xli_flushwstream (qse_xli_t* xli, qse_xli_io_arg_t* arg);
|
||||
|
||||
qse_xli_list_link_t* qse_xli_makelistlink (qse_xli_t* xli, qse_xli_list_t* parlist);
|
||||
void qse_xli_freelistlink (qse_xli_t* xli, qse_xli_list_link_t* link);
|
||||
|
||||
|
||||
qse_xli_str_t* qse_xli_makestrval (qse_xli_t* xli, const qse_cstr_t* value, const qse_char_t* strtag);
|
||||
qse_xli_list_t* qse_xli_makelistval (qse_xli_t* xli);
|
||||
void qse_xli_freeval (qse_xli_t* xli, qse_xli_val_t* val);
|
||||
|
||||
#if defined(__cplusplus)
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
#endif
|
1264
lib/xli/xli.c
Normal file
1264
lib/xli/xli.c
Normal file
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user